原文件是一篇<br>文章C++Builder 高手进阶 <br><br>(一) 编写弹出广告杀手<br><br>nxyc_twz@163.com<br><br>一、系统分析<br><br> 作为一名软件开发人员,必然会经常上网查找资料,可讨厌的广告真让人心烦。有没有办法自动关闭这些广告呢?答案是肯定的!也许你会说:“网上这类软件多的是,随便找一个不就行了?”,你说的确实不错,可作为一名软件开发人员,总不能什么都靠别人吧?自己动手作一个如何?说干就干,首先得找出弹出广告窗口的工作原理:<br><br>1. 使用自己的《系统窗口分析器》(在下一期讲述其开发过程),轻易地就发现弹出广告窗口的特征:其窗口类是CabinetWClass或IEFrame。<br><br>2. 如果找出窗口类是CabinetWClass或IEFrame的窗口,向其发送WM_CLOSE不就OK了吗?心里不由一阵窃喜!<br><br>3. 使用定时器定时进行检测,找到符合条件的窗口就向其发送WM_CLOSE消息。<br><br>二、实战前沿<br><br>道理讲明了,可如何找到窗口类名呢?这就需要使用API函数了:<br><br>GetWindowText :取得窗口文本<br><br>GetWindow :取得窗口句柄<br><br>GetClassName :取得类名<br><br>PostMessage :发送消息<br><br>SetWindowLong :设置方式<br><br>具体用法请查阅相关资料。<br><br>三、设计流程<br><br>1. 启动C++Builder 5,新建一个Application,将Form的Name设置为Form1,其BoldStyle设置为bsNone,在Form的Icon中设置好图标。<br><br>2. 定义数据变量:<br><br>TRAYICON_ID = 1;<br><br>ICONEVENT = WM_APP + 100;<br><br>TNotifyIconData TrayIcon ;<br><br>3. 拖放一个Timer控件和一个Popmenu控件到窗体上。Timer的属性:Name->Timer1,Interval->1000;Popmenu1上追加三个子项:继续->ContinueCmd,PauseCmd->暂停,ExitCmd->退出。<br><br>4. 使应用程序进入系统托盘:void TrayIconCmd(void);<br><br>TrayIcon.cbSize = sizeof(TNotifyIconData);<br><br>TrayIcon.hWnd = Handle;<br><br>TrayIcon.uID = ID_TRAYICON;<br><br>TrayIcon.uFlags = NIF_ICON | NIF_MESSAGE | NIF_TIP;<br><br>TrayIcon.uCallBackMessage = ICONEVENT;<br><br>TrayIcon.hIcon = Form1->Icon->Handle;<br><br>TrayIcon.szTip = “广告窗口杀手”;<br><br>Shell_NotifyIcon(NM_ADD, &TrayIcon);<br><br>5. 使应用程序退出系统托盘:void TrayIconExit(void);<br><br>Shell_NotifyIcon(NIM_DELETE, &TrayIcon );<br><br>6. 使应用程序启动后直接进入系统托盘:编写Form1的OnCreate事件<br><br>TrayIconCmd(); //在系统托盘中安装图标<br><br>Width = 0;<br><br> SetWindowLong(Application->Handle, GWL_EXSTYLE, WS_EX_TOOLWINDOW);<br><br>7. 从系统托盘中卸载图标:编写Form1的OnClose事件<br><br>TrayIconExit();<br><br>8. 自定义消息TrayIconOnClick:<br><br>在头文件.h中定义:<br><br>private:<br><br>MESSAGE void __fastcall TrayIconOnClick(Tmessage &message);<br><br>BEGIN_MESSAGE_MAP<br><br> MESSAGE_HANDLER(ICONEVENT, Tmessage, TrayIconOnClick);<br><br>END_MESSAGE_MAP(Tform)<br><br>9. 鼠标点击系统托盘图标(TrayIconOnClick),则弹出菜单:<br><br>Tpoint p;<br><br>if( message.lParam == WM_LBUTTONDOWN ||<br><br> message.lPram == WM_RBUTTONDOWN)<br><br>{<br><br>PopupMenu1->PopupComponent = Form1;<br><br> SetForegroundWindow(Handle); <br><br>GetCursorPos(p);<br><br> PopupMenu->Popup(p.x,p.y);<br><br>} <br><br>10. 枚举指定窗口句柄的所有子窗口:void MENumChildWindows(HWND hand);<br><br>char tempstr[255];<br><br>bool IsPopWindow = true;<br><br>HWND h = GetWindow(hand, GW_CHILD);<br><br>while(h)<br><br>{<br><br> GetClassName(h, tempstr, 256);<br><br> if( !strcmp(tempstr, “WorkerA”) || !strcmp( tempstr, “WorkerW”))<br><br> if( IsWindowVisible(h) ) IsPopWindow = false;<br><br>h = GetWindow(h, GW_HWNDNEXT);<br><br>}<br><br>if( IsPopWindw )PostMessage(hand, WM_CLOSE, 0, 0); <br><br>11. 双击Timer控件,添加其事件OnTimer:<br><br>char Text[255];<br><br>HWnd h := GetWindow(Handle, GW_HWNDFIRST);<br><br>while(h)<br><br>{<br><br> if (GetWindowText(h, Text, 255) > 0)<br><br> if (GetClassName(h,Text , 255)>0)<br><br> if ( !strcmp( Text, “CabinetWClass”) || !strcmp(Text, “IEFrame”))<br><br> EnumChildWindows(h);<br><br> h = GetWindow(h, GW_HWNDNEXT); <br><br>}<br><br>12.编写菜单响应事件:<br><br>ContinueCmd:继续<br><br>Timer1->Enabled = true;<br><br> PauseCmd->Checked = false;<br><br> ContinueCmd->Checked = true;<br><br>PauseCmd:暂停<br><br>Timer1->Enabled = false;<br><br> PauseCmd->Checked = true;<br><br> ContinueCmd->Checked = false;<br><br>ExitCmd::退出<br><br>Close();<br><br>