MFC-钩子(拦截消息)

原理:

 

 

wincore.cpp

是VC++消息处理函数(WindowProc的函数)的文件

文件路径:C:\Program Files (x86)\Microsoft Visual Studio\2019\Professional\VC\Tools\MSVC\14.29.30133\atlmfc\src\mfc         

 

钩子类型:

一共有两种类型的钩子:局部的和远程的。
局部钩子仅钩挂您自己进程的事件。
远程的钩子还可以将钩挂其它进程发生的事件。
远程的钩子又有两种:一种是系统级的全局钩子; 一种是线程级的钩子.全局钩子函数需要定义在 DLL 中, 线程级的钩子开始比较简单.
其实钩子函数就三个:
设置钩子: SetWindowsHookEx
释放钩子: UnhookWindowsHookEx
继续钩子: CallNextHookEx
在线程级的钩子中经常用到 GetCurrentThreadID 函数来获取当前线程的 ID

 

 

实例:应用程序内部钩子(局部钩子) 

功能:拦截鼠标

          程序运行时,无法用鼠标关闭本程序和操作按钮,当可以通过键盘操作按钮

创建项目 

 

 

添加钩子 

 

复制代码
// InnerHookDlg.cpp: 实现文件
//

#include "pch.h"
#include "framework.h"
#include "InnerHook.h"
#include "InnerHookDlg.h"
#include "afxdialogex.h"

#ifdef _DEBUG
#define new DEBUG_NEW
#endif


// 用于应用程序“关于”菜单项的 CAboutDlg 对话框

class CAboutDlg : public CDialogEx
{
public:
    CAboutDlg();

// 对话框数据
#ifdef AFX_DESIGN_TIME
    enum { IDD = IDD_ABOUTBOX };
#endif

    protected:
    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持

// 实现
protected:
    DECLARE_MESSAGE_MAP()
};

CAboutDlg::CAboutDlg() : CDialogEx(IDD_ABOUTBOX)
{
}

void CAboutDlg::DoDataExchange(CDataExchange* pDX)
{
    CDialogEx::DoDataExchange(pDX);
}

BEGIN_MESSAGE_MAP(CAboutDlg, CDialogEx)
END_MESSAGE_MAP()


// CInnerHookDlg 对话框



CInnerHookDlg::CInnerHookDlg(CWnd* pParent /*=nullptr*/)
    : CDialogEx(IDD_INNERHOOK_DIALOG, pParent)
{
    m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}

void CInnerHookDlg::DoDataExchange(CDataExchange* pDX)
{
    CDialogEx::DoDataExchange(pDX);
}

BEGIN_MESSAGE_MAP(CInnerHookDlg, CDialogEx)
    ON_WM_SYSCOMMAND()
    ON_WM_PAINT()
    ON_WM_QUERYDRAGICON()
    ON_WM_CLOSE()
    ON_BN_CLICKED(IDC_BUTTON1, &CInnerHookDlg::OnBnClickedButton1)
END_MESSAGE_MAP()


LRESULT CALLBACK MouseProc(_In_ int    nCode, _In_ WPARAM wParam, _In_ LPARAM lParam)
//全局鼠标钩子函数[不属于任何类]
{


    return 1; //返回1:表示消息已经处理,不在向下传递[起到拦截鼠标消息的功能]
    //
}


HHOOK hMouseHook = NULL; //定义一个钩子句柄变量--全局变量


// CInnerHookDlg 消息处理程序

BOOL CInnerHookDlg::OnInitDialog()  //对话框初始化函数
{
    CDialogEx::OnInitDialog();

    // 将“关于...”菜单项添加到系统菜单中。

    // IDM_ABOUTBOX 必须在系统命令范围内。
    ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
    ASSERT(IDM_ABOUTBOX < 0xF000);

    CMenu* pSysMenu = GetSystemMenu(FALSE);
    if (pSysMenu != nullptr)
    {
        BOOL bNameValid;
        CString strAboutMenu;
        bNameValid = strAboutMenu.LoadString(IDS_ABOUTBOX);
        ASSERT(bNameValid);
        if (!strAboutMenu.IsEmpty())
        {
            pSysMenu->AppendMenu(MF_SEPARATOR);
            pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
        }
    }

    // 设置此对话框的图标。  当应用程序主窗口不是对话框时,框架将自动
    //  执行此操作
    SetIcon(m_hIcon, TRUE);            // 设置大图标
    SetIcon(m_hIcon, FALSE);        // 设置小图标

    // TODO: 在此添加额外的初始化代码

    //1.设置钩子
    hMouseHook=SetWindowsHookEx(WH_MOUSE, MouseProc,NULL, GetCurrentThreadId());
    //参数1:int idHook 钩子类型,即它处理的消息类型
//       WH_CBT=5(窗口动作消息),在以下事件之前,系统都会调用WH_CBT Hook子程,这些事件包括:
//                1)激活,建立,销毁,最小化,最大化,移动,改变尺寸等窗口事件;
//                2)完成系统指令;
//                3)来自系统消息队列中的移动鼠标,键盘事件;
//                4)设置输入焦点事件;
//                5)同步系统消息队列事件。
//       WH_MOUSE=7   鼠标消息
//       WH_KEYBOARD=2  键盘消息
//参数2:HOOKPROC lpfn  回调函数地址(处理函数)
//       注意:不同钩子类型,处理函数是不同的
//            如果是WH_MOUSE类型   处理函数是MouseProc
//            如果是WH_KEYBOARD类型   处理函数是KeyboardProc
//            如果是WH_CBT类型   处理函数是CBTProc
//参数3:
//       NULL  钩子只关注本进程
//       如果是远程钩子,参数2就得是一个单独的动态库dll
//             参数3就是动态库的句柄,参数4为NULL
//        注意:远程钩子,杀毒软件都会认为是病毒
//参数4:DWORD dwThreadId  线程ID
//返回值:如果函数成功,则返回值是钩子句柄。如果函数失败,则返回值为 NULL




    return TRUE;  // 除非将焦点设置到控件,否则返回 TRUE
}

void CInnerHookDlg::OnSysCommand(UINT nID, LPARAM lParam)
{
    if ((nID & 0xFFF0) == IDM_ABOUTBOX)
    {
        CAboutDlg dlgAbout;
        dlgAbout.DoModal();
    }
    else
    {
        CDialogEx::OnSysCommand(nID, lParam);
    }
}

// 如果向对话框添加最小化按钮,则需要下面的代码
//  来绘制该图标。  对于使用文档/视图模型的 MFC 应用程序,
//  这将由框架自动完成。

void CInnerHookDlg::OnPaint()
{
    if (IsIconic())
    {
        CPaintDC dc(this); // 用于绘制的设备上下文

        SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), 0);

        // 使图标在工作区矩形中居中
        int cxIcon = GetSystemMetrics(SM_CXICON);
        int cyIcon = GetSystemMetrics(SM_CYICON);
        CRect rect;
        GetClientRect(&rect);
        int x = (rect.Width() - cxIcon + 1) / 2;
        int y = (rect.Height() - cyIcon + 1) / 2;

        // 绘制图标
        dc.DrawIcon(x, y, m_hIcon);
    }
    else
    {
        CDialogEx::OnPaint();
    }
}

//当用户拖动最小化窗口时系统调用此函数取得光标
//显示。
HCURSOR CInnerHookDlg::OnQueryDragIcon()
{
    return static_cast<HCURSOR>(m_hIcon);
}



void CInnerHookDlg::OnClose()  //窗口关闭消息
{
    // TODO: 在此添加消息处理程序代码和/或调用默认值

    if (hMouseHook != NULL)  //如果钩子句柄不等于NULL
    {

    UnhookWindowsHookEx(hMouseHook);//卸载钩子
    //参数:要移除的挂钩的句柄
    //返回值:如果该函数成功,则返回值为非零值。如果函数失败,则返回值为零

    }
    
    CDialogEx::OnClose();
}


void CInnerHookDlg::OnBnClickedButton1()
{
    

    AfxGetMainWnd()->SendMessage(WM_CLOSE); //退出程序


}
复制代码

 

 

 卸载钩子

说明:在窗口关闭时卸载钩子

项目-->类向导

 

 

复制代码
void CInnerHookDlg::OnClose()  //窗口关闭消息
{
    // TODO: 在此添加消息处理程序代码和/或调用默认值

    if (hMouseHook != NULL)  //如果钩子句柄不等于NULL
    {

    UnhookWindowsHookEx(hMouseHook);//卸载钩子
    //参数:要移除的挂钩的句柄
    //返回值:如果该函数成功,则返回值为非零值。如果函数失败,则返回值为零

    }
    
    CDialogEx::OnClose();
}
复制代码

 

 本工程下载:

链接:https://pan.baidu.com/s/1o1mITr-4OtoHHWwaI_bFGQ     提取码:6666

 

 

实例:远程钩子

1.写动态库

Dll.cpp

复制代码
#include <Windows.h>


HHOOK hMouseHook = NULL;//钩子句柄

LRESULT CALLBACK MouseProc(_In_ int    nCode, _In_ WPARAM wParam, _In_ LPARAM lParam)
{
    return 1; 
}



//1设置钩子
void SetHook(void)
{
    SetWindowsHookEx(WH_MOUSE, MouseProc,GetModuleHandle(L"Dll"),NULL);
    //GetModuleHandle  获取一个应用程序或动态链接库的模块句柄。只有在当前进程的场景中,这个句柄才会有效
    //参数:模块的文件名
    //返回值:HMODULE,如执行成功,则返回模块句柄。零表示失败

}

//2卸载钩子
void UnHook(void)
{
    if (hMouseHook != NULL) {
        UnhookWindowsHookEx(hMouseHook);
    }
}
复制代码

 

2.模块定义导出函数

右击工程-->添加-->新建项

 

 

LIBRARY Dll   //即将导出的库名
EXPORTS
    SetHook @1
    UnHook @2

 

 

3.生成dll

钩子库实例下载:

链接:https://pan.baidu.com/s/1p9OZgy06GJORK2V2d18OMQ    提取码:6666

4.新建MFC工程hook

dll库的调用看:https://www.cnblogs.com/liming19680104/p/17146631.html 

复制代码
// hookDlg.cpp: 实现文件
//

#include "pch.h"
#include "framework.h"
#include "hook.h"
#include "hookDlg.h"
#include "afxdialogex.h"

#ifdef _DEBUG
#define new DEBUG_NEW
#endif

#pragma comment(lib,"Dll.lib")

void SetHook(void);
void UnHook(void);



// 用于应用程序“关于”菜单项的 CAboutDlg 对话框

class CAboutDlg : public CDialogEx
{
public:
    CAboutDlg();

// 对话框数据
#ifdef AFX_DESIGN_TIME
    enum { IDD = IDD_ABOUTBOX };
#endif

    protected:
    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持

// 实现
protected:
    DECLARE_MESSAGE_MAP()
};

CAboutDlg::CAboutDlg() : CDialogEx(IDD_ABOUTBOX)
{
}

void CAboutDlg::DoDataExchange(CDataExchange* pDX)
{
    CDialogEx::DoDataExchange(pDX);
}

BEGIN_MESSAGE_MAP(CAboutDlg, CDialogEx)
END_MESSAGE_MAP()


// ChookDlg 对话框



ChookDlg::ChookDlg(CWnd* pParent /*=nullptr*/)
    : CDialogEx(IDD_HOOK_DIALOG, pParent)
{
    m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}

void ChookDlg::DoDataExchange(CDataExchange* pDX)
{
    CDialogEx::DoDataExchange(pDX);
}

BEGIN_MESSAGE_MAP(ChookDlg, CDialogEx)
    ON_WM_SYSCOMMAND()
    ON_WM_PAINT()
    ON_WM_QUERYDRAGICON()
    ON_BN_CLICKED(IDC_BUTTON1, &ChookDlg::OnBnClickedButton1)
    ON_BN_CLICKED(IDC_BUTTON2, &ChookDlg::OnBnClickedButton2)
END_MESSAGE_MAP()


// ChookDlg 消息处理程序

BOOL ChookDlg::OnInitDialog()
{
    CDialogEx::OnInitDialog();

    // 将“关于...”菜单项添加到系统菜单中。

    // IDM_ABOUTBOX 必须在系统命令范围内。
    ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
    ASSERT(IDM_ABOUTBOX < 0xF000);

    CMenu* pSysMenu = GetSystemMenu(FALSE);
    if (pSysMenu != nullptr)
    {
        BOOL bNameValid;
        CString strAboutMenu;
        bNameValid = strAboutMenu.LoadString(IDS_ABOUTBOX);
        ASSERT(bNameValid);
        if (!strAboutMenu.IsEmpty())
        {
            pSysMenu->AppendMenu(MF_SEPARATOR);
            pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
        }
    }

    // 设置此对话框的图标。  当应用程序主窗口不是对话框时,框架将自动
    //  执行此操作
    SetIcon(m_hIcon, TRUE);            // 设置大图标
    SetIcon(m_hIcon, FALSE);        // 设置小图标

    // TODO: 在此添加额外的初始化代码

    return TRUE;  // 除非将焦点设置到控件,否则返回 TRUE
}

void ChookDlg::OnSysCommand(UINT nID, LPARAM lParam)
{
    if ((nID & 0xFFF0) == IDM_ABOUTBOX)
    {
        CAboutDlg dlgAbout;
        dlgAbout.DoModal();
    }
    else
    {
        CDialogEx::OnSysCommand(nID, lParam);
    }
}

// 如果向对话框添加最小化按钮,则需要下面的代码
//  来绘制该图标。  对于使用文档/视图模型的 MFC 应用程序,
//  这将由框架自动完成。

void ChookDlg::OnPaint()
{
    if (IsIconic())
    {
        CPaintDC dc(this); // 用于绘制的设备上下文

        SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), 0);

        // 使图标在工作区矩形中居中
        int cxIcon = GetSystemMetrics(SM_CXICON);
        int cyIcon = GetSystemMetrics(SM_CYICON);
        CRect rect;
        GetClientRect(&rect);
        int x = (rect.Width() - cxIcon + 1) / 2;
        int y = (rect.Height() - cyIcon + 1) / 2;

        // 绘制图标
        dc.DrawIcon(x, y, m_hIcon);
    }
    else
    {
        CDialogEx::OnPaint();
    }
}

//当用户拖动最小化窗口时系统调用此函数取得光标
//显示。
HCURSOR ChookDlg::OnQueryDragIcon()
{
    return static_cast<HCURSOR>(m_hIcon);
}



void ChookDlg::OnBnClickedButton1()
{
    SetHook();


}


void ChookDlg::OnBnClickedButton2()
{
    UnHook();


}
复制代码

实例工程下载:

链接:https://pan.baidu.com/s/1phSHCATgrQm68M02b0nLzA      提取码:6666   

 

 

 

 

posted @   天子骄龙  阅读(968)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· 别再用vector<bool>了!Google高级工程师:这可能是STL最大的设计失误
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 单元测试从入门到精通
· 上周热点回顾(3.3-3.9)
历史上的今天:
2022-01-21 halcon-dev_update_off()和dev_update_on()
2022-01-21 qt5-使用MSVC编译器中文时
2022-01-21 halcon-Qt 5.13联合halcon的配置
2022-01-21 halcon-read_image读入图像
2022-01-21 halcon-HDevelop使用
2022-01-21 对比色
2020-01-21 python-opencv-图像毛玻璃模糊效果
点击右上角即可分享
微信分享提示

目录导航