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