QGraphicsItem 在添加到组时自定义选择 - 奇怪的不一致结果

QGraphicsItem customize selection when adding to a group - weird inconsistent results

这个问题可能与 QGraphicsItemGroup 相关,也可能不相关 - 虽然我以前从未见过这种行为....
简要说明:我正在取消选择一个项目,但除非我调用只读 scene().selectedItems() - 即使我不使用它,否则该操作不会发生。

详情:

我有一个自定义 QGraphicsScene class,必须对 selectedItems().

执行很多操作

如果项目在一个组中,则不应在任何操作中使用它们 - 仅应使用该组。

所以我创建了 addToGroup() 方法来执行我需要的操作:

void addToGroup(QList<QGraphicsItem *> children) {
    foreach(QGraphicsItem* child, children)
    {
        child->setSelected(false);
        QGraphicsItemGroup::addToGroup(child);
    }
}

很遗憾,项目拒绝取消选择!

然后,我在每一行之后添加调试消息 - 发现添加调试消息会改变结果!

void addToGroup(QList<QGraphicsItem *> children) {
    foreach(QGraphicsItem* child, children)
    {
        child->setSelected(false);
        scene()->selectedItems();        // this makes it work !
        QGraphicsItemGroup::addToGroup(child);
    }
}

调用 scene()->selectedItems(); - 这应该是只读的 - 使项目实际上被取消选择!

请允许我理解这一点!

完整示例代码:

#include <QApplication>
#include <QGraphicsView>
#include <QGraphicsRectItem>
#include <QDebug>

class Item {
public:
    Item(int id) {
        m_id = id;
    }
private:
    int m_id;
};

class RectItem: public QGraphicsRectItem, public Item
{
public:
    RectItem(int id) : Item(id) {
        setFlags(QGraphicsItem::ItemIsSelectable | QGraphicsItem::ItemIsMovable | QGraphicsItem::ItemIsFocusable);
        setRect(QRectF(0,0,100,100));
        setPos(QPointF(10+110*id,10));
        setSelected(true);
    }
};

class GroupItem: public QGraphicsItemGroup, public Item
{
public:

    GroupItem(int id) : Item(id) {
        setFlags(QGraphicsItem::ItemIsMovable | QGraphicsItem::ItemIsFocusable | QGraphicsItem::ItemIsSelectable);
    }
    void addToGroup(QList<QGraphicsItem *> children) {
        foreach(QGraphicsItem* child, children)
        {
            child->setSelected(false);
            //scene()->selectedItems().size();
            QGraphicsItemGroup::addToGroup(child);
        }
    }
};

class MyScene: public QGraphicsScene
{
public:
    MyScene() {}
    void group() {
        GroupItem* g = new GroupItem(items().size());
        addItem(g);
        g->addToGroup(selectedItems());
        g->setSelected(true);
    }
};

int main(int argc, char *argv[])
{
    QApplication app(argc, argv);
    MyScene s;
    QGraphicsView view(&s);
    s.setSceneRect(0, 0, 230, 120);
    view.show();
    RectItem* r0 = new RectItem(0);
    s.addItem(r0);
    RectItem* r1 = new RectItem(1);
    s.addItem(r1);
    s.group();
    qDebug() << "Should only have 1 (group) selected out of 3\nTotal items:" << s.items().size() << "; Selected items:" << s.selectedItems().size();
    return app.exec();
}

关于 scene().selectedItems() 的良好观察。当 QGraphicsItemGroup 添加到 QGraphicsItemGroup.

时,实际上还有一个与 QGraphicsItem 项目选择状态不一致的地方

It appears that the internal private set selectedItems accumulates items when a QGraphicsItem is selected.但是,如果未选择此类项目,则不会从该集中删除它。

QGraphicsItem 仅在调用函数 QGraphicsScene::selectedItems() 并且该项目的成员函数 QGraphicsItem::isSelected() 时从该集合中删除 returns false.

在你的情况下,项目首先被添加到私人场景选择集中。然后更改这些项目的选定标志,并将组对象添加到处于选定状态的场景中。因此,集合中有三个对象。诀窍是现在 isSelected() 对于所有项目都是 true,因为 isSelected() 对于组中的项目总是 return 与组 isSelected() 独立于项目选择状态。因此,所有三个项目都出现在列表 selectedItems().

如果在 child->setSelected(false) 之后调用 scene().selectedItems(),则重新创建场景选择集并删除子项。这给出了 selectedItems().

中只有一项的最终结果

函数QGraphicsScene::selectedItems()非常繁重,因为它总是生成新的私有集。在 foreach 循环中调用它只是为了删除子项目是不好的。通过更改 group() 函数中的调用顺序也可以实现相同的目标:

void group() {
    GroupItem* g = new GroupItem(items().size());
    // at first add items to group
    g->addToGroup(selectedItems());
    // add group to scene
    addItem(g);
    g->setSelected(true);
}

这只给出了 selectedItems() 中的一项。发生这种情况是因为当将子项目添加到不在场景中的组时,这些项目将从场景中删除。因此,它们也从选定集中删除。然后将三个未选中的项目添加到场景中,但 g->setSelected(true) 仅将组添加到选中的集合中。

但是,如果在上面的示例中您在 setSelected(true) 之后调用 addItem(g),它将再次给出 selectedItems() 中的所有三个项目:

void group() {
    GroupItem* g = new GroupItem(items().size());
    g->addToGroup(selectedItems());
    g->setSelected(true);
    addItem(g);
}

发生这种情况是因为 addItem(g) 同时向场景添加了三个项目。所有三个项目都有 isSelected() return 值 true.


另一个不一致与内部项目选择状态有关。看起来如果将一个项目添加到组中,该项目的状态将冻结在添加到该组之前的状态。这意味着,如果您将所选项目添加到组 ,即使未选择组 ,它也始终显示为已选中(带有虚线边框)。无法通过为该项目调用 setSelected(false) 来更改此类视图,因为现在该函数仅更改组的选定状态。即使项目 isSelected() returns false.

也可以显示为选中状态

看起来这两种不一致都不是设计有意的,在 Qt 错误跟踪器上报告它们是有意义的。