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 return o.asReturnedValue();
79 }
80
81 void setValue(const QString &role, const QVariant &value) override;
82 bool resolveIndex(const QQmlAdaptorModel &model, int idx) override;
83
86
87private:
89 QVariant cachedData;
90
91 // Gets cleaned when the metaobject has processed it.
92 bool cachedDataClean = false;
93};
94
95
100{
101public:
102 VDMListDelegateDataType(QQmlAdaptorModel *model)
103 : model(model)
104 {
105 QQmlAdaptorModelEngineData::setModelDataType<QQmlDMListAccessorData>(&builder, this);
106 metaObject.reset(builder.toMetaObject());
107 *static_cast<QMetaObject *>(this) = *metaObject.data();
108 }
109
110 void cleanup(QQmlAdaptorModel &) const override
111 {
112 release();
113 }
114
115 int rowCount(const QQmlAdaptorModel &model) const override
116 {
117 return model.list.count();
118 }
119
120 int columnCount(const QQmlAdaptorModel &model) const override
121 {
122 switch (model.list.type()) {
123 case QQmlListAccessor::Invalid:
124 return 0;
125 case QQmlListAccessor::StringList:
126 case QQmlListAccessor::UrlList:
127 case QQmlListAccessor::Integer:
128 return 1;
129 default:
130 break;
131 }
132
133 // If there are no properties, we can get modelData itself.
134 return std::max(1, propertyCount() - propertyOffset);
135 }
136
137 static const QMetaObject *metaObjectFromType(QMetaType type)
138 {
139 if (const QMetaObject *metaObject = type.metaObject())
140 return metaObject;
141
142 // NB: This acquires the lock on QQmlMetaTypeData. If we had a QQmlEngine here,
143 // we could use QQmlGadgetPtrWrapper::instance() to avoid this.
144 if (const QQmlValueType *valueType = QQmlMetaType::valueType(type))
145 return valueType->staticMetaObject();
146
147 return nullptr;
148 }
149
150 template<typename String>
151 static QString toQString(const String &string)
152 {
153 if constexpr (std::is_same_v<String, QString>)
154 return string;
155 else if constexpr (std::is_same_v<String, QByteArray>)
156 return QString::fromUtf8(string);
157 else if constexpr (std::is_same_v<String, const char *>)
158 return QString::fromUtf8(string);
159 Q_UNREACHABLE_RETURN(QString());
160 }
161
162 template<typename String>
163 static QByteArray toUtf8(const String &string)
164 {
165 if constexpr (std::is_same_v<String, QString>)
166 return string.toUtf8();
167 else if constexpr (std::is_same_v<String, QByteArray>)
168 return string;
169 else if constexpr (std::is_same_v<String, const char *>)
170 return QByteArray::fromRawData(string, qstrlen(string));
171 Q_UNREACHABLE_RETURN(QByteArray());
172 }
173
174 template<typename String>
175 static QVariant value(const QVariant *row, const String &role)
176 {
177 const QMetaType type = row->metaType();
178 if (type == QMetaType::fromType<QVariantMap>())
179 return row->toMap().value(toQString(role));
180
181 if (type == QMetaType::fromType<QVariantHash>())
182 return row->toHash().value(toQString(role));
183
184 const QMetaType::TypeFlags typeFlags = type.flags();
185 if (typeFlags & QMetaType::PointerToQObject)
186 return row->value<QObject *>()->property(toUtf8(role));
187
188 if (const QMetaObject *metaObject = metaObjectFromType(type)) {
189 const int propertyIndex = metaObject->indexOfProperty(toUtf8(role));
190 if (propertyIndex >= 0)
191 return metaObject->property(propertyIndex).readOnGadget(row->constData());
192 }
193
194 return QVariant();
195 }
196
197 template<typename String>
198 void createPropertyIfMissing(const String &string)
199 {
200 for (int i = 0, end = propertyCount(); i < end; ++i) {
201 if (QAnyStringView(property(i).name()) == QAnyStringView(string))
202 return;
203 }
204
205 createProperty(toUtf8(string), nullptr);
206 }
207
208 void createMissingProperties(const QVariant *row)
209 {
210 const QMetaType type = row->metaType();
211 if (type == QMetaType::fromType<QVariantMap>()) {
212 const QVariantMap map = row->toMap();
213 for (auto it = map.keyBegin(), end = map.keyEnd(); it != end; ++it)
214 createPropertyIfMissing(*it);
215 } else if (type == QMetaType::fromType<QVariantHash>()) {
216 const QVariantHash map = row->toHash();
217 for (auto it = map.keyBegin(), end = map.keyEnd(); it != end; ++it)
218 createPropertyIfMissing(*it);
219 } else if (type.flags() & QMetaType::PointerToQObject) {
220 const QMetaObject *metaObject = row->value<QObject *>()->metaObject();
221 for (int i = 0, end = metaObject->propertyCount(); i < end; ++i)
222 createPropertyIfMissing(metaObject->property(i).name());
223 } else if (const QMetaObject *metaObject = metaObjectFromType(type)) {
224 for (int i = 0, end = metaObject->propertyCount(); i < end; ++i)
225 createPropertyIfMissing(metaObject->property(i).name());
226 }
227 }
228
229 template<typename String>
230 static void setValue(QVariant *row, const String &role, const QVariant &value)
231 {
232 const QMetaType type = row->metaType();
233 if (type == QMetaType::fromType<QVariantMap>()) {
234 static_cast<QVariantMap *>(row->data())->insert(toQString(role), value);
235 } else if (type == QMetaType::fromType<QVariantHash>()) {
236 static_cast<QVariantHash *>(row->data())->insert(toQString(role), value);
237 } else if (type.flags() & QMetaType::PointerToQObject) {
238 row->value<QObject *>()->setProperty(toUtf8(role), value);
239 } else if (const QMetaObject *metaObject = metaObjectFromType(type)) {
240 const int propertyIndex = metaObject->indexOfProperty(toUtf8(role));
241 if (propertyIndex >= 0)
242 metaObject->property(propertyIndex).writeOnGadget(row->data(), value);
243 }
244 }
245
246 QVariant value(const QQmlAdaptorModel &model, int index, const QString &role) const override
247 {
248 const QVariant entry = model.list.at(index);
249 if (role == QLatin1String("modelData") || role.isEmpty())
250 return entry;
251
252 return value(&entry, role);
253 }
254
256 QQmlAdaptorModel &model,
257 const QQmlRefPointer<QQmlDelegateModelItemMetaType> &metaType,
258 int index, int row, int column) override
259 {
260 const QVariant value = (index >= 0 && index < model.list.count())
261 ? model.list.at(index)
262 : QVariant();
263 return new QQmlDMListAccessorData(metaType, this, index, row, column, value);
264 }
265
266 bool notify(const QQmlAdaptorModel &model, const QList<QQmlDelegateModelItem *> &items, int index, int count, const QList<int> &) const override
267 {
268 for (auto modelItem : items) {
269 const int modelItemIndex = modelItem->modelIndex();
270 if (modelItemIndex < index || modelItemIndex >= index + count)
271 continue;
272
273 auto listModelItem = static_cast<QQmlDMListAccessorData *>(modelItem);
274 QVariant updatedModelData = model.list.at(listModelItem->modelIndex());
275 listModelItem->setModelData(updatedModelData);
276 }
277 return true;
278 }
279
280 void emitAllSignals(QQmlDMListAccessorData *accessor) const;
281
282 int metaCall(QObject *object, QMetaObject::Call call, int id, void **arguments) final;
283 int createProperty(const char *name, const char *) final;
284#if QT_VERSION >= QT_VERSION_CHECK(7, 0, 0)
286#else
287 QMetaObject *toDynamicMetaObject(QObject *accessors) final;
288#endif
289
294};
295
296QT_END_NAMESPACE
297
298#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
bool notify(const QQmlAdaptorModel &model, const QList< QQmlDelegateModelItem * > &items, int index, int count, const QList< int > &) const override
void createMissingProperties(const QVariant *row)
int rowCount(const QQmlAdaptorModel &model) const override
const QMetaObject * toDynamicMetaObject(QObject *accessors) const final
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