检测何时按下Ctrl(或其他修改键)的正确方式



我想根据是否按下修改键(Ctrl(来决定一个操作。我找到的一个解决方法是安装一个事件过滤器,并使用QApplication.queryKeyboardModifiers()检测何时按下Ctrl,使用QApplication.keyboardModifiers()检测何时释放Ctrl

from PySide6.QtCore import Qt, Signal
from PySide6.QtWidgets import QApplication, QMainWindow
class MainWindow(QMainWindow):
ctrl_signal = Signal(bool)
def __init__(self):
QMainWindow.__init__(self)
self.installEventFilter(self)
self.ctrl_signal.connect(self.ctrl_slot)
def eventFilter(self, _object, e):
if QApplication.queryKeyboardModifiers() == Qt.CTRL: # This runs twice, and only on key press (not release)
print("Ctrl pressed")
self.ctrl_signal.emit(True)
elif QApplication.keyboardModifiers() == Qt.CTRL: # This runs once, but only on release
print("Ctrl released")
self.ctrl_signal.emit(False)
return False
def ctrl_slot(self, e):
print("e: ", e)  # Do something
app = QApplication([])
window = MainWindow()
window.show()
app.exec_()

然而,我担心这是对.queryKeyboardModifiers().keyboardModifiers()功能的意外使用,因此以后可能会导致更多麻烦。是否有适当的方法来检测何时单独按下/释放修改键(即没有按下任何其他键(?

虽然我使用的是PySide6,但如果有帮助的话,我会接受C++或PyQt中的答案。

您目前正在做的是检查每次事件发生时是否按下Ctrl键(例如单击、移动、调整大小等(,这似乎不是您的目标,只是检测何时发生更改,因此您必须改进Qt.KeyPress或Qt.KeyRelease事件的过滤器。另一方面,如果你想检测另一个子窗口小部件何时消耗事件,你的方法将不起作用,因为它不会传播到父窗口,相反,最好将过滤器应用到QWindow,因为键盘事件是在它有焦点时到达的,不依赖于子窗口的逻辑。

from PySide6.QtCore import Qt, Signal, QObject, QEvent
from PySide6.QtWidgets import QApplication, QMainWindow

class ControlHelper(QObject):
ctrl_signal = Signal(bool)
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:
if event.type() == QEvent.KeyPress:
if event.key() == Qt.Key_Control:
self.ctrl_signal.emit(True)
if event.type() == QEvent.KeyRelease:
if event.key() == Qt.Key_Control:
self.ctrl_signal.emit(False)
return super().eventFilter(obj, event)

class MainWindow(QMainWindow):
ctrl_signal = Signal(bool)
def ctrl_slot(self, e):
print("e: ", e)

app = QApplication([])
window = MainWindow()
window.show()
helper = ControlHelper(window.windowHandle())
helper.ctrl_signal.connect(window.ctrl_slot)
app.exec_()

最新更新