首页 > 解决方案 > QStandardItemModel & 绑定到自定义对象

问题描述

我尝试使用 Qt 示例的城市标准项并将其调整为我的示例。我有一个奇怪的结果:

这是我的用户类:

class User{
public:
User();

QString getFirstname() const;
void setFirstname(const QString &value);

QString getLastname() const;
void setLastname(const QString &value);

int getAge() const;
void setAge(int value);

private:
QString firstname;
QString lastname;
int age;
};

我已经声明了一个usermodel.h:

class UserModel: public QStandardItemModel
{
    Q_OBJECT
public:
    UserModel(QList<User> users, QObject *parent = Q_NULLPTR);

    QHash<int, QByteArray> roleNames() const;
};

下面是构造函数和 roleNames 函数的实现:

enum ItemRoles {
    FirstnameRole,
    LastnameRole,
    AgeRole,
};


UserModel::UserModel(QList<User> users, QObject *parent) : QStandardItemModel(parent)
{    

    //this->setItemRoleNames(roleNames());
    this->setColumnCount(3);
    for (auto user: users) {
        QStandardItem *item = new QStandardItem();
        item->setData(user.getFirstname(), FirstnameRole);
        item->setData(user.getLastname(), LastnameRole);
        item->setData(user.getAge(), AgeRole);
        appendRow(item);
    }

    setSortRole(FirstnameRole);
}


QHash<int, QByteArray> UserModel::roleNames() const
{
    QHash<int, QByteArray> mapping = QStandardItemModel::roleNames();

    mapping[FirstnameRole] = "firstname";
    mapping[LastnameRole] = "lastname";
    mapping[AgeRole] = "age";

    return mapping;
}

我的表格视图仅显示使用该功能添加的最后一个角色: item->setData(user.getFirstname(), FirstnameRole);

如果它最后添加的年龄,它的年龄显示...任何线索?

标签: c++qtqstandarditemmodelqstandarditem

解决方案


假设您确实需要一个自定义模型并想要扩展现有模型。由于您的数据采用表格形式,因此我建议将其QAbstractTableModel用作基类。

所以让我们来上课:

class UserModel: public QAbstractTableModel
{
    Q_OBJECT
    QList<User> _users;
public:
    UserModel(QList<User> users, QObject *parent = Q_NULLPTR) : QAbstractTableModel(parent), _users(users){}

如您所见,该类本身存储了在构建时给出的用户列表。构造函数本身只对列表进行复制初始化。

然后你需要提供这些实现,至少:

int rowCount(const QModelIndex &) const override
{
    return _users.size();
}
int columnCount(const QModelIndex &) const override
{
    return 3;
}
QVariant data(const QModelIndex &index, int role) const override
{
    if(role == Qt::DisplayRole)
    {
        User user = _users.at(index.row());
        QVariant data[] = { user.getFirstname(), user.getLastname() , user.getAge() };
        return data[index.column()];
    }
    return {};
}

WhilecolumnCount是常量并且总是返回 3,rowCount将返回用户列表中的项目数。在data实现中,根据索引rowcolumn以及传递的role对传递的索引进行检查并返回一个值。重要的是要理解,当视图调用data传递一个role等于时Qt::DisplayRole,该函数应该返回将在单元格中显示的数据(index.row(), index.column()),在我们的例子中:三个User数据成员之一。

重新实现该sort功能也非常有用,即

void sort(int column, Qt::SortOrder order) override
{
    auto fnSort = [](const User & u1, const User & u2){ return u1.getFirstname() < u2.getFirstname(); };
    auto lnSort = [](const User & u1, const User & u2){ return u1.getLastname() < u2.getLastname(); };
    auto agSort = [](const User & u1, const User & u2){ return u1.getAge() < u2.getAge(); };

    std::function<bool (const User &, const User &)>  sortFn[] = {fnSort, lnSort, agSort};
    std::sort(_users.begin(), _users.end(), sortFn[column]);

    if(order == Qt::DescendingOrder)
    {
        std::reverse(_users.begin(), _users.end());
    }
}

这样您就可以让用户按列排序,如预期的那样:

myTableView->setModel(new UserModel(list));
myTableView->model()->sort(2, Qt::DescendingOrder); //sort by first age, in descending order

如果出于某种原因,您想使用自定义角色,请让您的枚举如下所示:

enum ItemRoles {
    FirstnameRole = Qt::UserRole,
    LastnameRole,
    AgeRole,
};

Qt::UserRole(为此目的)开始确保您的角色不会与内置角色发生冲突。

请注意,上面的代码旨在提出一个可能的解决方案,而不是解决方案本身(并且缺少许多重要的功能,如边界检查和内存管理)。


推荐阅读