0

ユーザーがcontact_nameなどの特定の変数を使用するように設計した単一ページ入力docxテンプレートファイルがあります。OpenXml SDK + Open-Xml-PowerTools を使用した処理中に、このテンプレートに基づいて docx ファイルの多くのインスタンスを作成し、変数を実際の値に置き換えます。最後に 1 つの docx 出力が必要なので、Open-Xml-PowerTools DocumentBuilder を使用して 1 つの docx にマージします。

これは、ユーザーが番号付きリストをテンプレートに入れるまで機能しているように見えました。私の最初の問題は、番号付きリストがマージ後のドキュメント インスタンス間で番号付けを継続することでした。つまり、リストの 2 ページ目の番号は、ドキュメントがすべて同じリスト ID を参照していると考えたため、1-10 ではなく 11-20 でした。

ドキュメント本文で num id が一意であることを確認することでこれを解決することができましたが、最初のページを超えてリストの書式設定が失われています。それらが適切な番号付きリストではないようなページ。スタイルと番号付けのセクションを更新して、これらの一致する新しい数値 ID を取得する必要があるようですが、これを機能させることはできません。

これについて ericwhite.com のフォーラムに投稿しましたが、最新の問題についてはまだ連絡がありません ( http://ericwhite.com/blog/forums/topic/list-numbering-on-merged-docs/ )。

これを修正するための私の最新の試みは、OpenXml-Power-Tools 内で例外をスローしているため、新しいリスト ID で一部のセクションを更新していないと思います。誰もこれを行う方法を知っていますか? 以下の例外を伴う以下のコード試行。

public bool Merge(List<InterchangeableWordProcessingDocument> inputFiles, string outputFilePath)
    {
        if (inputFiles == null)
        {
            logger.LogDebug("No files to merge.");
            return true;
        }
        try
        {

            List<OpenXmlPowerTools.Source> sources = new List<OpenXmlPowerTools.Source>();
            int highestListNumbering = 0;
            int highestAbstractListNumbering = 0;
            foreach (var inputFile in inputFiles)
            {
                //Sometimes merge puts start of next page onto end of previous one so prevent
                //Seems to cause extra blank page when there are labels so don't do on labels pages
                if (inputFile.DocType == DocType.Letter)
                {
                    using (var wordDoc = inputFile.GetAsWordProcessingDocument())
                    {
                        var para = wordDoc.MainDocumentPart.Document.Body.ChildElements.First<Paragraph>();

                        if (para.ParagraphProperties == null)
                        {
                            para.ParagraphProperties = new ParagraphProperties();
                        }

                        para.ParagraphProperties.PageBreakBefore = new PageBreakBefore();

                        //http://ericwhite.com/blog/forums/topic/list-numbering-on-merged-docs/
                        //Numberings should be unique to each page otherwise they continue from the previous
                        //Keep track of how many we have so we can add on to always have a unique number
                        var numIds = wordDoc.MainDocumentPart.Document.Body.Descendants<NumberingId>().ToList();

                        logger.LogDebug("Found " + numIds.Count + " num ids.");

                        foreach (var numId in numIds)
                            numId.Val += highestListNumbering;

                        var styleNumIds = wordDoc.MainDocumentPart.StyleDefinitionsPart.RootElement.Descendants<NumberingId>().ToList();

                        if (wordDoc.MainDocumentPart.StyleDefinitionsPart != null)
                        {

                            logger.LogDebug("Found " + styleNumIds.Count + " stlye num ids.");
                            foreach (var styleNumId in styleNumIds)
                                styleNumId.Val += highestListNumbering;
                        }

                        if (wordDoc.MainDocumentPart.NumberingDefinitionsPart != null)
                        {

                            var numberingNumIds = wordDoc.MainDocumentPart.NumberingDefinitionsPart.RootElement.Descendants<NumberingInstance>().ToList();

                            logger.LogDebug("Found " + numberingNumIds.Count + " numbering num ids.");
                            foreach (var numberingNumId in numberingNumIds)
                            {
                                numberingNumId.NumberID += highestListNumbering;
                                numberingNumId.AbstractNumId.Val += highestAbstractListNumbering;
                            }

                            var abstractNumberingNumIds = wordDoc.MainDocumentPart.NumberingDefinitionsPart.RootElement.Descendants<AbstractNumId>().ToList();

                            logger.LogDebug("Found " + abstractNumberingNumIds.Count + " abstract num ids." + wordDoc.MainDocumentPart.NumberingDefinitionsPart.RootElement.XName.LocalName);
                            foreach (var abstractNumberingNumId in abstractNumberingNumIds)
                                abstractNumberingNumId.Val += highestAbstractListNumbering;

                            //Keep the max nums up to date
                            if (abstractNumberingNumIds.Count > 0)
                                highestAbstractListNumbering = Math.Max(highestAbstractListNumbering, abstractNumberingNumIds.Max(ln => (ln.Val.HasValue ? ln.Val.Value : 0)));

                        }


                        if (numIds.Count > 0)
                            highestListNumbering = Math.Max(highestListNumbering, numIds.Max(ln => (ln.Val.HasValue ? ln.Val.Value : 0)));



                        wordDoc.MainDocumentPart.Document.Save();
                    }
                }
                sources.Add(new OpenXmlPowerTools.Source(inputFile.GetAsWmlDocument(), true));

            }
            DocumentBuilder.BuildDocument(sources, outputFilePath);
            return true;

        }
        catch (SystemException ex)
        {
            logger.LogError("Error occured while generating bereavement letters. ", ex);

            return false;
        }
        finally
        {
            foreach (var inputFile in inputFiles)
            {
                inputFile.Dispose();
            }
        }
    }

例外:

System.InvalidOperationException: Sequence contains no elements
at System.Linq.Enumerable.First[TSource](IEnumerable`1 source)
at OpenXmlPowerTools.DocumentBuilder.CopyNumbering(WordprocessingDocument sourceDocument, WordprocessingDocument newDocument, IEnumerable1 newContent, List1 images)
at OpenXmlPowerTools.DocumentBuilder.AppendDocument(WordprocessingDocument sourceDocument, WordprocessingDocument newDocument, List1 newContent, Boolean keepSection, String insertId, List1 images)
at OpenXmlPowerTools.DocumentBuilder.BuildDocument(List`1 sources, WordprocessingDocument output)
at OpenXmlPowerTools.DocumentBuilder.BuildDocument(List`1 sources, String fileName)
at BereavementMailing.TemplateEngine.Merge(List`1 inputFiles, String outputFilePath) in C:\caw\Underdog\Apps\Services\BereavementMailingEngine\BM_RequestProcessor\TemplateEngine.cs:line 508
4

1 に答える 1

1

各パスで同じ AbstractNumId 参照値を 2 回更新しているようです。代わりに、AbstractNum 定義 ID 値を更新する必要があります。

NumberingPart XML の参照値は次のようになります。

    <w:num w:numId="58">
        <w:abstractNumId w:val="2"/>
    </w:num>

そして、あなたはそれらを2回更新しています。

abstractNumber の定義は次のようになります。

<w:abstractNum w:abstractNumId="0"
               w15:restartNumberingAfterBreak="0">
    <w:nsid w:val="FFFFFF88"/>
    <w:multiLevelType w:val="singleLevel"/>
    <w:tmpl w:val="8EE6963C"/>
    <w:lvl w:ilvl="0">
        <w:start w:val="1"/>
        <w:numFmt w:val="decimal"/>
        <w:pStyle w:val="ListNumber"/>
        <w:lvlText w:val="%1."/>
        <w:lvlJc w:val="left"/>
        <w:pPr>
            <w:tabs>
                <w:tab w:val="num"
                       w:pos="360"/>
            </w:tabs>
            <w:ind w:left="360"
                   w:hanging="360"/>
        </w:pPr>
    </w:lvl>
</w:abstractNum>

このセクションを変更してみてください:

オリジナル

var abstractNumberingNumIds = wordDoc.MainDocumentPart.NumberingDefinitionsPart.RootElement.Descendants<AbstractNumId>().ToList();

logger.LogDebug("Found " + abstractNumberingNumIds.Count + " abstract num ids." + wordDoc.MainDocumentPart.NumberingDefinitionsPart.RootElement.XName.LocalName);
foreach (var abstractNumberingNumId in abstractNumberingNumIds)
abstractNumberingNumId.Val += highestAbstractListNumbering;

//Keep the max nums up to date
if (abstractNumberingNumIds.Count > 0)
    highestAbstractListNumbering = Math.Max(highestAbstractListNumbering, abstractNumberingNumIds.Max(ln => (ln.Val.HasValue ? ln.Val.Value : 0)));

新しい

var abstractNums = wordDoc.MainDocumentPart.NumberingDefinitionsPart.RootElement.Descendants<AbstractNum>().ToList();

logger.LogDebug("Found " + abstractNums.Count + " abstract nums." + wordDoc.MainDocumentPart.NumberingDefinitionsPart.RootElement.XName.LocalName);
foreach (var abstractNum in abstractNums)
    abstractNum.AbstractNumberId += highestAbstractListNumbering;

//Keep the max nums up to date
if (abstractNums.Count > 0)
    highestAbstractListNumbering = Math.Max(highestAbstractListNumbering, abstractNums.Select(a => a.AbstractNumberId).Max(n => n.HasValue ? n.Value : 0));
于 2016-09-22T18:58:37.967 に答える