2

使用 MS OpenXml Sdk,我能够将模板流复制到结果流并使用以下代码在正文末尾附加动态文本(wp>>wr>>wt):

var templateStream = File.OpenRead(templatePath);
templateStream.CopyTo(resultStream);

using (var resultPackage = WordprocessingDocument.Open(resultStream, true))
{
     var document = resultPackage.MainDocumentPart.Document;

     var body = document.Body;

     // Add new text.
     var para = body.AppendChild(new Paragraph());
     var run = para.AppendChild(new Run());
     run.AppendChild(new Text(firstName));
     document.Save();
}

我的下一个合乎逻辑的步骤是用 firstName 替换 resultStream 中文本框的内部文本,如下面的代码所示。

// replacing code in using statement from above
var document = resultPackage.MainDocumentPart.Document;
var textbox = document.Descendants<TextBox>().First();
const string firstNametag = "<<IH.FirstName>>";
if (textbox.InnerText.Contains(firstNametag))
{
   var textboxContent = textbox.Elements<TextBoxContent>().First();
   textboxContent.RemoveAllChildren();
   var paragraph = textboxContent.AppendChild(new Paragraph());
   var run = paragraph.AppendChild(new Run());
   run.AppendChild(new Text(firstName));
}
document.Save();

在第一个示例和一些附加代码中,结果流被适当地序列化为 docx,并且在 Word 中查看时,firstName 被附加到正文的末尾。在第二个示例中,尽管在调试器中进一步检查显示 textboxContent 的子项反映了上面所做的更改,但文本框及其内容保持不变。

我是 OpenXML 开发的新手,所以如果有什么明显的地方请指出。

4

1 回答 1

4

哦,哇,我什至不想认为我了解整个情况,但这里有一个快速的尝试。有更多openXml经验的人请加入...

当您在 docx 上的 word 中创建文本框时,document.xml 文件会得到以下标记:

<w.r>
    <mc.AlertnateContent>
        <mc.Choice Requires="wps">
            <wps:txbx>
                <w:txbxContent>
                    <w:r>
                        <w.t>
                            Text Goes Here
                        </w.t>
                    </w.r>
                </w:txbxContent>
            </wps:txbx>
        </mc.Choice>
        <mc.Fallback>
            <v.textbox>
                <w:txbxContent>
                    <w:r>
                        <w.t>
                            Text Goes Here
                        </w.t>
                    </w.r>
                </w:txbxContent>
            </v.textbox>
        </mc.Fallback>
    </mc.AlertnateContent>
</w.r>

请注意 mc.AlternateContent、mc.Choice 和 mc.Fallback 标记。这些是什么鬼???有人在我遇到的一篇博客文章中这样说 -

“据我了解——但不要把我的话当成福音——AlternateContent 可以出现在任何地方,并提供一种机制,如果消费应用程序可以处理它,则包含增强功能,如果不能处理,则提供回退。” -Tony Jollans - http://social.msdn.microsoft.com/Forums/en-US/worddev/thread/f8a5c277-7049-48c2-a295-199d2914f4ba/

在我的情况下,我只修改了后备文本框(v.txbx 不是 wps.txbx),因为我假设 Resharper 正确地要求我导入 DocumentFormat.OpenXml.Vml 命名空间以依赖于 TextBox 对象。不知道为什么在我已经包含的命名空间 DocumentFormat.OpenXml.Packaging 或 DocumentFormat.OpenXml.Wordprocessing 之一中没有文本框定义,但这超出了这个问题的范围。不用说,在意识到这一点并更新我的代码以寻找两者的共同 w.txbxContent 后,我​​实现了我想做的事情。

这是经过一些重构的更新代码,在原始问题的 using 语句中调用 ReplaceTag 方法并提供模型对象而不是字符串。此外,为方便起见,请使用 tagToValueSelector 字典。

private void ReplaceTags(Document document, SomeModel model)
{
    var textBoxContents = document.Descendants<TextBoxContent>().ToList();
    foreach (var textBoxContent in textBoxContents)
    {
        ReplaceTag(textBoxContent, model);
    }
}

private void ReplaceTag(TextBoxContent textBoxContent, SomeModel model)
{
    var tag = textBoxContent.InnerText.Trim();
    if (!tagsTomValues.ContainsKey(tag)) return;
    var valueSelector = tagsTomValues[tag];
    textBoxContent.RemoveAllChildren();
    var paragraph = textBoxContent.AppendChild(new Paragraph());
    var run = paragraph.AppendChild(new Run());
    run.AppendChild(new Text(valueSelector(model)));
}

// called in the ctor
private static void IntializeTags(IDictionary<string, Func<SomeModel, string>> dictionary)
{
    dictionary.Add("<<IH.Name>>", m => string.Format("{0} {1}", m.FirstName, m.LastName));
}

快乐的 openXmling :)

于 2012-05-03T03:53:00.633 回答