我有一个QML ListView,它使用QAbstractListModel子类作为模型。
ListView {
id: myListView
x: 208
y: 19
width: 110
height: 160
delegate: myListDelegate {}
model: MyListModel
opacity: 0
}
该模型是MyListItem
s。
class MyListModel : public QAbstractListModel
{
Q_OBJECT
public:
enum MyRoles {
HeadingRole = Qt::UserRole + 1,
DescriptionRole,
QuantityRole
};
explicit MyListModel(QObject *parent = 0);
void addMyListItem(const MyListItem &item);
int rowCount(const QModelIndex & parent = QModelIndex()) const;
QVariant data(const QModelIndex & index, int role = Qt::DisplayRole) const;
void dropList();
private:
QList<MyListItem> m_list;
};
在代表中我有一个鼠标区。
如何拦截鼠标区域的点击并从我的QList模型中选择那个 MyListItem
并将其发送到应用程序的C ++部分内的某个位置?
答案 0 :(得分:9)
评论提到从MyListItem
返回指向data()
的指针到QML,并在QML中访问和修改它。这需要您MyListItem
继承QObject
并为要在QML中访问的每个成员添加一个Q_PROPERTY
。它还需要密切关注对象所有权(QQmlEngine::ObjectOwnership
)。
还有另一种方法:实施QAbstractListModel::setData()
和QAbstractListModel::roleNames()
,模型内容可以从QML更改,例如model.roleName = foo
。
下面的最小工作示例,每次点击代表时数量加倍:
<强> C ++:强>
struct MyListItem
{
QString heading;
QString description;
int quantity;
};
class MyListModel : public QAbstractListModel
{
Q_OBJECT
Q_ENUMS(MyRoles)
public:
enum MyRoles {
HeadingRole = Qt::UserRole + 1,
DescriptionRole,
QuantityRole
};
using QAbstractListModel::QAbstractListModel;
QHash<int,QByteArray> roleNames() const override {
return { { HeadingRole, "heading" },
{ DescriptionRole, "description" },
{ QuantityRole, "quantity" },
};
}
int rowCount(const QModelIndex & parent = QModelIndex()) const override {
if (parent.isValid())
return 0;
return m_list.size();
}
bool setData(const QModelIndex &index, const QVariant &value, int role) override
{
if (!hasIndex(index.row(), index.column(), index.parent()) || !value.isValid())
return false;
MyListItem &item = m_list[index.row()];
if (role == DescriptionRole) item.description = value.toString();
else if (role == HeadingRole) item.heading = value.toString();
else if (role == QuantityRole) item.quantity = value.toInt();
else return false;
emit dataChanged(index, index, { role } );
return true ;
}
QVariant data(const QModelIndex & index, int role = Qt::DisplayRole) const override {
if (!hasIndex(index.row(), index.column(), index.parent()))
return {};
const MyListItem &item = m_list.at(index.row());
if (role == DescriptionRole) return item.description;
if (role == HeadingRole) return item.heading;
if (role == QuantityRole) return item.quantity;
return {};
}
private:
QVector<MyListItem> m_list = {
{ "heading 1", "description 1", 1 },
{ "heading 2", "description 2", 42 },
{ "heading 3", "description 3", 4711 }
};
};
<强> QML:强>
ListView {
id: listView
anchors.fill: parent
model: MyListModel {}
delegate: Item {
implicitHeight: text.height
width: listView.width
Text {
id: text
text: model.heading + " " + model.description + " " + model.quantity
}
MouseArea {
anchors.fill: text
onClicked: {
model.quantity *= 2;
}
}
}
}
答案 1 :(得分:2)
您还可以在委托中使用index
属性来操作数据。您只需要使用模型上的索引方法将QML索引转换为QModelIndex
。这是一个简单的例子,我们每次点击一个列表项时都会将显示值更改为字符串“3”。
ListView {
id: listView
anchors.fill: parent
model: my_model
delegate: Rectangle {
height: 50
width: listView.width
MouseArea {
anchors.fill: parent
onClicked: {
// Column is always zero as it's a list
var column_number = 0;
// get `QModelIndex`
var q_model_index = my_model.index(index, column_number);
// see for list of roles:
// http://doc.qt.io/qt-5/qabstractitemmodel.html#roleNames
var role = 1
var data_changed = my_model.setData(q_model_index, "3", role);
console.log("data change successful?", data_changed);
}
}
}
}
除了委托中的index
属性外,委托中还提供了所有默认角色名称。例如,我之前使用decoration
角色来设置color
委托的Rectangle
属性。有关详情,请参阅this list。
ListView {
delegate: Rectangle {
// list items have access to all default `roleNames`
// in addition to the `index` property.
// For example, using the decoration role, demo'd below
color: decoration
}
}
另请参阅this link Mitch Curtis建议使用qmlRegisterUncreatableType注册用户枚举。
答案 2 :(得分:1)
因为setRoleNames()在QAbstractListModel中是绝对的。您可以覆盖roleNames()并显式添加您的角色。继承QAbstractListModel的简单实现如下。
class BaseListModel : public QAbstractListModel
{
Q_OBJECT
Q_ENUMS(Roles)
public:
enum Roles {
Name = Qt::UserRole + 1
};
virtual QHash<int, QByteArray> roleNames() const;
virtual int rowCount(const QModelIndex &parent) const;
virtual QVariant data(const QModelIndex &index, int role) const override;
virtual bool setData(const QModelIndex &index, const QVariant &value, int role) override;
private:
QStringList _list;
};
// class
BaseListModel::BaseListModel(QObject *parent) :
QAbstractListModel(parent)
{
QHash<int, QByteArray> h = RecipeListModel::roleNames();
}
QHash<int, QByteArray> BaseListModel::roleNames() const {
return {
{ Name, "name" },
};
}
int BaseListModel::rowCount(const QModelIndex &parent) const {
if (parent.isValid())
return 0;
return _list.size();
}
QVariant BaseListModel::data(const QModelIndex &index, int role) const {
if (!hasIndex(index.row(), index.column(), index.parent()))
return {};
return _list.at(index.row())->data(role);
}
bool RecipeListModel::setData(const QModelIndex &index, const QVariant &value, int role) {
if (!hasIndex(index.row(), index.column(), index.parent()) || !value.isValid())
return false;
bool ret = _list.at(index.row())->setData(role, value);
if (ret) {
emit dataChanged(index, index, { role });
}
return ret;
}
QVariant BaseListModel::data(int role) const {
switch(role) {
case Name:
return name();
default:
return QVariant();
}
}
bool BaseListModel::setData(int role, const QVariant &value)
switch(role) {
case Name:
setName(value.toString());
return true;
default:
return false;
}
}