Как вставить титульный лист в MS Word в начале документа? - PullRequest
0 голосов
/ 06 декабря 2018

Мне нужно вставить титульную страницу в документ Word 2016.Строительный блок является титульным листом и имеет

InsertOptions = (int)WdDocPartInsertOptions.wdInsertPage; //= 2

Пока все хорошо.

Но VSTO может вставлять только так:

buildingBlock.Insert(range);

Он вставляется на местедиапазона.

Событие Application.ActiveDocument.BuildingBlockInsert не вызывается.

При использовании собственной титульной страницы вставки (вкладка вставка -> титульная страница) вставка выполняется правильно (и создается только одна отмена ввода.block).

        // -----------------------------------------------------------------
        // try 1
        var range = Application.ActiveDocument.Range();
        range.Collapse(WdCollapseDirection.wdCollapseStart);
        buildingBlock.Insert(range);
        // result: inserting on the existing first page
        //           one undo entry 
        //           event BuildingBlockInsert has not been raised
        // -----------------------------------------------------------------
        // try 2
        //object start = 0;
        //object end = 0;
        //var range = Application.ActiveDocument.Range(ref start, ref end);
        //buildingBlock.Insert(range);
        // result: inserting on the existing first page
        //           one undo entry 
        //           event BuildingBlockInsert has not been raised
        // -----------------------------------------------------------------
        // try 3
        //var range = Application.ActiveDocument.Range();
        //range.InsertParagraphBefore();
        //var p = Application.ActiveDocument.Paragraphs[1];
        //buildingBlock.Insert(p.Range);
        // result: inserting on the existing first page
        //           two undo entries 
        //           event BuildingBlockInsert has not been raised
        // -----------------------------------------------------------------

Аналогичная проблема описана в примечании: https://docs.microsoft.com/en-us/office/vba/word/concepts/working-with-word/working-with-building-blocks#inserting-a-building-block-into-a-document

Кажется, VSTO игнорирует любые параметры вставки, и нет способа параметризации параметра вставки путем вставки.

Как VSTO может вставить строительный блок в новую первую страницу как собственное действие Word?

Я использую надстройку VS 2017 Word 2016, .Net Framework 4.6.1.

1 Ответ

0 голосов
/ 12 декабря 2018

Сводка:

  1. VSTO просто "подражает" вставке встроенного стандартного блока Word, и не все функции (InsertOptions) и функциональные возможности (событие документа BuildingBlockInsert) связаны.
  2. Объединение встроенного WordПользовательский опыт пользовательского интерфейса (добавление / замена, удаление титульной страницы) и пользовательская надстройка VSTO (добавление / замена титульной страницы) возможны только условно.

Мой текущий код:

private static Microsoft.Office.Interop.Word.Application Application => Globals.ThisAddIn.Application;

private void InsertCoverPage(BuildingBlock buildingBlock)
{
    // validate not null
    if(buildingBlock == null) throw new ArgumentNullException(nameof(buildingBlock));

    // validate is a cover page
    if (buildingBlock.Type.Index != (int)WdBuildingBlockTypes.wdTypeCoverPage &&
        buildingBlock.Type.Index != (int)WdBuildingBlockTypes.wdTypeCustomCoverPage)
    {
        throw new ArgumentNullException(nameof(buildingBlock));
    }

    // validate insert option
    if (buildingBlock.InsertOptions != (int) WdDocPartInsertOptions.wdInsertPage) 
    {
        throw new Exception(
            "building block as a cover page must been inserted in a new page at the beginning of document");
    }

    Application.ScreenUpdating = false;

    Range range = GetCurrentCoverPageRange() ?? Application.ActiveDocument.Range(0, 0);// search a first existing cover page range

    range.InsertBreak(WdBreakType.wdPageBreak); // case existing cover page: replace by page break 
    // range.Start = 0; // = 0
    range.End = 0; // reset only end position
    buildingBlock.Insert(range, true);

    Marshal.ReleaseComObject(range);

    Application.ScreenUpdating = true;
}

private Range GetCurrentCoverPageRange()
{
    Range result = null;

    // Word insert natively a cover page with 2 paragraphs - so we need found these first 2 consecutive paragraphs marked as a cover page
    for (int i = 1; i < Application.ActiveDocument.Paragraphs.Count + 1; i++)
    {
        var paragraph = Application.ActiveDocument.Paragraphs[i];
        var isCoverPage = (bool)paragraph.Range.Information[WdInformation.wdInCoverPage];
        if (isCoverPage)
        {
            if (result == null)
            {
                result = paragraph.Range;
            }
            else
            {
                result.End = paragraph.Range.End;
            }
        }
        else
        {
            if (result != null)
            {
                break;
            }
        }
    }

    return result;
}
...