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 错误跟踪器上报告它们是有意义的。
这个问题可能与 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 错误跟踪器上报告它们是有意义的。