无法通过我的QAbstractItemModel派生类

时间:2016-05-16 03:37:57

标签: c++ qt model qtreeview qabstractitemmodel

我正在使用QAbstractItemModel实现自定义模型,我希望在QColumnView中显示该模型。模型中的数据是目录和文件的简单层次结构,其中每个目录都可以包含文件和其他目录。

我的QAbstractItemModel派生类名为MyModel。它依赖于另外两个QObject派生类,Directory和Item。目录有几种方法可以返回它包含的项目数(包括子目录),在特定索引处获取项目,等等。到目前为止,为了此代码的目的,File只有一个名称,并由Directory实例包含。 MyModel还包含一个名为invisibleTopLevelDirectory()的方法,该方法返回作为此层次结构根目录的Directory。

问题是我的QColumnView没有正确显示数据。每列最多显示一行,无论它应该有多少行。我不知道为什么!尽可能接近我已经正确地实现了我的QAbstractItemModel派生类。更令人困惑的是,如果我使用QTreeView而不是QColumnView,树视图显示所有数据。

以下是相关代码:

// Returns the instance of Directory at the given index, or NULL if the data
// is not a Directory instance.
Directory *directoryAtModelIndex(const QModelIndex &index)
{
    return qobject_cast<Directory *>((QObject *)index.internalPointer());
}

// Returns the instance of File at the given index, or NULL if the data
// is not a File instance.
File *fileAtModelIndex(const QModelIndex &index)
{
    return qobject_cast<File *>((QObject *)index.internalPointer());
}

QModelIndex MyModel::index(int row, int column, const QModelIndex &parent) const
{
    if (!hasIndex(row, column, parent)) {
        return QModelIndex();
    }

    Directory *parentDirectory;

    if (!parent.isValid()) {
        parentDirectory = invisibleTopLevelDirectory();
    } else {
        parentDirectory = (Directory *)(parent.internalPointer());
    }

    if (!parentDirectory) {
        return QModelIndex();
    }

    QObject *item = parentDirectory->itemAtIndex(row);

    if (item) {
        return createIndex(row, column, item);
    } else {
        return QModelIndex();
    }
}

QModelIndex MyModel::parent(const QModelIndex &index) const
{
    if (!index.isValid()) {
        return QModelIndex();
    }

    QObject *item = (QObject *)index.internalPointer();
    Directory *parentItem = qobject_cast<Directory *>(item->parent());

    if (!parentItem || parentItem == invisibleTopLevelDirectory()) {
        return QModelIndex();
    }

    int listIndex = parentItem->indexOfItem(item);

    if (listIndex < 0) {
        return QModelIndex();
    } else {
        return createIndex(listIndex, index.column(), parentItem);
    }
}

int MyModel::rowCount(const QModelIndex &parent) const
{
    if (parent.column() > 0) {
        return 0;
    }

    if (!parent.isValid()) {
        return invisibleTopLevelDirectory() ? invisibleTopLevelDirectory()->itemCount() : 0;
    }

    Directory *parentItem = directoryAtModelIndex(parent);

    if (parentItem) {
        return parentItem->itemCount();
    } else {
        return 0;
    }
}

int MyModel::columnCount(const QModelIndex &parent) const
{
    Q_UNUSED(parent);
    return 1;
}

QVariant MyModel::data(const QModelIndex &index, int role) const
{
    if (!index.isValid()) {
        return QVariant();
    }

    if (role == Qt::DisplayRole) {
        if (Directory *item = directoryAtModelIndex(index)) {
            return item->name;
        } else if (File *item = fileAtModelIndex(index)) {
            return item->name;
        }   
    }

    return QVariant();
}

我已经多次倾注了这段代码,我简直无法弄清楚为什么我的QColumnView不会为任何给定的列显示多行数据。我已经使用print语句确认为第一个传递的行调用了index()和data(),为所有父项调用了rowCount()并返回了正确的行数,而parent()返回正确的行,列和父QModelIndex。如果我在QTreeView中使用相同的模型,它可以正常工作。

任何人都可以弄清楚我做错了什么吗?

1 个答案:

答案 0 :(得分:1)

  1. 方法父级中的错误:您使用索引行而不是父级行。应该是:

    Directory* parent_of_parent = qobject_cast<Directory*>(parentItem->parent());
    if (!parent_of_parent) {
        return QModelIndex();
    }
    
    int listIndex = parent_of_parent->indexOfItem(parentItem);
    
  2. 您可以使用QFileSystemModel。