资源描述:
《实现菜单阴影效果》由会员上传分享,免费在线阅读,更多相关内容在行业资料-天天文库。
1、利用钩子实现菜单阴影效果程序运行效果截图: 也许有很多人曾和我一样,对OfficeXP里面的菜单的阴影效果羡慕不已,它不需要在WindowsXP中就可以在菜单后面显示阴影,当然在WindowsXP中,已经完全支持菜单阴影了。虽然我们不一定很有必要自己来实现这个较难实现的效果。但是正如有很多人想实现那种IE风格的菜单栏一样,尽管它们并不能为我们带来更多实用的功能,却可以使我们的程序看起来与众不同。:) 菜单也是一个窗口,假如我们能得到它的窗口的句柄,要实现像添加阴影这样的效果,就不会很难了。可惜我们根本找不到这个窗口是在哪里被创建的,也
2、没办法很容易地取得它的窗口句柄,甚至几乎难以相信它是一个窗口,因为我实在找不到它的窗口句柄啊。经过对许多别人已经做好的类的源代码的"研究",我终于找到了一个方法。那就是万能的钩子,如果说在Windows里面抓"人",连钩子也办不到的话,那我就不知道该用什么方法实现了,呵呵。 下面我就一起来看看如何抓到这些"可恶"的家伙吧。为了便于移植,我们就写一个专用的类吧,就取名为CMenuWndHook。添加两个静态成员先:staticCMapm_WndMenuMap;staticHHOOKm_hMenuHook; 被我们抓到的这些家伙肯定不止一个
3、,我们需要一个映射模板类来保存它们的句柄和对应的CMenuWndHook类对象的指针。m_hMenuHook则为我们将要创建的钩子的钩子句柄。再在CPP文件中初始化它们:CMapCMenuWndHook::m_WndMenuMap;HHOOKCMenuWndHook::m_hMenuHook=NULL;下面再添加两个函数来做安装与卸载hook之用,它们都是静态函数:voidCMenuWndHook::InstallHook(){ if(m_hMenuHook==NULL) { m_hMenuHook=::SetWindowsHook
4、Ex(WH_CALLWNDPROC, WindowHook, AfxGetApp()->m_hInstance, ::GetCurrentThreadId()); }}Windows之下一般用上面的SetWindowsHookExAPI函数来安装HOOK,它的函数原型如下:HHOOKSetWindowsHookEx(intidHook,//钩子的类型,即它处理的消息类型 HOOKPROClpfn, //子函数的入口地址,当钩子钩到任何消息后先调用这个函数。
5、 //(如果dwThreadId参数为0,或是一个由别的进程创建的线程的标识, //lpfn必须指向DLL中的钩子子程。除此以外,lpfn可以指向当前进 //程的一段钩子子程代码) HINSTANCEhMod,//应用程序实例的句柄。标识包含lpfn所指的子程的DLL。 //如果dwThreadId标识当前进程创建的一个线程, //而且子程代码位于当前进程,hMod必须为NULL。 //可以很简单的设定其为本应用程序的实例句柄。 DWORDdwThreadId//与安装的钩子子程相关联的线程的标识符。 //如果为0,钩子
6、子程与所有的线程关联,即为全局钩子。 //但这时,你钩子只能是放在DLL中。 ); 函数成功则返回钩子子程的句柄,失败返回NULL。我们用到的是WH_CALLWNDPROC类型的钩子,它使你可以监视发送到窗口过程的消息,系统在消息发送到接收窗口过程之前会调用你指定的WH_CALLWNDPROCHook子程,这样你就可以等它们自投罗网,然后就可以对它们为所欲为了。卸载钩子就简单多了,只需要调用UnhookWindowsHookEx即可,当然,我们还需要额外做一点清理工作:voidCMenuWndHook::UnInst
7、allHook(){ POSITIONpos=m_WndMenuMap.GetStartPosition();while(pos!=NULL) { HWNDhwnd; CMenuWndHook*pMenuWndHook; m_WndMenuMap.GetNextAssoc(pos,hwnd, pMenuWndHook); deletepMenuWndHook; pMenuWndHook=NULL;}m_WndMenuMap.RemoveAll();if(m_hMenuHook!=NULL){ ::UnhookWindows
8、HookEx(m_hMenuHook);} } 在介绍如何安装钩子时,提到要一个钩子子程,这个子程必须按下面的格式声明,否则不能使用:LRESULTCALLBACKWindowHook(in