在我的应用程序中,我想使用QGraphicsItemGroup将项目分组为一个项目。我稍微玩了一下,并不确定使用它,因为当我想捕获事件时,事件会合并在一起,但是我想与特定的孩子一起处理特定的事件。我该如何实现?
您需要呼叫QGraphicsItemGroup::setHandlesChildEvents(false)
。这将停止QGraphicsItemGroup
尝试处理该事件,并让子级QGraphicsItem
代替处理它们。
我认为这是QGraphicsItemGroup
的重点。从文档中判断,这旨在简化一次移动和变换多个项目的过程,例如设想以下情况:用户在应用程序中的多个项目周围绘制了一个选择矩形,因为他想移动所有项目。也许您还想要创建项目的层次结构,例如有一个父项和几个子项。这样,您将获得每个项目的个别事件。这可以通过调用QGraphicsItem::setParentItem()
;
该问题未指定涉及哪个版本的Qt,并且对于Qt4有正确的答案。这是Qt5的答案(它适用于PyQt5,我想它也可以在C ++中使用)。解决方案是重新实现sceneEvent
,重新实现一个专门的事件捕获程序,例如contextMenuEvent
是不够的。
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
from PyQt5 import QtCore
from PyQt5 import QtGui
from PyQt5 import QtWidgets
class GraphicsGroup(QtWidgets.QGraphicsItemGroup):
def __init__(self, parent: QtWidgets.QGraphicsItem = None):
super().__init__(parent)
self.setFiltersChildEvents(False) # False is already the default.
def sceneEvent(self, event: QtCore.QEvent):
print(f'sceneEvent in GraphicsGroup {id(self)}')
event.ignore()
return True
def contextMenuEvent(self, event: QtWidgets.QGraphicsSceneContextMenuEvent):
print(f'contextMenuEvent in GraphicsGroup {id(self)}')
event.ignore()
class GraphicsItem(QtWidgets.QGraphicsItem):
def __init__(self,
rect: QtCore.QRectF,
name: str,
parent: QtWidgets.QGraphicsItem = None):
super().__init__(parent)
self._name = name
self._rect = rect
def boundingRect(self):
return self._rect
def paint(self,
painter: QtGui.QPainter,
option: QtWidgets.QStyleOptionGraphicsItem,
widget: QtWidgets.QWidget = None):
painter.setPen(QtGui.QPen(QtCore.Qt.NoPen))
painter.setBrush(QtGui.QBrush(QtCore.Qt.red))
painter.drawRect(self._rect)
def sceneEvent(self, event: QtCore.QEvent):
print(f'sceneEvent in "{self._name}"')
if (event.type() == QtCore.QEvent.GraphicsSceneContextMenu):
self.contextMenuEvent(event)
event.accept()
return True
def contextMenuEvent(self, event: QtWidgets.QGraphicsSceneContextMenuEvent):
print(f'contextMenuEvent in "{self._name}"')
class MainWindow(QtWidgets.QMainWindow):
def __init__(self):
super().__init__()
self._scene = QtWidgets.QGraphicsScene()
layout = QtWidgets.QHBoxLayout()
self._view = QtWidgets.QGraphicsView(self._scene)
layout.addWidget(self._view)
self._widget = QtWidgets.QWidget()
self._widget.setLayout(layout)
group = GraphicsGroup()
self._scene.addItem(group)
scene_item = GraphicsItem(QtCore.QRectF(0, 0, 100, 100), 'in scene')
self._scene.addItem(scene_item)
group_item = GraphicsItem(QtCore.QRectF(150, 0, 100, 100), 'in group')
group.addToGroup(group_item)
group_item = GraphicsItem(QtCore.QRectF(300, 0, 100, 100), '2nd in group')
group.addToGroup(group_item)
self.setCentralWidget(self._widget)
self.setWindowTitle('contextMenuEvent with QGraphicsItemGroup')
if __name__ == '__main__':
import sys
app = QtWidgets.QApplication(sys.argv)
mainWindow = MainWindow()
mainWindow.setGeometry(100, 100, 800, 500)
mainWindow.show()
sys.exit(app.exec_())