没有找到合适的产品?
联系客服协助选型:023-68661681
提供3000多款全球软件/控件产品
针对软件研发的各个阶段提供专业培训与技术咨询
根据客户需求提供定制化的软件开发服务
全球知名设计软件,显著提升设计质量
打造以经营为中心,实现生产过程透明化管理
帮助企业合理产能分配,提高资源利用率
快速打造数字化生产线,实现全流程追溯
生产过程精准追溯,满足企业合规要求
以六西格玛为理论基础,实现产品质量全数字化管理
通过大屏电子看板,实现车间透明化管理
对设备进行全生命周期管理,提高设备综合利用率
实现设备数据的实时采集与监控
利用数字化技术提升油气勘探的效率和成功率
钻井计划优化、实时监控和风险评估
提供业务洞察与决策支持实现数据驱动决策
翻译|其它|编辑:郝浩|2006-03-08 11:39:00.000|阅读 1453 次
概述:
# 界面/图表报表/文档/IDE等千款热门软控件火热销售中 >>
static CMap m_WndMenuMap; static HHOOK m_hMenuHook; 被我们抓到的这些家伙肯定不止一个,我们需要一个映射模板类来保存它 们的句柄和对应的CMenuWndHook 类对象的指针。m_hMenuHook则为我们将要创建的钩子的钩子句柄。再在CPP文件中初始化它们:
CMap CMenuWndHook::m_WndMenuMap; HHOOK CMenuWndHook::m_hMenuHook = NULL;下面再添加两个函数来做安装与卸载hook之用, 它们都是静态函数:
void CMenuWndHook::InstallHook() { if (m_hMenuHook == NULL) { m_hMenuHook = ::SetWindowsHookEx(WH_CALLWNDPROC, WindowHook, AfxGetApp()->m_hInstance, ::GetCurrentThreadId()); } }Windows之下一般用上面的SetWindowsHookEx API函数来安装HOOK,它的函数原型如下:
HHOOK SetWindowsHookEx(int idHook, //钩子的类型,即它处理的消息类型 HOOKPROC lpfn, //子函数的入口地址,当钩子钩到任何消息后先调用这个函数。 // (如果dwThreadId参数为0,或是一个由别的进程创建的线程的标识, //lpfn必须指向DLL中的钩子子程。除此以外,lpfn可以指向当前进 //程的一段钩子子程代码) HINSTANCE hMod, //应用程序实例的句柄。标识包含lpfn所指的子程的DLL。 // 如果dwThreadId标识当前进程创建的一个线程, //而且子程代码位于当前进程,hMod必须为NULL。 //可以很简单的设定其为本应用程序的实例句柄。 DWORD dwThreadId //与安装的钩子子程相关联的线程的标识符。 //如果为0,钩子子程与所有的线程关联,即为全局钩子。 //但这时,你钩子只能是放在DLL中。 );
函数成功则返回钩子子程的句柄,失败返回NULL。 我们用到的是WH_CALLWNDPROC类型的钩子,它使你可以监视发送到窗口过程的消息, 系统在消息发送到 接收窗口过程之前会调用你指定的WH_CALLWNDPROC Hook 子程,这样你就可以等它们自投罗网,然后就可以 对它们为所欲为了。 卸载钩子就简单多了,只需要调用UnhookWindowsHookEx即可,当然,我们还需要额外做一点清理工作:
void CMenuWndHook::UnInstallHook() { POSITION pos = m_WndMenuMap.GetStartPosition(); while (pos != NULL) { HWND hwnd; CMenuWndHook *pMenuWndHook; m_WndMenuMap.GetNextAssoc(pos, hwnd, pMenuWndHook); delete pMenuWndHook; pMenuWndHook= NULL; } m_WndMenuMap.RemoveAll(); if (m_hMenuHook != NULL) { ::UnhookWindowsHookEx(m_hMenuHook); } }在介绍如何安装钩子时,提到要一个钩子子程,这个子程必须按下面的格式声明,否则不能使用:
LRESULT CALLBACK CMenuWndHook::WindowHook(int code, WPARAM wParam, LPARAM lParam) { //如果你安装的是WH_CALLWNDPROC类型的钩子的话,系统就会传递一个这个家伙的指针: CWPSTRUCT* pStruct = (CWPSTRUCT*)lParam; while (code == HC_ACTION) { HWND hWnd = pStruct->hwnd; // 截获 WM_CREATE 消息, 为了保证不抓错"人",我们必须严格确定这是否是我们要抓的家伙, // 这样我们就可以在它们刚出头就把它们逮住: if(pStruct->message != WM_CREATE &&pStruct->message != 0x01E2) { break; } // 是否为菜单类 ---------------------------------------- TCHAR strClassName[10]; int Count = ::GetClassName(hWnd, strClassName, sizeof(strClassName) / sizeof(strClassName[0])); // 再次确认它的身份(菜单窗口类的类名为"#32768",且为6个字符长): if (Count != 6 || _tcscmp(strClassName, _T("#32768")) != 0 ) { // 对不起,认错人了,pass :-) break; } //是否已经被子类化------------------------------------ // 我们抓到一个之后,会给它用SetProp挂个牌(后面会介绍) if(::GetProp(pStruct->hwnd, CoolMenu_oldProc) ! = NULL ) { // 已经在编? pass. break; } // 抓到一个,给它登记注册(这个函数我会在后面介绍), 而且不能登记失败, :) VERIFY(AddWndHook(pStruct->hwnd) != NULL); //下面该叫它去洗心革面了----------------- //取得原来的窗口过程 ---------------------------------- WNDPROC oldWndProc = (WNDPROC)(long)::GetWindowLong(pStruct->hwnd, GWL_WNDPROC); if (oldWndProc == NULL) { break; } ASSERT(oldWndProc != CoolMenuProc); //这个过程一样不能出错 // 保存到窗口的属性中 ---------------------------------- // 哈哈,给它打个记号吧 (SetProp API函数是用来给一个窗口加上一个属性的, // RemoveProp 则是删除一个属性,GetProp 是取得一个属性的值) // CoolMenu_oldProc 为一字符数组, 我在CPP文件的开头声明了它,表示你要 // 添加的属性名: const TCHAR CoolMenu_oldProc[]=_T("CoolMenu_oldProc"); // 这里保存的是它的原来的窗口过程,这种该随身带的东西还是让它自己拿着比较好 if (!SetProp(pStruct->hwnd,CoolMenu_oldProc, oldWndProc)) { break; } // 子类化---------------------------------------------- // 这个不用我说了吧,这里我们用了偷梁换柱的方法,呵呵,这可是子类化的惯技了: if (!SetWindowLong(pStruct->hwnd, GWL_WNDPROC,(DWORD)(ULONG)CoolMenuProc) ) { //没有成功!!唉,就放过他吧,虽然忙了半天了,不过这种情况我想是不可能发生的! ::RemoveProp(pStruct->hwnd, CoolMenu_oldProc); break; } } // 这句可是绝对不能少的,叫那些闲杂人等该干什么就干什么去,不要? // 嘿嘿,看你的程序怎么死吧! return CallNextHookEx(m_hMenuHook, code, wParam, lParam); }我们再来看看,怎么"登记"它们:
CMenuWndHook* CMenuWndHook::AddWndHook(HWND hwnd) { CMenuWndHook* pWnd = NULL; if (m_WndMenuMap.Lookup(hwnd, pWnd)) { // 有这个人了,不用再登记了。 return pWnd; } // 给它分配个房间(牢房! 嘿嘿) pWnd = new CMenuWndHook(hwnd); if (pWnd != NULL) { m_WndMenuMap.SetAt(hwnd, pWnd); } return pWnd; } // 另外还可有一个对应的查找函数: CMenuWndHook* CMenuWndHook::GetWndHook(HWND hwnd) { CMenuWndHook* pWnd = NULL; if (m_WndMenuMap.Lookup(hwnd, pWnd)) { return pWnd; } return NULL; }上面的函数和变量大部分都是静态成员,因为hook系统只要有一套就可以了到 这里为止,坚巨的任务已经完成了一半,做下面的事,就得心应手多了。下面是窗口的新过程,依然为一个静态的函数。
LRESULT CALLBACK CMenuWndHook::CoolMenuProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { WNDPROC oldWndProc = (WNDPROC)::GetProp(hWnd, CoolMenu_oldProc); CMenuWndHook* pWnd = NULL; switch (uMsg) { // 计算非客户区的大小-------------------------- case WM_NCCALCSIZE: { LRESULT lResult = CallWindowProc(oldWndProc, hWnd, uMsg, wParam, lParam); if ((pWnd = GetWndHook(hWnd)) != NULL) { pWnd->OnNcCalcsize((NCCALCSIZE_PARAMS*)lParam); } return lResult; } break; // 当窗口的位置将要发生改变, 在这里它一般发生在菜单被弹出之前, // 给你最后一次机会设置它的位置. case WM_WINDOWPOSCHANGING: { if ((pWnd = GetWndHook(hWnd)) != NULL) { pWnd->OnWindowPosChanging((LPWINDOWPOS)lParam); } } break; // 为什么要响应这个消息呢? 我也不知道啊,我只知道,当菜单是以动画的方式弹出的时候 // 系统是通过发送这个消息来绘制菜单的,wParam是对应的设备上下文句柄,不过我也不知 // 道它到底是属于谁的. case WM_PRINT: { LRESULT lResult = CallWindowProc(oldWndProc, hWnd, uMsg, wParam, lParam); if ((pWnd = GetWndHook(hWnd)) != NULL) { pWnd->OnPrint(CDC::FromHandle((HDC)wParam)); } return lResult; } break; //这个就不同说了吧. case WM_NCPAINT: { if ((pWnd = GetWndHook(hWnd)) != NULL) { pWnd->OnNcPaint(); return 0; } } break; // 菜单窗口被隐藏的时候,我也不知道这种情况会不会发生, :(, 主要是看到人家这样处理了. case WM_SHOWWINDOW: { if ((pWnd = GetWndHook(hWnd)) != NULL) { pWnd->OnShowWindow(wParam != NULL); } } break; // 菜单窗口被销毁的时候 case WM_NCDESTROY: { if ((pWnd = GetWndHook(hWnd)) != NULL) { pWnd->OnNcDestroy(); } } break; } return CallWindowProc(oldWndProc, hWnd, uMsg, wParam, lParam); }下面就看如何慢慢实现这些消息的响应函数吧:
void CMenuWndHook::OnWindowPosChanging(WINDOWPOS *pWindowPos) { if (!IsShadowEnabled()) { //加一块区域来显示阴影------- pWindowPos->cx += 4; pWindowPos->cy += 4; } // 为了绘制阴影,我们须要先保存这个区域的图像,以便绘制半透明的阴影. if (!IsWindowVisible(m_hWnd) && !IsShadowEnabled()) { if (m_bmpBack.m_hObject != NULL) { m_bmpBack.DeleteObject(); } m_bmpBack.Attach(GetScreenBitmap(CRect(pWindowPos->x, pWindowPos->y, pWindowPos->cx, pWindowPos->cy))); } } void CMenuWndHook::OnNcCalcsize(NCCALCSIZE_PARAMS* lpncsp) { if (!IsShadowEnabled()) { //留出一点区域来显示阴影------- lpncsp->rgrc[0].right -= 4; lpncsp->rgrc[0].bottom -= 4; } }
标签:本站文章除注明转载外,均为本站原创或翻译。欢迎任何形式的转载,但请务必注明出处、不得修改原文相关链接,如果存在内容上的异议请邮件反馈至chenjj@evget.com
面对“数字中国”建设和中国制造2025战略实施的机遇期,中车信息公司紧跟时代的步伐,以“集约化、专业化、标准化、精益化、一体化、平台化”为工作目标,大力推进信息服务、工业软件等核心产品及业务的发展。在慧都3D解决方案的实施下,清软英泰建成了多模型来源的综合轻量化显示平台、实现文件不失真的百倍压缩比、针对模型中的大模型文件,在展示平台上进行流畅展示,提升工作效率,优化了使用体验。
本站的模型资源均免费下载,登录后即可下载。模型仅供学习交流,勿做商业用途。
本站的模型资源均免费下载,登录后即可下载。模型仅供学习交流,勿做商业用途。
本站的模型资源均免费下载,登录后即可下载。模型仅供学习交流,勿做商业用途。
服务电话
重庆/ 023-68661681
华东/ 13452821722
华南/ 18100878085
华北/ 17347785263
客户支持
技术支持咨询服务
服务热线:400-700-1020
邮箱:sales@evget.com
关注我们
地址 : 重庆市九龙坡区火炬大道69号6幢
慧都科技 版权所有 Copyright 2003-
2025 渝ICP备12000582号-13 渝公网安备
50010702500608号