Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qqmldmlistaccessordata_p.h
Go to the documentation of this file.
1// Copyright (C) 2023 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3// Qt-Security score:significant
4
5#ifndef QQMLDMLISTACCESSORDATA_P_H
6#define QQMLDMLISTACCESSORDATA_P_H
7
8//
9// W A R N I N G
10// -------------
11//
12// This file is not part of the Qt API. It exists purely as an
13// implementation detail. This header file may change from version to
14// version without notice, or even be removed.
15//
16// We mean it.
17//
18
19#include <private/qqmladaptormodelenginedata_p.h>
20#include <private/qqmldelegatemodel_p_p.h>
21#include <private/qobject_p.h>
22
23QT_BEGIN_NAMESPACE
24
25class VDMListDelegateDataType;
26
28{
29 Q_OBJECT
30 Q_PROPERTY(QVariant modelData READ modelData WRITE setModelData NOTIFY modelDataChanged)
32public:
36 const QVariant &value);
38
40 {
41 return cachedData;
42 }
43
44 void setModelData(const QVariant &data);
45
46 static QV4::ReturnedValue get_modelData(const QV4::FunctionObject *b, const QV4::Value *thisObject, const QV4::Value *, int)
47 {
48 QV4::ExecutionEngine *v4 = b->engine();
49 const QQmlDelegateModelItemObject *o = thisObject->as<QQmlDelegateModelItemObject>();
50 if (!o)
51 return v4->throwTypeError(QStringLiteral("Not a valid DelegateModel object"));
52
53 return v4->fromVariant(static_cast<QQmlDMListAccessorData *>(o->d()->item)->cachedData);
54 }
55
56 static QV4::ReturnedValue set_modelData(const QV4::FunctionObject *b, const QV4::Value *thisObject, const QV4::Value *argv, int argc)
57 {
58 QV4::ExecutionEngine *v4 = b->engine();
59 const QQmlDelegateModelItemObject *o = thisObject->as<QQmlDelegateModelItemObject>();
60 if (!o)
61 return v4->throwTypeError(QStringLiteral("Not a valid DelegateModel object"));
62 if (!argc)
63 return v4->throwTypeError();
64
65 static_cast<QQmlDMListAccessorData *>(o->d()->item)->setModelData(
66 QV4::ExecutionEngine::toVariant(argv[0], QMetaType {}));
67 return QV4::Encode::undefined();
68 }
69
70 QV4::ReturnedValue get() override
71 {
72 QV4::Scope scope(metaType->v4Engine);
73 QQmlAdaptorModelEngineData *data = QQmlAdaptorModelEngineData::get(scope.engine);
74 QV4::ScopedObject o(
75 scope, scope.engine->memoryManager->allocate<QQmlDelegateModelItemObject>(this));
76 QV4::ScopedObject p(scope, data->listItemProto.value());
77 o->setPrototypeOf(p);
78 ++scriptRef;
79 return o.asReturnedValue();
80 }
81
82 void setValue(const QString &role, const QVariant &value) override;
83 bool resolveIndex(const QQmlAdaptorModel &model, int idx) override;
84
87
88private:
90 QVariant cachedData;
91
92 // Gets cleaned when the metaobject has processed it.
93 bool cachedDataClean = false;
94};
95
96
101{
102public:
103 VDMListDelegateDataType(QQmlAdaptorModel *model)
104 : model(model)
105 {
106 QQmlAdaptorModelEngineData::setModelDataType<QQmlDMListAccessorData>(&builder, this);
107 metaObject.reset(builder.toMetaObject());
108 *static_cast<QMetaObject *>(this) = *metaObject.data();
109 }
110
111 void cleanup(QQmlAdaptorModel &) const override
112 {
113 release();
114 }
115
116 int rowCount(const QQmlAdaptorModel &model) const override
117 {
118 return model.list.count();
119 }
120
121 int columnCount(const QQmlAdaptorModel &model) const override
122 {
123 switch (model.list.type()) {
124 case QQmlListAccessor::Invalid:
125 return 0;
126 case QQmlListAccessor::StringList:
127 case QQmlListAccessor::UrlList:
128 case QQmlListAccessor::Integer:
129 return 1;
130 default:
131 break;
132 }
133
134 // If there are no properties, we can get modelData itself.
135 return std::max(1, propertyCount() - propertyOffset);
136 }
137
138 static const QMetaObject *metaObjectFromType(QMetaType type)
139 {
140 if (const QMetaObject *metaObject = type.metaObject())
141 return metaObject;
142
143 // NB: This acquires the lock on QQmlMetaTypeData. If we had a QQmlEngine here,
144 // we could use QQmlGadgetPtrWrapper::instance() to avoid this.
145 if (const QQmlValueType *valueType = QQmlMetaType::valueType(type))
146 return valueType->staticMetaObject();
147
148 return nullptr;
149 }
150
151 template<typename String>
152 static QString toQString(const String &string)
153 {
154 if constexpr (std::is_same_v<String, QString>)
155 return string;
156 else if constexpr (std::is_same_v<String, QByteArray>)
157 return QString::fromUtf8(string);
158 else if constexpr (std::is_same_v<String, const char *>)
159 return QString::fromUtf8(string);
160 Q_UNREACHABLE_RETURN(QString());
161 }
162
163 template<typename String>
164 static QByteArray toUtf8(const String &string)
165 {
166 if constexpr (std::is_same_v<String, QString>)
167 return string.toUtf8();
168 else if constexpr (std::is_same_v<String, QByteArray>)
169 return string;
170 else if constexpr (std::is_same_v<String, const char *>)
171 return QByteArray::fromRawData(string, qstrlen(string));
172 Q_UNREACHABLE_RETURN(QByteArray());
173 }
174
175 template<typename String>
176 static QVariant value(const QVariant *row, const String &role)
177 {
178 const QMetaType type = row->metaType();
179 if (type == QMetaType::fromType<QVariantMap>())
180 return row->toMap().value(toQString(role));
181
182 if (type == QMetaType::fromType<QVariantHash>())
183 return row->toHash().value(toQString(role));
184
185 const QMetaType::TypeFlags typeFlags = type.flags();
186 if (typeFlags & QMetaType::PointerToQObject)
187 return row->value<QObject *>()->property(toUtf8(role));
188
189 if (const QMetaObject *metaObject = metaObjectFromType(type)) {
190 const int propertyIndex = metaObject->indexOfProperty(toUtf8(role));
191 if (propertyIndex >= 0)
192 return metaObject->property(propertyIndex).readOnGadget(row->constData());
193 }
194
195 return QVariant();
196 }
197
198 template<typename String>
199 void createPropertyIfMissing(const String &string)
200 {
201 for (int i = 0, end = propertyCount(); i < end; ++i) {
202 if (QAnyStringView(property(i).name()) == QAnyStringView(string))
203 return;
204 }
205
206 createProperty(toUtf8(string), nullptr);
207 }
208
209 void createMissingProperties(const QVariant *row)
210 {
211 const QMetaType type = row->metaType();
212 if (type == QMetaType::fromType<QVariantMap>()) {
213 const QVariantMap map = row->toMap();
214 for (auto it = map.keyBegin(), end = map.keyEnd(); it != end; ++it)
215 createPropertyIfMissing(*it);
216 } else if (type == QMetaType::fromType<QVariantHash>()) {
217 const QVariantHash map = row->toHash();
218 for (auto it = map.keyBegin(), end = map.keyEnd(); it != end; ++it)
219 createPropertyIfMissing(*it);
220 } else if (type.flags() & QMetaType::PointerToQObject) {
221 const QMetaObject *metaObject = row->value<QObject *>()->metaObject();
222 for (int i = 0, end = metaObject->propertyCount(); i < end; ++i)
223 createPropertyIfMissing(metaObject->property(i).name());
224 } else if (const QMetaObject *metaObject = metaObjectFromType(type)) {
225 for (int i = 0, end = metaObject->propertyCount(); i < end; ++i)
226 createPropertyIfMissing(metaObject->property(i).name());
227 }
228 }
229
230 template<typename String>
231 static void setValue(QVariant *row, const String &role, const QVariant &value)
232 {
233 const QMetaType type = row->metaType();
234 if (type == QMetaType::fromType<QVariantMap>()) {
235 static_cast<QVariantMap *>(row->data())->insert(toQString(role), value);
236 } else if (type == QMetaType::fromType<QVariantHash>()) {
237 static_cast<QVariantHash *>(row->data())->insert(toQString(role), value);
238 } else if (type.flags() & QMetaType::PointerToQObject) {
239 row->value<QObject *>()->setProperty(toUtf8(role), value);
240 } else if (const QMetaObject *metaObject = metaObjectFromType(type)) {
241 const int propertyIndex = metaObject->indexOfProperty(toUtf8(role));
242 if (propertyIndex >= 0)
243 metaObject->property(propertyIndex).writeOnGadget(row->data(), value);
244 }
245 }
246
247 QVariant value(const QQmlAdaptorModel &model, int index, const QString &role) const override
248 {
249 const QVariant entry = model.list.at(index);
250 if (role == QLatin1String("modelData") || role.isEmpty())
251 return entry;
252
253 return value(&entry, role);
254 }
255
257 QQmlAdaptorModel &model,
258 const QQmlRefPointer<QQmlDelegateModelItemMetaType> &metaType,
259 int index, int row, int column) override
260 {
261 const QVariant value = (index >= 0 && index < model.list.count())
262 ? model.list.at(index)
263 : QVariant();
264 return new QQmlDMListAccessorData(metaType, this, index, row, column, value);
265 }
266
267 bool notify(const QQmlAdaptorModel &model, const QList<QQmlDelegateModelItem *> &items, int index, int count, const QVector<int> &) const override
268 {
269 for (auto modelItem : items) {
270 const int modelItemIndex = modelItem->modelIndex();
271 if (modelItemIndex < index || modelItemIndex >= index + count)
272 continue;
273
274 auto listModelItem = static_cast<QQmlDMListAccessorData *>(modelItem);
275 QVariant updatedModelData = model.list.at(listModelItem->modelIndex());
276 listModelItem->setModelData(updatedModelData);
277 }
278 return true;
279 }
280
281 void emitAllSignals(QQmlDMListAccessorData *accessor) const;
282
283 int metaCall(QObject *object, QMetaObject::Call call, int id, void **arguments) final;
284 int createProperty(const char *name, const char *) final;
285#if QT_VERSION >= QT_VERSION_CHECK(7, 0, 0)
287#else
288 QMetaObject *toDynamicMetaObject(QObject *accessors) final;
289#endif
290
295};
296
297QT_END_NAMESPACE
298
299#endif // QQMLDMLISTACCESSORDATA_P_H
void setModelData(const QVariant &data)
bool resolveIndex(const QQmlAdaptorModel &model, int idx) override
void setValue(const QString &role, const QVariant &value) override
static QV4::ReturnedValue get_modelData(const QV4::FunctionObject *b, const QV4::Value *thisObject, const QV4::Value *, int)
static QV4::ReturnedValue set_modelData(const QV4::FunctionObject *b, const QV4::Value *thisObject, const QV4::Value *argv, int argc)
QV4::ReturnedValue get() override
void emitAllSignals(QQmlDMListAccessorData *accessor) const
void createMissingProperties(const QVariant *row)
int rowCount(const QQmlAdaptorModel &model) const override
const QMetaObject * toDynamicMetaObject(QObject *accessors) const final
bool notify(const QQmlAdaptorModel &model, const QList< QQmlDelegateModelItem * > &items, int index, int count, const QVector< int > &) const override
VDMListDelegateDataType(QQmlAdaptorModel *model)
int createProperty(const char *name, const char *) final
static const QMetaObject * metaObjectFromType(QMetaType type)
static QString toQString(const String &string)
QQmlDelegateModelItem * createItem(QQmlAdaptorModel &model, const QQmlRefPointer< QQmlDelegateModelItemMetaType > &metaType, int index, int row, int column) override
int metaCall(QObject *object, QMetaObject::Call call, int id, void **arguments) final
static QVariant value(const QVariant *row, const String &role)
void createPropertyIfMissing(const String &string)
QVariant value(const QQmlAdaptorModel &model, int index, const QString &role) const override
static QByteArray toUtf8(const String &string)
int columnCount(const QQmlAdaptorModel &model) const override
static void setValue(QVariant *row, const String &role, const QVariant &value)
void cleanup(QQmlAdaptorModel &) const override