首页 > 解决方案 > 带有 Qwidget 的 KeyPressEvent 不起作用?

问题描述

如果我按任何键,什么都不会发生。如何实施?而且我注意到如果我按下了某个键(第一次/非常开始/第一个字符),在文本框中显示该按下的字符需要更多时间。如何解决它?

from PyQt5 import QtWidgets,QtCore,QtGui

class MyKey(QtWidgets.QWidget):
    def __init__(self):
        super(). __init__()
        self.setWindowTitle(" My Key Board")
        self.ui()

    def ui(self):
        self.tb = QtWidgets.QLineEdit()
        self.vb = QtWidgets.QVBoxLayout()
        self.vb.addWidget(self.tb)
        self.setLayout(self.vb)

    def keyPressEvent(self, e):
        print("key pressed")
        print(e.key())
        if type(e) == QtGui.QKeyEvent:
            if e.key() == QtCore.Qt.Key_A:
                print("ypu pressed 'A'")


if __name__ == "__main__":
    import sys
    myapp = QtWidgets.QApplication(sys.argv)
    mywindow = MyKey()
    mywindow.show()
    sys.exit(myapp.exec_())

标签: pythonpyqtpyqt5

解决方案


在 Qt 中,KeyPressEvent 仅发送到具有焦点的小部件,如果它使用它,则事件不会发送到父小部件,否则也会通知父小部件。

在这种情况下,QLineEdit 是具有焦点并使用具有文本、数字和一些特殊键的事件的小部件,因此在这些情况下不会通知父小部件。QLineEdit 不使用其他键,例如父小部件接收的 , 等F2Escape

如果您想收听特殊键或键组合,则必须使用 aQShortcut但是因为 OP 想要收听所有事件,那么一个可能的解决方案是在与窗口关联的 QWindow 上安装一个事件过滤器,如果它接收到窗口具有焦点时的键盘事件,无论哪个小部件具有焦点。

from PyQt5 import QtWidgets, QtCore, QtGui


class MyKey(QtWidgets.QWidget):
    def __init__(self):
        super().__init__()
        self.setWindowTitle(" My Key Board")
        self.ui()

    def ui(self):
        self.tb = QtWidgets.QLineEdit()

        vb = QtWidgets.QVBoxLayout(self)
        vb.addWidget(self.tb)

    def handle_key_press(self, key, text):
        if key == QtCore.Qt.Key_A:
            print("ypu pressed 'A'")


class KeyHelper(QtCore.QObject):
    key_pressed = QtCore.pyqtSignal(int, str)

    def __init__(self, window):
        super().__init__(window)
        self._window = window
        self.window.installEventFilter(self)

    @property
    def window(self):
        return self._window

    def eventFilter(self, obj, event):
        if obj is self._window and event.type() == QtCore.QEvent.KeyPress:
            self.key_pressed.emit(event.key(), event.text())
        return super().eventFilter(obj, event)


if __name__ == "__main__":
    import sys

    myapp = QtWidgets.QApplication(sys.argv)

    mywindow = MyKey()
    mywindow.show()

    helper = KeyHelper(mywindow.windowHandle())
    helper.key_pressed.connect(mywindow.handle_key_press)

    sys.exit(myapp.exec_())

推荐阅读