HOOK - 低级鼠标Hook
这里使用的钩子类型idHook 是WH_MOUSE_LL。
在使用SetWindowsHook截获鼠标事件时,如果使用WH_MOUSE参数,则只能监控钩子所在模块的鼠标事件。
如需要截获整个系统的鼠标事件,那么使用WH_MOUSE_LL参数。
二、自定义消息通信
这里和参考的一样,将其写成MFC DLL然后使用导出函数。
通过发送消息SendMessage的方式,让dll和主程序进行通信(SendMessage 同步),
这里要注意使用的是自定义消息的方式来进行通信,就像IPC中的自定义消息一样,.dll与MFC中的消息定义必须完全一样:
//自定义消息,用于处理dll发来的消息
//该消息的定义和dll中定义的消息必须一样
#define WM_MOUSEMESSAGE WM_USER + 100
DLL部分代码:
.dll中
// 安装低级鼠标子函数,截获系统所有的鼠标消息
BOOL WINAPI StartHookMouse(HWND hwnd)
{
//这里自身的窗口句柄传过来 __hWnd = hwnd; /*
在使用SetWindowsHook截获鼠标事件时,如果使用WH_MOUSE参数,则只能监控钩子所在模块的鼠标事件。
如需要截获整个系统的鼠标事件,那么使用WH_MOUSE_LL参数。 */
__hhkMouse = SetWindowsHookEx(
WH_MOUSE_LL,
LowLevelMouseProc,
__hInstance, ); if (NULL == __hhkMouse)
{
return FALSE;
}
else
{
return TRUE;
} }
// 定义低级鼠标子函数
LRESULT CALLBACK LowLevelMouseProc(int nCode, WPARAM wParam, LPARAM lParam)
{
// 有鼠标消息时,将其发给主程序 if (__hWnd != NULL && nCode == HC_ACTION)//HC_ACTION 有鼠标信息
{
//通知MFC
::SendMessage(__hWnd, WM_MOUSEMESSAGE, wParam, lParam);
} return CallNextHookEx(__hhkMouse, nCode, wParam, lParam); }
然后,主程序收到该消息后,将其显示在程序界面上。
这边为了给主程序发送消息,在主程序调用安装鼠标钩子的函数时,我们需要将主窗口的句柄通过参数传递到dll中,这样我们就可以发送消息给主程序了。
三、共享代码段,所有线程共享
如果一个程序运行了两个实例,很遗憾,这两个程序的代码并不会共享,因为程序代码在程序运行时就被装载到内存,第二个程序再启动时,系统没法判断也不会判断这个程序是否还是原来的程序(如果程序被更新或更改),所以仍然会把程序代码重新加载到内存的另一块区域。
代码的共享体现应该是链接库中,特别是动态链接库中。因为库被设计成动态链接的,所以程序可以在运行后再确定代码所在的内存地址,这也就能实现多个程序调用同一块代码了。
// 共享代码段,所有线程共享
#pragma data_seg("SHARED")
static HHOOK __hhkMouse = NULL; // 鼠标钩子句柄
static HINSTANCE __hInstance = NULL; // 本DLL的实例句柄
static HWND __hWnd = NULL; // 调用DLL的主窗口句柄
#pragma data_seg()
#pragma comment(linker,"/section:SHARED,rws")
四、卸载hook
// 卸载低级鼠标钩子
VOID WINAPI StopHookMouse()
{
if (__hhkMouse != NULL)
{
::UnhookWindowsHookEx(__hhkMouse);
}
}
/***********************************/
五、MFC中的自定义消息
首先必须得和dll中的完全一样,定义一个自定义消息,必须完全一样
(这里可以参考博客http://www.cnblogs.com/hcxblog/archive/2012/10/02/2710261.html)
然后要写自定义消息的三句话:
1.消息映射
2.消息处理函数的声明
3.消息处理函数的实现
ON_MESSAGE(WM_MOUSEMESSAGE, &CLowLevelHookMouseDlg::OnMouseMessage) //消息映射
afx_msg LRESULT OnMouseMessage(WPARAM wParam, LPARAM lParam);
LRESULT CLowLevelHookMouseDlg::OnMouseMessage(
WPARAM wParam, LPARAM lParam)
----------------------------------------------------------------------------------------
使用函数:
HINSTANCE __hInstanceDll = NULL; typedef BOOL (CALLBACK *StartHookMouse)(HWND hwnd);
typedef VOID (CALLBACK *StopHookMouse)();
//启动鼠标钩子
void CLowLevelHookMouseDlg::OnBnClickedButtonStarthook()
{
// TODO: 在此添加控件通知处理程序代码
__hInstanceDll = LoadLibrary(_T("HookDll.dll"));
if (__hInstanceDll == NULL)
{
::MessageBox(NULL,(LPCTSTR)GetLastError(),L"LoadLibrary() Error",);
return;
} StartHookMouse StartHook;
StartHook = (StartHookMouse)::GetProcAddress(
__hInstanceDll, "StartHookMouse");
if (StartHook == NULL)
{
::MessageBox(NULL, (LPCTSTR)GetLastError(), L"GetProcAddress() Error", );
} if (StartHook(this->m_hWnd))
{
m_List.InsertItem(m_List.GetItemCount(), _T(""));
m_List.SetItemText(m_List.GetItemCount() - , , _T(""));
m_List.SetItemText(m_List.GetItemCount() - , , _T("启动鼠标钩子成功")); CButton *pButton = (CButton *)GetDlgItem(IDC_BUTTON_STARTHOOK)->EnableWindow(FALSE); CButton *Button = (CButton *)GetDlgItem(IDC_BUTTON_STOPHOOK)->EnableWindow(TRUE); }
else
{
m_List.InsertItem(m_List.GetItemCount(), _T(""));
m_List.SetItemText(m_List.GetItemCount() - , , _T(""));
m_List.SetItemText(m_List.GetItemCount() - , , _T("启动鼠标钩子失败"));
}
} //结束鼠标钩子
void CLowLevelHookMouseDlg::OnBnClickedButtonStophook()
{
// TODO: 在此添加控件通知处理程序代码 StopHookMouse StopHook;
__hInstanceDll = LoadLibrary(_T("HookDll.dll"));
if (__hInstanceDll == NULL)
{
::MessageBox(NULL, (LPCTSTR)GetLastError(), L"LoadLibrary() Error", );
return;
} StopHook = (StopHookMouse) ::GetProcAddress(
__hInstanceDll, "StopHookMouse"); if (StopHook == NULL)
{
m_List.InsertItem(m_List.GetItemCount(), _T(""));
m_List.SetItemText(m_List.GetItemCount() - , , _T(""));
m_List.SetItemText(m_List.GetItemCount() - , , _T("获取函数 StopHookMouse 失败")); return;
}
else
{
StopHook();
m_List.InsertItem(m_List.GetItemCount(), _T(""));
m_List.SetItemText(m_List.GetItemCount() - , , _T(""));
m_List.SetItemText(m_List.GetItemCount() - , , _T("停止HOOKMOUSE成功"));
CButton *pButton = (CButton *)GetDlgItem(IDC_BUTTON_STOPHOOK)->EnableWindow(FALSE); CButton *Button = (CButton *)GetDlgItem(IDC_BUTTON_STARTHOOK)->EnableWindow(TRUE); } if (__hInstanceDll != NULL)
{
::FreeLibrary(__hInstanceDll);
} // 确保list control 最后一行可见 跳转到stop那行
m_List.EnsureVisible(m_List.GetItemCount() - , FALSE); }
六、最后
剩下的都是一些简单的实现了,这里觉得重要的写在了上面。其他的实现比较常规。
1.其中有两个checkbox的实现做了发现没有达到想要的效果就废掉了。
2.还有就是源博主中如果多点了几次hook,在unhook的时候会出问题,所以这里
CButton *pButton = (CButton *)GetDlgItem(IDC_BUTTON_STOPHOOK)->EnableWindow(FALSE);
Ring3 x86 x64 通过.
以上就是本次的一个小总结,如有不妥还请批评指正。
再次感谢。
代码:
链接:http://pan.baidu.com/s/1dEPx03J 密码:y463
IPC自定义消息代码:
链接:http://pan.baidu.com/s/1o795ZRW 密码:dt6p
HOOK - 低级鼠标Hook的更多相关文章
- 基于 VirtualApp 结合 whale hook框架实现hook第三方应用
要点 1. whale hook framework 使用示例: 2. 参考项目:VirtualHook: 3. 按照 VirtualHook 修改 VirtualApp: 4. 编写 hook pl ...
- Android so注入(inject)和Hook技术学习(三)——Got表hook之导出表hook
前文介绍了导入表hook,现在来说下导出表的hook.导出表的hook的流程如下.1.获取动态库基值 void* get_module_base(pid_t pid, const char* modu ...
- windows hook + pyhook3 + python win32api hook + C 键盘hook
安装pyhook3见:https://www.cnblogs.com/lqerio/p/12096710.html 使用见:https://www.cnblogs.com/lqerio/p/12106 ...
- 如何通过HOOK改变windows的API函数(找到函数的相对偏移)
我们知道,系统函数都是以DLL封装起来的,应用程序应用到系统函数时,应首先把该DLL加载到当前的进程空间中,调用的系统函数的入口地址,可以通过GetProcAddress函数进行获取.当系统函数进行调 ...
- 窗口置顶 - 仿TopWind
前置学习:低级鼠标hook,获得鼠标状态. 这个在原来获得鼠标状态的基础上,加上一个事件处理即可. TopWind就是一个可以置顶窗口的文件,避免复制粘贴的时候的来回切换(大窗口与小窗口),算是一个实 ...
- HOOK API (一)——HOOK基础+一个鼠标钩子实例
HOOK API (一)——HOOK基础+一个鼠标钩子实例 0x00 起因 最近在做毕业设计,有一个功能是需要实现对剪切板的监控和进程的防终止保护.原本想从内核层实现,但没有头绪.最后决定从调用层入手 ...
- 钩子编程(HOOK) 屏蔽全部按键、鼠标及系统功能键 (4)
摘要:上篇文章<钩子编程(HOOK) 安装系统全局钩子>已经具体的解说了全局钩子的安装.本文将增强一下钩子的功能.实现屏蔽全部按键鼠标与系统功能键.要实现这个功能.须要安装两个全局钩子,& ...
- c# 使用hook来监控鼠标键盘事件的示例代码
如果这个程序在10几年前,QQ刚刚兴起的时候,有了这个代码,就可实现盗号了. 当然使用钩子我们更多的是实现"全局快捷键"的需求. 比如 程序最小化隐藏后要"某快捷键&qu ...
- 键盘Hook【Delphi版】
原文:https://www.cnblogs.com/edisonfeng/archive/2012/05/18/2507858.html 一.钩子的基本概念 a) Hook作用:监视windows消 ...
随机推荐
- 详细解析HTML基础结构
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- 修改Anaconda3中jupyter的工作目录
1.打开Anaconda Prompt,运行jupyter notebook --generate-config 2.找到当前用户下的.jupter文件夹,打开jupyter_notebook_con ...
- css预处理器--sass学习($变量名)
sass有两种形式1.scss 2.sass 一:代码的基本用法 1.变量 如果变量需要镶嵌在字符串之中,就必须需要写在#{}之中. $side : left; .rounded { border- ...
- linux下配置mysql 与错误解决
1.下载mysql wget https://cdn.mysql.com//Downloads/MySQL-8.0/mysql-8.0.11-el7-x86_64.tar.gz 2.检查linux是否 ...
- 微服务-网关-node.js by 大雄daysn
目录 序言 一.node.js入门1.1 下载并安装1.2 从helloworld到一个web应用1.3 Express框架二.node.js搭建网关 三.node.js集群搭建 序言 首先一个问 ...
- python全栈开发笔记---------字符串格式化
字符串格式化 %s 可以接收任何值, %d只能接收整形 .其他类型报错 msg ='i am %s my body' %'ales' print(msg) #i am ales my body msg ...
- PHP_CodeSniffer 安装和phpstorm配置
安装 1.mac安装 sudo pear install PHP_CodeSniffer phpstorm配置 1. 点击菜单:File->Settings 或 按快捷键 Ctrl+Alt+S ...
- 第三视角Beta答辩总结
第三视角Beta答辩总结 博客链接以及团队信息 组长博客链接 成员信息(按拼音排序) 姓名 学号 备注 张扬 031602345 组长 陈加伟 031602204 郭俊彦 031602213 洪泽波 ...
- jvm-垃圾收集
概述 说起垃圾收集,大部分人都把这项技术当做Java语言的伴生产物.其实,GC主要就是考虑完成三件事情: 哪些内存需要回收 什么时候回收 如何回收. 经过半个多世纪的发展,目前内存的动态分配与内存的回 ...
- obspy下载地震波数据
Retrieving Data from Data Centers(从数据中心检索数据) PS:此部分提供了使用obspy下载数据的推荐方式,但是由于数据中心和web服务在不断更新变化,所有有些建议可 ...