c++ - 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);
如果它最后添加的年龄,它的年龄显示...任何线索?
解决方案
假设您确实需要一个自定义模型并想要扩展现有模型。由于您的数据采用表格形式,因此我建议将其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
实现中,根据索引row和column以及传递的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
(为此目的)开始确保您的角色不会与内置角色发生冲突。
请注意,上面的代码旨在提出一个可能的解决方案,而不是解决方案本身(并且缺少许多重要的功能,如边界检查和内存管理)。
推荐阅读
- swagger - Spring InterceptorRegistry - 允许 Swagger / Actuator
- sql-server - 使用 INSERT 查询的结果更新
- javascript - 空手道 - 可以使用 Karate.set 插入具有多个值的新键,并在 POST 调用中迭代这些值吗?
- windows - 在 VS Code 启动时.. 自动连接到 WSL-Ubuntu,打开项目文件夹和个人文件
- asp.net - “System.Collections.Generic.List`1[API.Entities.CompanySetting]”类型的表达式不能用于“System.Linq.IQueryable”类型的参数
- blazor - Blazor 组件动态授权
- python - 为什么容器内容会因我是 `docker run ...` 还是 `docker-compose run ...` 而有所不同?
- java - Java TicTacToe:计算机或玩家的第一步
- python - 是否有包包含功能可以获得文件大小的稀疏性?
- vba - 表格列字VBA中的整数