簡體   English   中英

QWebEngineView 中的 Qt 事件傳播

[英]Qt Event Propagation in QWebEngineView

我有一個名為generate_input_event的函數。 我正在嘗試使用此函數來模擬 QWebEngineView 中的按鍵操作。

def generate_input_event(window_id, key_code, modifiers, low_level_data, x, y):
    modifiers_flag = create_modifiers_flag(modifiers)
    logging.info("generate input, window: {} code: {}, modifiers {}".format(
        window_id, key_code, modifiers_flag))
    event = QKeyEvent(QEvent.KeyPress, key_code, modifiers_flag)
    event.artificial = True
    event_window = window.get_window(window_id)
    QCoreApplication.sendEvent(event_window.qtwindow, event)

每當我運行我的程序並在我的 QWebEngineView 中突出顯示一個輸入字段,並調用generate_input_event ,預計它會將該字母輸入到輸入字段中。

我還設置了一個事件過濾器來捕獲所有按鍵操作,除了我人工生成的按鍵操作。

class EventFilter(QWidget):
    def __init__(self, parent=None):
        super(EventFilter, self).__init__(parent)
        qApp.installEventFilter(self)

    def eventFilter(self, obj, event):
        if (event.type() == QEvent.KeyPress and hasattr(event, 'artificial')):
            logging.info("artificial event")
            return False. # send to widget
        elif (event.type() == QEvent.KeyPress and not is_modifier(event.key())):
            modifiers = create_modifiers_list(event.modifiers())
            key_string = create_key_string(event)
            key_code = event.key()
            logging.info("send code: {} string: {} modifiers {}".format(
                key_code, key_string, modifiers))
            return True. # do not forward to widgets
        return False

但是,當我實際運行我的代碼時,這是我得到的以下輸出:

INFO:root:send code: 65 string: a modifiers ['']
INFO:root:generate input, window: 1 code: 65, modifiers <PyQt5.QtCore.Qt.KeyboardModifiers object at 0x106a4ea58>
INFO:root:artificial event

輸出看起來正確,但是, QWebEngineView 的輸入字段實際上從未獲得由generate_input_event人為生成的字母。

PS 如果您出於上下文原因希望查看整個文件/項目,請在此處查看此分支/文件: https : //github.com/atlas-engineer/next/blob/generate_events/ports/pyqt-webengine /實用程序.py

Qt Webengine 使用 RenderWidgetHostViewQtDelegateWidget 進行渲染,這是在加載頁面后創建的,因此您必須在 load() 或 setHtml() 之后訪問它,以便小部件必須發送這些事件。

以下示例將顯示 QWebEngineView 和 QLineEdit,在顯示兩個窗口后,您在 QLineEdit 中鍵入的內容將顯示在 QWebEngineView 中。

from PyQt5 import QtCore, QtGui, QtWidgets, QtWebEngineWidgets


class EventFilter(QtCore.QObject):
    def eventFilter(self, obj, event):
        if event.type() == QtCore.QEvent.KeyPress and hasattr(
            event, "artificial"
        ):
            print("event:", event.key(), event.text())
            return False
        return super().eventFilter(obj, event)


class ForwardKeyEvent(QtCore.QObject):
    def __init__(self, sender, receiver, parent=None):
        super(ForwardKeyEvent, self).__init__(parent)
        self.m_sender = sender
        self.m_receiver = receiver
        self.m_sender.installEventFilter(self)

    def eventFilter(self, obj, event):
        if self.m_sender is obj and event.type() == QtCore.QEvent.KeyPress:
            # self.m_receiver.setFocus()
            new_event = QtGui.QKeyEvent(
                QtCore.QEvent.KeyPress,
                event.key(),
                event.modifiers(),
                event.text(),
            )
            new_event.artificial = True
            QtCore.QCoreApplication.postEvent(self.m_receiver, new_event)
        return super().eventFilter(obj, event)


if __name__ == "__main__":
    import sys

    app = QtWidgets.QApplication(sys.argv)
    ef = EventFilter()
    app.installEventFilter(ef)

    lineedit = QtWidgets.QLineEdit()
    lineedit.show()

    view = QtWebEngineWidgets.QWebEngineView()
    view.resize(640, 480)
    view.show()

    view.load(QtCore.QUrl("https://www.google.com/"))
    # RenderWidgetHostViewQtDelegateWidget is created after loading a page 
    # so you must access it after load() or setHtml().
    render_widget = view.findChild(QtWidgets.QWidget)
    print(render_widget.metaObject().className())
    assert(render_widget)
    fe = ForwardKeyEvent(lineedit, render_widget)
    sys.exit(app.exec_())

可以使用以下最小示例演示為 QWebengineview 發布和偵聽事件的正確方法:


from PyQt5 import QtCore, QtGui, QtWidgets, QtWebEngineWidgets
from PyQt5.QtCore import Qt


class ForwardKeyEvent(QtCore.QObject):
    def __init__(self, sender, receiver, parent=None):
        super(ForwardKeyEvent, self).__init__(parent)
        self.m_sender = sender
        self.m_receiver = receiver
        self.m_sender.installEventFilter(self)

    def eventFilter(self, obj, event):
        if self.m_sender is obj and event.type() == QtCore.QEvent.KeyPress:
            new_event = QtGui.QKeyEvent(
                QtCore.QEvent.KeyPress,
                65,
                Qt.KeyboardModifiers(),
                "a",
            )
            new_event.artificial = True
            QtCore.QCoreApplication.postEvent(self.m_receiver.focusProxy(), new_event)
            return True
        return False


if __name__ == "__main__":
    import sys
    app = QtWidgets.QApplication(sys.argv)
    lineedit = QtWidgets.QLineEdit()
    lineedit.show()
    view = QtWebEngineWidgets.QWebEngineView()
    view.resize(640, 480)
    view.show()
    view.load(QtCore.QUrl("https://www.google.com/"))
    # RenderWidgetHostViewQtDelegateWidget is created after loading a page
    # so you must access it after load() or setHtml().
    fe = ForwardKeyEvent(lineedit, view)
    sys.exit(app.exec_())

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM