Автор Мирончик Игорь – независимый разработчик и преподаватель курсов по Delphi, C++Builder, Oracle. E-maile EKSKHQ@Elsite.ru, тел. (09657) 334-22Доступ к COM серверам Microsoft Office из Delphi 5ВведениеВ статье рассматривается вопрос доступа к общеизвестным приложениям Microsoft Office, таким как Word, Excel, Outlook и другим, через новый набор компонент, представленных в Delphi 5. Для начала работы нам предстоит установить на компьютере приложения Microsoft Office 97 – Excel, Word, Outlook, PowerPoint. Если считаете необходимым, то можно добавить и Access (но с ним у меня особые счеты). Ну и конечно устанавливается новый продукт Delphi 5. Кроме множества изменений в нем имеется одно, для нас сейчас необходимое – новая закладка на палитре инструментов – Servers.Через эти компоненты мы будем получать доступ к COM серверам приложений Office, использующих автоматизацию (прежде известную как OLE Automation). Мы рассмотрим несколько примеров построения контроллеров автоматизации для создания отчетов в MS Word, производство расчетов и построение диаграмм в MS Excel, а так же формирование рассылки писем адресатам через MS Outlook.^ Приложения и объекты MS Office. Обзор Office – это среда, в которой большинство задач можно решать без какого либо программирования. Но вся ценность приложений Office для разработчика заключается в том, что все, что можно сделать руками, можно сделать программным путем с использованием средств VBA (Visual Basic for Application). Кроме того, приложения Office поставляют сервера COM, которые предоставляют интерфейс доступа к приложению и его объектам. Благодаря этому, разработчик в среде Delphi имеет возможность, создав контроллер автоматизации, управлять сервером. Так как устроено приложение в Office? На самом деле приложение рассматривается как совокупность объектов со своими методами, свойствами, событиями, которые обеспечивают скелет приложения. Программист Office является не создателем приложения, как например это делается в Delphi, а он принимает участие в создании системы документов. Таким образом ДОКУМЕНТ, а не программа являются целью разработки.Наследование – мощный инструмент построения нового класса, однако программистам известен еще один способ получения класса – встраивание. Как и наследование, встраивание транзитивное отношение. В объектной модели Office нет наследования в полном смысле этого слова, а есть только встраивание.Всегда существует задающий приложение корневой объект, он всегда называется Application. Каждое приложение Office имеет свой собственный окорневой обект – Word.Application, Excel.Application. Outlook приложение само является корневым объектом, несмотря на это в объект Application встраиваются все остальные объекты (участники), которые являются свойствами главного объекта. У участником могут быть свои участники и так далее. В документе методов очень много, причем самых разных, но имеются и одинаковые методы в различных приложениях Office. Среди таковых – Run, Quit, Activate; но даже и они в разных приложениях имеют различный набор параметров, а зачастую выполняют не адекватные действия. Такое многообразие различных реализаций классов отпугивало программистов контроллеров автоматизации. Однако используя компоненты доступа к серверам автоматизации – положение резко улучшилось. На следующем рисунке показан очень маленький фрагмент структуры объекта Microsoft Word Objects. Более полное представление об объектной архитектуре приложений Office можно найти в соответствующих файлах помощи. Как только открывается новый документ, будь то PowerPoint, Excel, Word, автоматически создается каркас нового документа, который представляет собой набор библиотек с классами. Объекты этих классов будут доступны в данном документе. Задачей разработчика контроллера автоматизации является получить доступ к корневому объекту сервера, выстроить цепочку доступа к объектам – участникам (встроенным объектам), правильно передать параметры. ^ Объекты Application и организация доступа к ним из Delphi.Я наверное утомил читателя устройством приложений Office. Но все-таки необходимо еще раз отметить, что всех объектов этого семейства не перечесть, но главные необходимо знать и уметь ими пользоваться. Фундаментальным объектом любого приложения является Application. Давайте получим к нему доступ из Delphi. Создаем новый проектНа главную форму выкладываем компоненту с закладки Servers, которая называется WordApplicationУстанавливаем свойства компоненты AutoConnect и AutoQuit в TrueЗапускаем приложение на выполнение.Казалось бы ничего не произошло, запустилась форма и отобразилась на экране, на самом деле наше приложение запустило сервер автоматизации Microsoft Word, этот факт можно обнаружить запустив на выполнение Task Manager и выбрав закладку Processes. Среди прочих процессов мы обнаруживаем WINWORD.EXE. На самом деле была приложением проделана следующая работа:При создании формы, в системном реестре, по идентификатору CLSID был найден сервер Word.ApplicationЗапущено на выполнение приложение, находящееся по адресу в реестре (ProgID)Сервер предоставил нашему приложению, которое и является контроллером автоматизации интерфейс, через который мы и получим доступ к объекту Application. Интерфейс Idispatch унаследован от Iunknown, который в свою очередь имеет три метода, один из которых _ADDRef умеет считать количество клиентов, в настоящий момент использующих сервер. Как только от сервера отсоединиться последний клиент, он автоматически будет выгружен из памяти компьютера. Закройте наше приложение и загляните в Task Manager - Word выгрузился из памяти.Точно такой же результат можно было бы получить прописав следующий код: procedure TForm1.Button1Click(Sender: TObject);var wd:OleVariant; fileName:string;begintry fileName:=ExtractFilePath(Application.EXEName)+'report.DOC';//^ Создаем объект интерфейса для доступа к серверу COM wd:=CreateOleObject('Word.Application');//Проверка наличия методов и правильность передачи параметров будет осуществляться на стадии выполнения приложения wd.application.documents.add; wd.application.activedocument.range.insertAfter(now); wd.application.activedocument.saveas(fileName);//выгрузаем сервер из памяти компьютера wd.application.quit(true,0);….Не забудьте добавить в раздел uses модуль COMOBJ. ^ Базовый класс TOLEServerНа закладке Service находится набор компонент для доступа к серверам автоматизации, не все компоненты возвращают ссылку на объект Application, то есть могут быть получены интерфейсы для доступа к вложенным объектам, таким как Документ Word или рабочая книга Excel. Все компоненты унаследованы от класса TOLEServer, который наследует свойства класса Tcomponent. TOLEServer является базовым классом всех COM серверов, которые можно получить через среду IDE следующим образом: Project | Import Type Library. Кроме этого этот класс имеет еще несколько свойств и методов для управления связью с COM сервером. Среди таковых уже знакомое нам свойство AutoConnect, которое автоматически запускает COM сервер и производит извлечение из него интерфейса, обеспечивающего связь с контроллером. Еще одно важное свойство класса TOLEServer – ConnectKind – указывающее тип процесса к которому устанавливается связь. Свойство используется методом Connect, который вызывается автоматически, если свойство AutoConnect истинно. Ниже описаны значения, которые может принимать ConnectKind: Значение свойства ConnectKind Характеристика CkRunningOrNew Контроллер производит подключение к уже существующему процессу, или запускает новый процесс, при отсутствии такового. Этот вид взаимодействия между COM сервером и контроллером наиболее часто применяется на практике. Такое значение свойства установлено по умолчанию. CkNewInstance При соединении с сервером каждый раз создается новый экземпляр CkRunningInstance Соединение устанавливается с уже запущенным COM сервером. Если таковой отсутствует – будет создан соответствующий объект ошибки, который необходимо обработать CkRemote Это значение используется совместно со свойством RemoteMachineName, если необходимо подключиться к серверу на удаленной машине ckAttachToInterface При установке этого значения интерфейс не создается и соответственно нельзя указывать значение True для свойства AutoConnect. Соединение с сервером производится с помощью метода ConnectTo На последнем значении свойства ConnectKind равном ckAttachToInterface, хотелось бы остановиться более подробно. Действительно мы соединяемся с сервером через главный интерфейс, представленный в объекте Application. Но, например, возникает необходимость подключить к нашему проекту такие компоненты, как WordDocument или WordParagraphFormat, в этом случае мы просто производим подключение к уже существующему интерфейсу, а не создаем его заново. Также это может быть необходимо, когда контроллер должен отслеживать события, происходящие в COM сервере. Я предлагаю вам провести маленький эксперимент:Создайте новое приложениеВыложите на форму компонент WordApplication и WordDocumentСвойства AutoConnect и AutoQuit для WordApplication установите в TrueСвойство ConnectKind для WordDocument установите в ckAttachToInterfaceДля события onDokumentChange и onFormCreate пропишите следующий код:procedure TForm1.WordApplication1DocumentChange(Sender: TObject);begin//производим подключение к текущему документуWordDocument1.ConnectTo( WordApplication1.ActiveDocument);//Контроллер добавляет новую строку в текущий документWordDocument1.Range.InsertAfter(#13+'Переход к документу'+#13+WordApplication1.ActiveDocument.Get_FullName+' произведен :'+ DateTimeToStr(Now));end;procedure TForm1.FormCreate(Sender: TObject);begin//COM сервер отображает себя на экране WordApplication1.Visible:=true;end; После запуска приложения будет автоматически загружен Word, создайте в нем несколько новых документов и переключайтесь между ними с помощью меню Window. Вы увидите, что контроллер автоматизации добавляет новые строки в текущий активный документ. Аналогично можно управлять и сервером ExcelApplication. При создании новой рабочей книги на сервере, в контроллере будет проинициализировано событие onNewWorkBook, которое можно обработать аналогично предыдущему примеру.^ Классы – наследники ToleObjectТеперь давайте заглянем во внутренний мир компонент закладки Servers. Как мы уже убедились, все классы унаследованы от ToleObject. Кроме того, еще наследуется и интерфейс из библиотеки TLB. Как это происходит? Проведем следующее упражнение:Из среды программирования Delphi удаляем пакет с компонентами COM серверов (Component | Install Packages | Borland Sample Automation Servers Component | Remove). После этого закладка Servers удалилась. Создаем новый пакет с использованием библиотеки типовВыберите Project | Type libraryИз списка зарегистрированных серверов выберите библиотеку типов Excel (Microsoft Excel 8.0 Object Library)Укажите имя закладки палитры компонент (Pallete Page), куда будут установлены новые классы – TexcelQueryTable, TexcelApplication, TexcelChart, TexcelWorksheet, TexcelWorkbook, TExcelOLEObject . Выберите закладку Servers.В боксе - General Component Wraper установите флажок для генерации компонеты на основе библиотеки типов и размещении ее на палитре компонент.Нажмите кнопку InstallСпецифицируйте имя пакета, где будет сгенерирован новый классУстановите сервер на палитру компонент.После проделанных манипуляций вам стал доступен COM сервер Excel, в состав которого входят шесть классов. Аналогичным образом восстановите компоненты для сервера Word и Outlook.Теперь мы можем посмотреть на объявление класса TwordApplication и его предка -ToleServer :TOleServer = class(TComponent, IUnknown)TWordApplication = class(TOleServer)Благодаря такому объявлению, класс TwordApplication наследует свойства и методы класса Tcomponent (способен устанавливаться на палитре компонент и прочие…), а так же знает все о доступе к интерфейсам COM серверов, благодаря наследованию интерфейса IUNknown. В библиотеке типов прописаны все доступные методы и свойсва COM сервера. Когда создается контроллер автоматизации (выкладываем на форму соответствующий компонент из палитры), то приложение получает доступ к Dual Interface описанный в библиотеке типов. Dual интерфейс – есть совокупность пользовательского интерфейса, описанного в библиотеке типов и dispinterface, который доступен в момент выполнения приложения. Это реализовано с помощью ^ COM VTABLE интерфейса, унаследованного от IDISPATCH. Использование Vtable интерфейса имеет ряд преимуществ:Передаваемые параметры, их типы и количество проверяется на стадии проектирования и редактор кода сопровождает разработчика всевозможными хинтами и подсказками.Через Vtable интерфейс осуществляется значительно быстрый доступ к серверу автоматизации, чем через DispInterfaceВ то же время, не всегда разработчик получает доступ к библиотеке типов и соответственно к V – таблице, поэтому приходится иногда пользоваться Idispatch интерфейсомНиже приведен пример использования Dual интерфейсаprocedure TForm1.FormCreate(Sender: TObject);var foo:TWordApplication; foo1:_Application;begin foo:=WordApplication1; foo1:=CoWordApplication.Create; ShowMessage(foo.UserName+#13+foo1.Get_Name);end;Пример использования интерфейса с поздним связыванием был показан в начале этой статьи, когда доступ к COM серверу был осуществлен с помощью функции CreateOleObject. Компилятор в этом случае ничего не знает о методах и параметрах сервера, информация о них извлекается на стадии выполнения приложения, отсюда и потеря скорости выполнения приложения и всевозможные ошибки, которые компилятор не в состоянии обработать. При такой разработке приложения программист достает SDK от Microsoft office и начинает старательно изучать большие тома литературы.Подводя итог можно говорить о том, что для доступа к COM серверу автоматизации существует три способаVtableIdispatch Позднее связывание (CreateOleObject)Наиболее прогрессивный – первый способ, через который работают компоненты Delphi 5 для доступа к COM серверам приложений Office.Пример использования Vtable интерфейсаПостановка задачи: Имеется шаблон документа – shablon.DOC, подготовленный в MS Word, поля, которые должны быть заменены, помечены символом @. Необходимо прочитать данные из источника информации и заменить метки, на реальные данные, после чего распечатать полученный документ.Воспользуемся методом Vtable . Выложим на форму компоненту WordApplication, WordDocument и кнопку Button. Для события OnClick компоненты Button1 пропишем следующий код:procedure TForm1.Button1Click(Sender: TObject);var//Объявление переменных, для передачи их в качестве формальных параметров в сервер автоматизацииShablon,FileName,oldStr,newStr,replace,ext:OleVariant;beginTable1.Active:=false; Table1.Active:=true; Shablon:=ExtractFilePath(Application.EXEName)+'shablon.DOC'; FileName:=ExtractFilePath(Application.EXEName)+'report.DOC';//Открываем шаблон документаWordApplication1.Documents.Open(Shablon,EmptyParam, EmptyParam,EmptyParam,EmptyParam,EmptyParam, EmptyParam,EmptyParam,EmptyParam,EmptyParam);//Связываем компоненту с существующим интерфейсомWordDocument1.ConnectKind:=ckAttachToInterface; WordDocument1.ConnectTo(WordApplication1.ActiveDocument);//Следующие переменные понадобятся нам для выполнения методов сервераreplace:=1; oldStr:='@1'; newStr:=DateTimeToStr(Now);//Находим в документе метки и производим их заменыWordDocument1.Range.Find.Execute(oldStr,EmptyParam,EmptyParam, EmptyParam,EmptyParam,EmptyParam,EmptyParam, EmptyParam,EmptyParam,newStr,replace); oldStr:='@2'; newStr:=WordApplication1.UserName; WordDocument1.Range.Find.Execute(oldStr,EmptyParam,EmptyParam, EmptyParam,EmptyParam,EmptyParam,EmptyParam, EmptyParam,EmptyParam,newStr,replace); ………//сохранение документа и отображение его в OLE контейнере (предварительный просмотр) WordDocument1.SaveAs(FileName); WordDocument1.Close; OleContainer1.CreateLinkToFile(FileName,false); OleContainer1.Refresh;end;Исходный код этого примера имеется в приложении к данной статьи. Обратите внимание на передачу параметров в методы. Все параметры описаны как OleVariant и передаются в методы по ссылке (такая передача параметров характерна лишь для сервера Word, Excel и OutLook, например, принимают в основном фактические параметры). Сервер автоматизации сам будет разбираться, какой тип вы фактически подставили в качестве значения переменной, поэтому необходимо при разработке приложения особое внимание уделять хинтам, и почаще заглядывать в библиотеку типов сервера, с которым вы работаете в настоящее время. К стати, после того как мы установили компоненты, библиотеки типов разместились в директории …Delphi5\Imports\. В файлах библиотек типов можно найти, например все константы, необходимые для передачи параметров в Excel и OutLook, названия их интуитивно понятны. В модуле System – Delphi 5 описана переменная EmptyParam, которую необходимо использовать для передачи “пустышек” в качестве параметров. varEmptyParam: OleVariant; { "Пустой параметр" который должен опционально использоваться в DUAL интерфейсе. }Описание методов в библиотеке типов заставляет нас очень аккуратно соблюдать порядок передачи параметром, это несколько затрудняет процесс программирования, но зато Dual интерфейс работает значительно быстрее чем передача именованных параметров при позднем связывании сервера и контроллера автоматизации.^ Пример использования Dispatch интерфейсаПостановка задачи: получить информацию о документе, который открывается MS Word, при этом не должны использоваться никакие компоненты для доступа к серверам автоматизации.Создадим новое приложение и выложим на форму кнопку. Пропишем событие onDoubleClick для кнопки следующим образом:procedure TForm1.Button1Click(Sender: TObject);var Shablon:OleVariant;word:_ApplicationDisp;begin Shablon:=ExtractFilePath(Application.EXEName)+'shablon.DOC'; word:=CoWordapplicaTion.Create as _ApplicationDisp; (Word.Documents as DocumentsDisp).Open(Shablon,EmptyParam, EmptyParam,EmptyParam,EmptyParam,EmptyParam, EmptyParam,EmptyParam,EmptyParam,EmptyParam); showmessage((Word.Application as _application).Get_Name+#13+ ((Word.Application as _application).ActiveDocument as _documentDisp).Path ); word.quit(EmptyParam,EmptyParam,EmptyParam);end;Переменная word имеет тип ApplicationDisp = dispinterface ['{00020970-0000-0000-C000-000000000046}'], который описан в библиотеки типов Word97_TLB.Pas. Строка word:=CoWordapplicaTion.Create as _ApplicationDisp; создает экземпляр сервера и возвращает DispInterface. Благодаря наличию библиотеки и жесткому приведению типов разработчик имеет возможность делать меньше ошибок, так как компилятор имеет определенную информацию о типах данных. В сравнении с использованием метода позднего связывания через функцию CreateOleObject этот метод более прогрессивный. По крайней мере, хотя бы часть информации компилятору да известна.Подводя итог выше сказанному, можно отметить, что действительно использование компонент для доступа к COM серверам через Vtable интерфейс значительно ускоряет работу приложения и ограждает разработчика от ошибок.^ Другие компоненты для доступа к COM серверамТеперь поговорим о сервере Excel. После импортирования библиотеки типов, на закладке палитры компонент появились шесть иконок, основная из них ExcelApplication - компонента, обеспечивающая доступ к объекту Application сервера Excel. Следующий пример демонстрирует работу с объектом Application и WorkBook сервера Excel. Параметры серверу передаются по значению, кроме того, библиотека типов предоставляет все основные константы для работы с сервером:const xlDays = $00000000; xlMonths = $00000001; xlYears = $00000002;procedure TForm1.FormCreate(Sender: TObject);beginExcelApplication1.Workbooks.add(ExtractFilePath(Application.EXEName)+'customs',0); ExcelWorkbook1.ConnectTo(ExcelApplication1.ActiveWorkbook); ExcelApplication1.Visible[0]:=true;end; Таким образом получив доступ к серверу мы можем легко формировать всевозможные отчеты, строить графики. Другими словами из Delphi пользоваться ресурсами MS Excel. ^ Работа с Outlook из DelphiТак же как и в других серверах, в Outlook имеется класс Application, в который встраивается класс NameSpace. Последний предоставляет доступ к данным через объект класса MAPIFolders, представляющий собой коллекцию папок пользователя. Получив доступ к Outlook через компоненту OutLookApplication, извлекается объект доступа к MAPI папкам.mapi:=OutlookApplication1.GetNamespace('MAPI'); for i:=1 to mapi.Folders.Count doListBox1.Items.Add(mapi.Folders.Item(i).Name);Добраться до соответствующей папки теперь, нет проблем, так как мы имеем дело с вложенными объектами. В приложении к статье имеется пример извлечения содержимого всех папок сервера Microsoft OutLook.Следующий пример производит рассылку писем, извлекая адреса из таблицы базы данных. Я вам предлагаю его самостоятельно доработать и привести к “товарному виду”.procedure TForm1.Button1Click(Sender: TObject);var mapi:NameSpace;begin Table1.Active:=false;Table1.Active:=true;//Получаем доступ к папкеmapi:=OutlookApplication1.GetNamespace('MAPI');while not Table1.Eof dobegin//Подключаем объект класса TMailItem к новому елементу исходящих писем , для работы через Vtable интерфейсMailItem1.ConnectTo(_DMailItem(mapi.Folders.Item(olPersonal). Folders.Item(olFolderOutbox). Items.Add(olPostItem) as iDispatch));//Наполняем новое письмо информациейMailItem1.Subject:='test for '+Table1Common_Name.Value; MailItem1.to_:=Table1Category.Value+'@elsite.ru'; Table1Graphic.SaveToFile(Table1SpeciesNo.AsString+'.BMP');MailItem1.Attachments.Add(ExtractFilePath(Application.EXEName)+Table1SpeciesNo.AsString+ '.BMP',1,1,Table1SpeciesName.Value);MailItem1.Body:=#13#13+'Письмо с вложенным документом '+#13+ MailItem1.SenderName; MailItem1.Sensitivity:=olConfidential ; MailItem1.FlagStatus:=olFlagMarked;//Сохраняем письмо. OutLook самостоятельно его отошлет по почтеMailItem1.Save; Table1.next;end;end;В заключении хочется еще раз подчеркнуть, что использование Vtable интерфейса значительно упростило процесс разработки контроллеров автоматизации COM серверов. Используя мощный язык Delphi 5 и открытый интерфейс серверов MS Office можно строить очень большие и серьезные приложения, работающие совместно. Copyright © Inprise Representative Office, Moscow, 1999.http://www.borland.ru , http://www.inprise.ru , http://www.inprise.com , http://www.borland.com , http://www.interbase.com