从热键启动时,SendInput为Ctrl + C文本不起作用

时间:2017-07-31 23:15:59

标签: c++ winapi sendinput

目前使用此代码复制Windows 10中当前打开的窗口中的所选文本。如果我的目标程序(记事本)具有焦点,我可以自行运行此代码。记事本中的选定文本将复制到数据变量OK。

wchar_t title[MAX_PATH];
HWND target_window = GetForegroundWindow();
GetWindowText(target_window, title, MAX_PATH);
std::wcout << "Target window is '" << title << "'" << std::endl;

// Send Control + C
int key_count = 4;

INPUT* input = new INPUT[key_count];
for (int i = 0; i < key_count; i++)
{
    input[i].ki.dwFlags = 0;
    input[i].type = INPUT_KEYBOARD;
}

input[0].ki.wVk = VK_CONTROL;
input[0].ki.wScan = MapVirtualKey(VK_CONTROL, MAPVK_VK_TO_VSC);
input[1].ki.wVk = 0x43; // Virtual key code for 'c'
input[1].ki.wScan = MapVirtualKey(0x43, MAPVK_VK_TO_VSC);
input[2].ki.dwFlags = KEYEVENTF_KEYUP;
input[2].ki.wVk = input[0].ki.wVk;
input[2].ki.wScan = input[0].ki.wScan;

input[3].ki.dwFlags = KEYEVENTF_KEYUP;
input[3].ki.wVk = input[1].ki.wVk;
input[3].ki.wScan = input[1].ki.wScan;

if (!SendInput(key_count, (LPINPUT)input, sizeof(INPUT)))
{
    // TODO: error handling
}
else
{
    // not ideal but not sure of another way to wait for SendInput to complete
    Sleep(100); 
    if (OpenClipboard(NULL))
    {
        HGLOBAL hglb = GetClipboardData(CF_UNICODETEXT);
        LPWSTR lpwstr = (LPWSTR)(GlobalLock(hglb));
        std::wstring data(lpwstr);
        GlobalUnlock(hglb);
        CloseClipboard();
        // do something with selected text in data
    }
    else
    {
        // TODO: error handling
    }
}

但是,如果我通过Hotkey启动完全相同的代码,则不起作用:

if (RegisterHotKey(
    NULL,
    1,
    MOD_CONTROL | MOD_ALT | MOD_NOREPEAT,
    VK_OEM_2))  // back slash question mark key
{
    std::cout << "Hotkey 'Ctrl+Alt+/' registered, using MOD_NOREPEAT flag\n";
}

MSG msg = { 0 };
while (GetMessage(&msg, NULL, 0, 0) != 0)
{
    if (msg.message == WM_HOTKEY)
    {
        std::cout << "WM_HOTKEY received\n";

        // Call function to COPY TEXT here

        if (RegisterHotKey(
            NULL,
            1,
            MOD_CONTROL | MOD_ALT | MOD_NOREPEAT,
            VK_OEM_2))  // back slash question mark key
        {
            std::cout << "Hotkey 'Ctrl+Alt+/' registered, using MOD_NOREPEAT flag\n";
        }
    }
}

现在,在这两种情况下,GetWindowText()都会显示我要复制文本的程序的标题。

另外,我写了一个简单的测试实用程序来检查 Ctrl + C 是否正在传递给Window。似乎正在传递 Ctrl + C ,但副本没有发生。

1 个答案:

答案 0 :(得分:1)

Alt 是否因为热键而仍然按下并且您实际上正在发送 Ctrl + Alt + C < /骨节病>? SendInput将输入直接插入全局输入队列。

您可以尝试设置一个计时器以响应热键,并在计时器处理程序中调用GetAsyncKeyState,直到所有修改键都生成,然后再生成输入。

更好的选择是使用UI Automation而不是像这样的黑客。