我注意到 QApplication 处理关键事件时出现了一些异常行为,这危及了我希望制作的小游戏。
按住某个键会导致
keyPressEvent
和 keyReleaseEvent
方法被重复(并且非常频繁地)调用,而不是触发 keyPressEvent
一次并等待按键被释放来触发另一个(这是期望的)以及无可否认的预期行为)。
这会导致巨大的性能问题,按住多个键会导致程序完全忽略某些键,大概是因为事件队列受到影响。
本程序演示了重复调用:
from PyQt4 import QtGui
import sys
class Window(QtGui.QWidget):
def __init__(self):
super(Window, self).__init__()
def keyPressEvent(self, event):
print 'PRESSED'
event.accept()
def keyReleaseEvent(self, event):
print 'RELEASED'
event.accept()
app = QtGui.QApplication(sys.argv)
Window().show()
sys.exit(app.exec_())
这个程序(我用来测试 Qt 游戏潜力的一个愚蠢的图形化程序)演示了在按住现有键时忽略新按下的键。
from PyQt4 import QtGui
import sys
class Window(QtGui.QWidget):
def __init__(self):
super(Window, self).__init__()
self.resize(100,300)
self.lower, self.upper = 10, -10
self.keys = [81, 65, 90, 87, 83, 88, 69, 68, 67, 82, 70, 86, 84, 71,
66, 89, 72, 78, 85, 74, 77, 73, 75, 44, 79, 76, 46, 80, 59, 47]
self.dots = [self.lower] * len(self.keys)
def keyPressEvent(self, event):
pressed = event.key()
if (pressed in self.keys):
index = self.keys.index(pressed)
self.dots[index] = self.height()+self.upper
self.repaint()
event.accept()
def keyReleaseEvent(self, event):
pressed = event.key()
if (pressed in self.keys):
index = self.keys.index(pressed)
self.dots[index] = self.lower
self.repaint()
event.accept()
def paintEvent(self, event):
step = self.width() / (len(self.dots) + 1)
painter = QtGui.QPainter()
painter.begin(self)
x, y = 0, 0
for w in self.dots:
i, j = x + step, w
painter.drawLine(x, self.height() - y, i, self.height() - j)
x, y = i, j
painter.end()
app = QtGui.QApplication(sys.argv)
Window().show()
sys.exit(app.exec_())
通过运行上述程序可以观察到,按住 3 或 4 个以上的键将导致不会建立新的尖峰,直到释放当前按住的键为止。
如何防止这种行为,使得
keyPressEvent
只针对尚未物理释放的按键触发一次?
使用 event.isAutoRepeat() 可能会有所帮助
例如
def keyPressEvent(self, event):
if event.isAutoRepeat():
return
pressed = event.key()
if (pressed in self.keys):
index = self.keys.index(pressed)
self.dots[index] = self.height()+self.upper
self.repaint()
event.accept()