OpenXml Force图像适合父容器

时间:2018-06-01 14:32:43

标签: c# ms-word openxml openxml-sdk

我使用以下代码将一段hmtl解析为word文档

//Need the following packages
//<package id="DocumentFormat.OpenXml" version="2.7.2" targetFramework="net471" />
//<package id = "HtmlToOpenXml.dll" version="2.0.1" targetFramework="net471" />

using System.Linq;
using DocumentFormat.OpenXml;
using DocumentFormat.OpenXml.Packaging;

using DocumentFormat.OpenXml.Wordprocessing;
using HtmlToOpenXml;

namespace ConsoleAppHtmlParse
{
    class Program
    {
        static void Main(string[] args)
        {
            string fileName = @"C:\temp\myDoc.docx";

            using (WordprocessingDocument document = WordprocessingDocument.Create(fileName, WordprocessingDocumentType.Document))
            {
                document.AddMainDocumentPart();
                document.MainDocumentPart.Document = new Document(new Body());
                HtmlConverter conveter = new HtmlConverter(document.MainDocumentPart);

                var compositeElements = conveter.Parse(Html);

                Paragraph p = compositeElements[0] as Paragraph;
                p.ParagraphProperties = new ParagraphProperties();
                p.ParagraphProperties.FrameProperties = new FrameProperties();
                p.ParagraphProperties.FrameProperties.Width = new StringValue("3200");

                document.MainDocumentPart.Document.Body.Append(compositeElements);
            }
        }

        const string Html = "<p>SomeText<img  src=\"\" alt=\"Screenshot_3\" />moretext</p>";
    }
}

上面的代码示例产生如下内容:
   enter image description here
我想&#34;收缩&#34;图像是这样的:
enter image description here

如果我知道&#34;父母&#34;我可以这样做吗?容器大小?

由于

1 个答案:

答案 0 :(得分:5)

了解OpenXml的修改:

w:drawing元素有两个部分来管理图像大小

  1. wp:extent节点,wp:inline的子节点,它确定将包含图像的文档区域的范围。它在这里没有帮助我们。
  2. 注意:这对自由办公室来说至关重要

    extent

    1. spPr节点用于定义包含xfrm节点的形状属性,该节点将变换应用于对象。它定义了偏移量和范围。
    2. 注意:这对OpenXml至关重要 - Google文档等

      inline extent

      我在下面编写了一个程序,对您的代码进行了两处修改。

      1. 定义页面宽度:设置PageSizePageMargin:定义我们需要填充图像的可用区域。在我们的例子中,可用区域可以定义如下:

          

        可用页面宽度 = PageSize.Width - PageMargin.Left - PageMargin.Right

        // Define Constants for Page Width and Page Margin
        private const int PageWidth = 17000;
        private const int PageHeight = 10000;
        private const int PageMarginLeft = 1000;
        private const int PageMarginRight = 1000;
        private const int PageMarginTop = 1000;
        private const int PageMarginBottom = 1000;
        private const double DocumentSizePerPixel = 15;
        private const double EmuPerPixel = 9525;
        
        // Set Page Size and Page Margin so that we can place the image as desired.
        // Available Width = PageWidth - PageMarginLeft - PageMarginRight (= 17000 - 1000 - 1000 = 15000 for default values)
        var sectionProperties = new SectionProperties();
        sectionProperties.AppendChild(new PageSize { Width = PageWidth, Height = PageHeight });
        sectionProperties.AppendChild(new PageMargin { Left = PageMarginLeft, Bottom = PageMarginBottom, Top = PageMarginTop, Right = PageMarginRight });
        document.MainDocumentPart.Document.Body.AppendChild(sectionProperties);
        
      2. 更新图片宽度:使用图片的更新宽度和高度值更新Extents

        1. 计算宽高比,可用于更新图像的高度。

        2. 计算Emu中的新宽度,使用下面的

          来理解单词
            

          Page 15宽度= 1像素宽度= 9525 EMU每像素

               

          可用页面宽度 = 15000页面宽度= 15000/15像素= 1000像素= 1000 * 9525鸸= = 9525000鸸

        3. 使用Extents在两个不同的位置更新网页的宽度和高度,以供Google文档和自由办公室等使用。

                   // Search for Extents used by the word present in Drawing > Inline > Extent
                  var inlineEnumerable = p.ChildElements.Where(e => e is DocumentFormat.OpenXml.Wordprocessing.Run)
                      .Where(r => r.GetFirstChild<Drawing>() != null).Select(r => r.GetFirstChild<Drawing>())
                      .Where(r => r.GetFirstChild<Inline>() != null).Select(r => r.GetFirstChild<Inline>());
          
                  // Update Visible Extent
                  var inlineChildren = inlineEnumerable as Inline[] ?? inlineEnumerable.ToArray();
                  foreach (var inlineChild in inlineChildren)
                  {
                      var inlineElement = inlineChild.Extent;
                      UpdateExtent(inlineElement);
                  }
          
                  // Search for Extents used by the word present in Drawing > Inline > Graphic > GraphicData > Picture > ShapeProperties > Transform2D > Extents
                  var extentsEnumerable = inlineChildren
                      .Where(r => r.GetFirstChild<Graphic>() != null).Select(d => d.GetFirstChild<Graphic>())
                      .Where(r => r.GetFirstChild<GraphicData>() != null).Select(r => r.GetFirstChild<GraphicData>())
                      .Where(r => r.GetFirstChild<DocumentFormat.OpenXml.Drawing.Pictures.Picture>() != null)
                      .Select(r => r.GetFirstChild<DocumentFormat.OpenXml.Drawing.Pictures.Picture>())
                      .Where(r => r.GetFirstChild<DocumentFormat.OpenXml.Drawing.Pictures.ShapeProperties>() != null)
                      .Select(r => r.GetFirstChild<DocumentFormat.OpenXml.Drawing.Pictures.ShapeProperties>())
                      .Where(r => r.GetFirstChild<Transform2D>() != null).Select(r => r.GetFirstChild<Transform2D>())
                      .Where(r => r.GetFirstChild<Extents>() != null).Select(r => r.GetFirstChild<Extents>());
          
                  // Modify all images in Extents to the desired size here, to be stretched out on available page width
                  foreach (var extents in extentsEnumerable)
                  {
                      // Set Image Size: We calculate Aspect Ratio of the image and then calculate the width and update the height as per aspect ratio
                      var inlineElement = extents;
          
                      UpdateExtent(inlineElement);
                  }
          
      3. 更新范围方法以更新范围值:

            private static void UpdateExtent(dynamic inlineElement)
            {
                // Read Default Cx and Cy Values provided in Emu
                var extentCx = inlineElement.Cx;
                var extentCy = inlineElement.Cy;
        
                // Aspect ratio used to set image height after calculation of width
                double aspectRatioOfImage = (double)extentCy / extentCx;
        
                // We know 15 width of Page = 1 width of image in pixel = 9525 EMUs per pixel, and we convert document size to pixel and then to EMU
                // For Default Values Available page width = 15000 page width = 15000/ 15 pixels = 1000 pixels = 1000 * 9525 Emu = 9525000 Emu
                double newExtentCx = EmuPerPixel * ((PageWidth - PageMarginLeft - PageMarginRight) / DocumentSizePerPixel);
                // Maintain the Aspect Ratio for height
                double newExtentCy = aspectRatioOfImage * newExtentCx;
        
                // Update the values
                inlineElement.Cx = (long)Math.Round(newExtentCx);
                inlineElement.Cy = (long)Math.Round(newExtentCy);
            } 
        

        完整计划

        namespace Solutions
        {
            using System;
            using System.Linq;
            using DocumentFormat.OpenXml;
            using DocumentFormat.OpenXml.Drawing;
            using DocumentFormat.OpenXml.Drawing.Wordprocessing;
            using DocumentFormat.OpenXml.Packaging;
            using DocumentFormat.OpenXml.Wordprocessing;
            using HtmlToOpenXml;
        
            using Paragraph = DocumentFormat.OpenXml.Wordprocessing.Paragraph;
        
            public class WordProcessorClass
            {
                const string Html = "<p>SomeText<img src=\"\" alt=\"Screenshot_3\" />moretext</p>";
        
                // Define Constants for Page Width and Page Margin
                private const int PageWidth = 17000;
                private const int PageHeight = 10000;
                private const int PageMarginLeft = 1000;
                private const int PageMarginRight = 1000;
                private const int PageMarginTop = 1000;
                private const int PageMarginBottom = 1000;
                private const double DocumentSizePerPixel = 15;
                private const double EmuPerPixel = 9525;
        
                public static void Main1()
                {
                    string fileName = @"f:\myDoc.docx";
        
                    using (WordprocessingDocument document = WordprocessingDocument.Create(fileName, WordprocessingDocumentType.Document))
                    {
                        document.AddMainDocumentPart();
                        document.MainDocumentPart.Document = new Document(new Body());
        
                        HtmlConverter converter = new HtmlConverter(document.MainDocumentPart);
                        var compositeElements = converter.Parse(Html);
                        var p = compositeElements[0] as Paragraph;
        
        
                        // Set Page Size and Page Margin so that we can place the image as desired.
                        // Available Width = PageWidth - PageMarginLeft - PageMarginRight (= 17000 - 1000 - 1000 = 15000 for default values)
                        var sectionProperties = new SectionProperties();
                        sectionProperties.AppendChild(new PageSize { Width = PageWidth, Height = PageHeight });
                        sectionProperties.AppendChild(new PageMargin { Left = PageMarginLeft, Bottom = PageMarginBottom, Top = PageMarginTop, Right = PageMarginRight });
                        document.MainDocumentPart.Document.Body.AppendChild(sectionProperties);
        
                        if (p != null)
                        {
                            // Search for Extents used by the word present in Drawing > Inline > Extent
                            var inlineEnumerable = p.ChildElements.Where(e => e is DocumentFormat.OpenXml.Wordprocessing.Run)
                                .Where(r => r.GetFirstChild<Drawing>() != null).Select(r => r.GetFirstChild<Drawing>())
                                .Where(r => r.GetFirstChild<Inline>() != null).Select(r => r.GetFirstChild<Inline>());
        
                            // Update Visible Extent
                            var inlineChildren = inlineEnumerable as Inline[] ?? inlineEnumerable.ToArray();
                            foreach (var inlineChild in inlineChildren)
                            {
                                var inlineElement = inlineChild.Extent;
                                UpdateExtent(inlineElement);
                            }
        
                            // Search for Extents used by the word present in Drawing > Inline > Graphic > GraphicData > Picture > ShapeProperties > Transform2D > Extents
                            var extentsEnumerable = inlineChildren
                                .Where(r => r.GetFirstChild<Graphic>() != null).Select(d => d.GetFirstChild<Graphic>())
                                .Where(r => r.GetFirstChild<GraphicData>() != null).Select(r => r.GetFirstChild<GraphicData>())
                                .Where(r => r.GetFirstChild<DocumentFormat.OpenXml.Drawing.Pictures.Picture>() != null)
                                .Select(r => r.GetFirstChild<DocumentFormat.OpenXml.Drawing.Pictures.Picture>())
                                .Where(r => r.GetFirstChild<DocumentFormat.OpenXml.Drawing.Pictures.ShapeProperties>() != null)
                                .Select(r => r.GetFirstChild<DocumentFormat.OpenXml.Drawing.Pictures.ShapeProperties>())
                                .Where(r => r.GetFirstChild<Transform2D>() != null).Select(r => r.GetFirstChild<Transform2D>())
                                .Where(r => r.GetFirstChild<Extents>() != null).Select(r => r.GetFirstChild<Extents>());
        
                            // Modify all images in Extents to the desired size here, to be stretched out on available page width
                            foreach (var extents in extentsEnumerable)
                            {
                                // Set Image Size: We calculate Aspect Ratio of the image and then calculate the width and update the height as per aspect ratio
                                var inlineElement = extents;
        
                                UpdateExtent(inlineElement);
                            }
                        }
        
                        document.MainDocumentPart.Document.Body.Append(compositeElements);
                    }
                }
        
                private static void UpdateExtent(dynamic inlineElement)
                {
                    // Read Default Cx and Cy Values provided in Emu
                    var extentCx = inlineElement.Cx;
                    var extentCy = inlineElement.Cy;
        
                    // Aspect ratio used to set image height after calculation of width
                    double aspectRatioOfImage = (double)extentCy / extentCx;
        
                    // We know 15 width of Page = 1 width of image in pixel = 9525 EMUs per pixel, and we convert document size to pixel and then to EMU
                    // For Default Values Available page width = 15000 page width = 15000/ 15 pixels = 1000 pixels = 1000 * 9525 Emu = 9525000 Emu
                    double newExtentCx = EmuPerPixel * ((PageWidth - PageMarginLeft - PageMarginRight) / DocumentSizePerPixel);
                    // Maintain the Aspect Ratio for height
                    double newExtentCy = aspectRatioOfImage * newExtentCx;
        
                    // Update the values
                    inlineElement.Cx = (long)Math.Round(newExtentCx);
                    inlineElement.Cy = (long)Math.Round(newExtentCy);
                }
            }
        }
        

        <强>输出:

        自由办公室

        Libre Office Document

        Google文档

        Google Document

        参考:Output Document