问题描述
我想根据是否按下修饰键 (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_()