不能在 QPlainTextEdit 中使用其他键与 keyPressEvent



我正在尝试在QPlainTextEdit小部件上安装keyPressEvent,这样当键入时,我正常输入,当我按回车键时,它会将文本添加到QPlainTextEdit中。我必须文件,QtDes.py 由Qt Designer创建,另一个文件,QtTextEvent.py。这些是我的文件:

QtDes.py:

# -*- coding: utf-8 -*-
# Form implementation generated from reading ui file 'QtDes.ui'
#
# Created by: PyQt5 UI code generator 5.13.1
#
# WARNING! All changes made in this file will be lost!
from PyQt5 import QtGui, QtWidgets, QtCore
class Ui_MainWindow(object):
def __init__(self, *args, **kwargs):
super(Ui_MainWindow, self).__init__(*args, **kwargs)
self.exactAns = ""
self.approxAns = 0
def setupUi(self, MainWindow):
MainWindow.setObjectName("MainWindow")
MainWindow.resize(800, 569)
self.centralwidget = QtWidgets.QWidget(MainWindow)
self.centralwidget.setObjectName("centralwidget")
self.gridLayout = QtWidgets.QGridLayout(self.centralwidget)
self.gridLayout.setObjectName("gridLayout")
self.plainTextEdit = QtWidgets.QPlainTextEdit(self.centralwidget)
font = QtGui.QFont()
font.setFamily("Courier New")
self.plainTextEdit.setFont(font)
self.plainTextEdit.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAsNeeded)
self.plainTextEdit.setTabChangesFocus(False)
self.plainTextEdit.setLineWrapMode(QtWidgets.QPlainTextEdit.NoWrap)
self.plainTextEdit.setOverwriteMode(False)
self.plainTextEdit.setTabStopWidth(40)
self.plainTextEdit.setTabStopDistance(40.0)
self.plainTextEdit.setObjectName("plainTextEdit")
self.plainTextEdit.appendPlainText("First Line: ")
self.plainTextEdit.keyPressEvent = self.keyPressEvent
self.gridLayout.addWidget(self.plainTextEdit, 0, 0, 1, 1)
MainWindow.setCentralWidget(self.centralwidget)
self.menubar = QtWidgets.QMenuBar(MainWindow)
self.menubar.setGeometry(QtCore.QRect(0, 0, 800, 21))
self.menubar.setObjectName("menubar")
MainWindow.setMenuBar(self.menubar)
self.statusbar = QtWidgets.QStatusBar(MainWindow)
self.statusbar.setObjectName("statusbar")
MainWindow.setStatusBar(self.statusbar)
self.retranslateUi(MainWindow)
QtCore.QMetaObject.connectSlotsByName(MainWindow)
def retranslateUi(self, MainWindow):
_translate = QtCore.QCoreApplication.translate
MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow"))

if __name__ == "__main__":
import sys
app = QtWidgets.QApplication(sys.argv)
MainWindow = QtWidgets.QMainWindow()
ui = Ui_MainWindow()
ui.setupUi(MainWindow)
MainWindow.show()
sys.exit(app.exec_())

QtTextEvent.py:

from PyQt5 import QtCore, QtGui, QtWidgets
from QtDes import Ui_MainWindow
class MainWindow(QtWidgets.QMainWindow, Ui_MainWindow):
def __init__(self, *args, **kwargs):
QtWidgets.QMainWindow.__init__(self, *args, **kwargs)
self.setupUi(self)
def keyPressEvent(self, event):
if event.key() == QtCore.Qt.Key_Return or event.key() == QtCore.Qt.Key_Enter:
print("Enter pressed")
self.plainTextEdit.appendPlainText("New Line: ")
else:
super(MainWindow, self).keyPressEvent(event)
if __name__ == "__main__":
import sys
app = QtWidgets.QApplication(sys.argv)
w = MainWindow()
w.show()
sys.exit(app.exec_())

按 Enter 会执行它应该执行的操作,但按其他按钮不起作用。我从这个问题和这个问题中得到了答案。我的实现有问题吗,我该如何解决?

解释:

当您按下某个键时,keyPressEvent 方法实现向 QPlaintTextEdit 添加文本的任务,但您正在为其分配另一个不实现此逻辑的 keyPressEvent(从 QMainWindow(,即它不添加文本。因此,将一种方法分配给另一种方法是不正确的,因为您删除了小部件默认具有的行为。

溶液:

在您的情况下,只需要听键盘,如果您按 Enter 或返回,然后添加文本,然后要收听事件,您只需要一个事件过滤器。

为此,您必须删除 QtDes.py 文件中的self.plainTextEdit.keyPressEvent = self.keyPressEvent。同时在 QtTextEvent.py 文件中实现事件过滤器:

from PyQt5 import QtCore, QtGui, QtWidgets
from QtDes import Ui_MainWindow

class MainWindow(QtWidgets.QMainWindow, Ui_MainWindow):
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self.setupUi(self)
self.plainTextEdit.installEventFilter(self)
def eventFilter(self, obj, event):
if obj is self.plainTextEdit and event.type() == QtCore.QEvent.KeyPress:
if event.key() in (QtCore.Qt.Key_Return, QtCore.Qt.Key_Enter):
print("Enter pressed")
self.plainTextEdit.appendPlainText("New Line: ")
return True
return super(MainWindow, self).eventFilter(obj, event)

if __name__ == "__main__":
import sys
app = QtWidgets.QApplication(sys.argv)
w = MainWindow()
w.show()
sys.exit(app.exec_())

另一种可能的解决方案是从 QPlainTextEdit 继承并重写 keyPressEvent 方法。

plaintextedit.py

class PlainTextEdit(QtWidgets.QPlainTextEdit):
def keyPressEvent(self, event):
if event.key() in (QtCore.Qt.Key_Return, QtCore.Qt.Key_Enter):
print("Enter pressed")
self.appendPlainText("New Line: ")
return
super(PlainTextEdit, self).keyPressEvent(event)

然后更改为:

QtDes.py

from plaintextedit import PlainTextEdit
# ...
class Ui_MainWindow(object):
def setupUi(self, MainWindow):
# ...
self.plainTextEdit =PlainTextEdit(self.centralwidget)
# ...

(您也可以按照我在此答案中的指示进行推广(

你不能"安装keyPressEvent",即使你可以,它也不适用于你的方法。

通过这样做:

self.plainTextEdit.keyPressEvent = self.keyPressEvent

你实际上正在做这样的事情:

mainWindowInstance.plainTextEdit.keyPress = mainWindowInstance.keyPressEvent

结果是 plainTextEdit 不会接收事件,而是主窗口接收,并且由于如果不处理事件总是发送回父级,因此不会发生任何其他事情。

理论上的解决方案是针对 QPlainTextEdit 小部件调用基类实现:

def keyPressEvent(self, event):
if event.key() == QtCore.Qt.Key_Return or event.key() == QtCore.Qt.Key_Enter:
print("Enter pressed")
self.plainTextEdit.appendPlainText("New Line: ")
else:
QtWidgets.QPlainTextEdit.keyPressEvent(self.plainTextEdit, event)

请注意,我没有调用self.plainTextEdit.keyPressEvent(event),因为它会导致递归。

无论如何,这个想法并不好,因为通过这种方式,您也会覆盖QMainWindow keyPressEvent(如果您需要它,这可能是一个问题,但这不是重点(。

有两种可能(和更"优雅"(的解决方案:

1. 子类QPlainText在设计器中编辑和推广

此方法允许您在设计器中创建 UI,并为可以使用自己的代码扩展的自定义小部件设置基本参数;有关该过程的说明,请参阅此答案。

有了这个,您可以执行以下操作:

class MyTextEdit(QtWidget.QPlainTextEdit):
def keyPressEvent(self, event):
if event.key() == QtCore.Qt.Key_Return or event.key() == QtCore.Qt.Key_Enter:
print("Enter pressed")
self.appendPlainText("New Line: ")
else:
super().keyPressEvent(event)

额外的优点是,通过这种方式,代码也更干净,更容易实现。

2. 在小部件上安装事件过滤器

事件过滤器能够"拦截"小部件接收的任何事件,并可能对其做出反应。在您的情况下,它可能是这样的:

class MainWindow(QtWidgets.QMainWindow, Ui_MainWindow):
def __init__(self, *args, **kwargs):
QtWidgets.QMainWindow.__init__(self, *args, **kwargs)
self.setupUi(self)
self.plainTextEdit.installEventFilter(self)
def eventFilter(self, source, event):
if (event.type() == QtCore.QEvent.KeyPress and 
event.key() in (QtCore.Qt.Key_Enter, QtCore.Qt.Key_Return)):
# ensure that the widget receives the key event
super().eventFilter(source, event)
print("Enter pressed")
self.plainTextEdit.appendPlainText("New Line: ")
# the event is accepted and not sent to its ancestors
return True
return super().eventFilter(source, event)

最新更新