c++ - 升级到 Qt 5.15 后,ListView 委托中的父级为空
问题描述
如果尝试设置其委托属性并滚动列表(这会使委托被销毁/创建),则ListView
具有最简单委托的A会产生大量警告。在 Qt 5.12 或 5.9 中并非如此。"21:35:31.911 warning T#16084047 unknown - qrc:/main.qml:15: TypeError: Cannot read property 'left' of null"
anchors
文件main.qml
import QtQuick 2.15
import QtQuick.Window 2.15
Window {
width: 640
height: 480
visible: true
title: qsTr("Hello World")
ListView {
anchors.fill: parent
model: cppModel
delegate: Rectangle {
anchors.left: parent.left
anchors.right: parent.right
height: 50
Text { text: model.itemName }
}
}
}
文件main.cpp
:
#include <QAbstractListModel>
#include <QGuiApplication>
#include <QQmlApplicationEngine>
#include <QtGlobal>
#include <QQmlContext>
#include <iostream>
void myMessageHandler(QtMsgType type, const QMessageLogContext &context, const QString &msg) {
QString logLine = qFormatLogMessage(type, context, msg);
std::cout << logLine.toStdString() << std::endl;
}
class CppModel: public QAbstractListModel {
// QAbstractItemModel interface
public:
virtual int rowCount(const QModelIndex &parent) const override { return 100; }
virtual QVariant data(const QModelIndex &index, int role) const override {
if (role == (Qt::DisplayRole + 1)) {
return QString("Element %1").arg(index.row());
}
return QVariant();
}
virtual QHash<int, QByteArray> roleNames() const override {
return {{(Qt::DisplayRole+1), "itemName"}};
}
};
int main(int argc, char *argv[])
{
qSetMessagePattern("%{time hh:mm:ss.zzz} %{type} T#%{threadid} %{function} - %{message}");
qInstallMessageHandler(myMessageHandler);
QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
QGuiApplication app(argc, argv);
QQmlApplicationEngine engine;
CppModel cppModel;
engine.rootContext()->setContextProperty("cppModel", &cppModel);
const QUrl url(QStringLiteral("qrc:/main.qml"));
QObject::connect(&engine, &QQmlApplicationEngine::objectCreated,
&app, [url](QObject *obj, const QUrl &objUrl) {
if (!obj && url == objUrl)
QCoreApplication::exit(-1);
}, Qt::QueuedConnection);
engine.load(url);
return app.exec();
}
我做错了什么以及如何正确设置anchors
委托元素?
解决方案
这是 Qt 5.15 中行为更改的结果。第一个问题在这里报告,这里有更详细的总结。新文档说:
委托会根据需要进行实例化,并且可能随时被销毁。因此,状态永远不应该存储在委托中。委托通常是 ListView 的 contentItem 的父级,但通常取决于它是否在视图中可见,父级可以更改,有时为 null。因此,不建议从委托中绑定到父级的属性。如果您希望委托填写 ListView 的宽度,请考虑改用以下方法之一:
ListView { id: listView // ... delegate: Item { // Incorrect. width: parent.width // Correct. width: listView.width width: ListView.view.width // ... } }
因此,您可以:
- 给出
ListView
anid
并在绑定中使用它而不是parent
. - 使用附加属性 (
ListView.view
) 访问视图。 - 检查空值 (
anchors.left: parent ? parent.left : undefined
)。
选项 1 和 2 将产生更清晰的代码。
选项 1 会导致QObject
创建更少的对象(每个附加对象都是一个QObject
),但会将委托与该特定视图联系起来。
选项 2 更适合作为独立组件的委托,这些组件将与其他ListView
s 一起重用。
推荐阅读
- php - 如何根据 Laravel 中的其他列在数据库中添加唯一记录
- reactjs - 我没有很好地关闭 --yarn dev- (端口 4444 已经在使用中。)
- jmeter - 如何让 JMeter 为具有令牌的 N 个用户中的每一个运行多个操作
- junit - 如何使用 JUnit、Spring Boot 和 Flyway 进行测试?
- flutter - 如何检查一个元素是否存在于 xml RSS 提要中
- apache - 哪个是正确的 ALPN 行为?
- apache-kafka - Kafka 消费者静态成员资格的 group.instance.id 应该是什么?
- python - Python 问题:TypeError:__init__() 缺少 1 个必需的位置参数:'brett'
- php - 在 PHP 中确定输出格式的最有效方法
- javascript - 如何将Javascript导入Angular2中的全局命名空间