这是 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)

我觉得我设置了ComboBoxtextRole后,显示角色应该命名为"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

所以这是我的问题:如果 ComboBoxtextRole 意味着 data role 设置为等于 textRole 的值。为什么 data role 有时会变成 Qt::DisplayRole?是ComboBox的bug吗?

角色名称到角色 ID 的映射不明确不是明智的做法。映射必须是 1:1。 Qt自带的DisplayRole已经命名为display。为您的名称命名仍然是有效的 js 标识符,或按原样使用 Qt。