使用 OpenXML 的格式的 Office 文档的元素,使用 p:cNvPr Non-Visual Drawing Properties 的 Id 属性作为标识,在标准协议这个标识是唯一的,但实际很多文档都存在重复的标识。本文告诉大家在使用 Office 2016 版本测试重复 id 的行为
在 OpenXML 的元素的标识一个好的设计是使用 uint 作为标识符,这样做的兼容能力更强。因为可以做到在存在标识符不存在以及重复的时候,自动处理
在 OpenXML 里面的元素在 xml 文档的顺序和元素的序号顺序没有关系,有很多文档的 xml 里面存放的元素的 id 标识的大小和元素所在 xml 的顺序是不同的
在 Office 里,将会尝试给存在重复的标识的元素重新按照在 xml 的顺序给定一个序号作为元素标识
重复 id 标识的规则如下
元素重复选第一个
如有两个元素的标识都是相同的,同时在动画被使用,那么动画将会选择在 xml 第一个读到相同的 id 的元素
<p:cSld>
<p:spTree>
<p:sp>
<p:nvSpPr>
<p:cNvPr id="3" name="林德熙的元素" />
<p:cNvSpPr />
<p:nvPr />
</p:nvSpPr>
<!-- 忽略代码 -->
</p:sp>
<p:sp>
<p:nvSpPr>
<p:cNvPr id="3" name="林德熙的元素" />
<p:cNvSpPr />
<p:nvPr />
</p:nvSpPr>
<!-- 忽略代码 -->
</p:sp>
</p:spTree>
</p:cSld>
<p:timing>
<p:tnLst>
<p:par>
<!-- 忽略代码 -->
<p:tgtEl>
<p:spTgt spid="3" />
</p:tgtEl>
<!-- 忽略代码 -->
</p:par>
</p:tnLst>
</p:timing>
可以看到上面的元素有两个元素使用了 id="3" 同时动画也使用 spid="3" 的元素,在 PPT 选择在 xml 第一个读到的元素
测试课件请点击 元素重复选第一个.pptx 下载
元素不存在按序号
如果有动画等引用的元素的 id 是不存在的,但是在页面里面存在元素的 id 是重复的,如下面代码
<p:cSld>
<p:spTree>
<p:sp>
<p:nvSpPr>
<p:cNvPr id="3" name="林德熙的元素" />
<p:cNvSpPr />
<p:nvPr />
</p:nvSpPr>
<!-- 忽略代码 -->
</p:sp>
<p:sp>
<p:nvSpPr>
<p:cNvPr id="3" name="林德熙的元素" />
<p:cNvSpPr />
<p:nvPr />
</p:nvSpPr>
<!-- 忽略代码 -->
</p:sp>
</p:spTree>
</p:cSld>
<p:timing>
<p:tnLst>
<p:par>
<!-- 忽略代码 -->
<p:tgtEl>
<p:spTgt spid="2" />
</p:tgtEl>
<!-- 忽略代码 -->
</p:par>
</p:tnLst>
</p:timing>
元素 id 序号列表是 1 3 3 4 5 6 但没有动画引用的 spid="2" 的元素
因为元素 id 是 uint 的,可以按照 xml 的顺序,将重复的元素重新给序号作为标识。例如上面的序号里面,就存在两个元素都是重复的 3 作为标识。此时第二个使用 3 作为标识的元素将会被作为无标识元素,再根据按照顺序所缺少的标识重新设置。上面代码缺少了一个 2 的标识,因此就将第二个使用 3 作为标识的元素的标识修改为 2 作为标识
测试课件请点击 元素不存在按序号.pptx 下载
按照这个规则,如果元素的 id 序号列表是 1 3 3 3 5 6 而有两个动画分别应用 spid="2" 和 spid="4" 的元素,按照上面的规则,将按照序号重新给定重复的标识的元素新的值。将会给第二个使用 3 作为标识的元素的标识修改为 2 作为标识,给第三个使用 3 作为标识的元素的标识修改为 4 作为标识
本文的属性是依靠 dotnet OpenXML 解压缩文档为文件夹工具 工具协助测试的,这个工具是开源免费的工具,欢迎小伙伴使用
我搭建了自己的博客 https://blog.lindexi.com/ 欢迎大家访问,里面有很多新的博客。
如果在博客看到有任何不懂的,欢迎交流