Delphi word копирование текста

Студворк — интернет-сервис помощи студентам

Здравствуйте, нужна помощь в решении следующей задачи:
Нужно скопировать достаточно большой фрагмент текста из вордовского документа А вордовский документ Б, не потеряв форматирования. В копируемых фрагментах текста присутствуют списки, но нет таблиц.
Должны быть определены хотябы места вставки(т.е. копирование всего документа А в определенное место в документе Б), в идеале конечно хотелось бы определить и места копирования. К примеру:
Найти и копировать из документа А ВСЕ что заключено между тегами [Copy][/Copy] в документ Б в место заключенное между тегами [Paste][/Paste] с заменой тэгов вставки.
Но в реализации я вас не ограничиваю если есть другие варианты с удовольсвием приму и их.
Помогите пожалуйста, заранее спасибо

Добавлено через 11 часов 21 минуту
ап, —…—

Добавлено через 4 минуты
Вот почва для размышлений: код копирует весь документ 1 в определенное место в документе 2, но единственная проблема в том что он полностью заменяет весь текст в документе 2 на весь текст из документа 1.
А нужно чтобы одно ключевое слово из документа 2 заменилось на весь текст документа 1…
Надеюсь вы меня поняли

Pascal
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
Function FindAndReplace(ADocument1, ADocument2:TFileName; Str:string):boolean;
  Var WordDoc: OleVariant;
  const wdReplaceAll=1;
  Begin
   WordDoc:=CreateOLEObject('Word.Application');
   WordDoc.Visible:=false;
   WordDoc.Documents.Add(ADocument1);
   WordDoc.Documents.Add(ADocument2);
   WordDoc.Documents.Item(2).Activate;
  WordDoc.Selection.Find.MatchSoundsLike := False;
  WordDoc.Selection.Find.MatchAllWordForms := False;
  WordDoc.Selection.Find.MatchWholeWord := False;
  WordDoc.Selection.Find.Format := False;
  WordDoc.Selection.Find.Forward := True;
  WordDoc.Selection.Find.ClearFormatting;
  WordDoc.Selection.Find.Text:=Str;
  WordDoc.Selection.Find.Replacement.Text:=WordDoc.Documents.Item(1).Range.Text;
  FindAndReplace:=WordDoc.Selection.Find.Execute(Replace:=wdReplaceAll);
  WordDoc.ActiveDocument.SaveAs(ADocument2);
   WordDoc.Quit;
   WordDoc:=Unassigned;
  End;
procedure TForm1.Button1Click(Sender: TObject);
begin
FindAndReplace('C:/1.doc','C:/2.doc', '#paste');
end;
 
end.

 
O’ShinW ©
 
(2013-01-22 16:19)
[0]

Запущен Ворд W
W := CreateOleObject(«Word.Application»);

Есть документ образец D1
D1 := W.Documents.Open(Pattern);

Есть новый документ D2
D2 := W.Documents.Add;

Хочу в новом документе D2 наделать страниц по образцу D1

  while not(DS.Eof) do
   begin
     Copy1Paste2(W, D1, D2, a, b);
     W.Selection.InsertBreak(OleVariant(7));
     W.Selection.Start := a;
     W.Selection.End := b;
     ReplaceByRecord(W, DS);
     DS.Next;
   end;

procedure TWordExporter.Copy1Paste2(W, D1, D2: Variant; a,b:Integer);
begin
 D1.Activate;
 D1.Range(a, b).Select;
 W.Selection.Copy;
 D2.Activate;
 W.Selection.Paste;
end
;

все работает. Но есть нюанс..
W.Selection.Copy помещает копируемое в системный буфер обмена, и если переключится на любое другое приложение, что-то скопировать в буфер и вставить, то вставится не ожидаемое, а содержимое шаблонного документа D1(равно как и скопированное в другом приложении может быть вставлено в D2).

Пытался писать
COPY_TEMPL := W.Selection;
 но потом в  
COPY_TEMPL.paste;
 получаю ошибку «Объект уничтожен»


 
sniknik ©
 
(2013-01-22 16:26)
[1]

> Пытался писать
> …
нафига??? че то смысл ускользает. напиши комментарии к своим действиям…

> помещает копируемое в системный буфер обмена
ну так, сохрани «системный буфер обмена» до, восстанови после… не?


 
clickmaker ©
 
(2013-01-22 16:33)
[2]

а просто значение из Д1 нельзя присвоить в Д2 что-ли?


 
sniknik ©
 
(2013-01-22 16:36)
[3]

у selection-ов куча методов кроме Copy и Paste вообще то (работать н вариантами а интерфейсами)… например есть Text т.е. —
W2.Selection.Text:= W1.Selection.Text;
должно сработать.


 
O’ShinW ©
 
(2013-01-22 16:38)
[4]


> нафига??? че то смысл ускользает. напиши комментарии к своим
> действиям…

D1 := W.Documents.Open(Pattern);
D1.Range(a, b).Select;
COPY_TEMPL := W.Selection; //один раз, не в буфер взять, а в переменную

и потом
 while not(DS.Eof) do
  begin
    COPY_TEMPL.paste; // только вставлять, не копируя каждый раз
    W.Selection.InsertBreak(OleVariant(7));
    W.Selection.Start := a;
    W.Selection.End := b;
    ReplaceByRecord(W, DS);
    DS.Next;
  end;
———-


> ну так, сохрани «системный буфер обмена» до, восстанови
> после… не?

т.е. имеешь ввиду, аля так:

procedure TWordExporter.Copy1Paste2(W, D1, D2: Variant; a,b:Integer);
begin
D1.Activate;
PUSH_BUFF;
D1.Range(a, b).Select;
W.Selection.Copy;
D2.Activate;
W.Selection.Paste;
POP_BUFF;
end;

?
но могут поместить что-то в буфер м/д этими действиями.


 
O’ShinW ©
 
(2013-01-22 16:42)
[5]


> а просто значение из Д1 нельзя присвоить в Д2 что-ли?

нет

d1 содержит всякое форматирование и разметку, на одном листе
#ACCOUNT#
#NAME#
#INN#                                                 #HOUSE_ID#

это все вставляется на первый лист нового документа
потом #FIELD# меняется по значению записи совпадающего поля DS
только на первом листе (что бы не парсил ворд все листы, т.к. долго. Там до тысячи может быть листов)

в d2 на выходе много листов, оформленных по образцу из d1

>> W2.Selection.Text:= W1.Selection.Text;
тогда разметка сбивается


 
O’ShinW ©
 
(2013-01-22 16:45)
[6]


> но могут поместить что-то в буфер м/д этими действиями.

тогда  хук?
ну ,на изменение буфера..

а нельзя ли как-то все-таки куда то скопировать, не в буфер
а потом из него вставлять?


 
O’ShinW ©
 
(2013-01-22 16:46)
[7]


> у selection-ов куча методов кроме Copy и Paste

смотрел.. Все через буфер идет


 
sniknik ©
 
(2013-01-22 16:48)
[8]

> COPY_TEMPL := W.Selection; //один раз, не в буфер взять, а в переменную
Selection это объект в документе, т.что это не «взять в переменную» содержимое как ты похоже считаешь, а скопировать ссылку на этот объект к себе, «содержимое» где было там и осталось.

> COPY_TEMPL.paste; // только вставлять, не копируя каждый раз
а оно и не копируется каждый раз, если ты этого не сделал. (а бы делаешь см. Copy1Paste2).


 
sniknik ©
 
(2013-01-22 16:50)
[9]

>>> W2.Selection.Text:= W1.Selection.Text;
> тогда разметка сбивается
еще раз
> у selection-ов куча методов кроме Copy и Paste вообще то (работать н вариантами а интерфейсами)… например есть Text т.е. —
не? т.е. пример работает, т.е. правда, почему не посмотреть остальное?


 
sniknik ©
 
(2013-01-22 16:53)
[10]

> смотрел.. Все через буфер идет
врешь. давай возьмем например Text … буфер меняется ???


 
O’ShinW ©
 
(2013-01-22 17:05)
[11]


> врешь. давай возьмем например Text … буфер меняется ??
> ?

нет. Но и не работает, форматирование другое получается.


> Selection это объект в документе,
т.что это не «взять в переменную» содержимое как ты похоже считаешь, а скопировать ссылку на этот объект к себе, «содержимое» где было там и осталось.

понимаю. Я взял только указатель. Я об этом сказал, когда написал что получил ошибку.
Надеялся,что Variant может как-то сам выделится / создаться.


> а оно и не копируется каждый раз, если ты этого не сделал.

Copy1Paste2. — это я так сейчас написал.
Раньше постоянно не копировал заново, копировал один раз.
и любое использование буфера плодило тысячи страниц с содержимым отличным от D1, равным тому ,что скопировали в буфер.


>  у selection-ов куча методов кроме Copy и Paste вообще то

Хорошо, взял я текст. Он вставится. только будет не

           (Логотип(картинка))

Адрес                    
                       Дорогой, ФИО

а

(Логотип(картинка))
Адрес
Дорогой, ФИО


 
clickmaker ©
 
(2013-01-22 17:11)
[12]

так форматирование, вроде как, отдельно можно взять-присвоить.


 
sniknik ©
 
(2013-01-22 17:13)
[13]

> нет. Но и не работает
работает. так как ему положено, это текст. не самообманывайся, называй вещи своими именами.

> Хорошо, взял я текст. Он вставится. только будет не
я тебе не навязываю текст, я его как пример привел. и ты не делай одолжений «хорошо беря» его, тебе очевидно нужно другое … еще раз — там КУЧА методов.


 
sniknik ©
 
(2013-01-22 17:15)
[14]

> так форматирование, вроде как, отдельно можно взять-присвоить.
там не только форматирование, там вообще все что ворд делает можно извне «дергая за веревочки» реализовать… теоретически. т.к. как конкретно это надо разбираться, а это дело пишущего.


 
O’ShinW ©
 
(2013-01-22 17:26)
[15]


> работает. так как ему положено, это текст. не самообманывайся,
>  называй вещи своими именами.

Что так будет работать — я знаю. Поэтому и даже не рассматривал вариант.


> тебе очевидно нужно другое … еще раз — там КУЧА методов

Открыл макрос, набрал Selection(.) посмотрел.
не очень много.
Некоторые уже пробовал (класс писался мной еще год назад, тогда ни к чему н пришел, помнится. Просто сейчас решил дописать по-человечески)


> и ты не делай одолжений

мне совершенно незачем повышать шрифт :)  на кого бы то ни было, тем более на такого уважаемого мной человека, как ты.
Сказал же так, потому чт ты настойчиво предлагал использовать TEXT (хоть и , например)


> так форматирование, вроде как, отдельно можно взять-присвоить.

наверное..
но потом, скорее всего, придется прыгать по абзацам, строкам вручную..
т.е. не получится вроде такого
format := такой, text := такой
и 1:1 по образцу.


 
sniknik ©
 
(2013-01-22 17:34)
[16]

> Открыл макрос, набрал Selection(.) посмотрел.
> не очень много.
не поленился, повторил, посчитал — 159 свойств/методов вперемешку, разделать/считать вложенные от «подобьектов» не стал.
да, мизер совсем.


 
O’ShinW ©
 
(2013-01-22 17:35)
[17]

да..
воот :)

поэтому, я сразу и спросил, как бы мне
>Копировать из одного документа MSWord в другой, без буфера обмена

Например, один из вариантов, который представляется простым, имхо:
Весь selection насильно переписать в другую, не глобальную переменную,
может ей выделить, установить размер подходящий, и просто переместить туда из selection всё.

т.е. тот же
COPY_TEMPL := W.Selection;
но «вручную»
не приравнять указатель, а создать/ наполнить


 
O’ShinW ©
 
(2013-01-22 17:37)
[18]

17 к 14

к 16
159  свойств/методов — это всего.
Там про Copy|PAste и аналогичных — немного


 
sniknik ©
 
(2013-01-22 17:41)
[19]

> Там про Copy|PAste и аналогичных — немного
где логика? ты пишешь о проблеме с ними, хочешь по другому, а ищешь среди аналогичных… хотя искать нужно как раз во всех оставшихся.


 
sniknik ©
 
(2013-01-22 17:44)
[20]

> всех оставшихся
НАПРИМЕР в «Text аналогичных», раз он работает так тебе нужно но без форматирования…


 
clickmaker ©
 
(2013-01-22 17:49)
[21]

S1 := D1.Selection.FormattedText;
D2.Selection.FormattedText:= S1;

не?


 
O’ShinW ©
 
(2013-01-22 18:14)
[22]

намекаешь на FormattedText?
Да, это почти как нужно..

Но, почти. Потому что еще картинки-логотипы не берутся и размеры листа не те, а по-умолчанию.
Картинки можно, конечно, и потом вставить. И размеры задать.

А если первый раз все-таки целиком вставить, как раньше, то и размеры задавать не надо, по размерам первого листа идет.


 
sniknik ©
 
(2013-01-22 18:19)
[23]

> Но, почти.
рассмотрели 2, осталось 157 свойств/методов (155 без 2-х в [0]).


 
O’ShinW ©
 
(2013-01-22 18:20)
[24]


> S1 := D1.Selection.FormattedText;
> D2.Selection.FormattedText:= S1;
>

ну да, почти

  W.Selection.Copy;
  COPY_TEMPL := W.Selection.FormattedText;
  D2 := W.Documents.Add;
  W.Selection.Paste; // для размеров листов
  while not(DS.Eof) do
  begin
     W.ActiveDocument.Range.FormattedText := COPY_TEMPL;
     W.Selection.InsertBreak(OleVariant(7));
и т.п.

В принципе, картинками теперь попадать осталось..
и таблицами
и еще чем-нибудь
но сначала проверить есть ли они..

блин, проще, наверное, буфер обмена залочить..


 
O’ShinW ©
 
(2013-01-22 19:12)
[25]

нет, все-таки надо сделать уж

> sniknik ©


> clickmaker ©

Спасибо!


 
брат Птибурдукова
 
(2013-01-22 19:50)
[26]


> O»ShinW ©   (22.01.13 16:42) [5]
Хе, делал такое же, когда лотусом занимался, только там не ворд был, а эксель. Если мне не изменяет склероз, мы тупо копировали файл шаблона (не через буфер, естественно), а потом копию наполняли данными.


 
O’ShinW ©
 
(2013-01-22 20:42)
[27]


> брат Птибурдукова   (22.01.13 19:50) [26]

У екселя есть хороший метод copyto(destination), в обход буфера


 
O’ShinW ©
 
(2013-01-24 13:31)
[28]

D2.Range(0,0).InsertFile( Pattern );
никакого буфера..


> sniknik ©

и не надо ничего говорить :)
ps
Спасибо!


    msm.ru

    Нравится ресурс?

    Помоги проекту!

    Пожалуйста, выделяйте текст программы тегом [сode=pas] … [/сode]. Для этого используйте кнопку [code=pas] в форме ответа или комбобокс, если нужно вставить код на языке, отличном от Дельфи/Паскаля.


    Следующие вопросы задаются очень часто, подробно разобраны в FAQ и, поэтому, будут безжалостно удаляться:
    1. Преобразовать переменную типа String в тип PChar (PAnsiChar)
    2. Как «свернуть» программу в трей.
    3. Как «скрыться» от Ctrl + Alt + Del (заблокировать их и т.п.)
    4. Как прочитать список файлов, поддиректорий в директории?
    5. Как запустить программу/файл?
    … (продолжение следует) …


    Вопросы, подробно описанные во встроенной справочной системе Delphi, не несут полезной тематической нагрузки, поэтому будут удаляться.
    Запрещается создавать темы с просьбой выполнить какую-то работу за автора темы. Форум является средством общения и общего поиска решения. Вашу работу за Вас никто выполнять не будет.


    Внимание
    Попытки открытия обсуждений реализации вредоносного ПО, включая различные интерпретации спам-ботов, наказывается предупреждением на 30 дней.
    Повторная попытка — 60 дней. Последующие попытки бан.
    Мат в разделе — бан на три месяца…

    >
    Работа с WORD, копирование с форматом

    • Подписаться на тему
    • Сообщить другу
    • Скачать/распечатать тему



    Сообщ.
    #1

    ,
    19.04.12, 10:20

      Member

      **

      Рейтинг (т): 2

      есть, скажем, 3 документа word с картинками, таблицами и тд, в общем с активным использованием оформлений.
      нужно эти три документа объединить в 1. т.е скопировать первый, вставить в новый. скопировать второй, вставить в конец нового..
      вся проблема в том, что копирует он просто текст. т.е без оформления, без стилей, без таблиц.
      причём, как-то раньше не замечал, но оказывается даже если вручную скопировать и вставить, то вставит без оформления..
      как это обойти? может можно внаглую залезть в ворд, и скопипастить вместе с тегами (если они там есть).
      или может сохранить в формате xml, и уже тогда скопипастить вместе с тегами?
      в общем согласен на любой метод, где-то же ворд эти стили хранит

      Добавлено 19.04.12, 10:59
      может даже картинкой сделать?
      т.е заскринить значимую часть документа. но как сделать скрин ворда в таком случае?


      Игорь Акопян



      Сообщ.
      #2

      ,
      19.04.12, 11:07

        я бы порыл в направлении команды меню — Вставка/Файл.
        Как раз в позицию курсора вставляет содержимое файла. Для начала записал макрос с этими действиями, потом адаптировал под delphi


        Dart_Sitius



        Сообщ.
        #3

        ,
        19.04.12, 12:53

          Member

          **

          Рейтинг (т): 2

          спасибо! в сочетании с закладками это то, что нужно

          0 пользователей читают эту тему (0 гостей и 0 скрытых пользователей)

          0 пользователей:

          • Предыдущая тема
          • Delphi: Общие вопросы
          • Следующая тема

          Рейтинг@Mail.ru

          [ Script execution time: 0,0446 ]   [ 16 queries used ]   [ Generated: 13.04.23, 22:37 GMT ]  

          Предположим, у нас уже
          открыт файл. Вопросы открытия и сохранения документов уже были в других статьях,
          так что подробно на этом останавливаться не будем. Просто по ходу дела будет
          приведено то, чего раньше не встречалось — выход из документа без сохрания
          изменений. Как-то забыл, извините:)

          Текст

          Сначала о самом простом
          — добавлении в документ Word нужной строки текста. Поместим на форму компоненты
          WordDocument, WordApplicationи WordParagraphFormat с
          палитры Servers. Нас интересуют в первую очередь свойство Range
          компонента WordDocument и свойство Selection компонента
          WordApplication
          . Классики утверждают, что они являются ссылкой на объекты
          Range
          и Selection. Range представляет из себя, проще говоря,
          кусок текста, это может быть как весь текст документа, так и любая его часть.
          Его пределы задаются двумя (или меньше) параметрами типа OleVariant.
          Например:

          var range1, range2, range3, a, b : OleVariant;
          ...
          range1:=WordDocument1.Range;
          a:=5;
          b:=15;
          range2:=WordDocument1.Range(a,b);
          range3:=WordDocument1.Range(a);
          

          Первый наш объект включает в себя весь текст документа, у второго мы ограничили
          пределы 5-м и 15-м символами, третий представляет из себя весь последующий текст
          документа, начиная с 5-го символа. Объект имеет несколько полезных методов,
          например, с его помощью можем добавить текст в документ:

          range2.InsertAfter('MS Word');
          

          Это
          мы вставили текст после выделенного Range. Точно также можем вставить
          текст и перед ним, для этого служит метод InsertBefore(). Текст,
          заключенный в объекте Range, можем получить так:
          WordDocument1.Range(a,b).Text; Кроме
          того, с помощью Range можем изменить шрифт в пределах объекта. Пример:

          a:=5;
          b:=15;
          WordDocument1.Range(a,b).Font.Bold:=1;
          WordDocument1.Range(a,b).Font.Size:=14;
          WordDocument1.Range(a,b).Font.Color:=clRed;
          

          Если
          хотим отменить выделение жирным шрифтом, присваиваем 0. Аналогично можно сделать
          шрифт курсивом, подчеркнутым — наберите WordDocument1.Range.Font., и
          среда сама подскажет, какие могут быть варианты. Методы Select, Cut, Copy и
          Paste работают как в обычном тексте. С помощью Paste можем на место выбранного
          Range вставить не только строки, но и рисунок, находящийся в буфере
          обмена.

          WordDocument1.Range(a,b).Select;
          WordDocument1.Range(a,b).Cut;
          WordDocument1.Range(a,b).Copy;
          WordDocument1.Range(a,b).Paste;
          

          С
          помощью Range можем найти в документе нужную строку. Пусть в тексте содержится
          слово «picture». Например, нам на его место надо будет вставить рисунок.

          var a, b, vstart, vend: OleVariant;
           j, ilengy: Integer;
          ...
          ilengy:=Length(WordDocument1.Range.Text);
          for j:=0 to ilengy-8 do begin
           a:=j;
           b:=j+7;
           if WordDocument1.Range(a,b).Text='picture' then begin
           vstart:=j;
           vend:=j+7;
           end;
          end;
          WordDocument1.Range(vstart,vend).Select;
          

          Такая
          процедура находит и выделяет нужный кусок текста.Теперь про Selection, представляющий из себя выделенный фрагмент
          документа. Если выделения нет, это текущая позиция курсора в документе. С его
          помощью можем вставить что-либо на место выделенного фрагмента, сделать
          выравнивание, изменить шрифт. Он также имеет методы InsertAfter() и
          InsertBefore()
          :

          WordApplication1.Selection.InsertAfter("text1");
          WordApplication1.Selection.InsertBefore("text2");

          Форматирование выделенного текста происходит аналогично Range, например:

          WordApplication1.Selection.Font.Bold:=1;
          WordApplication1.Selection.Font.Size:=16;
          WordApplication1.Selection.Font.Color:=clGreen;
          

          Для
          выравнивания проще воспользоваться компонентом WordParagraphFormat.
          Сначала только нужно «подключить» его к выделенному фрагменту текста:

          WordParagraphFormat1.ConnectTo(WordApplication1.Selection.ParagraphFormat);
          WordParagraphFormat1.Alignment:=wdAlignParagraphCenter;
          

          Значения его свойства Alignment может принимать значения
          wdAlignParagraphCenter, wdAlignParagraphLeft, wdAlignParagraphRight
          , смысл
          которых очевиден. Имеются и методы Cut, Copy и Paste, которые в пояснениях вряд
          ли нуждаются:

          WordApplication1.Selection.Cut;
          WordApplication1.Selection.Copy;
          WordApplication1.Selection.Paste;
          

          Убираем выделение с помощью метода Collapse.
          При этом необходимо указать, в какую сторону сместится курсор, будет ли он до
          ранее выделенного фрагмента или после:

          var vcol: OleVariant;
          ...
          vcol:=wdCollapseStart;
          WordApplication1.Selection.Collapse(vcol);
          

          При
          этом выделение пропадет, а курсор займет позицию перед фрагментом текста. Если
          присвоить переменной значение wdCollapseEnd, то курсор переместится назад. Можно
          просто поставить в скобках «пустышку»:

          WordApplication1.Selection.Collapse(EmptyParam);
          

          Тогда
          свертывание выделения производится по умолчанию, к началу выделенного текста.
          РисункиЛогично было бы предположить, что рисунки документа будут представлять из себя
          коллекцию, аналогичную таблицам, и мы, обратившись к конкретной картинке, сможем
          менять ее свойства — обтекание, размер и т.д. Однако ничего подобного в WordDocument не обнаруживается. Потому возможности управления встраиваемыми в
          документ изображениями сильно ограничены. Простейший метод вставить в документ рисунок — по упомянутым причинам он же и
          единственный — скопировать его в Word из буфера обмена. Предположим, рисунок у
          нас находится в компоненте DBImage. Сначала нужно загнать его в буфер обмена:

          Clipboard.Assign(DBImage1.Picture);
          

          Теперь для его вставки следует воспользоваться методом Paste объектов Range или
          Selection: WordApplication1.Selection.Paste или WordDocument1.Range(a,b).Paste.
          Оставить для рисунка достаточное количество пустых строк и попасть в нужное
          место — это уже наша забота. Если он попадет посреди текста, вид будет довольно
          противный — при такой вставке обтекание текстом рисунка происходит как-то
          странно. Можно приготовить для отчета шаблон, где заменяем рисунком какое-либо
          ключевое слово. О том, как найти в документе нужный текст, см. выше.
          А
          теперь о несколько ином способе вставки рисунка, который устраняет проблемы с
          обтеканием и дает нам возможность перемещать его по документу, масштабировать и
          задавать отступы между рисунком и текстом. Способ, собственно, тот же — копируем
          из буфера обмена, но не прямо в документ, а в «рамку» — текстовую вставку. В ней
          может находиться не только текст, но и картинка, чем и воспользуемся.
          «Рамки» образуют коллекцию Frames, нумеруются целым индексом, пробегающим
          значения от 1 до WordDocument1.Frames.Count. Добавим в документ рамку,
          изменим ее размер и вставим рисунок:

          Clipboard.Assign(DBImage1.Picture);
          vstart:=1;
          vend:=2;
          WordDocument1.Frames.Add(WordDocument1.Range(vstart,vend));
          i:=1;
          WordDocument1.Frames.Item(i).Height:=DBImage1.Height;
          WordDocument1.Frames.Item(i).Width:=DBImage1.Width;
          WordDocument1.Frames.Item(i).Select;
          WordApplication1.Selection.Paste;
          

          Здесь
          для простоты предполагается, что размер DBImage равен размеру самой
          картинки, а также что до этого рамок у нас в документе не было. Обратить
          внимание следует на несколько моментов. Размер рамки надо задавать до того, как
          копировать в нее рисунок. Иначе она будет иметь размер по умолчанию, под который
          замасштабируется и наша картинка. При попытке изменить размер рамки задним
          числом размер картинки уже не изменится. Кроме того, параметр Range при
          добавлении рамки часто никакой роли не играет. Рамка изначально все равно
          появится в левом верхнем углу документа, а указанный кусок текста при этом не
          пострадает. Но это только в том случае, если он не выделен. Если в документе
          есть выделение, рамка появится вместо выделенного фрагмента. Таким образом можем
          ее вставить в нужное место взамен какого-то ключевого слова.
          При желании можем ее подвигать в документе и «вручную». Для этого служат
          свойства горизонтального и вертикального позиционирования, которые задают ее
          отступ от левого верхнего «угла» документа:

          i:=1;
          WordDocument1.Frames.Item(i).VerticalPosition:=30;
          WordDocument1.Frames.Item(i).HorizontalPosition:=50;
          Отступ между краями рамки и текстом задается следующим образом:
          WordDocument1.Frames.Item(i).HorizontalDistanceFromText:=10;
          WordDocument1.Frames.Item(i).VerticalDistanceFromText:=10;
          

          А
          теперь о масштабировании. Для этого достаточно длину и ширину рамки умножить на
          одно и то же число. Например:

          WordDocument1.Frames.Item(i).Height:=DBImage1.Height*1.5;
          WordDocument1.Frames.Item(i).Width:=DBImage1.Width*1.5;
          

          При
          этом наша картинка в полтора раза пропорционально растянется. Точно также можно
          и уменьшить, но делить, как и множить, следует на одно число. Растягивать длину
          и ширину по-разному у меня лично не получалось. Задавать размер опять-таки надо
          еще до вставки рисунка. Ну и, наконец, удаление рамки:

          WordDocument1.Frames.Item(i).Delete;
          

          Списки Списки в документе образуют коллекцию Lists, к отдельному списку
          обращаемся WordDocument1.Lists.Item(i), где i целое число от 1 до
          WordDocument1.Lists.Count
          … на этом все. Нет методов, позволяющих не то
          что создать новый список, а даже добавить пункт к уже существующему. Ничего
          страшного, настоящие герои всегда идут в обход:)) Сейчас мы все же проделаем и
          то, и другое. Все что нам понадобится — свойство Range отдельного списка, то
          есть его текст без разделения на пункты, а также возможность его выделить:

          WordDocument1.Lists.Item(i).Range.Select;
          

          Для
          этого в любом случае потребуется заготовка. Неважно, вставлена она в общий
          шаблонный документ или хранится в отдельном файле. Заготовку делаем так:
          выбираем в меню Формат/Список, и сохраняем, если это отдельный шаблон списка. У
          нас появляется пустой список без текста с одним маркером. Далее вспоминаем, как
          мы делали списки вручную — писали текст, нажимали «Enter», появлялся новый
          элемент списка. Теперь то же самое, только программно. Предположим, у нас уже
          открыт документ с заготовкой, и мы хотим внести в список пункты «Item 1» и «Item
          2»:

          var i: Integer;
           vcol: OleVariant;
          ...
          i:=1;
          vcol:=wdCollapseEnd;
          WordDocument1.Lists.Item(i).Range.Select;
          WordApplication1.Selection.Collapse(vcol);
          WordApplication1.Selection.InsertAfter('Item 1');
          WordDocument1.Lists.Item(i).Range.Select;
          WordApplication1.Selection.Collapse(vcol);
          WordApplication1.Selection.InsertAfter(#13);
          WordDocument1.Lists.Item(i).Range.Select;
          WordApplication1.Selection.Collapse(vcol);
          WordApplication1.Selection.InsertAfter('Item 2');
          WordDocument1.Lists.Items(i).Range.Select;
          WordApplication1.Selection.Copy;
          

          То
          есть мы вставляем в документ текст первого пункта списка, он попадает на свое
          место. Потом посылаем в Word символ перехода строки, он честно переходит и тем
          самым сам создает нам второй пункт списка, куда и вставляем нужную строку. Ну и
          так далее, нужное количество раз. Последние две строки нужны, если список
          заготовлен в отдельном файле — после их выполнения список оказывается в буфере
          обмена. Здесь выгода в том, что можем иметь заготовки списков разных стилей и по
          ходу дела выбирать, какой список создать. Затем открываем документ, где должен
          быть список, выделяем с помощью Range нужный кусок, копируем из буфера обмена
          через WordDocument1.Range(a,b).Paste. Чтобы не испортить файл с заготовкой,
          можем сразу после открытия пересохранить его под другим именем, а можем просто
          выйти из него без сохранения изменений

          var vsave: OleVariant;
          ...
          vsave:=wdDoNotSaveChanges;
          WordDocument1.Close(vsave);
          

          Константа сохранения изменений может принимать значения

          Символьное
          обозначение


          Шестнадцатеричное

           

          wdSaveChanges

          $FFFFFFFF

          wdDoNotSaveChanges

          $00000000

          wdPromptToSaveChanges

          $FFFFFFFE

          Первое значение
          сохраняет изменения, второе дает возможность выйти без сохранения изменений.
          Последняя константа вызывает при выходе стандартный диалог сохранения изменений.
          Можем сделать и несколько по-другому. Хотя мы не можем создать новый элемент
          списка, но текст в уже существующем изменить можно:

          var i,j: Integer;
          ...
          i:=1;
          j:=1;
          WordDocument1.Lists.Item(i).ListParagraphs.Item(j).Range.Text:='Item 1';

          Так что можно с помощью переходов строки создать нужное количество элементов, а затем их заполнить:

          WordDocument1.Lists.Item(i).Range.Select;
          WordApplication1.Selection.Collapse(vcol);
          WordApplication1.Selection.InsertAfter(#13);
          j:=1;
          WordDocument1.Lists.Item(i).ListParagraphs.Item(j).Range.Text:='Item 1';
          j:=2;
          WordDocument1.Lists.Item(i).ListParagraphs.Item(j).Range.Text:='Item 2';
          

          Это
          было в предположении, что у нас один элемент списка в заготовке уже есть. Ну
          вот, в общем-то, и все про текст, списки и картинкиСтатистика документовВ
          данном небольшом материале рассматривается вопрос подсчета статистики файлов
          *.doc и *.rtf. Такой вопрос у меня возник, когда пришлось сделать небольшую базу
          данных по учету документов, куда надо было заносить и статистику документа —
          число знаков, слов и т.п. Открывать каждый раз Word, считать статистику и
          забивать ее в форму ввода было лень, так что пришла в голову мысль это дело
          автоматизировать. Информации по данному вопросу найти так и не удалось, так что
          основным источником знаний служили заголовочный файл Word2000.pas и
          справка по Visual Basic for Applications. Ну и, конечно, множество разных
          экспериментов. Сразу
          оговорюсь, что я не профессиональный программист, так что в тонкости интерфейсов
          вникать не будем — сам в них не особо разбираюсь. Потому, не мудрствуя лукаво,
          просто поместим на форме компоненты WordApplication и WordDocument
          с палитры Servers. Для работы используются свойства и методы этих
          компонентов. Встроенная статистика Word подсчитывает статистику обычного текста, обычных и
          концевых сносок. Для подсчета статистики используется метод компонента
          WordDocument ComputeStatistic()
          . Он имеет один параметр, характеризующий,
          что именно считать, представляющий из себя шестнадцатеричную константу.
          Константы описаны в заголовочном файле Word2000.pas, он лежит обычно в
          /Delphi/Ocx/Servers
          .


          Шестнадцатеричная

          Символьное обозначение

          Смысл

          $00000000

          wdStatisticWords

          Количество слов

          $00000001

          wdStatisticLines

          Количество строк

          $00000002

          wdStatisticPages

          Количество страниц

          $00000003

          wdStatisticCharacters

          Знаки без пробелов

          $00000004

          wdStatisticParagraphs

          Количество разделов

          $00000005

          wdStatisticCharactersWithSpaces

          Знаки с пробелами

          Это было основное, что
          надо знать. Ну а теперь по порядку. Поместив на форму упомянутые компоненты, видим, что свойств и методов у них
          совсем мало. В первую очередь следует определиться с методом ConnectKind
          компонента WordApplication. Оно может принимать различные значения, но мы
          оставим присваемое по умолчанию значение ckRunningOrNew. Это означает, что
          соединение происходит с уже работающим сервером, при его отсутствии запускается
          новый. Как правило, это вполне устраивает. Первым делом откроем документ. Предварительно надо объявить переменную FileName,
          она будет типа OleVariant, которой присвоим строку с именем файла.

          WordApplication1.Connect;
          WordApplication1.Documents.Open(FileName,
          EmptyParam,EmptyParam,EmptyParam,
          EmptyParam,EmptyParam,EmptyParam,
          EmptyParam,EmptyParam,EmptyParam,
          EmptyParam,EmptyParam);
          WordDocument1.ConnectTo(WordApplication1.ActiveDocument);
          

          Обратите внимание на количество параметров-«пустышек».
          Их число больше того, которое обычно
          приводится в книжках. Ну, в моих, во всяком случае. Объясняется это тем, что
          «книжные» функции предназначены для MS Word 97, а такая запись для работы с Word
          2000 и Word XP.
          «Plain Text»
          Объявив нужное количество переменных типа LongInt (в очень большом файле или при
          суммировании по нескольким документам в принципе может оказаться больше знаков,
          чем пределы обычного целого типа), можем уже и приступать к подсчету. Например,
          посчитаем число слов, знаков с пробелами и без пробелов обычного текста, а также
          количество страниц в документе. Результаты сохраним соответственно в «длинных»
          переменных

          WCount:=WordDocument1.ComputeStatistics($00000000);
          CCount:=WordDocument1.ComputeStatistics($00000003);
          SCount:=WordDocument1.ComputeStatistics($00000005);
          PCount:=WordDocument1.ComputeStatistics($00000002);
          

          Открыв нужный документ в Word’е и вызвав диалог подсчета статистики, нетрудно
          увидеть, что значения переменных равны параметрам вордовской статистики со
          сброшенным флажком «Учитывать все сноски».
          Сноски
          Сноски в документах могут быть обычные и концевые. То есть если первые
          располагаются внизу данной страницы, то концевые — строго в конце документа.
          Кроме того, они могут отличаться и нумерацией — автоматической или заданной
          пользователем. Начнем с обычных сносок как с самого простого. В терминологии
          объектной модели Word — Footnotes. Сначала надо вычислить количество самих
          сносок:

          ifcount:=WordDocument1.DefaultInterface.Footnotes.Count;
          

          Подсчет статистики текста в сноске производится так:

          FWCount:=WordDocument1.DefaultInterface.Footnotes.Item(ifoot).Range.ComputeStatistics($00000000);
          

          Здесь
          ifoot — целое число, «нумерующее» сноску. Для того, чтобы учесть сами номера
          сносок, сделаем так:

          FWCount:=FWCount+WordDocument1.DefaultInterface.Footnotes.Item(ifoot).Reference.ComputeStatistics($00000000);
          

          Это
          мы посчитали для примера количество слов в сноске с номером ifoot и ее метке —
          при пользовательской нумерации в качестве «номера» может быть целое предложение.
          Далее начинаем перебирать их одну за другой. При этом следует учесть, что кроме
          статистики сносок необходимо получить и статистику их «номеров». То есть:

          for ifoot:=1 to ifcount do begin
           FWCount:=FWCount+WordDocument1.DefaultInterface.Footnotes.Item(ifoot).Range.ComputeStatistics($00000000);
           FCCount:=FCCount+WordDocument1.DefaultInterface.Footnotes.Item(ifoot).Range.ComputeStatistics($00000003);
           FSCount:=FSCount+WordDocument1.DefaultInterface.Footnotes.Item(ifoot).Range.ComputeStatistics($00000005);
           FCCount:=FCCount+WordDocument1.DefaultInterface.Footnotes.Item(ifoot).Reference.ComputeStatistics($00000003);
           FSCount:=FSCount+WordDocument1.DefaultInterface.Footnotes.Item(ifoot).Reference.ComputeStatistics($00000005)+1;
           if WordDocument1.DefaultInterface.Footnotes.Item(ifoot).Reference.Text<>IntToStr(ifoot)
           then begin
           FWCount:=FWCount+
           WordDocument1.DefaultInterface.Footnotes.Item(ifoot).Reference.ComputeStatistics($00000000);
           end;
          end;
          

          Прибавление единицы появляется оттого, что сумма статистики сносок и номеров не
          совпадает с тем, что выдает встроенная статистика Word. Между номером сноски и
          текстом сноски Word ставит пробел, который почему-то не учитывается. Условный
          оператор определяет, как пронумерована данная сноска — по умолчанию или нет. В
          последнем случае следует проверить количество слов в обозначении сноски. Такая
          схема дает результат, совпадающий со показаниями встроенной статистики. Кроме
          того, цикл у нас идет от 1 — так начинается нумерация сносок в MS Word, да и
          практически всех остальных объектов тоже.Теперь перейдем к концевым сноскам. Теоретически все то же самое, только вместо
          слова «Footnotes» пишем «Endnotes». И тут наталкиваемся на сюрприз — почему-то
          оно считает неточно. Я в данном случае поступил так: сохраняю документ под
          другим именем, переконвертирую концевые сноски в обычные и далее все, как
          сказано выше. Сохранение документа:

          WordDocument1.SaveAs(FileName, FileFormat),
          

          где в скобках стоят два параметра типа OleVariant — имя файла и шестнадцатеричная
          константа, задающая формат файла. Некоторые константы:


          Шестнадцатеричная


          Символьное
          обозначение


          Смысл

          $00000000

          wdFormatDocument

          Документ Word

          $00000004

          wdFormatDOSText

          Простой текст

          $00000006

          wdFormatRTF

          Файл RTF

          Полный список констант
          формата можно найти все в том же файле Word2000.pas. И еще один интересный
          момент — если просто поставить в скобки обе константы, работать не будет.
          Следует предварительно объявить две переменных, присвоить им соответствующие
          значения и только потом сохранять. Ну, а
          теперь, собственно, можем вернуться к сноскам. Конвертирование концевых сносок в
          обычные происходит так:

          WordDocument1.DefaultInterface.Endnotes.Convert;
          

          Теперь мы имеем документ, в котором содержатся только обычные сноски. С ними
          никаких проблем не возникает, пример, как с ними работать, см. выше. Если
          интересует статистика отдельно разных типов сносок, считаем предварительно
          статистику обычных сносок, сохраняем ее в «буферных» переменных и считаем еще
          раз после конвертирования. Разница даст статистику концевых сносок по
          отдельности. Сложив статистику сносок и простого текста, получаем статистику
          документа с учетом сносок так, как ее дает сам Word.
          Дополнительно…
          Тут
          по традиции несколько покритикуем Microsoft. Как оказалось, Word показывает не
          все, что содержится в документе. Не принимаются в расчет колонтитулы. А ведь в
          них может содержаться изрядный кусок текста, особенно в справках, бланках и т.п.
          Оказывается, Word их на самом деле считает, но нам не показывает. Вот и
          посмотрим, как же его можно заставить это сделать. Колонтитулы в документе тесно связаны с несколько загадочной штукой под
          названием «разделы» — Sections. Каждый раздел может иметь верхние и нижние
          колонтитулы. Потому первым делом определяем количество абзацев.

          isectct:=WordDocument1.DefaultInterface.Sections.Count;
          

          Здесь
          у нас целые переменные isectct, icofct, icohct обозначают
          соответственно количество разделов как таковых, количество нижних и верхних
          колонтитулов данного раздела. Переменная isec служит «номером» раздела,
          переменные icof, icoh «нумеруют» соответственно нижние и верхние
          колонтитулы в пределах данного раздела. Количество колонтитулов в разделе
          определяем так:

          icofct:=WordDocument1.DefaultInterface.Sections.Item(isec).Footers.Count;
          icohct:=WordDocument1.DefaultInterface.Sections.Item(isec).Headers.Count;
          

          Теперь уже можем «достать» текст из колонтитула:

          CBWCount:=
          WordDocument1.DefaultInterface.Sections.Item(isec).Footers.Item(icof).Range.ComputeStatistics($00000000);
          

          В
          данном случае мы для примера посчитали число слов, содержащихся в нижнем
          колонтитуле под номером icof, принадлежащем разделу под номером isec.
          Теперь можем написать «двойной» цикл для подсчета статистики верхних и нижних
          колонтитулов.

          18th
          Окт

          Posted by micher under Delphi

          Как данные из Edit переместить в Word?

          NURKZ


          procedure TForm1.Button3Click(Sender: TObject);
          var
          word : variant;
          procedure FindAndReplace (SearchStr, ReplaceStr : string);
          begin
          word.Selection.Find.Text := SearchStr;
          word.Selection.Find.Replacement.Text := ReplaceStr;
          word.Selection.Find.Execute (Replace := 2);
          end;
          begin
          try
          Word := CreateOleObject('Word.Application');
          except
          MessageBox (Handle, 'Не установлен Microsoft Office Word. Формирование ' +
          'диакарты невозможно.', 'Ошибка',
          MB_OK or MB_ICONERROR);
          exit;
          end;
          word.documents.open ('C:LTKSoftDiagnostic.docx');
          FindAndReplace ('%NAMECTO%', Edit8.Text);
          FindAndReplace ('%GRNZ%', Edit1.Text);
          FindAndReplace ('%VLADELETS%', Edit13.Text);
          FindAndReplace ('%MODELTS%', Edit2.Text);
          FindAndReplace ('%GODVYPUSKA%', Edit3.Text);
          FindAndReplace ('%SERNUMSRTS%', Edit4.Text);
          FindAndReplace ('%RAMA%', Edit6.Text);
          FindAndReplace ('%KUZOV%', Edit5.Text);
          FindAndReplace ('%DKNUMBER%', Edit7.Text);
          FindAndReplace ('%MINMASS%', Edit10.Text);
          FindAndReplace ('%MAXMASS%', Edit11.Text);
          FindAndReplace ('%PROBEG%', Edit12.Text);
          FindAndReplace ('%TOPLIVO%', Edit14.Text);
          FindAndReplace ('%DATE%', Edit9.Text);
          word.Visible := true;
          word := Unassigned;
          end;

          тема на форуме

          Похожие статьи

          • Добавить строку с верха вниз StringGrid
          • Среднеарифметическое значение из Edit
          • Как запретить ввод определённых символов в Edit?
          • Вставка произвольных непечатных символов перед шифрованием и удаление рандомных знаков после расшифровки
          • Маска на авторизацию в Edit
          • Как программно при открытии ворда скрыть все панели инструментов?
          • Как сделать прозрачный edit?
          • Пример как можно загрузить через TStringList в Edit’ы
          • Изменение цвета поля edit при попадании на него курсора
          • Как сделать чтобы TEdit стал браузерной строкой TWebBrowser?

          Когда я делаю автоматизацию Word из Delphi XE, у меня одновременно открываются два документа. Я хочу скопировать содержимое заданного диапазона одного документа в другой диапазон другого документа. Как я могу это сделать?

          Рассмотрим следующий код:

          procedure TForm1.ManipulateDocuments;
          var
            vDoc1,vDoc2 : TWordDocument;
            vFilename : olevariant;
            vRange1,vRange2 : Range;
          begin
            vDoc1 := TWordDocument.Create(nil);
            vDoc2 := TWordDocument.Create(nil);
            try
              vFilename := 'c:temptest1.doc';
              vDoc1.ConnectTo(FWordApp.Documents.Open(vFilename,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam));
          
              vFilename := 'c:temptest2.doc';
              vDoc2.ConnectTo(FWordApp.Documents.Open(vFilename,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam,EmptyParam));
          
              vRange1 := GetSourceRange(vDoc1);
              vRange2 := GetDestinationRange(vDoc2);
          
              vRange2.CONTENTS := vRange1.CONTENTS; //What should I substitute for CONTENTS?
            finally
              vDoc1.Free;
              vDoc2.Free;
            end;
          end;
          

          Могу ли я чем-то заменить СОДЕРЖАНИЕ? Я не могу использовать текст, так как хочу скопировать форматирование, закладки, коды полей и т. Д. Нужно ли мне вообще делать это по-другому? Какие-либо предложения?

          4 ответа

          Лучший ответ

          Я не знаю способа для более ранних версий Word, но для более новых версий (2007 и новее) вы можете экспортировать диапазон из документа в файл фрагмента, а затем импортировать его из другого документа. Если вам нужна ранняя привязка, вам может потребоваться импортировать библиотеку типов (msword.olb), я не знаю, есть ли она в Delphi XE. В противном случае код мог бы выглядеть так:

          function GetTempFileName(Prefix: string): string;
          begin
            SetLength(Result, MAX_PATH);
            GetTempPath(MAX_PATH, PChar(Result));
            windows.GetTempFileName(PChar(Result), PChar(Prefix), 0, PChar(Result));
          end;
          
          procedure TForm2.Button1Click(Sender: TObject);
          const
          //  wdFormatDocument = 0;
            wdFormatRTF = $00000006;
          var
            WordApp : OleVariant;
            fragment: string;
            vDoc1, vDoc2: OleVariant;
            vRange1, vRange2: OleVariant;
          begin
            try
              WordApp := GetActiveOleObject('Word.Application');
            except
              WordApp := CreateOleObject('Word.Application');
            end;
            WordApp.Visible := True;
          
            vDoc1 := WordApp.Documents.Open(ExtractFilePath(Application.ExeName) + 'test1.doc');
            vRange1 := vDoc1.Range(20, 120);     // the export range
            fragment := GetTempFileName('frg');
            vRange1.ExportFragment(fragment, wdFormatRTF);
            try
              vDoc2 := WordApp.Documents.Open(ExtractFilePath(Application.ExeName) + 'test2.doc');
              vRange2 := vDoc2.Range(15, 15);    // where to import
              vRange2.ImportFragment(fragment);
            finally
              DeleteFile(fragment);
            end;
          end;
          

          В моем тесте формат «документ» вызывал ошибку (что-то вроде невозможности вставить форматирование XML), следовательно, использовался формат RTF.

          изменить:

          В более ранних версиях кажется возможным вставить именованный выбор из одного документа в выбор в другом документе. Результат кажется не идеальным с точки зрения форматирования, если один из выделенных фрагментов находится в середине некоторого текста. Но в остальном вроде работает хорошо.

            ...
            WordApp.Visible := True;
          
            vDoc1 := WordApp.Documents.Open(ExtractFilePath(Application.ExeName) + 'test1.doc');
            vRange1 := vDoc1.Range(20, 188);                 // the transfer range
            vDoc1.Bookmarks.Add('TransferSection', vRange1); // arbitrary bookmark name
          
            vDoc2 := WordApp.Documents.Open(ExtractFilePath(Application.ExeName) + 'test2.doc');
            vRange2 := vDoc2.Range(103, 104);           // where to import the bookmark
            vRange2.Select;
            vDoc2.ActiveWindow.Selection.InsertFile(vDoc1.FullName, 'TransferSection');
          
            vDoc1.Bookmarks.Item('TransferSection').Delete; // no need for the bookmark anymore
           
          


          3

          Sertac Akyuz
          6 Апр 2011 в 02:39

          Если вы можете использовать формат Office Open XML (т. Е. Формат файла docx, представленный в Word 2007), то вы можете сделать это без автоматизации.

          Версии Word до 2007 должны устанавливать пакет совместимости, который включает файлы docx для Word 2003, 2002 и 2000.

          На самом деле docx-файл представляет собой zip-файл, содержащий несколько xml-файлов. Попробуйте изменить расширение docx-файла с .docx на .zip и открыть этот файл, например. WinZip.

          Итак … Разархивируйте docx-файл и возьмите нужную xml-часть. Как чистая строка или как XML-документ. Затем вы можете вставить эту xml-часть в другой docx-файл. Однако вам нужно знать где в xml-структуре, чтобы получить / вставить xml. Это будет зависеть от того, насколько хорошо вы знаете структуру документа и насколько пользователю разрешено редактировать документ.

          Я не знаю, как Word при таком подходе будет обрабатывать повторяющиеся имена закладок и т. Д.


          1

          Jørn E. Angeltveit
          24 Мар 2011 в 12:51

          Кажется, я нашел каноническое решение этого вопроса, копаясь в аналогичной проблеме. Свойство FormattedText объекта Range — это именно то, что вам нужно. Просто используйте:

          vRange2.FormattedText := vRange1;
          

          И содержимое vRange1 будет скопировано в vRange2. Кроме того, это тоже работает:

          vRange2 := vRange1;
          

          Хотя второй оператор не копирует форматирование.


          0

          Danatela
          21 Май 2014 в 09:07

          Почему бы не использовать буфер обмена? Если весь текст выделен в vDoc1, то для его копирования в буфер обмена требуется один простой вызов: vDoc1.copy. Точно так же для копирования содержимого буфера обмена во второй документ требуется один простой вызов: vDoc2.paste. Буфер обмена будет содержать всю информацию о форматировании.


          -3

          No’am Newman
          23 Мар 2011 в 13:06

          Like this post? Please share to your friends:
        • Delphi word как закрыть word
        • Delphi excel количество строк
        • Delphi excel как число
        • Delphi excel как скопировать диапазон
        • Delphi excel имя всех листов