利用Delphi编写IE扩展
就是如何使IE扩展组件可以响应事件。
在自己的程序中使用过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 := to dps.cArgs - do
pDispIds^[i] := dps.cArgs - - i;
if (dps.cNamedArgs <= ) then Exit;
for i := to dps.cNamedArgs - 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 := ;
bHasParams := (dps.cArgs > );
if (bHasParams) then
begin
iDispIdsSize := dps.cArgs * SizeOf(TDispId);
GetMem(pDispIds, iDispIdsSize);
end;
try
if (bHasParams) then BuildPositionalDispIds(pDispIds, dps);
case DispId of
:
begin
DoStatusTextChange(dps.rgvarg^[pDispIds^[]].bstrval);
Result := S_OK;
end;
:
begin
DoProgressChange(dps.rgvarg^[pDispIds^[]].lval, dps.rgvarg^[pDispIds^[]].lval);
Result := S_OK;
end;
:
begin
DoCommandStateChange(dps.rgvarg^[pDispIds^[]].lval, dps.rgvarg^[pDispIds^[]].vbool);
Result := S_OK;
end;
:
begin
DoDownloadBegin();
Result := S_OK;
end;
:
begin
DoDownloadComplete();
Result := S_OK;
end;
:
begin
DoTitleChange(dps.rgvarg^[pDispIds^[]].bstrval);
Result := S_OK;
end;
:
begin
DoPrtype
POleVariant = ^OleVariant;
var
dps: TDispParams absolute Params;
bHasParams: boolean;
pDispIds: PDispIdList;
iDispIdsSize: integer;
begin
Result := DISP_E_MEMBERNOTFOUND;
pDispIds := nil;
iDispIdsSize := ;
bHasParams := (dps.cArgs > );
if (bHasParams) then
begin
iDispIdsSize := dps.cArgs * SizeOf(TDispId);
GetMem(pDispIds, iDispIdsSize);
end;
try
if (bHasParams) then BuildPositionalDispIds(pDispIds, dps);
case DispId of
:
begin
DoStatusTextChange(dps.rgvarg^[pDispIds^[]].bstrval);
Result := S_OK;
end;
:
begin
DoProgressChange(dps.rgvarg^[pDispIds^[]].lval, dps.rgvarg^[pDispIds^[]].lval);
Result := S_OK;
end;
:
begin
DoCommandStateChange(dps.rgvarg^[pDispIds^[]].lval, dps.rgvarg^[pDispIds^[]].vbool);
Result := S_OK;
end;
:
begin
DoDownloadBegin();
Result := S_OK;
end;
:
begin
DoDownloadComplete();
Result := S_OK;
end;
:
begin
DoTitleChange(dps.rgvarg^[pDispIds^[]].bstrval);
Result := S_OK;
end;
:
begin
DoPropertyChange(dps.rgvarg^[pDispIds^[]].bstrval);
Result := S_OK;
end;
:
begin
DoBeforeNavigate2(IDispatch(dps.rgvarg^[pDispIds^[]].dispval), POleVariant(dps.rgvarg^[pDispIds^[]].pvarval)^, POleVariant(dps.rgvarg^[pDispIds^[]].pvarval)^, POleVariant(dps.rgvarg^[pDispIds^[]].pvarval)^, POleVariant(dps.rgvarg^[pDispIds^[]].pvarval)^, POleVariant(dps.rgvarg^[pDispIds^[]].pvarval)^, dps.rgvarg^[pDispIds^[]].pbool^);
Result := S_OK;
end;
:
begin
DoNewWindow2(IDispatch(dps.rgvarg^[pDispIds^[]].pdispval^), dps.rgvarg^[pDispIds^[]].pbool^);
Result := S_OK;
end;
:
begin
DoNavigateComplete2(IDispatch(dps.rgvarg^[pDispIds^[]].dispval), POleVariant(dps.rgvarg^[pDispIds^[]].pvarval)^);
Result := S_OK;
end;
:
begin
DoDocumentComplete(IDispatch(dps.rgvarg^[pDispIds^[]].dispval), POleVariant(dps.rgvarg^[pDispIds^[]].pvarval)^);
Result := S_OK;
end;
:
begin
DoOnQuit();
Result := S_OK;
end;
:
begin
DoOnVisible(dps.rgvarg^[pDispIds^[]].vbool);
Result := S_OK;
end;
:
begin
DoOnToolBar(dps.rgvarg^[pDispIds^[]].vbool);
Result := S_OK;
end;
:
begin
DoOnMenuBar(dps.rgvarg^[pDispIds^[]].vbool);
Result := S_OK;
end;
:
begin
DoOnStatusBar(dps.rgvarg^[pDispIds^[]].vbool);
Result := S_OK;
end;
:
begin
DoOnFullScreen(dps.rgvarg^[pDispIds^[]].vbool);
Result := S_OK;
end;
:
begin
DoOnTheaterMode(dps.rgvarg^[pDispIds^[]].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 := ;
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(SoftwareMicrosoftWindowsCurrentVersionexplorerBrowser 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(SoftwareMicrosoftWindowsCurrentVersionexplorerBrowser 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 事件中编写代码访问服务器并转到正确的站点上去。
利用Delphi编写IE扩展的更多相关文章
- 利用Delphi编写Socket通信程序
一.Delphi与Socket 计算机网络是由一系列网络通信协议组成的,其中的核心协议是传输层的TCP/IP和UDP协议.TCP是面向连接的,通信双方保持一条通路,好比目前的电话线,使用telnet登 ...
- Delphi编写Shell扩展
用delphi创建一个外壳扩展(Shell Extension)程序的基本步骤如下: (1) 创建一个 ActiveX Library 工程,命名为“CloudUpload“(2) 创建一个新的自动化 ...
- Delphi 编写ActiveX控件(OCX控件)的知识和样例(有详细步骤)
一.ActiveX应用情况简介: ActiveX控件也就是一般所说的OCX控件,它是 ActiveX技术的一部分.ActiveX是微软公司推出的基于组件对象模型COM的技术,包括对Windows 32 ...
- 转 : 用Delphi编写安装程序
http://www.okbase.net/doc/details/931 还没有亲自验证过,仅收藏 当你完成一个应用软件的开发后,那么你还需要为该软件做一个规范化的安装程序,这是程序设计的最后一步 ...
- 利用Delphi监视注册表的变化
转帖:利用Delphi监视注册表的变化 2009-12-23 11:53:51 分类: 利用Delphi监视注册表的变化 我们在编写软件的时候,常常需要把一些信息保存到系统的注册表中.如果 ...
- 品味性能之道<九>:利用Loadrunner编写socket性能测试脚本简述
一.概述 Loadrunner拥有极为丰富的工具箱,供予我们制造出各种奇妙魔法的能力.其中就有此次要讨论的socket套接字操作. 二.socket概述 ...
- 一步步入门编写PHP扩展
1.写在最前 随着互联网飞速发展,lamp架构的流行,php支持的扩展也越来越多,这样直接促进了php的发展. 但是php也有脚本语言不可避免的问题,性能比例如C等编译型语言相差甚多,所以在考虑性能问 ...
- Delphi编写WebService体会
源:Delphi编写WebService体会 Dispatch: 派遣,分派 Invoke: 调用 Invokable: 可调用接口 TReomtable: WebService中自定义类都是继承自该 ...
- 利用Delphi的File Of Type创建并管理属于你自己的数据库
http://www.360doc.com/content/16/1128/19/28222077_610249962.shtml 利用Delphi的File Of Type创建并管理属于你自己的数据 ...
随机推荐
- Linux常用命令2(远程文件下载+查看文件内容)
一.远程文件下载的两种方法:ftp命令 + scp命令 ftp命令: 服务器若安装了ftp Server,另外一台Linux可以使用ftp的client程序来进行文件的远程拷贝读取下载和写入上载. 1 ...
- 03-MySql安装和基本管理
本节掌握内容: MySQL的介绍安装.启动 windows上制作服务 MySQL破解密码 MySQL中统一字符编码 MySQL是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,目前属于 O ...
- Node.js Error: listen EADDRNOTAVAIL
1 前言 nodejs部署在云服务器,外网用域名加端口访问不进来,但在服务器本地用127.0.0.1加端口可以访问,并且端口已经放开,然后只能排查配置.此文章仅作为记录使用. 如果端口和另一个的端口一 ...
- JS算法之二分查找
二分查找法主要是解决「在一堆有序的数中找出指定的数」这类问题,不管这些数是一维数组还是 多维数组,只要有序,就可以用二分查找来优化. 二分查找是一种「分治」思想的算法,大概流程如下: 1.数组中排在中 ...
- java :: Java中的双冒号操作符
java中的双冒号操作符 定义 双冒号运算操作符是类方法的句柄,lambda表达式的一种简写,这种简写的学名叫eta-conversion或者叫η-conversion. 通常的情况下: 把 x -& ...
- (无)webservice执行过程深入理解
前面我们搞了1,2个DEMO,基本对webservice服务发布,调用 ,执行 有一定的了解. 今天的话,我们再系统的梳理下webservice执行过程. 首先我们在webservice服务器端开发w ...
- canvas扩散圆环
最近看了很多牛的动画,想想自己的canvas的确很菜. 公式在那里,但是不是太会套.找demo发现都是很难的 于是找了个简单的效果 圆环从中间扩散的效果 关键是 globalCompositeOper ...
- Linux使用netstat命令查看并发连接数
我们的网站部署在linux的服务器上,特别是web服务器,我们可能有时候做为运维人员,肯定是要查看网站的并发连接数是不是达到瓶颈等,所以在linux下,我们如何查看服务器的并发连接数呢?使用以下命令即 ...
- js获取、修改url中参数
//获取url的参数 function getParam(paramKey){ //获取当前URL var url = location.href; //获取要取得的get参数位置 var get = ...
- 计蒜客 无脑博士的试管们 【dfs】
题目链接:https://nanti.jisuanke.com/t/31 题目大意: 无脑博士有三个容量分别是A,B,C 升的试管,A,B,C 分别是三个从 1 到20 的整数,最初,A 和 B 试管 ...