抢先式弹出窗口杀手

类别:.NET开发 点击:0 评论:0 推荐:

抢先式弹出窗口杀手

 

演示下载:

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*/);

}

 

本文地址:http://com.8s8s.com/it/it45854.htm