全局低级键盘钩子[通俗易懂]

全局低级键盘钩子[通俗易懂]/*========================================================================文件:kbevent.h说明:全局消息钩子时间:2005-03-20编写:oshj||oshj@21cn.c

大家好,又见面了,我是你们的朋友全栈君。 /*========================================================================

文件: kbevent.h

说明:全局消息钩子

时间:2005-03-20

编写:oshj ||
oshj@21cn.com

环境:VC6.0/Win2000 Pro/SP4/1024*768

特别说明:NT5.0以上操作系统才能支持,WIN9X不支持!

=========================================================================*/

//————————————————————————-

#ifndef _INC_KEEVENT

#define _INC_KEEVENT

#include <windows.h>
#include <stdio.h>
//————————————————————————-
#define DLLEXPORT extern “C” __declspec(dllexport)
//————————————————————————-
#define MAX_TEXTLEN 1024
#define UM_GETTEXT WM_USER + 0x392
//————————————————————————-
//低级键盘钩子回调函数
LRESULT CALLBACK LowLevelKBProc(int nCode, WPARAM wParam, LPARAM lParam);
//按ctrl+alt+del时系统管理器无效
void RegKeyEnable(BOOL bEnable=TRUE);

///
//键盘空虚时间回调函数
LRESULT CALLBACK MyKbdHook(int code, WPARAM wParam, LPARAM lParam);
//鼠标空虚时间回调函数
LRESULT CALLBACK MyMouseHook(int code, WPARAM wParam, LPARAM lParam);
//————————————————————————-

#endif // _INC_KEEVENT

//.cpp

//————————————————————————-
#define _WIN32_WINNT  0x0500    //仅NT5.0以上系统可用

#include “kbevent.h”
//————————————————————————-
// Magic registry key/value for “Remove Task Manager” policy.
#define HKCU HKEY_CURRENT_USER
LPCTSTR KEY_DisableTaskMgr =
 “Software\\Microsoft\\Windows\\CurrentVersion\\Policies\\System”;
LPCTSTR VAL_DisableTaskMgr = “DisableTaskMgr”;
//————————————————————————-
#pragma data_seg(“.sdata”)
HHOOK g_hHookKeybd = NULL;  
HHOOK g_hHookMouse = NULL;  
DWORD g_dwLastInputTick = 0; 
#pragma data_seg()
#pragma comment(linker,”-section:.sdata,rws”)
//————————————————————————-
// 全局变量
HINSTANCE     g_hInstance = NULL;              // 模块实例句柄
HHOOK         g_hHook = NULL;                  // 钩子句柄
//————————————————————————-
//dll main
BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved)
{

 // 保存模块实例句柄
    g_hInstance = (HINSTANCE)hinstDLL;

    switch (fdwReason)
 {

  case DLL_PROCESS_ATTACH:
   break;

  case DLL_THREAD_ATTACH:
    break;

  case DLL_THREAD_DETACH:
    break;
  
  case DLL_PROCESS_DETACH:   
   if( NULL!=g_hHook )
    UnhookWindowsHookEx( g_hHook );
   if( NULL!=g_hHookKeybd )
    UnhookWindowsHookEx( g_hHookKeybd );
   if( NULL!=g_hHookMouse )
    UnhookWindowsHookEx( g_hHookMouse );
   break;
    }
    return TRUE;
}

//————————————————————————-
// 底层键盘钩子函数
LRESULT CALLBACK LowLevelKBProc(int nCode, WPARAM wParam, LPARAM lParam)
{

 if (nCode == HC_ACTION)
 {

   PKBDLLHOOKSTRUCT p;

  switch (wParam)
  {

   case WM_KEYDOWN: 
   case WM_SYSKEYDOWN:
   case WM_KEYUP:   
   case WM_SYSKEYUP:
    p = (PKBDLLHOOKSTRUCT)lParam;

    if( //(p->vkCode==VK_TAB) ||
     (p->vkCode==VK_ESCAPE) ||
     (p->vkCode==VK_LWIN) ||
     (p->vkCode==VK_RWIN) ||
     (p->vkCode==VK_CONTROL) ||
     (p->vkCode==VK_SHIFT) ||
     (p->vkCode==VK_DELETE) ||
     ( (p->vkCode==VK_F4) && ((p->flags & LLKHF_ALTDOWN)!=0) ) ||    //ALT+F4
     ( (p->vkCode==VK_TAB) && ((p->flags & LLKHF_ALTDOWN)!=0) ) ||    //ALT+TAB
     ( (p->vkCode==VK_ESCAPE) && ((p->flags & LLKHF_ALTDOWN)!=0) ) ||   //ALT+ESC
     ( (p->vkCode==VK_ESCAPE) && ((GetKeyState(VK_CONTROL) & 0x8000)!=0) ) || //CTRL+ESC
     ( (p->vkCode==VK_DELETE) && ((p->flags & LLKHF_ALTDOWN)!=0) && ((GetKeyState(VK_CONTROL) & 0x8000)!=0)  ) //CTRL+ESC

      )
    {

     return TRUE;
    }

   break;
  }

 }

 return ::CallNextHookEx( g_hHook, nCode, wParam, lParam );

}

//————————————————————————-
//开始钩子
DLLEXPORT BOOL WINAPI StartKBEvent()
{

 //如果已经安装键盘钩子则返回 FALSE
 if ( NULL!=g_hHook ) return FALSE;

 RegKeyEnable(FALSE);
 //安装底层键盘钩子
 //NT5.0以上操作系统才能支持
 //WIN98不支持!
 g_hHook = SetWindowsHookEx( WH_KEYBOARD_LL, LowLevelKBProc, g_hInstance, NULL );
 
 if ( g_hHook==NULL) return FALSE;

 return TRUE;

}
 
//————————————————————————-
//结束钩子
DLLEXPORT BOOL WINAPI StopKBEvent()
{

 RegKeyEnable(TRUE);
 //卸载钩子
    if( UnhookWindowsHookEx(g_hHook)==0 )
  return FALSE;
 g_hHook = NULL;

    return TRUE;

}
//————————————————————————-
//按ctrl+alt+del时系统管理器无效
void RegKeyEnable(BOOL bEnable)
{

 HKEY hk;
 if( RegOpenKey(HKCU, KEY_DisableTaskMgr,&hk)!=ERROR_SUCCESS )
 {

  RegCreateKey(HKCU, KEY_DisableTaskMgr, &hk);
 }
 //
 if( TRUE==bEnable )
 {

  RegDeleteValue(hk,VAL_DisableTaskMgr);

 }
 else
 {

  DWORD val=1;
  RegSetValueEx(hk, VAL_DisableTaskMgr, NULL,REG_DWORD, (BYTE*)&val, sizeof(val));

 }

}

///
//以下是获取键盘鼠标空闲时间的钩子部分
//————————————————————————-
//键盘空闲时间回调函数
LRESULT CALLBACK MyKbdHook(int code, WPARAM wParam, LPARAM lParam)
{

 if( code==HC_ACTION )
 {

  g_dwLastInputTick = GetTickCount();
  PKBDLLHOOKSTRUCT p;
  p = (PKBDLLHOOKSTRUCT) lParam;
  //MessageBox(0,”111″,”aaa”,MB_OK);
  //Z键连续按下2次以上
  if( (p->vkCode==VK_ESCAPE) )// && ((p->flags & KF_REPEAT)>=2) )
  {

   //MessageBox(0,”yes”,”aaa”,MB_OK);
   return TRUE;
  }
 }
 return ::CallNextHookEx( g_hHookKeybd, code, wParam, lParam );
}
//————————————————————————-
//鼠标空闲时间回调函数
LRESULT CALLBACK MyMouseHook(int code, WPARAM wParam, LPARAM lParam)
{

 if (code==HC_ACTION)
 {

  g_dwLastInputTick = GetTickCount();
 }
 return ::CallNextHookEx( g_hHookMouse, code, wParam, lParam );
}
//————————————————————————-
//开始钩子
DLLEXPORT BOOL WINAPI IdleInit()
{

 if( NULL!=g_hHookKeybd && NULL!=g_hHookMouse )  return FALSE;

 g_hHookKeybd = SetWindowsHookEx( WH_KEYBOARD, MyKbdHook, g_hInstance, 0 );
 g_hHookMouse = SetWindowsHookEx( WH_MOUSE, MyMouseHook, g_hInstance, 0 );
 g_dwLastInputTick = GetTickCount();

 return TRUE;

}
//————————————————————————-
//结束钩子
DLLEXPORT BOOL WINAPI IdleExit()
{

 if( NULL==g_hHookKeybd && NULL==g_hHookMouse )  return FALSE;

 UnhookWindowsHookEx( g_hHookKeybd );
 UnhookWindowsHookEx( g_hHookMouse );

 return TRUE;

}

//————————————————————————-
//获得空闲时间
DLLEXPORT DWORD IdleGetLastTime()
{

 return g_dwLastInputTick;
}

文章转自:http://blog.vckbase.com/flowsand/archive/2006/11/30/23312.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/140934.html原文链接:https://javaforall.net

(0)
上一篇 2022年5月8日 下午8:00
下一篇 2022年5月8日 下午8:20


相关推荐

  • 五种聚类方法_聚类分析是一种降维方法吗

    五种聚类方法_聚类分析是一种降维方法吗本文为雷锋字幕组编译的技术博客,原标题The5ClusteringAlgorithmsDataScientistsNeedtoKnow,作者为GeorgeSeif。聚类是一种关于数据点分组的机器学习技术。给出一组数据点,我们可以使用聚类算法将每个数据点分类到特定的组中。理论上,同一组中的数据点应具有相似的属性或特征,而不同组中的数据点应具有相当不同的属性或特征(即类内差异小,…

    2022年10月20日
    5
  • 腾讯,开始新一轮「谨慎扩张」

    腾讯,开始新一轮「谨慎扩张」

    2026年3月12日
    2
  • 关于数据连接配置connectionStrings的写法[通俗易懂]

    关于数据连接配置connectionStrings的写法[通俗易懂]参考http://www.connectionstrings.com/1、SQLServer<addname="ApplicationName"connectionSt

    2022年7月1日
    48
  • webpack开发环境和生产环境_环境问题是如何产生和发展的

    webpack开发环境和生产环境_环境问题是如何产生和发展的前言如果我们需要使用webpack,就需要依赖node环境nvmnodenpmwebpack@cliwebpacknvm安装nvm是一个用来管理node版本的工具。我们之所以需要使用n

    2022年7月30日
    8
  • 软件是bs架构还是cs架构_数据库为什么cs架构

    软件是bs架构还是cs架构_数据库为什么cs架构从定义上:CS即客户端到服务器架构;BS即浏览器到服务器架构从效率上:C/S效率高,大部分的数据已经安装在系统上,B/S效率低,每次都要加载最新的数据从迭代升级上:C/S需要删除老版本在安装新版本(在升级完成桌面图标会有一个刷新的动作);B/S则无缝升级;从安全上:C/S更安全,需要安装\注册\登录;B/S有浏览器就可以使用,安全程度低;从开发成本上:B/S成本低;C/S需要不同的系统开发人员,成本高…

    2025年10月9日
    7
  • 分布式Session共享解决方案「建议收藏」

    Session是服务器用来保存用户操作的一系列会话信息,由Web容器进行管理。单机情况下,不存在Session共享的情况,分布式情况下,如果不进行Session共享会出现请求落到不同机器要重复登录的情况,一般来说解决Session共享有以下几种方案。1、session复制session复制是早期的企业级的使用比较多的一种服务器集群session管理机制。应用服务器开启web容器的sessi…

    2022年4月4日
    41

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号