在 OpenXML 中,有着大量的不同的单位,本文记录 Office Open XML (OOXML) 的测量单位,以及提供一个专门用来处理这些单位换算的开源库给大家
咱先来介绍一下在 OpenXML 中的各个单位,在本文最后提供给大家一个我团队开源的 dotnetCampus.OpenXMLUnitConverter 库用来处理单位换算
DXA
在 Office Open XML 默认单位是 dxa 单位,也就是像素点的 20 倍,如 ISO 216 A4 (210x297mm ~ 8.3×11.7in) 的大小可以使用下面代码表示
<w:pgSz w:w="11906" w:h="16838"/>
在页面大小 Page width 和 Page height 和边距 margin 和缩进 tabs 使用的单位都是 DXA 单位
单位计算可以使用下面公式
像素 Points = dxa/20英寸 Inches = Points/72厘米 Centimeters = Inches*2.54
在 OpenXML 因为 dxa 是像素点的 20 倍,所以也叫二十分之一点,另外这里说的像素点是 Point 而不是像素 Pixel 哦
缩写如下
- Points
- Inches
- Centimeters
以 A4 为例
Width = 11906 dxa = 595.3 point = 8.27 in = 21 cm
Half-points
用来表示字体大小的半点,一个点等于两个半点,如表示 12pt 可以这样写
// run properties<w:rPr> // 24 = 12pt <w:sz w:val="24"/></w:rPr>
Fiftieths of a Percent
表示百分比相对值,用于表示表格的宽度和相对宽度,他的值和百分比换算如下
n/100 * 5000
如百分之50可以表示为 50/100 * 5000 pct 的大小,如表格的宽度是百分之50宽度
<w:tbl> <w:tblPr> <!-- 表格宽度是百分之50宽度 --> <w:tblW w:w="2500" w:type="pct"/> </w:tblPr> <w:tblGrid/> <w:tr> <w:tc> <w:p> <w:r> <w:t>Hello, World!</w:t> </w:r> </w:p> </w:tc> </w:tr></w:tbl>
English Metric Unit
这也是最常用的单位,使用 EMUs (English Metric Unit) 用来表示图片和其他元素的宽度,换算如下
1 in = 914400 EMUs1 cm = 360000 EMUs
如用于 w:drawing
绘制,表示绘制画布的宽度 <wp:extent cx="1530350" cy="2142490"/>
用这么大的数是可以提高精度和性能,不需要通过浮点计算
在 PPT 和 Word 等专业软件里面,为了保持精度,就需要减少浮点数的计算。为此设计出的 EMU 单位就是刚好能被英寸和厘米和像素以及 96 之间的转换整除,从而可以实现计算过程中,采用整数计算,实现高精度
关于 EMU 的定义,请看 ECMA 376 的 20.1.2.1 内容
Degree
表示 OpenXML 里面的角度单位,使用 60000.0 表示 360° 的圆
对应角度如下
180° = 10800000 Degree
90° = 5400000 Degree
45° = 2700000 Degree
更多请看 dotnet OpenXML 测量单位的角度和弧度值
千倍百分比
在 OpenXML 的百分比有千倍百分比的方式,使用每1000个单位代表百分之一的值,也就是对应比例是 1 比 100000 的值
代码
测量单位的转换代码请看 C# dontet Office Open XML Unit Converter
开源库
我所在的团队开源了 dotnetCampus.OpenXMLUnitConverter 包含了本文的转换方法
工具获取方法是通过以下命令安装 dotnet 工具
dotnet tool update -g dotnetCampus.OfficeDocumentZipper
启动工具方法是在命令行输入下面代码
OfficeDocumentZipper
另外,在项目使用,可以通过 NuGet 安装 dotnetCampus.OpenXMLUnitConverter 这个库
dotnet add package dotnetCampus.OpenXMLUnitConverter
这个库同时包含使用 SourceYard 打包的源代码 NuGet 包,可以使用下面代码安装
dotnet add package dotnetCampus.OpenXMLUnitConverter.Source
也可以在 csproj 添加下面代码
<PackageReference Include="dotnetCampus.OpenXMLUnitConverter.Source" Version="1.0.2-alpah01"> <PrivateAssets>all</PrivateAssets> <IncludeAssets>runtime; build; native; contentfiles; analyzers</IncludeAssets></PackageReference>
使用方法
可以将某个类型,在知道对应的单位的时候,创建对应的单位的对象,然后采用 ToXxx 的方式转换单位
例如 dotnet OpenXML 形状的 Outline 的 LineWidth 线条轮廓粗细宽度的行为 这一篇用到的转换线条宽度的代码
private static void ReadShape(Shape shape) { // 读取线条宽度的方法 var outline = shape.ShapeProperties?.GetFirstChild<Outline>(); if (outline != null) { var lineWidth = outline.Width; var emu = new Emu(lineWidth); var pixel = emu.ToPixel();
Console.WriteLine($"线条宽度 {pixel.Value}"); } else { // 这形状没有定义轮廓 } }
获取出来对应的属性,如 Outline 的 Width 属性,通过查阅文档了解到这是 EMU 单位的。期望转换为 Pixel 像素,可以使用如下面的代码转换
var lineWidth = outline.Width; var emu = new Emu(lineWidth); var pixel = emu.ToPixel();
用此方法的优势在于方便解决代码逻辑中的各个单位换算问题,比采用 double 或 int 等这些不带单位的类型会更好。详细请看 程序猿修养 给属性一个单位
更多请看 Office 使用 OpenXML SDK 解析文档博客目录

原文链接: http://blog.lindexi.com/post/Office-Open-XML-%E7%9A%84%E6%B5%8B%E9%87%8F%E5%8D%95%E4%BD%8D
本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。 欢迎转载、使用、重新发布,但务必保留文章署名 林德熙 (包含链接: https://blog.lindexi.com ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请与我 联系。