抢先式弹出窗口杀手
演示下载:
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*/);
}