使用 Delphi 开发Office Word插件

    在Office 2000中提供了基于COM的插件开发框架,这使得我们可以利用Delphi来扩展Office的功能。

    在Delphi 3,4中编写基于COM的插件,我们需要自己创建COM接口的封装类,更糟糕的是要想支持事件的话还需要使用连接点(connection points)对象来实现事件回调,这是非常麻烦的。但在Delphi 5中这一切就变得非常轻松了,Delphi 5的类型库引入工具提供了/L+的开关,可以自动为我们生成封装好的OLE Server。这下子再也没有什么好抱怨的了。

Office 2000 插件框架

    在Microsoft'的网站上,知识库文章(Knowledge Base article Q230689)中有一篇:Office 2000 COM Add-In Written in Visual C++ 。文章中提供了一个例子(http://support.microsoft.com/download/support/mslfiles/ COMADDIN.EXE)。这篇文章详细地描述了插件框架中的COM接口。仔细研究一下C++代码就可以了解如何编写Office 2000插件。

    Office 2000插件其实就是一个实现了IDTExtensibility2接口的自动化对象。IDTExtensibility2 接口相当简单,插件需要实现接口定义的全部5个函数:

    OnConnection:当应用程序连接到插件时会调用这个函数。插件在函数中接收下列初始化信息——应用程序对象模型进入点的指针,连接模式(是手工加入还是通过命令行载入), 应用程序的对象模型指针和用户自定义的信息。

    OnDisconnection:当应用程序断开插件时被调用,插件应该在这里清除先前分配的资源,删除它添加到应用程序的界面元素。

    OnStartupComplete:这个函数是当应用程序自动启动插件时被调用的。调用时,其他的插件都已经被加载到了内存,这时可以同其他插件进行通信。这个函数还适合添加用户界面元素。

    OnBeginShutdown:当应用程序准备关闭并将要断开插件时会被调用,这时插件应该停止接收用户输入。

    OnAddInsUpdate:当注册的插件列表被改变后会被调用。如果我们的插件不依赖于其他插件,这个函数可以为空。

接口、类型库和常数

    创建插件前,我们需要引入COM对象的接口类型库。这里使用Delphi 5带的TlibImp.exe (Delphi5\Bin目录下)来引入类型库。新版的TlibImp.exe支持新的/L+开关,可以自动创建一个OLE Server的Delphi封装。IDTExtensibility2接口是在MSADDNDR.DLL文件中声明的,位于\Program Files\Common Files\Designer\ 目录下。调用TLIBIMP\L+\Program Files\Common Files\ Designer\MSADDNDR.DLL会生成AddInDesignerObjects_TLB.pas 和 AddInDesignerObjects_TLB.dcr两个文件。在项目的uses部分加上对上面文件的引用以便使用接口。clause of our project to gain access to the interface.使用时注意:TLIBIMP重命名接口为_IDTExtensibility2。

    本文中将使用Word 2000作为例子,如果想编写Outlook、Excel或其他Office程序的插件需要引入相应特定的类型库。比如Word的类型库是定义在\ Program Files\Microsoft Office\Office\MSWORD9.OLB文件中。类似的,Excel、Access和OutLook类型库分别定义在EXCEL9.OLB、 MSACC9.OLB和MSOUTL9.OLB文件中。引入的接口生成在Office_TLB.pas和Word_TLB.pas单元中。

    注意:Office 2000的插件无法工作在Office 97的应用程序中。

最简单的插件

    现在让我们来实现一个最简单的插件,它只实现了IDTExtensibility2接口而没有实现任何比较有意义的功能,但对于演示如何实现插件是一个很好的开始。

    插件可以以进程内或进程外COM服务器的形式实现,在本文中,我们创建的是进程内COM服务器。在Delphi中,选择菜单File | New命令,然后创建一个ActiveX Library,保存生成的文件,再创建一个自动化对象(Automation Object),类名定义为AddIn,把实现单元保存为AddInMain.pas。在AddInMain.pas单元的uses部分添加对 AddinDesignerObjects_TLB,Word_TLB和Office_TLB单元的引用。最后添加 IDTExtensibility2 接口到类定义部分定义类要实现的接口。类定义如下:

    type

      TAddIn = class(TAutoObject, IAddIn, IDTExtensibility2)

    ...

    在类声明的protected部分,添加IDTExtensibility2 接口声明的方法定义,代码示意如下:

    // IDTExtensibility2 methods

    procedure OnConnection(const Application: IDispatch;

      ConnectMode: ext_ConnectMode; const AddInInst: IDispatch;

    var custom: PSafeArray); safecall;

    procedure OnDisconnection(RemoveMode: ext_DisconnectMode;

    var custom: PSafeArray); safecall;

    procedure OnAddInsUpdate(var custom: PSafeArray); safecall;

    procedure OnStartupComplete(var custom: PSafeArray);  safecall;

    procedure OnBeginShutdown(var custom: PSafeArray); safecall;

    使用快捷键[Ctrl][Shift][C]来完成类定义,并添加方法的实现部分的框架到单元中。为了测试插件,可添加下面代码到OnConnection方法中:

    ShowMessage('连接到' + WordApp.Name);

    添加下面代码到OnDisconnection方法的实现部分:

    ShowMessage('断开插件');

    这样就完成了一个最简单的插件了,接下来就是编译并注册插件到Word中去。

注册Office插件

    同其他COM对象一样,一个Office插件必须在系统中注册后才能使用。在Delphi中选择Run | Register ActiveX Server菜单命令,就可以注册我们刚才创建的插件。除了标准的COM注册,还需要进行Office 相关的注册,这需要在注册表中创建一个新的键值:

    HKEY_CURRENT_USER\Software\Microsoft\Office\

      <AppName>\Addins\<AddInProgID>

    <AppName>就是插件宿主应用程序的名字(这里是Word),<AddInProgID>是自动化对象的名字(这里是DIWordAddIn.AddIn,ActiveX library和类名的组合)。

    HKEY_CURRENT_USER \ Software \ Microsoft \ Office \ Word \ Addins\DIWord AddIn.AddIn

    我们还需要在这个键值下创建几个值:一个DWORD类型的名为LoadBehavior的值决定插件是如何加载及被应用程序调用的。在本文中我们设定它为 3–相当于1和2的结合就是应用程序连接插件并在启动时自动加载。值的意义列在表1.2中,各种值可以相互组合。

表1.2

意    义

$0

  断开,不加载

$1

  连接,加载

$2

  自动启动加载

$8

  只有当用户请求时才加载

$16

  只在下次程序启动时加载一次

    还有一些其他的值可以出现在注册表键值下,比如定义出现在应用程序COM管理器对话框中的名字,以及设定是否可以从命令行激活插件。

Office 2000用户界面

    Office应用程序共享一组通用的用户界面元素对象、菜单条、工具条通用控件(比如工具条按钮和组合编辑框)以及Office小助手。

    此前引入的Word类型库就包括了这些通用对象的类型库,但是Delphi引入时并没有像通常那样建立一个封装好的OLE Server,我们不得不手工创建一个Office公开的CommandBarButton对象的Delphi封装。这个对象对应于Office应用程序 的一个简单的菜单项或工具条按钮。

    对大多数的Microsoft的应用程序来说,Application对象代表对象模型的切入点。Office Application类提供了对CommandBars属性的引用。CommandBar对象包括工具条、浮动工具条和菜单。Office对象模型允许 我们创建或更新已有的CommandBars对象。Office_TLB.pas单元包含了ICommandBar接口,它可以被用来修改 CommandBar对象。

    CommandBar有一个Controls集合属性,对应于一组CommandBarControl控件。CommandBarControl控件对应 于放置在工具条上的控件,比如一个CommandBarButton对应一个简单的工具条按钮(或菜单项),CommandBarCombo控件对应组合 编辑框,CommandBarPopup对应于下拉菜单和CommandBarActiveX对应于ActiveX控制。

    在Office_TLB.pas单元中,除了ICommandBarButton接口,还有一个ICommandBarButtonEvents接口用来 提供对工具条上控件的事件支持。事件的支持通常是通过连接点、事件接收连接到可连接对象来实现。但这比较麻烦,我们还可以通过更简单的办法来实现事件支 持。检查一下Delphi在word_tlb.pas单元创建的TWordApplication的实现代码可以发现Delphi封装了每一个可连接对 象,自动实现了事件接收机制。这个单元可以作为一个范本用来创建自定义的对接口对象的封装。

    BtnSvr.pas单元包含了一个手工创建的Delphi封装。除了按标准的Delphi属性方式实现了CommandBarButton对象的属性 外,还实现了InitServerData、InvokeEvent、Connect、ConnectTo和Disconnect方法。可以注意到这部分 完全是模仿TWordApplication实现部分编写的CommandBarButton事件实现。主要就是在InitServerData方法中定 义服务器数据。根据Office_TLB.pas中不同的接口GUID,定义一个CommandBarButton接口的内部的接口Fintf,设定 InvokeEvent方法来激活基于定义在事件接口部分的DispID的Delphi事件支持。最后,Connect、ConnectTo和 Disconnect方法设定Fintf给需要的接口并接收相应的事件。

    定义在BtnSvr.pas单元中的Delphi封装类命名为TButtonServer。它需要从TOleServer对象继承以便支持事件处理。

同应用程序连接

    有了工具条按钮封装类后,接下来要声明一个TWordApplication域来保存对Word Application对象的引用。此外还需要为新的工具条定义一个接口指针以及两个域使用新的TButtonServer类来保存我们要创建的新的工具 条按钮和菜单项。

    在插件类的private部分添加:

    FWordApp : TWordApplication;

    DICommandBar : CommandBar;

    DIBtn : TButtonServer;

    DIMenu : TButtonServer;

    在OnConnection方法中,保存应用程序指针:

    var

      WA : Word_TLB._Application;

    begin

      FWordApp := TWordApplication.Create(nil);

      WA := Application as Word_TLB._Application;

      WordApp.ConnectTo(WA);

      …………………………..

    TWordApplication是Delphi 5中带的Server组件,ConnectTo 方法是用来连接插件和Word提供的接口。因为TWordApplication 把接口事件映射成了Delphi事件,我们可以直接使用标准的Delphi语法来设定事件处理过程。示意如下:

    WordApp.OnEventX := EventXHandler;

    比如我们如果想在Word的选区发生改变时实现某项功能,就可以设定OnWindowSelectionChange 事件。

插件如何创建新的工具条、按钮和菜单

    在创建新的工具条和按钮前,需要为按钮的OnClick过程先创建事件处理函数,下面就是简单的处理函数例子:

    procedure TAddIn.TestClick(const Ctrl: OleVariant;

      var CancelDefault: OleVariant);

    begin

      ShowMessage('有人点我了!');

      CancelDefault := True;

    end;

    CancelDefault参数用来设定是否替代缺省的菜单或工具条按钮的处理过程。这里不需要设定这个参数,因为我们将在插件中创建一个新的按钮。插件 注册为在程序启动时被加载,所以OnStartupComplete方法一定会被调用,用这个方法创建用户界面元素是比较合适的。这里定义BtnIntf 为CommandBarControl接口类型(要创建的CommandBarButton的父类接口)。接下来的任务是确定自定义的工具条是否已经被创 建了

    DICommandBar := nil;

    for i := 1 to WordApp.CommandBars.Count do

      if (WordApp.CommandBars.Item[i].Name ='Delphi') then

        DICommandBar := WordApp.CommandBars.Item[i];

        // 确定是否已经注册了命令条

      if (not Assigned(DICommandBar)) then begin

        DICommandBar:=

WordApp.CommandBars.Add('Delphi',EmptyParam,EmptyParam,EmptyParam);

        DICommandBar.Set_Protection(msoBarNoCustomize)

    end;

    先给工具条起一个唯一的名字“Delphi”,然后在启动时检查工具条是否已经被创建了。如果是的话就把它赋值给DICommandBar,否则调用 Word的CommandBars属性的Add方法创建一个新的工具条。接着给工具条添加msoBarNoCustomize的保护,这可以防止用户添加 或删除工具条上的按钮。这时DICommandBar指向一个有效的工具条,我们可以从接口的Controls集合中获得工具条按钮接口指针。如果工具条 上没有控件,就创建一个新的按钮。

    if (DICommandBar.Controls.Count > 0) then

      BtnIntf := DICommandBar.Controls.Item[1]

     else

      BtnIntf := DICommandBar.Controls.Add(msoControlButton,

      EmptyParam, EmptyParam, EmptyParam, EmptyParam);

    注意:集合中第一项是以1为底的,而不像Delphi中那样通常以0为底。现在我们获得了需要的工具条按钮接口,然后要创建一个基于按钮接口的TButtonServer 类封装。

     DIBtn := TButtonServer.Create(nil);

     DIBtn.ConnectTo(BtnIntf as _CommandBarButton);

     DIBtn.Caption := 'Delphi Test';

     DIBtn.Style := msoButtonCaption;

     DIBtn.Visible := True;

     DIBtn.OnClick := TestClick;

    这里使用ConnectTo 方法连接按钮的事件并设定先前创建的OnClick事件处理过程。最后,要确认使工具条可见。

    DICommandBar.Set_Visible(True);

    TLIBIMP程序创建了一个只读的而非可读写的工具条Visible 属性,但可以使用Set_Visible 方法来设定显示属性(不能生成可读写的属性可能是TLIBIMP的bug)。添加新的菜单项类似于前面,首先创建菜单的OnClick事件处理函数,下面 这个过程遍历被选文本的第一段,并设定其边框样式:

    procedure TAddIn.MenuClick(const Ctrl: OleVariant;

      var CancelDefault: OleVariant);

    var

      Sel : Word_TLB.Selection;

      Par : Word_TLB.Paragraph;

    begin

      Sel := WordApp.ActiveWindow.Selection;

      if (Sel.Type_ in [wdSelectionNormal,

        wdSelectionIP]) then begin

        Par := Sel.Paragraphs.Item(1);

      if (Par.Borders.OutsideLineStyle < wdLineStyleInset) then

        Par.Borders.OutsideLineStyle := 1 + Par.Borders.OutsideLineStyle

      else

        Par.Borders.OutsideLineStyle := wdLineStyleNone;

      end;

    end;

    在OnStartupComplete方法中,添加下面的代码来获得工具菜单的接口指针,查找自定义的的菜单项,如果没有就创建新的,然后设定它的OnClick事件:

    ToolsBar := WordApp.CommandBars['Tools'];

    MenuIntf := ToolsBar.FindControl(EmptyParam, EmptyParam,

      'DIMenu', EmptyParam, EmptyParam);

    if (not Assigned(MenuIntf)) then

      MenuIntf := ToolsBar.Controls.Add(msoControlButton,

      EmptyParam, EmptyParam, EmptyParam, EmptyParam);

      DIMenu := TButtonServer.Create(nil);

      DIMenu.ConnectTo(MenuIntf as _CommandBarButton);

      DIMenu.Caption := 'Delp&hi Menu';

      DIMenu.ShortcutText := '';

图1.34

      DIMenu.Tag := 'DIMenu';

      DIMenu.Visible := True;

      DIMenu.OnClick := MenuClick;

    CommandBar接口的FindControl方法使用唯一的标识来查找菜单项,如果找到了控件就赋值给 MenuIntf,如果没有找到就创建一个新的菜单项。图1.34显示了自定义的工具条。

清理资源

    注意应该在OnBeginShutdown 方法中清理用户界面元素:

      if (Assigned(DIBtn)) then

      begin

        DIBtn.Free;

        DIBtn := nil;

      end;

      if (Assigned(DIMenu)) then

      begin

        DIMenu.Free;

        DIMenu := nil;

      end;

      if (Assigned(DICommandBar)) then begin

        DICommandBar.Delete;

        DICommandBar := nil;

      end;

    因为插件的框架是通用的,我们可以将同样的OLE Server DLL用于多个应用程序,方法就是确定将激活插件的应用程序,并使用合适的对象模型。最简单的判断方法是在OnConnection中把应用程序的 IDispatch的接口指针赋值给一个OleVariant变量,然后使用相应的Name 属性来确定相应的程序:

    var

      AppVar : OleVariant;

    begin

      AppVar := Application;

      if (AppVar.Name = 'Outlook') then

      begin

        ...

      end

      else if (AppVar.Name = 'Microsoft Word') then

      begin

        ...

      end else ...

    最后,要想获得关于Office开发和Office 2000插件创建更详细的资料,可以查阅microsoft.public.officedev新闻组上的信息。