<form id="hz9zz"></form>
  • <form id="hz9zz"></form>

      <nobr id="hz9zz"></nobr>

      <form id="hz9zz"></form>

    1. 明輝手游網中心:是一個免費提供流行視頻軟件教程、在線學習分享的學習平臺!

      用Delphi編寫IE擴展

      [摘要]在自己的程序中使用過WebBrowser控件的朋友都知道,WebBrowser控件定義了諸如BeforeNavigate、DownloadComplete 等事件,我們可以通過編寫事件處理代碼實現對...
      在自己的程序中使用過WebBrowser控件的朋友都知道,WebBrowser控件定義了諸如BeforeNavigate、DownloadComplete 等事件,我們可以通過編寫事件處理代碼實現對WebBrowser控件的操作。那么如何實現對IE的事件響應和處理呢?同建立IE面板一樣。我們需要建立一個實現IObjectWithSite接口的COM組件,不同的是,我們還需要實現IDispatch接口,在IObjectWithSite接口的SetSite方法中獲得IE的WebBrowser接口并建立自身與WebBrowser的連接,然后如果在IE的Webbrowser對象中發生什么事件的話,那么IE就會回調連接的IDispatch接口的Invoke方法。我們通過在Invoke方法中編寫代碼就可以獲得IE事件了。這個利用的是COM編程的回調接口原理。
      下面我們首先來實現代碼。點擊Delphi菜單 File New 。在 ActiveX 頁面中選擇Active Library ,然后點擊 OK 按鈕。然后用同樣的方法建立一個COM Object。在COM Object Wizard 窗口中,將復選框 Included type library 去掉。然后在Class Name中輸入IEHelper,在Implemented Interface 中輸入:IDispatch;IObjectwithSite 。然后點擊 OK 按鈕建立一個COM組件。

        保存工程,將工程保存為IEHelper.dpr,將Unit1保存為IEHelperUnit.pas。下面是IEHelperUnit.pas的具體代碼:

      unit iehelperunit;

      interface

      uses
      WIndows, Comobj, ActiveX, SHDOCVW, MSHTML,Dialogs;


      type

      TIEHelperFactory = class(TComObjectFactory)
      private
      procedure AddKeys;
      procedure RemoveKeys;
      public
      procedure UpdateRegistry(Register: Boolean); override;
      end;


      TIEHelper = class(TComObject, IDispatch, IObjectWithSite)
      public
      function GetTypeInfoCount(out Count: Integer): HResult; stdcall;
      function GetTypeInfo(Index, LocaleID: Integer; out TypeInfo): HResult; stdcall;
      function GetIDsOfNames(const IID: TGUID; Names: Pointer;
      NameCount, LocaleID: Integer; DispIDs: Pointer): HResult; stdcall;
      function Invoke(DispID: Integer; const IID: TGUID; LocaleID: Integer;
      Flags: Word; var Params; VarResult, ExcepInfo, ArgErr: Pointer): HResult; stdcall;
      function SetSite(const pUnkSite: IUnknown): HResult; stdcall;
      function GetSite(const riid: TIID; out site: IUnknown): HResult; stdcall;
      private
      IE: IWebbrowser2;
      Cookie: Integer;
      end;

      const
      Class_IEHelper: TGUID = '{3D898C55-74CC-4B7C-B5F1-45913F368388}';


      implementation

      uses ComServ, Registry, SysUtils;


      procedure DoStatusTextChange(const Text: WideString);
      begin

      end;

      procedure DoProgressChange(Progress: Integer; ProgressMax: Integer);
      begin

      end;

      procedure DoCommandStateChange(Command: Integer; Enable: WordBool);
      begin

      end;

      procedure DoDownloadBegin;
      begin

      end;

      procedure DoDownloadComplete;
      begin

      end;

      procedure DoTitleChange(const Text: WideString);
      begin

      end;

      procedure DoPropertyChange(const szProperty: WideString);
      begin

      end;

      procedure DoBeforeNavigate2(const pDisp: IDispatch; var URL: OleVariant; var Flags: OleVariant; var TargetFrameName: OleVariant; var PostData: OleVariant; var Headers: OleVariant; var Cancel: WordBool);
      begin
      if URL<>'http://www.applevb.com/'then begin
      Showmessage('你不可以瀏覽其它站點');
      Cancel:=True;
      URL:='http://www.applevb.com';
      (pDisp as IWebbrowser2).Navigate2(URL,Flags,TargetFrameName,PostData,Headers);
      end;
      end;

      procedure DoNewWindow2(var ppDisp: IDispatch; var Cancel: WordBool);
      begin

      end;

      procedure DoNavigateComplete2(const pDisp: IDispatch; var URL: OleVariant);
      begin

      end;

      procedure DoDocumentComplete(const pDisp: IDispatch; var URL: OleVariant);
      begin

      end;

      procedure DoOnQuit;
      begin

      end;

      procedure DoOnVisible(Visible: WordBool);
      begin

      end;

      procedure DoOnToolBar(ToolBar: WordBool);
      begin

      end;

      procedure DoOnMenuBar(MenuBar: WordBool);
      begin

      end;

      procedure DoOnStatusBar(StatusBar: WordBool);
      begin

      end;

      procedure DoOnFullScreen(FullScreen: WordBool);
      begin

      end;

      procedure DoOnTheaterMode(TheaterMode: WordBool);
      begin

      end;


      procedure BuildPositionalDispIds(pDispIds: PDispIdList; const dps: TDispParams);
      var
      i: integer;
      begin
      Assert(pDispIds <> nil);
      for i := 0 to dps.cArgs - 1 do
      pDispIds^[i] := dps.cArgs - 1 - i;
      if (dps.cNamedArgs <= 0) then Exit;
      for i := 0 to dps.cNamedArgs - 1 do
      pDispIds^[dps.rgdispidNamedArgs^[i} := i;
      end;

      function TIEHelper.Invoke(DispID: Integer; const IID: TGUID; LocaleID: Integer;
      Flags: Word; var Params; VarResult, ExcepInfo, ArgErr: Pointer): HResult;
      type
      POleVariant = ^OleVariant;
      var
      dps: TDispParams absolute Params;
      bHasParams: boolean;
      pDispIds: PDispIdList;
      iDispIdsSize: integer;
      begin
      Result := DISP_E_MEMBERNOTFOUND;
      pDispIds := nil;
      iDispIdsSize := 0;
      bHasParams := (dps.cArgs > 0);
      if (bHasParams) then
      begin
      iDispIdsSize := dps.cArgs * SizeOf(TDispId);
      GetMem(pDispIds, iDispIdsSize);
      end;
      try
      if (bHasParams) then BuildPositionalDispIds(pDispIds, dps);
      case DispId of
      102:
      begin
      DoStatusTextChange(dps.rgvarg^[pDispIds^[0}.bstrval);
      Result := S_OK;
      end;
      108:
      begin
      DoProgressChange(dps.rgvarg^[pDispIds^[0}.lval, dps.rgvarg^[pDispIds^[1}.lval);
      Result := S_OK;
      end;
      105:
      begin
      DoCommandStateChange(dps.rgvarg^[pDispIds^[0}.lval, dps.rgvarg^[pDispIds^[1}.vbool);
      Result := S_OK;
      end;
      106:
      begin
      DoDownloadBegin();
      Result := S_OK;
      end;
      104:
      begin
      DoDownloadComplete();
      Result := S_OK;
      end;
      113:
      begin
      DoTitleChange(dps.rgvarg^[pDispIds^[0}.bstrval);
      Result := S_OK;
      end;
      112:
      begin
      DoPropertyChange(dps.rgvarg^[pDispIds^[0}.bstrval);
      Result := S_OK;
      end;
      250:
      begin
      DoBeforeNavigate2(IDispatch(dps.rgvarg^[pDispIds^[0}.dispval), POleVariant(dps.rgvarg^[pDispIds^[1}.pvarval)^, POleVariant(dps.rgvarg^[pDispIds^[2}.pvarval)^, POleVariant(dps.rgvarg^[pDispIds^[3}.pvarval)^, POleVariant(dps.rgvarg^[pDispIds^[4}.pvarval)^, POleVariant(dps.rgvarg^[pDispIds^[5}.pvarval)^, dps.rgvarg^[pDispIds^[6}.pbool^);
      Result := S_OK;
      end;
      251:
      begin
      DoNewWindow2(IDispatch(dps.rgvarg^[pDispIds^[0}.pdispval^), dps.rgvarg^[pDispIds^[1}.pbool^);
      Result := S_OK;
      end;
      252:
      begin
      DoNavigateComplete2(IDispatch(dps.rgvarg^[pDispIds^[0}.dispval), POleVariant(dps.rgvarg^[pDispIds^[1}.pvarval)^);
      Result := S_OK;
      end;
      259:
      begin
      DoDocumentComplete(IDispatch(dps.rgvarg^[pDispIds^[0}.dispval), POleVariant(dps.rgvarg^[pDispIds^[1}.pvarval)^);
      Result := S_OK;
      end;
      253:
      begin
      DoOnQuit();
      Result := S_OK;
      end;
      254:
      begin
      DoOnVisible(dps.rgvarg^[pDispIds^[0}.vbool);
      Result := S_OK;
      end;
      255:
      begin
      DoOnToolBar(dps.rgvarg^[pDispIds^[0}.vbool);
      Result := S_OK;
      end;
      256:
      begin
      DoOnMenuBar(dps.rgvarg^[pDispIds^[0}.vbool);
      Result := S_OK;
      end;
      257:
      begin
      DoOnStatusBar(dps.rgvarg^[pDispIds^[0}.vbool);
      Result := S_OK;
      end;
      258:
      begin
      DoOnFullScreen(dps.rgvarg^[pDispIds^[0}.vbool);
      Result := S_OK;
      end;
      260:
      begin
      DoOnTheaterMode(dps.rgvarg^[pDispIds^[0}.vbool);
      Result := S_OK;
      end;
      end;
      finally
      if (bHasParams) then FreeMem(pDispIds, iDispIdsSize);
      end;
      end;


      function TIEHelper.GetIDsOfNames(const IID: TGUID; Names: Pointer;
      NameCount, LocaleID: Integer; DispIDs: Pointer): HResult;
      begin
      Result := E_NOTIMPL;
      end;

      function TIEHelper.GetTypeInfo(Index, LocaleID: Integer;
      out TypeInfo): HResult;
      begin
      Result := E_NOTIMPL;
      pointer(TypeInfo) := nil;
      end;

      function TIEHelper.GetTypeInfoCount(out Count: Integer): HResult;
      begin
      Result := E_NOTIMPL;
      Count := 0;
      end;


      function TIEHelper.GetSite(const riid: TIID; out site: IUnknown): HResult;
      begin
      // Result := S_OK;
      if Assigned(IE) then result:=IE.QueryInterface(riid, site)
      else
      Result:= E_FAIL;
      end;

      function TIEHelper.SetSite(const pUnkSite: IUnknown): HResult;
      var
      cmdTarget: IOleCommandTarget;
      Sp: IServiceProvider;
      CPC: IConnectionPointContainer;
      CP: ICOnnectionPoint;
      begin
      if Assigned(pUnkSite) then begin
      cmdTarget := pUnkSite as IOleCommandTarget;
      Sp := CmdTarget as IServiceProvider;

      if Assigned(Sp)then
      Sp.QueryService(IWebbrowserApp, IWebbrowser2, IE);
      if Assigned(IE) then begin
      IE.QueryInterface(IConnectionPointContainer, CPC);
      CPC.FindConnectionPoint(DWEBbrowserEvents2, CP);
      CP.Advise(Self, Cookie)
      end;
      end;
      Result := S_OK;
      end;


      procedure TIEHelperFactory.AddKeys;
      var S: string;
      begin
      S := GUIDToString(CLASS_IEHelper);
      with TRegistry.Create do
      try
      RootKey := HKEY_LOCAL_MACHINE;
      if OpenKey('Software\Microsoft\Windows\CurrentVersion\explorer\Browser Helper Objects\' + S, TRUE)
      then CloseKey;
      finally
      free;
      end;
      end;

      procedure TIEHelperFactory.RemoveKeys;
      var S: string;
      begin
      S := GUIDToString(CLASS_IEHelper);
      with TRegistry.Create do
      try
      RootKey := HKEY_LOCAL_MACHINE;
      DeleteKey('Software\Microsoft\Windows\CurrentVersion\explorer\Browser Helper Objects\' + S);
      finally
      free;
      end;
      end;

      procedure TIEHelperFactory.UpdateRegistry(Register: Boolean);
      begin
      inherited UpdateRegistry(Register);
      if Register then AddKeys else RemoveKeys;
      end;

      initialization
      TIEHelperFactory.Create(ComServer, TIEHelper, Class_IEHelper,
      'IEHelper', '', ciMultiInstance, tmApartment);
      end.


      代碼很長,但是關鍵的是TIEHelper.SetSite方法以及TIEHelper.Invoke方法。在TIEHelper.SetSite方法中注意以下語句:
      if Assigned(Sp)then
      Sp.QueryService(IWebbrowserApp, IWebbrowser2, IE);
      if Assigned(IE) then begin
      IE.QueryInterface(IConnectionPointContainer, CPC);
      CPC.FindConnectionPoint(DWEBbrowserEvents2, CP);
      CP.Advise(Self, Cookie)


        上面的語句作用是,首先獲得IE的Webbrowser接口,然后尋找到連接點。并通過Advise方法建立COM自身與連接點的連接。
        當連接建立成功后,IE在有事件引發后,會調用連接到自身的IDispatch接口對象的Invoke方法。不同的事件對應不同的DispID編碼,我們可以在程序中判斷DispID并做相應的處理。在上面的程序中,我們只處理了BeforeNavigate2 事件,處理函數是DoBeforeNavigate2,在該函數中,如果瀏覽的站點不是'http://www.applevb.com/'的話,程序會提示:'你不可以瀏覽其它站點'并強行轉到http://www.applevb.com。
      很多的軟件,象“護花使者”以及“3721”一類的中文網址”都是利用上面的原理來實現對IE瀏覽器事件響應的,例如3721,當用戶輸入一個中文詞并瀏覽時,COM組件可以在BeforeNavigate2 事件中編寫代碼訪問服務器并轉到正確的站點上去。
        以上程序在Win2K、Delphi 5下編寫 Win98、Win2K下編輯通過,如果大家需要源程序或者對于COM編程需要有什么的指教的話,歡迎到我的主頁 http://www.applevb.com 訪問,我愿意同大家一起探討。


      日韩精品一区二区三区高清