抢先式弹出窗口杀手

** 抢先式弹出窗口杀手 **

演示下载 :

http://www.arstdesign.com/articles/preemptivepopupkiller_demo.zip

代码下载 :

http://www.arstdesign.com/articles/preemptivepopupkiller_src.zip

首先 ,本文不是其他弹出窗口程序的复制,本文的独特之处在于讲述如何在窗口弹出之前来杀死他们.这是本文于其他弹出窗口杀手程序的不同之处.
为了更有效的说明问题,本文讲述了如何用C++和C#来实现.


** 什么是抢先是弹出窗口杀手 **

普通的弹出窗口杀手是每隔一定时间就检测 IE窗口的标题.这就是说,他们不能在弹出窗口弹出之前做任何的事情.同样,以前被禁止(放在禁止列表中)显示的窗口也会再弹出来,除此之外,由于弹出窗口的标题有时是动态连接的url,所以需要查询已经禁止的窗口的字典.虽然这类程序都提供系统热键,但也需要用户一定的行动.

本文所讲述的方法能够在弹出窗口出现之前,就杀死他们.

** 利用 IE的事件 **

首先 , 我们从浏览网页时IE发生的事件看起,事件原型是 ** OnNewWindow2([out] IDispatch*, [out] BOOL *bCancel) ** ,每当打开一个新的窗口时候都会触发这个事件.即:

  • 当用户在一个连接上 ,点击右键(或者是图片等),然后选择”在新窗口中打开”.

  • 用 javascript打开窗口,执行了window.open("http://doubleshit.com/...","ad window", ...)

除了这两个选项,开发人员也许打算自始至终的管理鼠标的右键事件,但是这有一个缺点,不仅仅是右键点击才会出右键菜单,键盘上的”windows右键”也会可以弹出右键.实际上,一个比较聪明的办法是理解浏览器在解析HTML后如何显示弹出窗口,从这个地方入手.我们就可以作出这样:一旦OnNewWindow2在html文档还没有处理完毕的时候被调用,我们设置*bCancel = true,就会取消弹出窗口.

首先 ,让我们来看一下C++的代码实现

我们使用 CHtmlView MFC 类,这是一个简单包装的浏览器控件. OnNewWindow2 事件已经包含在其中了.

void CHtmlViewEx::OnNewWindow2( LPDISPATCH* ppDisp, BOOL* Cancel )

{

// GetBrowserInstance() is a member method of MFC's CHtmlView

if ( m_bFilterPopups && GetBrowserInstance() )

{

//检查document的状态还没有完结之前,是否有弹出窗口

//typedef enum tagREADYSTATE{

// READYSTATE_UNINITIALIZED = 0,

// READYSTATE_LOADING = 1,

// READYSTATE_LOADED = 2,

// READYSTATE_INTERACTIVE = 3,

// READYSTATE_COMPLETE = 4

//} READYSTATE;

READYSTATE nReadyState;

GetBrowserInstance()->get_ReadyState(&nReadyState);

if (nReadyState!=READYSTATE_COMPLETE)

{

*Cancel = TRUE;

return;

}

}

// 否则,一切正常,运行产生弹出窗口

((CWwApp*)AfxGetApp())->NewDocument();

CMDIFrameWnd pFrame = (CMDIFrameWnd)AfxGetApp()->m_pMainWnd;

CMDIChildWnd *pChild = (CMDIChildWnd *) pFrame->GetActiveFrame();

CWwView pNewView = (CWwView) pChild->GetActiveView();

*ppDisp = pNewView->GetBrowserInstance();

}

** 用 C#实现弹出窗口杀手 **

让我们一步一步来实现所要的功能 ,首先从最初缺省的窗体开始.

首先 ,将浏览器控件添加到工具箱中,在工具箱中,右键,自定义工具箱,然后在com组件中,选择” Microsoft web 浏览器(shdocvw.dll) ”,然后把它拖放到窗体中.

完成这一步,VS.Net的集成环境做了不少事情.它用 aximp (在VS.Net的 tool目录下)导入浏览器ActiveX控件,它导出了.Net下的安全组件,在你的obj目录下产生了两个文件 Interop.SHDocVw.dll和AxInterop.SHDocVw.dll . 用VS.Net的自带工具 ildasm (也在tool目录下),我们来看看它的工作机理

从上面的屏幕截图我们可以看到 , _Invoke([in][out] object & marshal( idispatch) ppDisp, [in][out] bool& Cancel) _ 会在另一个时间处理器中被重写.

接着,我们打开另一个dll文件( AxInterop.SHDocVw.dll), 来寻找 DWebBrowserEvents2_NewWindow2EventHandle 相对应符号:

现在我们看到 , AxInterop.SHDocVw.dll 就是我们要使用的dll.我们只要对这个事件处理,就可以阻止窗口弹出.

剩下要做的事情就是在窗体中添加浏览器控件 ,然后对NewWindows2分配事件对于函数.参考下面的图形.

代码骨架和前面的 C++很相似,我们在代码中解释一下.

// 描述 : NewWindow2 事件处理

//

// 用途 :阻止弹出窗口

//

private void OnNewWindow2EventHandler(object sender, AxSHDocVw.DWebBrowserEvents2_NewWindow2Event e)

{

if ( axWebBrowser1.ReadyState != SHDocVw.tagREADYSTATE.READYSTATE_COMPLETE)

{

e.ppDisp = null;

e.cancel = true;

return;

}

// 这种情况下是合理的,所以要产生一个新窗口

Form1 newwindow = new Form1();

newwindow.Text = "(new browser window)";

e.ppDisp = newwindow.axWebBrowser1.Application;

// and finally show the new window

newwindow.Show();

}

// command handlers

//

private void button_GoBack(object sender, System.EventArgs e)

{ //上一步

axWebBrowser1.GoBack();

}

private void button_Refresh(object sender, System.EventArgs e)

{ //刷新

axWebBrowser1.CtlRefresh();

}

// called when the user hits VK_ENTER in the address bar

private void OnNewUrl(object sender, System.Windows.Forms.KeyEventArgs e)

{ //在地址栏中敲回车

if (e.KeyCode==Keys.Enter)

Navigate( textBox1.Text );

}

protected void SyncUI(String sURL)

{

textBox1.Text = sURL; // update UI

}

public void Navigate(string sURL)

{

SyncUI(sURL);

object o = new object();

object oURL = (object) sURL;

this.axWebBrowser1.Navigate2( ref oURL,

ref o/ref object flags/,

ref o/ref object targetframe/,

ref o/ref object postdata/,

ref o/ref object headers/);

}

Published At
Categories with Web编程
Tagged with
comments powered by Disqus