美文网首页Qt学习
Qt全局热键实现

Qt全局热键实现

作者: 玖零儛 | 来源:发表于2020-06-08 10:33 被阅读0次

    Qt 中没有实现好的跨平台全局热键,需要自己调用系统 API 来实现。

    监听系统原生事件

    Qt 作为跨平台的框架,实现的功能是在每个平台都共有的部分,而有的功能可能由于某些原因不能跨平台就没有,这时候就需要调用系统原生的 API,比如全局热键的功能、电池休眠事件监听等。

    窗口的原生事件

    窗口的原生事件可以通过重载QWidget::nativeEvent(const QByteArray &eventType, void *message, long *result)来实现。在函数中返回 true 会停止传递,如果返回 false 会继续由 Qt 处理事件。当窗口有句柄的时候,触发了平台原生的事件就会调用此函数。由 eventType 标识本机平台事件,这些事件的参数在 message 中传递。不同平台的 eventType 、message 如下:

    Platform Event Type Identifier Message Type Result Type
    Windows "windows_generic_MSG" MSG * LRESULT
    macOS "NSEvent" NSEvent *
    XCB "xcb_generic_event_t" xcb_generic_event_t *

    下面举个例子,比如当前窗口在做一些操作,笔记本合上盖子进入休眠,要监听电池相关的事件:

    bool MainWidget::nativeEvent(const QByteArray &eventType, void *message, long *result)
    {
        if (eventType == "windows_generic_MSG" || eventType == "windows_dispatcher_MSG") {
            MSG* msg = static_cast<MSG*>(message);
            if (msg->message == WM_POWERBROADCAST) {
                if(msg->wParam == PBT_APMSUSPEND) {
                    qDebug("系统休眠");
                }
                else if(msg->wParam == PBT_APMRESUMESUSPEND || msg->wParam == PBT_APMRESUMEAUTOMATIC) {
                    qDebug("系统唤醒");
                }
            }
        }
    
        return QWidget::nativeEvent(eventType, message, result);
    }
    

    应用级原生事件

    像全局热键这样的事件是应用级别的,事件的监听就要放在 app 上。通过继承QAbstractNativeEventFilter来实现自己的原生事件过滤器并安装到 app 上即可。当有原生事件触发后就会调用事件过滤器的virtual bool nativeEventFilter(const QByteArray &eventType, void *message, long *)函数。
    要处理热键的话重写函数如下:

    bool MyWinEventFilter::nativeEventFilter(const QByteArray &eventType, void *message, long *)
    {
       if(eventType == "windows_generic_MSG") {
            MSG *msg = static_cast<MSG *>(message);
            if(msg->message == WM_HOTKEY) {
                //判断是否为指定热键按下
            }
    
        }
        return false;
    }
    

    热键的注册

    上面处理好了事件发生后的步骤,还需要告诉系统哪些算是热键。注册用到的原生 API 有:

    BOOL   RegisterHotKey(
    HWND   hWnd,         //响应该热键的窗口句柄, 由于是应用级的,可以不传
    Int   id,            //该热键的唯一标识
    UINT   fsModifiers,  //该热键的辅助按键,Ctrl、Alt、Shift
    UINT   vk            //该热键的键值
    );
    
    BOOL UnregisterHotKey(
    HWND hWnd,           //响应该热键的窗口句柄,同不传
    Int id               //该热键的唯一标识
    );
    
    

    注册时按键的虚拟码可以去官网查询
    注册按键的函数一般都是字符串的形式,可以自己写个转换函数,也可以使用 Qt 的QKeySequence先得到热键的 Qt 的键盘码,再映射到 windwos 的虚拟码。

    具体实现

    具体代码就不贴了,在 Github 上有个封装好各个平台的全局热键的仓库,QHotkey

    windows 相关实现在QHotkey/QHotkey/qhotkey_win.cpp中,里面就是调用 RegisterHotKey 来实现的。

    相关文章

      网友评论

        本文标题:Qt全局热键实现

        本文链接:https://www.haomeiwen.com/subject/picltktx.html