您的位置:首页 > 其它

弹出窗口杀手(上)

2008-04-24 14:49 190 查看
弹出窗口杀手是一个可以自动关闭IE弹出窗口的程序,它工作在系统的托盘中,按照一定的间隔来检测IE窗口,然后关闭弹出窗体。最后,还提供了用热键来杀掉弹出窗口的功能。
虽然已经有类似的用C++写的程序,但是本文讲述的是用C#来实现这些功能,并且本文所讲的方案在查找窗口上的方法要比更快一些。
这是一个崭新的话题,在Internet上我们还可以看到许多类似的程序。但是我也还是要借这个机会来讲述一些下面的技术在C#中如何实现:l系统托盘l程序切换l计时控件l查找窗口l系统热键

生成一个系统托盘程序

首先,产生一个新的C#WindowsForm程序,将NotifyIcon控件从工具箱中拖到窗体中,如下图所示:

在C#windowsForm程序中添加托盘为了保证系统托盘的图标和应用程序的图标一致,我们用一个共同的图标文件a.ico来设置系统托盘的图标和应用程序的图标。
为了使程序不显示在工具栏上,我们可以设置窗体的visible属性为false.这个可以在窗体属性窗口中直接实现
this.ShowInTaskbar=false;
到目前为止,系统托盘已基本好了,但是我们还没有设置右键菜单,也没有使程序显示和隐藏的功能.

程序切换

首先,程序的主窗体可以根据不同的状态来选择显示或者是隐藏,除此之外,我们可以用WindowState设置窗体的状态:
publicvoidHideApp()
{
this.WindowState=FormWindowState.Minimized;
Hide();
}
publicvoidShowApp()
{
Show();
this.WindowState=FormWindowState.Normal;
}
一个非常有趣的功能是让用户关闭窗体的时候程序并不是退出,为了实现这个功能,我们必须要重写窗体的OnClosing事件.
protectedoverridevoidOnClosing(CancelEventArgse)
{
//用最小化来代替关闭操作d
e.Cancel=true;
//最小化,并且隐藏窗体
this.WindowState=FormWindowState.Minimized;
Hide();
}
当然,我们必须要提供一个必须的退出方法.这个可以在托盘的右键菜单的exit中实现,
privatevoidmenu_App_Exit(objectsender,System.EventArgse)
{
NativeWIN32.UnregisterHotKey(Handle,100);
//隐藏托盘
notifyIcon1.Visible=false;
Application.Exit();
}

添加右键菜单

添加一个右键菜单和添加托盘基本一样,从工具箱中添加contextmenu就可以.右键菜单在你鼠标右键按下的时候是会自动弹出的.
当设置好右键菜单以后,我们必要要根据不同的情况来启用或停用右键菜单,这个可以通过在菜单的BeforePopup设置.Enabled属性来实现.
privatevoidmenu_App_BeforePopup(objectsender,System.EventArgse)
{
if(this.WindowState==FormWindowState.Minimized)
{
App_Show.Enabled=true;
App_Hide.Enabled=false;
}
else
{
App_Show.Enabled=false;
App_Hide.Enabled=true;
}
}

计时工具

.NetFramework的Timer能和系统的Win32timer实现一样的功能.我们要做的就是设置一个timer,然后合理的设置属性.
m_Timer=newSystem.Timers.Timer();//explicitnamespace(TimeralsoinSystem.Threading)
m_Timer.Elapsed+=newElapsedEventHandler(OnTimerKillPopup);
m_Timer.Interval=m_nInterval;//forinstance3000milliseconds
m_Timer.Enabled=true;//starttimer
protectedvoidOnTimerKillPopup(Objectsource,ElapsedEventArgse)
{
m_Timer.Enabled=false;//pausethetimer
FindPopupToKill();
m_Timer.Enabled=true;
}

本地win32窗体查找

本程序的实现原理是这样,先检查所有的IE窗口标题,然后于已经有的列表来比较,如果有相同的,我们就关闭这个窗口.

按照上面的方法,我们每n妙使用KillPopup()来检查.比较遗憾的是我们无法使用安全代码来完成所有的工作.我们可以使用System.Diagnostics.Proces来检查所有的IE进程,然后得到主窗体.但是每一个IE进程可以打开好几个窗口,虽然每一个窗体都于一个进程相关,但是还没有办法来使每一个窗体于进程对应起来.一个可行的办法使用System.Diagnostics.Process列举出所有的运行的进程,然后System.Diagnostics.ProcessThreadCollection来得到他们的.Threads属性,为了得到threadId,我们使用Win32APIEnumThreadWindows(DWORDthreadId,WNDENUMPROClpfn,LPARAMlParam)来实现,这是一个回调(callback)函数,他可以列举出于进程相关的窗体.当我们得到了窗体的句柄以后,我们可以使用另一个API函数GetWindowText(HWNDhwnd,/*out*/LPTSTRlpString,intnMaxCount)来得到窗体的标题,然后根据已经有的窗体,调用API函数SendMessage(HWNDhWnd,intmsg,intwParam,intlParam)来关闭窗口.下面使演示代码

Process[]myProcesses=Process.GetProcessesByName("IEXPLORE");
foreach(ProcessmyProcessinmyProcesses)
{
FindPopupToKill(myProcess);
}
protectedvoidFindPopupToKill(Processp)
{
//traverseallthreadsandenumallwindowsattachedtothethread
foreach(ProcessThreadtinp.Threads)
{
intthreadId=t.Id;
NativeWIN32.EnumThreadProccallbackProc=
newNativeWIN32.EnumThreadProc(MyEnumThreadWindowsProc);
NativeWIN32.EnumThreadWindows(threadId,callbackProc,IntPtr.Zero/*lParam*/);
}
}
//callbackusedtoenumerateWindowsattachedtooneofthethreads
boolMyEnumThreadWindowsProc(IntPtrhwnd,IntPtrlParam)
{
publicconstintWM_SYSCOMMAND=0x0112;
publicconstintSC_CLOSE=0xF060;
//getwindowcaption
NativeWIN32.STRINGBUFFERsLimitedLengthWindowTitle;
NativeWIN32.GetWindowText(hwnd,outsLimitedLengthWindowTitle,256);
StringsWindowTitle=sLimitedLengthWindowTitle.szText;
if(sWindowTitle.Length==0)returntrue;
//findthiscaptioninthelistofbannedcaptions
foreach(ListViewItemiteminlistView1.Items)
{
if(sWindowTitle.StartsWith(item.Text))
NativeWIN32.SendMessage(hwnd,NativeWIN32.WM_SYSCOMMAND,
NativeWIN32.SC_CLOSE,
IntPtr.Zero);//trysoftkill
}
returntrue;
}
publicclassNativeWIN32
{
publicdelegateboolEnumThreadProc(IntPtrhwnd,IntPtrlParam);
[DllImport("user32.dll",CharSet=CharSet.Auto)]
publicstaticexternboolEnumThreadWindows(intthreadId,EnumThreadProcpfnEnum,IntPtrlParam);
//usedforanoutputLPCTSTRparameteronamethodcall
[StructLayout(LayoutKind.Sequential,CharSet=CharSet.Auto)]
publicstructSTRINGBUFFER
{
[MarshalAs(UnmanagedType.ByValTStr,SizeConst=256)]
publicstringszText;
}
[DllImport("user32.dll",CharSet=CharSet.Auto)]
publicstaticexternintGetWindowText(IntPtrhWnd,outSTRINGBUFFERClassName,intnMaxCount);

                                            
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: