3

这是我得到的:

  • 一个QTreeView小部件(*);
  • 源模型MainModel继承自QStandardItemModel. 没有重新实现虚data() const方法;
  • 代理MainFilterProxyModel继承自QSortFilterProxyModel

那个树:

[PERIOD 1]
   [CHILD 1]
   [CHILD 2]
      [SUBCHILD 2.1]
      ...
   [CHILD N]
[PERIOD 2]
...
[PERIOD N]

所以主要问题是当我试图添加一个像 (**) 代码那样的CHILD-row时。过滤器代理模型,在文档被添加到源模型后,不知道新行,也没有在树上显示。

我确定代理在方法完成工作QStandardItemModel时没有收到信号appendRow,因此代理无法过滤新行,也没有使其可见。

有什么帮助吗?

谢谢。

PS:如果我关闭代理,一切都很好。但问题不在于代理。代理只是没有得到关于附加到主源模型的新行的信号......


(*)这里是QTreeView

MainView::MainView( QWidget* parent /* = 0 */ ) : QTreeView( parent )
{
     if( !model_ )
     {
          model_ = new MainModel( this );
     }

     if( !proxy_ )
     {
          proxy_ = new MainFilterProxyModel( this );
          proxy_->setDynamicSortFilter( true );
          proxy_->setSourceModel( model_ );

          setModel( proxy_ );
     }
}

(**)这是我的附加功能

void MainModel::addRow( const DocumentPtr& document, QStandardItem* parentItem )
{
     assert( document );

     QList< QStandardItem* > items;
     items << ( new QStandardItem );
     items << ( new QStandardItem );
     items << ( new QStandardItem );
     items << ( new QStandardItem );
     items << ( new QStandardItem );
     items << ( new QStandardItem );
     items << ( new QStandardItem );

     updateRow( document, items );

     if( !parentItem )
     {
          BOOST_FOREACH( const TimePeriod& period, TimePeriod::all() )
          {
               if( period.contains( QDateTime::fromTime_t( document->creationDate() ) ) )
               {
                    QStandardItem* periodItem = itemByPeriod( period );
                    Q_ASSERT( periodItem );

                    periodItem->appendRow( items );

                    break;
               }
          }
     }
     else
     {
          parentItem->appendRow( items );
     }
}
4

1 回答 1

0

建模的基类是QAbstractItemModel. 最好使用抽象类的方法来做你想做的事。QStandardItemModel是抽象方法的简单实现,QAbstractItemModel并且大多数QStandardItemModel新方法由重新实现的抽象函数使用。这是使用抽象类方法添加项目和子项目的代码:

QAbstractItemModel * pModel = new QStandardItemModel(parent);

int nRows = pModel->rowCount();
pModel->insertRow(nRows); // this will emit rowsAboutToBeInserted() and rowsInserted() signals
pModel->insertColumn(0); // this will emit columnsAboutToBeInserted() and columnsInserted() signals
const QModelIndex indexFirstItem = pModel->index(nRows, 0);
pModel->setData(indexFirstItem, "Item text"); // this will emit dataChanged() signal
int nChildRows = pModel->rowCount(indexFirstItem);
pModel->insertRow(nChildRows, indexFirstItem); // this will emit rowsInserted()
pModel->insertColumn(0, indexFirstItem); // we also need to do this for the item's children
const QModelIndex indexChild = pModel->index(nChildRows, 0, indexFirstItem);
pModel->setData(indexChild, "Child item text");

如果我们尝试使用 QStandardItemModel 方法做同样的事情,它看起来像:

QStandardItemModel *pModel = new QStandardItemModel(parent);
QStandardItem *pItem = new QStandardItem("Item text");
pItem->appendRow(new QStandardItem); // pItem is not yet added to pModel and rowsInserted won't be emitted
pModel->appendRow(pItem); // this will probably emit rowsInserted() signal but since we set tha text of the item when creating the pItem the dataChanged() signal won't be emitted.

因此,如果您确实pItem->appendRow()添加了子项目并且 pItem 尚未添加到模型中,您可能不会收到rowsInserted()信号,因此不会通知代理模型。根据我的经验,第一种方法效果更好,并且更健壮,尽管您需要编写几行额外的代码。直接使用QStandardItemModel方法通常会以丢失信号或其他问题而告终。您需要查看的只是文档QAbstractItemModelQModelIndex文档。

于 2013-04-30T17:37:06.367 回答