这是 Qt Quick ComboBox 中的错误吗?
Is this a bug in Qt Quick ComboBox?
我使用 QAbstractListModel
在 Qt Quick 中为 ComboBox
创建自定义模型。
头文件中的代码:
#ifndef JREFINDER_H
#define JREFINDER_H
#include <QObject>
#include <QMap>
#include <QString>
#include <QAbstractListModel>
enum BitSize
{
BitX86,
BitX64
};
class JreInformation : public QObject
{
Q_OBJECT
public:
..........
};
class JreFinder : public QAbstractListModel
{
Q_OBJECT
public:
enum JreFinderRoles
{
JavaPath = Qt::UserRole + 1,
JavaVerson,
JavaType,
Display
};
explicit JreFinder(QObject *parent = 0);
virtual QHash<int, QByteArray> roleNames() const;
virtual QVariant data(const QModelIndex &index, int role) const;
virtual int rowCount(const QModelIndex &parent = QModelIndex()) const;
Q_INVOKABLE void refresh();
Q_INVOKABLE void setIndex(int index);
Q_INVOKABLE QObject* getJavaInfo();
signals:
public slots:
private:
int m_index;
QList<JreInformation*> m_foundJre;
QHash<int, QByteArray> m_roleNames;
};
#endif // JREFINDER_H
源文件中的代码:
JreFinder::JreFinder(QObject *parent) : QAbstractListModel(parent)
,m_index(0)
{
m_roleNames.insert(JavaPath, "javaPath");
m_roleNames.insert(JavaVerson, "javaVersion");
m_roleNames.insert(JavaType, "javaType");
m_roleNames.insert(Display, "display");
QHash<int, QByteArray> defaultRoleNames = QAbstractListModel::roleNames();
QHashIterator<int, QByteArray> i(defaultRoleNames);
while (i.hasNext())
{
i.next();
m_roleNames.insert(i.key(), i.value());
}
}
QHash<int, QByteArray> JreFinder::roleNames() const
{
return m_roleNames;
}
QVariant JreFinder::data(const QModelIndex &index, int role) const
{
qDebug()<<"row"<<index.row()<<" role"<<role<<" size"<<m_foundJre.size();
if(index.row() >= m_foundJre.size() || index.row() < 0)
return QVariant();
if(role == JavaPath)
return m_foundJre[index.row()]->path;
else if(role == JavaVerson)
return m_foundJre[index.row()]->version;
else if(role == JavaType)
return m_foundJre[index.row()]->type;
else if(role == Display || role == Qt::DisplayRole)
{
QString d = m_foundJre[index.row()]->display();
qDebug()<<"display:"<<d;
return d;
}
return QVariant();
}
int JreFinder::rowCount(const QModelIndex &) const
{
qDebug()<<m_foundJre.size();
return m_foundJre.size();
}
void JreFinder::refresh()
{
beginResetModel();
foreach (JreInformation* info, m_foundJre)
{
delete info;
}
m_foundJre.clear();
ReadJreHome(m_foundJre);
endResetModel();
}
void JreFinder::setIndex(int index)
{
m_index = index;
}
QObject* JreFinder::getJavaInfo()
{
if(m_index >= m_foundJre.size() || m_index < 0)
return NULL;
return m_foundJre[m_index];
}
然后我使用以下方法将其公开到 QML 中:
JreFinder jreFinder;
jreFinder.refresh();
engine.rootContext()->setContextProperty("jreFinder", &jreFinder);
并使用此模型创建组合框:
ComboBox
{
textRole: "display"
model:jreFinder
}
此代码运行完美,但请查看源文件 data()
函数中的这一行:
else if(role == Display || role == Qt::DisplayRole)
我觉得我设置了ComboBox
的textRole
后,显示角色应该命名为"display",根据m_roleNames.insert(Display, "display");
。但它并不总是正确的。
这是上述代码的 qDebug()<<
输出:
1
1
row 0 role 260 size 1
display: "1.8.0_51 64bit"
和
1
1
row 0 role 0 size 1
display: "1.8.0_51 64bit"
两个输出是随机出现的。您会发现传递给 data()
函数的 role
参数有时会是 Zero。 零角色在Qt中表示Qt::DisplayRole
。
所以这是我的问题:如果 ComboBox
的 textRole
意味着 data role
设置为等于 textRole 的值。为什么 data role
有时会变成 Qt::DisplayRole
?是ComboBox
的bug吗?
角色名称到角色 ID 的映射不明确不是明智的做法。映射必须是 1:1。 Qt自带的DisplayRole
已经命名为display
。为您的名称命名仍然是有效的 js 标识符,或按原样使用 Qt。
我使用 QAbstractListModel
在 Qt Quick 中为 ComboBox
创建自定义模型。
头文件中的代码:
#ifndef JREFINDER_H
#define JREFINDER_H
#include <QObject>
#include <QMap>
#include <QString>
#include <QAbstractListModel>
enum BitSize
{
BitX86,
BitX64
};
class JreInformation : public QObject
{
Q_OBJECT
public:
..........
};
class JreFinder : public QAbstractListModel
{
Q_OBJECT
public:
enum JreFinderRoles
{
JavaPath = Qt::UserRole + 1,
JavaVerson,
JavaType,
Display
};
explicit JreFinder(QObject *parent = 0);
virtual QHash<int, QByteArray> roleNames() const;
virtual QVariant data(const QModelIndex &index, int role) const;
virtual int rowCount(const QModelIndex &parent = QModelIndex()) const;
Q_INVOKABLE void refresh();
Q_INVOKABLE void setIndex(int index);
Q_INVOKABLE QObject* getJavaInfo();
signals:
public slots:
private:
int m_index;
QList<JreInformation*> m_foundJre;
QHash<int, QByteArray> m_roleNames;
};
#endif // JREFINDER_H
源文件中的代码:
JreFinder::JreFinder(QObject *parent) : QAbstractListModel(parent)
,m_index(0)
{
m_roleNames.insert(JavaPath, "javaPath");
m_roleNames.insert(JavaVerson, "javaVersion");
m_roleNames.insert(JavaType, "javaType");
m_roleNames.insert(Display, "display");
QHash<int, QByteArray> defaultRoleNames = QAbstractListModel::roleNames();
QHashIterator<int, QByteArray> i(defaultRoleNames);
while (i.hasNext())
{
i.next();
m_roleNames.insert(i.key(), i.value());
}
}
QHash<int, QByteArray> JreFinder::roleNames() const
{
return m_roleNames;
}
QVariant JreFinder::data(const QModelIndex &index, int role) const
{
qDebug()<<"row"<<index.row()<<" role"<<role<<" size"<<m_foundJre.size();
if(index.row() >= m_foundJre.size() || index.row() < 0)
return QVariant();
if(role == JavaPath)
return m_foundJre[index.row()]->path;
else if(role == JavaVerson)
return m_foundJre[index.row()]->version;
else if(role == JavaType)
return m_foundJre[index.row()]->type;
else if(role == Display || role == Qt::DisplayRole)
{
QString d = m_foundJre[index.row()]->display();
qDebug()<<"display:"<<d;
return d;
}
return QVariant();
}
int JreFinder::rowCount(const QModelIndex &) const
{
qDebug()<<m_foundJre.size();
return m_foundJre.size();
}
void JreFinder::refresh()
{
beginResetModel();
foreach (JreInformation* info, m_foundJre)
{
delete info;
}
m_foundJre.clear();
ReadJreHome(m_foundJre);
endResetModel();
}
void JreFinder::setIndex(int index)
{
m_index = index;
}
QObject* JreFinder::getJavaInfo()
{
if(m_index >= m_foundJre.size() || m_index < 0)
return NULL;
return m_foundJre[m_index];
}
然后我使用以下方法将其公开到 QML 中:
JreFinder jreFinder;
jreFinder.refresh();
engine.rootContext()->setContextProperty("jreFinder", &jreFinder);
并使用此模型创建组合框:
ComboBox
{
textRole: "display"
model:jreFinder
}
此代码运行完美,但请查看源文件 data()
函数中的这一行:
else if(role == Display || role == Qt::DisplayRole)
我觉得我设置了ComboBox
的textRole
后,显示角色应该命名为"display",根据m_roleNames.insert(Display, "display");
。但它并不总是正确的。
这是上述代码的 qDebug()<<
输出:
1
1
row 0 role 260 size 1
display: "1.8.0_51 64bit"
和
1
1
row 0 role 0 size 1
display: "1.8.0_51 64bit"
两个输出是随机出现的。您会发现传递给 data()
函数的 role
参数有时会是 Zero。 零角色在Qt中表示Qt::DisplayRole
。
所以这是我的问题:如果 ComboBox
的 textRole
意味着 data role
设置为等于 textRole 的值。为什么 data role
有时会变成 Qt::DisplayRole
?是ComboBox
的bug吗?
角色名称到角色 ID 的映射不明确不是明智的做法。映射必须是 1:1。 Qt自带的DisplayRole
已经命名为display
。为您的名称命名仍然是有效的 js 标识符,或按原样使用 Qt。