介绍
sendinput在某些应用上面模拟按键无效,需要转换成unicode字符才可以模拟,特记录下。
代码
#include <stdio.h>
#include <stdint.h>
#include <windows.h>
const int SERVER_AUXILIARY_TRANSLATE_TABLE[] = {
{ VK_LSHIFT },
{ VK_RSHIFT },
{ VK_SHIFT },
{ VK_LCONTROL },
{ VK_RCONTROL },
{ VK_CONTROL },
{ VK_LMENU },
{ VK_RMENU },
{ VK_MENU },
{ VK_LEFT},
{ VK_UP},
{ VK_RIGHT},
{ VK_DOWN},
{ VK_TAB },
{ VK_BACK },
{ VK_RETURN },
{ VK_EXECUTE },
{ VK_PAUSE },
{ VK_ESCAPE },
{ VK_CLEAR },
{ VK_DELETE },
{ VK_HOME },
{ VK_PRIOR },
{ VK_NEXT },
{ VK_END },
{ VK_SELECT },
{ VK_SNAPSHOT },
{ VK_INSERT },
{ VK_HELP },
{ VK_CANCEL },
{ VK_SPACE },
{ VK_TAB },
{ VK_RETURN },
{ VK_F1 },
{ VK_F2 },
{ VK_F3 },
{ VK_F4 },
{ VK_HOME },
{ VK_LEFT },
{ VK_UP },
{ VK_RIGHT },
{ VK_DOWN },
{ VK_PRIOR },
{ VK_NEXT },
{ VK_END },
{ VK_CLEAR },
{ VK_INSERT },
{ VK_DELETE },
{ 0xbb }, // FIXME: What is it?
{ VK_MULTIPLY },
{ VK_ADD },
{ VK_SEPARATOR },
{ VK_SUBTRACT },
{ VK_DECIMAL },
{ VK_DIVIDE },
{ VK_NUMPAD0 },
{ VK_NUMPAD1 },
{ VK_NUMPAD2 },
{ VK_NUMPAD3 },
{ VK_NUMPAD4 },
{ VK_NUMPAD5 },
{ VK_NUMPAD6 },
{ VK_NUMPAD7 },
{ VK_NUMPAD8 },
{ VK_NUMPAD9 },
{ VK_F1 },
{ VK_F2 },
{ VK_F3 },
{ VK_F4 },
{ VK_F5 },
{ VK_F6 },
{ VK_F7 },
{ VK_F8 },
{ VK_F9 },
{ VK_F10 },
{ VK_F11 },
{ VK_F12 },
{ VK_F13 },
{ VK_F14 },
{ VK_F15 },
{ VK_F16 },
{ VK_F17 },
{ VK_F18 },
{ VK_F19 },
{ VK_F20 },
{ VK_F21 },
{ VK_F22 },
{ VK_F23 },
{ VK_F24 },
{ VK_LWIN },
{ VK_RWIN },
{ VK_APPS }
};
bool isSysKey(int32_t keyvalue) {
for (int i = 0; i < sizeof(SERVER_AUXILIARY_TRANSLATE_TABLE) / sizeof(int); ++ i) {
if (keyvalue == SERVER_AUXILIARY_TRANSLATE_TABLE[i]) {
return true;
}
}
return false;
}
void sendInputKey(int32_t keyevent,int32_t keyvalue)
{
/// 如果是系统键,则直接输出
if (isSysKey(keyvalue) || keyevent != KEYEVENTF_KEYUP) {
INPUT input;
memset(&input,0,sizeof(INPUT));
input.type = INPUT_KEYBOARD;
input.ki.dwFlags = (DWORD)keyevent;
input.ki.wVk = (WORD)keyvalue;
input.ki.wScan = MapVirtualKey(keyvalue, 0);
SendInput(1,&input, sizeof(INPUT));
}else {
/// 否则,则转化成unicode输出
int wscanCode = MapVirtualKey(keyvalue, 0);
WCHAR ch;
int ret = ToUnicode(keyvalue, wscanCode, nullptr, &ch, sizeof(WCHAR), 0);
if (1 == ret) {
/// 如果只有一个键则直接输出
INPUT input;
memset(&input,0,sizeof(INPUT));
input.type = INPUT_KEYBOARD;
input.ki.dwFlags = (DWORD)keyevent | KEYEVENTF_UNICODE;
input.ki.wVk = 0;
input.ki.wScan = ch;
SendInput(1,&input, sizeof(INPUT));
}else {
/// 有多个按键则默认输出
INPUT input;
memset(&input,0,sizeof(INPUT));
input.type = INPUT_KEYBOARD;
input.ki.dwFlags = (DWORD)keyevent;
input.ki.wVk = (WORD)keyvalue;
input.ki.wScan = MapVirtualKey(keyvalue, 0);
SendInput(1,&input, sizeof(INPUT));
}
}
}
int main()
{
Sleep(1000);
sendInputKey(0, 50);
sendInputKey(KEYEVENTF_KEYUP, 50);
getchar();
return 0;
}
总结
当是系统键值(比如ctrl,alt之类)时,则不需要添加KEYEVENTF_UNICODE标志进行输入,否则,需要转换成unicode字符,关添加KEYEVENTF_UNICODE字符对键值进行输入,因为在某些应用场景不添加KEYEVENTF_UNICODE标志时,模拟键盘消息的输入是无效的。