将QAbstractTableModel与Qml TableView一起使用仅显示第1列

Arm*_*ast 16 c++ qt model qml

我正在尝试使用QAbstractTableModel的一个(派生类)和一个Qml TableView;

但是,仅显示第1列.

原因是没有为非零列调用QVariant MyModel :: data(const QModelIndex&index,int role),但我不明白为什么.

但是,QTableView工作正常.

我做了一个单独的,简单的项目来重现我的问题:

MyModel.h:

#ifndef MYMODEL_H
#define MYMODEL_H

#include <QObject>
#include <QAbstractTableModel>
#include <QList>
#include <QString>
#include <QDebug>

struct SimpleData
{
    QString m_one;
    qint32 m_two;
    qreal m_three;
};

class MyModel : public QAbstractTableModel
{
    Q_OBJECT
public:
    explicit MyModel();//MyData *the_data);
    int rowCount(const QModelIndex & parent = QModelIndex()) const Q_DECL_OVERRIDE;
    int columnCount(const QModelIndex &parent = QModelIndex()) const Q_DECL_OVERRIDE;
    QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const Q_DECL_OVERRIDE;

    QHash<int, QByteArray> roleNames() const Q_DECL_OVERRIDE;
signals:

public slots:
    void theDataChanged();

private:
   QList<SimpleData> m_the_data;

};

#endif // MYMODEL_H
Run Code Online (Sandbox Code Playgroud)

mymodel.cpp:#include"mymodel.h"

MyModel::MyModel() : QAbstractTableModel(0)
{
    m_the_data << SimpleData{"Alpha", 10, 100.0}
               << SimpleData{"Beta", 20, 200.0}
               << SimpleData{"Gamma", 30, 300.0}
               << SimpleData{"Delta", 40, 400.0};
}

int MyModel::rowCount(const QModelIndex &parent) const
{
    Q_UNUSED(parent)
    return m_the_data.size();
}

int MyModel::columnCount(const QModelIndex &parent) const
{
    Q_UNUSED(parent)
    return 3;
}

QVariant MyModel::data(const QModelIndex &index, int role) const
{
    // Check DisplayRole
    if(role != Qt::DisplayRole)
    {
        return QVariant();
    }

    // Check boudaries
    if(index.column() < 0 ||
            columnCount() <= index.column() ||
            index.row() < 0 ||
            rowCount() <= index.row())
    {
        qDebug() << "Warning: " << index.row() << ", " << index.column();
        return QVariant();
    }

    // Nominal case
     qDebug() << "MyModel::data: " << index.column() << "; " << index.row();
    switch(index.column())
    {
        case 0:
            return m_the_data[index.row()].m_one;
        case 1:
            return  m_the_data[index.row()].m_two;
        case 2:
            return  m_the_data[index.row()].m_three;
        default:
            qDebug() << "Not supposed to happen";
            return QVariant();
    }
}

QHash<int, QByteArray> MyModel::roleNames() const
{
    QHash<int, QByteArray> roles;
    roles[0] = "one";
    roles[1] = "two";
    roles[2] = "three";
    return roles;

}

void MyModel::theDataChanged()
{
    //TODO
}
Run Code Online (Sandbox Code Playgroud)

main.qml:

import QtQuick 2.1
import QtQuick.Controls 1.0

ApplicationWindow {
    title: qsTr("Hello World")
    width: 640
    height: 480

    menuBar: MenuBar {
        Menu {
            title: qsTr("File")
            MenuItem {
                text: qsTr("Exit")
                onTriggered: Qt.quit();
            }
        }
    }
TableView {

    anchors.fill: parent

    TableViewColumn {title: "1"; role: "one"; width: 70 }
    TableViewColumn {title: "2"; role: "two"; width: 70   }
    TableViewColumn {title: "3"; role: "three"; width: 70 }

    model: theModel

}

}
Run Code Online (Sandbox Code Playgroud)

main.cpp中:

#include <QtQml>
#include <QQmlApplicationEngine>
#include <QApplication>
#include <QQuickWindow>
#include <QTableView>

#include "mymodel.h"

int main(int argc, char *argv[])
{
    // Application :
    QApplication app(argc, argv);

    // Data stuff :
    //MyData data(&app);
    MyModel model;



 // UI :
    QQmlApplicationEngine engine;
    engine.rootContext()->setContextProperty("theModel", &model);

    engine.load(QUrl("qrc:/qml/main.qml"));
    QList<QObject*> temp = engine.rootObjects();
    QObject *topLevel = temp.value(0);
    QQuickWindow *window = qobject_cast<QQuickWindow *>(topLevel);
    if ( !window ) {
        qWarning("Error: Your root item has to be a Window.");
        return -1;


 }

    // Display the main.qml, which show the model:
    window->show();

    // Same, using a QTableView:
    QTableView view;;
    view.setModel(&model);
    view.show();

    return app.exec();
}
Run Code Online (Sandbox Code Playgroud)

关于TableView的qDebug输出(行,然后是列):

MyModel::data:  0 ;  0 
MyModel::data:  0 ;  0 
MyModel::data:  0 ;  1 
MyModel::data:  0 ;  1 
MyModel::data:  0 ;  2 
MyModel::data:  0 ;  2 
MyModel::data:  0 ;  3 
MyModel::data:  0 ;  3 
Run Code Online (Sandbox Code Playgroud)

关于QTableVierw的qDebug输出:

MyModel::data:  0 ;  0 
MyModel::data:  0 ;  0 
MyModel::data:  0 ;  0 
MyModel::data:  1 ;  0 
MyModel::data:  2 ;  0 
MyModel::data:  0 ;  1 
MyModel::data:  1 ;  1 
MyModel::data:  2 ;  1 
MyModel::data:  0 ;  2 
MyModel::data:  1 ;  2 
MyModel::data:  2 ;  2 
MyModel::data:  0 ;  3 
MyModel::data:  1 ;  3 
MyModel::data:  2 ;  3 
Run Code Online (Sandbox Code Playgroud)

笔记/我试过的东西:

  • 在我给出的代码中,我可以使用QQmlListProperty; 但是我的实际代码更复杂

    • 实际查询数据,
    • 我没有SimpleData类
    • 我基本上使用QAbstractTableModel作为代理.因此,我无法切换到QQmlListProperty(或等效的)
  • 有一个项目来检查模型,ModelTest,但它不适用于Qt 5.2.

  • 数据类型(QString,qreal)不是问题:交换仍然只显示第1列.

相关链接:

http://qt-project.org/doc/qt-5/QAbstractTableModel.html

http://qt-project.org/doc/qt-5/qtquick-modelviewsdata-cppmodels.html#qabstractitemmodel

提前致谢!

规格:Windows 7,Qt 5.2,Mingw 4.8,Qt Creator 3.0

Fra*_*eld 20

TableViewColumn API建议通过角色而不是列来检索列中的数据,即"one","two"和"three",而传递的列将始终为0.除Qt之外的所有内容都返回QVariant() :: DisplayRole.Qt :: DisplayRole为0,转换为int.在角色名中,您将0的名称设置为"1",这就是为什么您碰巧看到"one"(DisplayRole)的内容.

所以要解决这个问题,你必须返回一,二和三的东西.我建议在标题中定义自定义角色枚举:

//In class MyModel:
enum Role {
    OneRole=Qt::UserRole,
    TwoRole,
    ThreeRole
};
Run Code Online (Sandbox Code Playgroud)

像这样定义roleNames:

QHash<int, QByteArray> MyModel::roleNames() const
{
    QHash<int, QByteArray> roles;
    roles[OneRole] = "one";
    roles[TwoRole] = "two";
    roles[ThreeRole] = "three";
    return roles;
}
Run Code Online (Sandbox Code Playgroud)

请注意,我从Qt :: UserRole开始而不是0.这避免了与预定义角色(如Qt :: DisplayRole)的冲突.

然后在data()中返回一,二和三的东西:

...
switch(role)
{
    case OneRole:
        return m_the_data[index.row()].m_one;
    case TwoRole:
        return m_the_data[index.row()].m_two;
    case ThreeRole:
        return m_the_data[index.row()].m_three;
}
...
Run Code Online (Sandbox Code Playgroud)

现在您应该看到数据了.

似乎来自QtQuickControls的TableView/TableViewColumn在角色和列的混合方面有点令人遗憾和困惑:虽然命名让我们想到模型列(但它们实际上是指这里的视图列),但是只能通过不同的角色检索数据,将列固定为0.(对我而言,TableViewColumn中应该有另一个可选属性"column".)这是C++ QAbstractItemModel/QTableView方式之间的一些冲突,其中多列是自然的事情和QtQuick视图,它们都只使用角色来引用数据,通常根本不支持多列(ListView,GridView).

  • 这个问题造成的混乱是巨大的,文件也很难确认/警告它.我不小心第一次学习了qt小部件,这增加了我的困惑. (5认同)
  • 好吧,它解决了这个问题:快速解决方法是删除if(role!= Qt :: DisplayRole)块,并通过交换机(角色)删除交换机(index.column()).我会为我的主项目做一个枚举.非常感谢! (3认同)
  • 在实现这个问题时,Humm看起来很匆忙,我希望他们能够更好地关注API. (2认同)