带有QGraphicsItemGroup的事件

问题描述 投票:12回答:3

在我的应用程序中,我想使用QGraphicsItemGroup将项目分组为一个项目。我稍微玩了一下,并不确定使用它,因为当我想捕获事件时,事件会合并在一起,但是我想与特定的孩子一起处理特定的事件。我该如何实现?

c++ qt events qt4 qgraphicsview
3个回答
14
投票

您需要呼叫QGraphicsItemGroup::setHandlesChildEvents(false)。这将停止QGraphicsItemGroup尝试处理该事件,并让子级QGraphicsItem代替处理它们。


4
投票

我认为这是QGraphicsItemGroup的重点。从文档中判断,这旨在简化一次移动和变换多个项目的过程,例如设想以下情况:用户在应用程序中的多个项目周围绘制了一个选择矩形,因为他想移动所有项目。也许您还想要创建项目的层次结构,例如有一个父项和几个子项。这样,您将获得每个项目的个别事件。这可以通过调用QGraphicsItem::setParentItem();

完成

0
投票

该问题未指定涉及哪个版本的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_())
© www.soinside.com 2019 - 2024. All rights reserved.