- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我正在尝试将(派生自)QAbstractTableModel 与 Qml TableView 一起使用;
但是,只显示第一列。
原因是非零列不调用 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
我的模型.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
}
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
}
}
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();
}
关于 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
关于 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
笔记/我试过的东西:
在我给出的代码中,我可以使用 QQmlListProperty;但是我的实际代码更复杂
有一个项目用于检查模型,ModelTest,但它不适用于 Qt 5.2。
数据类型(QString、qreal)不是问题:交换仍然只显示第一列。
相关链接:
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
最佳答案
TableViewColumn API 建议通过角色而不是列检索列中的数据,即“一”、“二”和“三”,而传递的列将始终为 0。您返回 QVariant()除了 Qt::DisplayRole 之外的一切。Qt::DisplayRole 为 0,转换为 int。在角色名称中,您将 0 的名称设置为“一”,这就是为什么您碰巧看到“一”的名称(DisplayRole)。
因此,要解决此问题,您必须为一、二和三返回一些内容。我建议在标题中定义一个自定义角色枚举:
//In class MyModel:
enum Role {
OneRole=Qt::UserRole,
TwoRole,
ThreeRole
};
像这样定义角色名称:
QHash<int, QByteArray> MyModel::roleNames() const
{
QHash<int, QByteArray> roles;
roles[OneRole] = "one";
roles[TwoRole] = "two";
roles[ThreeRole] = "three";
return roles;
}
请注意,我从 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;
}
...
现在您应该看到数据了。
QtQuickControls 中的 TableView/TableViewColumn 似乎对角色和列进行了一些不幸和令人困惑的混合:虽然命名让我们想到模型列(但它们实际上在这里指的是 View 的列),但只能检索数据通过不同的角色,列固定为 0。(对我来说,TableViewColumn 中应该有另一个可选属性“列”。)这在 C++ QAbstractItemModel/QTableView 方式之间有点冲突,其中多列是很自然的事情,而 QtQuick View 都只使用角色来引用数据,通常根本不支持多列(ListView, GridView)。
关于c++ - 将 QAbstractTableModel 与 Qml TableView 一起使用仅显示第一列,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21270969/
如果下一个元素的宽度超过指定布局的宽度,是否有 QML 布局或某些配置会自动将 QML 项目包装到下一行? 当我使用 QML GridLayout ,项目刚好离开窗口的边缘并被剪裁: GridLayo
如果下一个元素的宽度超过指定布局的宽度,是否有 QML 布局或某些配置会自动将 QML 项目包装到下一行? 当我使用 QML GridLayout ,项目刚好离开窗口的边缘并被剪裁: GridLayo
如何在 qml 文件之间发送变量或信号? http://i.stack.imgur.com/MChCG.png Mainwindow -> 创建组件Item2.qml MainWindow -> 创建
我正在做一些事情,我有一个名为“FloatingMenu”的类(它应该在 C++ 中管理菜单)及其在文件 FloatingMenu.qml 中用于 GUI 的 QML alter-ego。我有一个文件
我正在尝试做一些看似简单的事情,但失败了:定义一个简单的内联文本格式组件,然后用不同的文本多次实例化它。这是代码 Item { . . . Component { id: favButtonL
我可以在页面中使用 InvokeActionItem 轻松共享项目,但我需要能够在 ListView 项目中调用它。我设法触发了一个调用,但我不知道如何在触发它时添加数据。我不断收到错误消息 Invo
我如何在 QML 中检测 Window {} 之外的点击? Rectangle { id: topLevel height: 400; width: 400 Window {
我试过 : var child = grid.children[slot1]; grid.children[slot1] = grid.children[slot2]; grid.children[s
例如,我希望创建一个包含 100 个文本编辑器的 qml 窗口,如何在循环中创建它?那可能吗? 最佳答案 循环是命令式代码,所以它不是 QML,而是 Javascript 或 C++。所以当然,你可以
这是我的 QML 文件,其中包含一个文本组件: import QtQuick 2.0 Item { id: idItmWrapText Text { id: idTxt
我正在寻找一种方法来显示一个文本提示,说明预期的输入作为对用户的建议。以谷歌搜索栏为例: 是否有我缺少的属性,或者这是必须通过脚本来实现的? 最佳答案 Qt Quick 输入项上不存在该属性。您可以为
为 qml 项设置背景的最简单方法是让子矩形的 anchor 完全填满父项: Item { width: 320 height: 240 Rectangle {
我想将属性动态添加到 QML 元素: Item { id: dynamicProperty; property int first; Component.onCompleted: { /*
我用 PySide 和 QML 编写了某种安装程序。按照设计,它必须是多页的。而且我想将要从 QML 表单调用的插槽划分为不同的对象(理想情况下,划分为模块,但据我了解,带有插槽的对象必须继承 QOb
QML 中有没有办法用 opacity: 0 创建一个矩形?仍然有可见的边界?如果没有,有关如何解决它的任何建议? 谢谢 最佳答案 不,不透明度适用于项目的完整视觉方面(并且不透明度:0 使项目完全不
属性变体 a:{}似乎不起作用。 a 最终未定义,而不是一个空字典。 我对 Javascript 不是很有经验...初始化属性以保存空字典的正确方法是什么? 以下 qml 在控制台上打印“qrc:/m
我在 SO 上查看了大量关于 QML 内容边距的问题,但所有答案都指向缺少 spacing: 0 属性。我已经完成了所有这些,但仍然有一些我无法消除的奇怪空间。任何人都可以解释为什么这个 QML 代码
我有一个用于样式定义的 QML 单例,定义如下: pragma Singleton import QtQuick 2.2 import QtQuick.Controls 1.1 import QtQu
这是以下代码的结果: 主.qml import QtQuick 2.8 Item { Reusable { index: 1234 // reusable with
属性变体 a:{}似乎不起作用。 a 最终未定义,而不是一个空字典。 我对 Javascript 不是很有经验...初始化属性以保存空字典的正确方法是什么? 以下 qml 在控制台上打印“qrc:/m
我是一名优秀的程序员,十分优秀!