239void QQmlPropertyPrivate::initProperty(QObject *obj,
const QString &name,
240 QQmlPropertyPrivate::InitFlags flags)
242 QQmlRefPointer<QQmlTypeNameCache> typeNameCache = context ? context->imports() :
nullptr;
244 QObject *currentObject = obj;
245 QList<QStringView> path;
246 QStringView terminal(name);
248 if (name.contains(QLatin1Char(
'.'))) {
249 path = QStringView{name}.split(QLatin1Char(
'.'));
250 if (path.isEmpty())
return;
253 for (
int ii = 0; ii < path.size() - 1; ++ii) {
254 const QStringView &pathName = path.at(ii);
258 if (typeNameCache && !pathName.isEmpty() && pathName.at(0).isUpper()) {
259 QQmlTypeLoader *typeLoader = QQmlTypeLoader::get(engine.data());
260 QQmlTypeNameCache::Result r = typeNameCache->query(pathName, typeLoader);
262 if (r.type.isValid()) {
263 QQmlAttachedPropertiesFunc func = r.type.attachedPropertiesFunction(typeLoader);
266 currentObject = qmlAttachedPropertiesObject(currentObject, func);
267 if (!currentObject)
return;
268 }
else if (r.importNamespace) {
269 if (++ii == path.size())
273 r = typeNameCache->query<QQmlTypeNameCache::QueryNamespaced::No>(
274 path.at(ii), r.importNamespace, typeLoader);
276 if (!r.type.isValid())
279 QQmlAttachedPropertiesFunc func = r.type.attachedPropertiesFunction(typeLoader);
283 currentObject = qmlAttachedPropertiesObject(currentObject, func);
287 }
else if (r.scriptIndex != -1) {
290 Q_ASSERT(!
"Unreachable");
297 QQmlPropertyData local;
298 const QQmlPropertyData *property = currentObject
299 ? QQmlPropertyCache::property(currentObject, pathName, context, &local)
305 if (currentObject || !(flags & InitFlag::AllowId))
308 for (
auto idContext = context; idContext; idContext = idContext->parent()) {
309 const int objectId = idContext->propertyIndex(pathName.toString());
310 if (objectId != -1 && objectId < idContext->numIdValues()) {
311 currentObject = idContext->idValue(objectId);
320 }
else if (property->isFunction()) {
324 if (ii == (path.size() - 2) && QQmlMetaType::isValueType(property->propType())) {
326 const QMetaObject *valueTypeMetaObject = QQmlMetaType::metaObjectForValueType(property->propType());
327 if (!valueTypeMetaObject)
return;
329 int idx = valueTypeMetaObject->indexOfProperty(path.last().toUtf8().constData());
330 if (idx == -1)
return;
332 QMetaProperty vtProp = valueTypeMetaObject->property(idx);
334 Q_ASSERT(idx <= 0x0000FFFF);
336 object = currentObject;
338 valueTypeData.setFlags(QQmlPropertyData::flagsForProperty(vtProp));
339 valueTypeData.setPropType(vtProp.metaType());
340 valueTypeData.setCoreIndex(idx);
344 if (!property->isQObject()) {
345 if (
auto asPropertyMap = qobject_cast<QQmlPropertyMap*>(currentObject))
346 currentObject = asPropertyMap->value(path.at(ii).toString()).value<QObject*>();
350 property->readProperty(currentObject, ¤tObject);
353 if (!currentObject)
return;
359 terminal = path.last();
360 }
else if (!currentObject) {
364 auto findSignalInMetaObject = [&](
const QByteArray &signalName) {
365 const QMetaMethod method = findSignalByName(currentObject->metaObject(), signalName);
366 if (!method.isValid())
369 object = currentObject;
374 QQmlData *ddata = QQmlData::get(currentObject,
false);
375 auto findChangeSignal = [&](QStringView signalName) {
376 if (
auto propName = QQmlSignalNames::changedSignalNameToPropertyName(signalName)) {
377 const QQmlPropertyData *d =
378 ddata->propertyCache->property(*propName, currentObject, context);
379 while (d && d->isFunction())
380 d = ddata->propertyCache->overrideData(d);
382 if (d && d->notifyIndex() != -1) {
383 object = currentObject;
384 core = *ddata->propertyCache->signal(d->notifyIndex());
391 const auto findSignal = [&](
const QString &signalName) {
392 if (ddata && ddata->propertyCache) {
394 const QQmlPropertyData *d
395 = ddata->propertyCache->property(signalName, currentObject, context);
400 while (d && !d->isFunction())
401 d = ddata->propertyCache->overrideData(d);
404 object = currentObject;
409 return findChangeSignal(signalName);
412 return findSignalInMetaObject(signalName.toUtf8());
415 auto signalName = QQmlSignalNames::handlerNameToSignalName(terminal);
417 if (findSignal(*signalName))
420 signalName = QQmlSignalNames::badHandlerNameToSignalName(terminal);
422 if (findSignal(*signalName)) {
425 <<
"is not a properly capitalized signal handler name."
426 << QQmlSignalNames::signalNameToHandlerName(*signalName)
427 <<
"would be correct.";
433 if (ddata && ddata->propertyCache) {
434 const QQmlPropertyData *property = ddata->propertyCache->property(
435 terminal, currentObject, context);
438 while (property && !property->isSignal()) {
439 if (!property->isFunction()) {
440 object = currentObject;
442 nameCache = terminal.toString();
445 property = ddata->propertyCache->overrideData(property);
448 if (!(flags & InitFlag::AllowSignal))
452 Q_ASSERT(property->isSignal());
453 object = currentObject;
459 findChangeSignal(terminal);
462 const QByteArray propertyName = terminal.toUtf8();
463 const QMetaProperty prop = findPropertyByName(currentObject->metaObject(), propertyName);
465 if (prop.isValid()) {
466 object = currentObject;
471 if (flags & InitFlag::AllowSignal)
472 findSignalInMetaObject(terminal.toUtf8());
853 QObject *object, QQmlPropertyIndex index,
856 int coreIndex = index.coreIndex();
857 int valueTypeIndex = index.valueTypeIndex();
859 QQmlData *data = QQmlData::get(object,
false);
861 if (!data || !data->hasBindingBit(coreIndex))
864 QQmlAbstractBinding::Ptr oldBinding;
865 oldBinding = data->bindings;
867 while (oldBinding && (oldBinding->targetPropertyIndex().coreIndex() != coreIndex ||
868 oldBinding->targetPropertyIndex().hasValueTypeIndex())) {
869 oldBinding = oldBinding->nextBinding();
874 data->clearBindingBit(coreIndex);
878 if (valueTypeIndex != -1 && oldBinding->kind() == QQmlAbstractBinding::ValueTypeProxy) {
879 oldBinding =
static_cast<QQmlValueTypeProxyBinding *>(oldBinding.data())->binding(index);
884 if (oldBinding->isSticky() && !(flags & QQmlPropertyPrivate::OverrideSticky))
887 if (!(flags & QQmlPropertyPrivate::DontEnable))
888 oldBinding->setEnabled(
false, {});
889 oldBinding->removeFromObject();
918QQmlPropertyPrivate::binding(QObject *object, QQmlPropertyIndex index)
920 auto aliasTarget = findAliasTarget(object, index);
921 object = aliasTarget.targetObject;
922 index = aliasTarget.targetIndex;
924 QQmlData *data = QQmlData::get(object);
928 const int coreIndex = index.coreIndex();
929 const int valueTypeIndex = index.valueTypeIndex();
931 if (coreIndex < 0 || !data->hasBindingBit(coreIndex))
934 QQmlAbstractBinding *binding = data->bindings;
935 while (binding && (binding->targetPropertyIndex().coreIndex() != coreIndex ||
936 binding->targetPropertyIndex().hasValueTypeIndex()))
937 binding = binding->nextBinding();
939 if (binding && valueTypeIndex != -1) {
940 if (binding->kind() == QQmlAbstractBinding::ValueTypeProxy)
941 binding =
static_cast<QQmlValueTypeProxyBinding *>(binding)->binding(index);
947void QQmlPropertyPrivate::findAliasTarget(QObject *object, QQmlPropertyIndex bindingIndex,
948 QObject **targetObject,
949 QQmlPropertyIndex *targetBindingIndex)
951 QQmlData *data = QQmlData::get(object,
false);
953 int coreIndex = bindingIndex.coreIndex();
954 int valueTypeIndex = bindingIndex.valueTypeIndex();
956 const QQmlPropertyData *propertyData =
957 data->propertyCache?data->propertyCache->property(coreIndex):
nullptr;
958 if (propertyData && propertyData->isAlias()) {
959 QQmlVMEMetaObject *vme = QQmlVMEMetaObject::getForProperty(object, coreIndex);
961 QObject *aObject =
nullptr;
int aCoreIndex = -1;
int aValueTypeIndex = -1;
962 if (vme->aliasTarget(coreIndex, &aObject, &aCoreIndex, &aValueTypeIndex)) {
964 Q_ASSERT(valueTypeIndex == -1 || aValueTypeIndex == -1);
966 QQmlPropertyIndex aBindingIndex(aCoreIndex);
967 if (aValueTypeIndex != -1) {
968 aBindingIndex = QQmlPropertyIndex(aCoreIndex, aValueTypeIndex);
969 }
else if (valueTypeIndex != -1) {
970 aBindingIndex = QQmlPropertyIndex(aCoreIndex, valueTypeIndex);
973 findAliasTarget(aObject, aBindingIndex, targetObject, targetBindingIndex);
979 *targetObject = object;
980 *targetBindingIndex = bindingIndex;
1058void QQmlPropertyPrivate::takeSignalExpression(
const QQmlProperty &that,
1059 QQmlBoundSignalExpression *expr)
1061 if (!(that.type() & QQmlProperty::SignalProperty)) {
1067 if (!that.d->object)
1069 QQmlData *data = QQmlData::get(that.d->object,
nullptr != expr);
1073 QQmlBoundSignal *signalHandler = data->signalHandlers;
1075 while (signalHandler && signalHandler->signalIndex() != QQmlPropertyPrivate::get(that)->signalIndex())
1076 signalHandler = signalHandler->m_nextSignal;
1078 if (signalHandler) {
1079 signalHandler->takeExpression(expr);
1084 int signalIndex = QQmlPropertyPrivate::get(that)->signalIndex();
1085 QQmlBoundSignal *signal =
new QQmlBoundSignal(that.d->object, signalIndex, that.d->object,
1087 signal->takeExpression(expr);
1159QVariant QQmlPropertyPrivate::readValueProperty()
1161 auto doRead = [&](QQmlGadgetPtrWrapper *wrapper) {
1162 wrapper->read(object, core.coreIndex());
1163 return wrapper->readOnGadget(wrapper->property(valueTypeData.coreIndex()));
1166 if (isValueType()) {
1167 if (QQmlGadgetPtrWrapper *wrapper = QQmlGadgetPtrWrapper::instance(engine, core.propType()))
1168 return doRead(wrapper);
1169 if (QQmlValueType *valueType = QQmlMetaType::valueType(core.propType())) {
1170 QQmlGadgetPtrWrapper wrapper(valueType,
nullptr);
1171 return doRead(&wrapper);
1174 }
else if (core.isQList()) {
1175 auto coreMetaType = core.propType();
1178 if (coreMetaType.flags() & QMetaType::IsQmlList) {
1179 QQmlListProperty<QObject> prop;
1180 core.readProperty(object, &prop);
1181 return QVariant::fromValue(QQmlListReferencePrivate::init(prop, coreMetaType));
1184 QVariant result(coreMetaType);
1187 coreMetaType.construct(result.data());
1188 core.readProperty(object, result.data());
1192 }
else if (core.isQObject()) {
1194 QObject *rv =
nullptr;
1195 core.readProperty(object, &rv);
1196 return QVariant::fromValue(rv);
1200 if (!core.propType().isValid())
1201 return object->metaObject()->property(core.coreIndex()).read(object);
1205 void *args[] = {
nullptr, &value, &status };
1206 if (core.propType() == QMetaType::fromType<QVariant>()) {
1209 value = QVariant(core.propType(), (
void*)
nullptr);
1210 args[0] = value.data();
1212 core.readPropertyWithArgs(object, args);
1213 if (core.propType() != QMetaType::fromType<QVariant>() && args[0] != value.data())
1214 return QVariant(QMetaType(core.propType()), args[0]);
1221QList<QUrl> QQmlPropertyPrivate::urlSequence(
const QVariant &value)
1223 if (value.metaType() == QMetaType::fromType<QList<QUrl>>())
1224 return value.value<QList<QUrl> >();
1227 if (value.metaType() == QMetaType::fromType<QUrl>()) {
1228 urls.append(value.toUrl());
1229 }
else if (value.metaType() == QMetaType::fromType<QString>()) {
1230 urls.append(QUrl(value.toString()));
1231 }
else if (value.metaType() == QMetaType::fromType<QByteArray>()) {
1232 urls.append(QUrl(QString::fromUtf8(value.toByteArray())));
1233 }
else if (value.metaType() == QMetaType::fromType<QStringList>()) {
1234 QStringList urlStrings = value.value<QStringList>();
1235 const int urlStringsSize = urlStrings.size();
1236 urls.reserve(urlStringsSize);
1237 for (
int i = 0; i < urlStringsSize; ++i)
1238 urls.append(QUrl(urlStrings.at(i)));
1256bool QQmlPropertyPrivate::writeEnumProperty(
const QMetaProperty &prop,
int idx, QObject *object,
const QVariant &value,
int flags)
1258 if (!object || !prop.isWritable())
1262 if (prop.isEnumType() && v.metaType() != prop.metaType()) {
1263 QMetaEnum menum = prop.enumerator();
1264 if (v.userType() == QMetaType::QString) {
1266 if (prop.isFlagType())
1267 v = QVariant(menum.keysToValue(value.toByteArray(), &ok));
1269 v = QVariant(menum.keyToValue(value.toByteArray(), &ok));
1273 if (!v.convert(prop.metaType()))
1283 void *argv[] = { v.data(), &v, &status, &flags };
1284 QMetaObject::metacall(object, QMetaObject::WriteProperty, idx, argv);
1318 QObject *object,
const QQmlPropertyData &core,
1319 const QQmlRefPointer<QQmlContextData> &context, QQmlPropertyData::WriteFlags flags,
1320 int internalIndex, Op op)
1322 if (QQmlGadgetPtrWrapper *wrapper = context
1323 ? QQmlGadgetPtrWrapper::instance(context->engine(), core.propType())
1325 return changePropertyAndWriteBack(
1326 object, core.coreIndex(), wrapper, flags, internalIndex, op);
1329 if (QQmlValueType *valueType = QQmlMetaType::valueType(core.propType())) {
1330 QQmlGadgetPtrWrapper wrapper(valueType,
nullptr);
1331 return changePropertyAndWriteBack(
1332 object, core.coreIndex(), &wrapper, flags, internalIndex, op);
1338bool QQmlPropertyPrivate::writeValueProperty(
1339 QObject *object,
const QQmlPropertyData &core,
const QQmlPropertyData &valueTypeData,
1340 const QVariant &value,
const QQmlRefPointer<QQmlContextData> &context,
1341 QQmlPropertyData::WriteFlags flags)
1343 removeValuePropertyBinding(object, core, valueTypeData, flags);
1345 if (!valueTypeData.isValid())
1346 return write(object, core, value, context, flags);
1348 return changeThroughGadgetPtrWrapper(
1349 object, core, context, flags | QQmlPropertyData::HasInternalIndex,
1350 valueTypeData.coreIndex(), [&](QQmlGadgetPtrWrapper *wrapper) {
1351 return write(wrapper, valueTypeData, value, context, flags);
1355bool QQmlPropertyPrivate::resetValueProperty(
1356 QObject *object,
const QQmlPropertyData &core,
const QQmlPropertyData &valueTypeData,
1357 const QQmlRefPointer<QQmlContextData> &context, QQmlPropertyData::WriteFlags flags)
1359 removeValuePropertyBinding(object, core, valueTypeData, flags);
1361 if (!valueTypeData.isValid())
1362 return reset(object, core, flags);
1364 return changeThroughGadgetPtrWrapper(
1365 object, core, context, flags | QQmlPropertyData::HasInternalIndex,
1366 valueTypeData.coreIndex(), [&](QQmlGadgetPtrWrapper *wrapper) {
1367 return reset(wrapper, valueTypeData, flags);
1415 QObject *object,
const QQmlPropertyData &property,
const QVariant &value,
1416 QQmlPropertyData::WriteFlags flags, QMetaType propertyMetaType, QMetaType variantMetaType,
1417 bool isUrl, QQmlEnginePrivate *enginePriv) {
1420 || variantMetaType == QMetaType::fromType<QString>()
1421 || propertyMetaType == QMetaType::fromType<QList<QUrl>>()
1422 || property.isQList()) {
1423 return {
false,
false};
1427 switch (propertyMetaType.id()) {
1428 case QMetaType::Bool:
1429 if (value.canConvert(propertyMetaType)) {
1430 bool b = value.toBool();
1431 return {
true, property.writeProperty(object, &b, flags)};
1433 return {
false,
false};
1434 case QMetaType::Int: {
1436 int i = value.toInt(&ok);
1437 return {ok, ok && property.writeProperty(object, &i, flags)};
1439 case QMetaType::UInt: {
1441 uint u = value.toUInt(&ok);
1442 return {ok, ok && property.writeProperty(object, &u, flags)};
1444 case QMetaType::Double: {
1446 double d = value.toDouble(&ok);
1447 return {ok, ok && property.writeProperty(object, &d, flags)};
1449 case QMetaType::Float: {
1451 float f = value.toFloat(&ok);
1452 return {ok, ok && property.writeProperty(object, &f, flags)};
1454 case QMetaType::QString:
1455 if (value.canConvert(propertyMetaType)) {
1456 QString s = value.toString();
1457 return {
true, property.writeProperty(object, &s, flags)};
1459 return {
false,
false};
1460 case QMetaType::QVariantMap:
1461 if (value.canConvert(propertyMetaType)) {
1462 QVariantMap m = value.toMap();
1463 return {
true, property.writeProperty(object, &m, flags)};
1465 return {
false,
false};
1471 QVariant converted = QQmlValueTypeProvider::createValueType(
1472 value, propertyMetaType, enginePriv ? enginePriv->v4Engine.get() :
nullptr);
1473 if (!converted.isValid()) {
1474 converted = QVariant(propertyMetaType);
1475 if (!QMetaType::convert(value.metaType(), value.constData(),
1476 propertyMetaType, converted.data())) {
1477 return {
false,
false};
1480 return {
true, property.writeProperty(object, converted.data(), flags)};
1486 QMetaSequence::Iterable iterable;
1487 if (!QMetaType::convert(metaType, data, QMetaType::fromType<QMetaSequence::Iterable>(), &iterable))
1490 const QMetaSequence metaSequence = iterable.metaContainer();
1492 if (!metaSequence.hasConstIterator()
1493 || !metaSequence.canGetValueAtConstIterator()
1494 || !metaSequence.valueMetaType().flags().testFlag(QMetaType::PointerToQObject)) {
1498 const void *container = iterable.constIterable();
1499 void *it = metaSequence.constBegin(container);
1500 const void *end = metaSequence.constEnd(container);
1501 QObject *o =
nullptr;
1502 while (!metaSequence.compareConstIterator(it, end)) {
1503 metaSequence.valueAtConstIterator(it, &o);
1505 metaSequence.advanceConstIterator(it, 1);
1507 metaSequence.destroyConstIterator(it);
1508 metaSequence.destroyConstIterator(end);
1587 QQmlListProperty<QObject> *prop, QMetaType metaType,
const void *data, DoAppend &&doAppend)
1589 QMetaSequence::Iterable iterable;
1590 if (!QMetaType::convert(metaType, data, QMetaType::fromType<QMetaSequence::Iterable>(), &iterable))
1591 return AssignResult::TypeMismatch;
1593 const QMetaSequence metaSequence = iterable.metaContainer();
1594 if (!metaSequence.hasConstIterator()
1595 || !metaSequence.canGetValueAtConstIterator()
1596 || !metaSequence.valueMetaType().flags().testFlag(QMetaType::PointerToQObject)) {
1597 return AssignResult::TypeMismatch;
1600 const void *container = iterable.constIterable();
1601 const void *end = metaSequence.constEnd(container);
1603 QObject *o =
nullptr;
1605 if (prop->at && prop->count && metaSequence.hasSize()
1606 && metaSequence.size(container) == prop->count(prop)) {
1607 void *it = metaSequence.constBegin(container);
1610 while (!metaSequence.compareConstIterator(it, end)) {
1611 metaSequence.valueAtConstIterator(it, &o);
1612 if (o != prop->at(prop, ++i)) {
1616 metaSequence.advanceConstIterator(it, 1);
1618 metaSequence.destroyConstIterator(it);
1623 void *it = metaSequence.constBegin(container);
1624 while (!metaSequence.compareConstIterator(it, end)) {
1625 metaSequence.valueAtConstIterator(it, &o);
1627 metaSequence.advanceConstIterator(it, 1);
1629 metaSequence.destroyConstIterator(it);
1632 metaSequence.destroyConstIterator(end);
1633 return same ? AssignResult::WasEqual : AssignResult::Copied;
1644QQmlPropertyPrivate::ListCopyResult QQmlPropertyPrivate::convertToQQmlListProperty(QQmlListProperty<QObject> *listProperty, QMetaType actualListType,
const QVariant &value) {
1645 QQmlListProperty<QObject> &prop = *listProperty;
1646 QMetaType listValueType = QQmlMetaType::listValueType(actualListType);
1649 QQmlMetaObject elementMetaObject = QQmlMetaType::rawMetaObjectForType(listValueType);
1651 auto doAppend = [&](QQmlListProperty<QObject> *propPtr, QObject *o) {
1652 if (Q_UNLIKELY(o && (elementMetaObject.isNull()
1653 || !QQmlMetaObject::canConvert(o, elementMetaObject)))) {
1654 qCWarning(lcIncompatibleElement)
1655 <<
"Cannot append" << o <<
"to a QML list of" << listValueType.name();
1658 propPtr->append(propPtr, o);
1662 AssignResult result = AssignResult::TypeMismatch;
1663 QMetaType variantMetaType = value.metaType();
1664 if (variantMetaType == QMetaType::fromType<QQmlListReference>()) {
1665 result = assignListToListProperty(
1666 &prop,
static_cast<
const QQmlListReference *>(value.constData()),
1667 std::move(doAppend));
1668 }
else if (variantMetaType == QMetaType::fromType<QObjectList>()) {
1669 result = assignListToListProperty(
1670 &prop,
static_cast<
const QObjectList *>(value.constData()),
1671 std::move(doAppend));
1672 }
else if (variantMetaType == QMetaType::fromType<QVariantList>()) {
1673 result = assignListToListProperty(
1674 &prop,
static_cast<
const QVariantList *>(value.constData()),
1675 std::move(doAppend));
1677 result = assignMetaContainerToListProperty(
1678 &prop, variantMetaType, value.data(), doAppend);
1679 if (result == AssignResult::TypeMismatch) {
1681 doAppend(&prop, QQmlMetaType::toQObject(value));
1682 result = AssignResult::Copied;
1692 const QQmlPropertyData &property,
1693 const QMetaType propertyMetaType,
const QVariant &value)
1695 Q_ASSERT(propertyMetaType.flags() & QMetaType::IsQmlList);
1696 QQmlListProperty<QObject> prop;
1697 property.readProperty(object, &prop);
1700 if (!prop.clear || !prop.append)
1703 const bool useNonsignalingListOps = prop.clear == &QQmlVMEMetaObject::list_clear
1704 && prop.append == &QQmlVMEMetaObject::list_append;
1705 if (useNonsignalingListOps) {
1706 prop.clear = &QQmlVMEMetaObject::list_clear_nosignal;
1707 prop.append = &QQmlVMEMetaObject::list_append_nosignal;
1710 auto result = QQmlPropertyPrivate::convertToQQmlListProperty(&prop, propertyMetaType, value);
1712 if (useNonsignalingListOps && result == QQmlPropertyPrivate::ListCopyResult::Copied) {
1713 Q_ASSERT(QQmlVMEMetaObject::get(object));
1714 QQmlVMEResolvedList(&prop).activateSignal();
1717 return result != QQmlPropertyPrivate::ListCopyResult::TypeMismatch;
1723 const QQmlPropertyData &property, QQmlPropertyData::WriteFlags flags,
1724 const QMetaType propertyMetaType,
const QMetaType variantMetaType,
const QVariant &value,
1727 Q_ASSERT(!(propertyMetaType.flags() & QMetaType::IsQmlList));
1728 if (variantMetaType == propertyMetaType) {
1730 return property.writeProperty(object, v.data(), flags);
1732 QVariant outputList(propertyMetaType);
1733 const QQmlType type = QQmlMetaType::qmlListType(propertyMetaType);
1734 const QMetaSequence outputSequence = type.listMetaSequence();
1735 if (!outputSequence.canAddValue())
1736 return property.writeProperty(object, outputList.data(), flags);
1738 const QMetaType outputElementMetaType = outputSequence.valueMetaType();
1739 const bool outputIsQVariant = (outputElementMetaType == QMetaType::fromType<QVariant>());
1741 QMetaSequence::Iterable inputIterable;
1742 QVariant inputList = value;
1743 if (QMetaType::view(
1744 inputList.metaType(), inputList.data(),
1745 QMetaType::fromType<QMetaSequence::Iterable>(), &inputIterable)) {
1747 const QMetaSequence inputSequence = inputIterable.metaContainer();
1748 const QMetaType inputElementMetaType = inputSequence.valueMetaType();
1749 const bool inputIsQVariant = (inputElementMetaType == QMetaType::fromType<QVariant>());
1751 QVariant outputElement
1752 = outputIsQVariant ? QVariant() : QVariant(outputElementMetaType);
1753 QVariant inputElement
1754 = inputIsQVariant ? QVariant() : QVariant(inputElementMetaType);
1756 void *it = inputSequence.constBegin(inputList.constData());
1757 void *end = inputSequence.constEnd(inputList.constData());
1759 for (; !inputSequence.compareConstIterator(it, end);
1760 inputSequence.advanceConstIterator(it, 1)) {
1762 if (inputIsQVariant)
1763 inputSequence.valueAtIterator(it, &inputElement);
1765 inputSequence.valueAtIterator(it, inputElement.data());
1767 if (outputIsQVariant) {
1768 outputSequence.addValue(outputList.data(), &inputElement);
1769 }
else if (inputElement.metaType() == outputElement.metaType()) {
1770 outputSequence.addValue(outputList.data(), inputElement.constData());
1773 inputElement.metaType(), inputElement.constData(),
1774 outputElementMetaType, outputElement.data());
1775 outputSequence.addValue(outputList.data(), outputElement.constData());
1779 inputSequence.destroyConstIterator(it);
1780 inputSequence.destroyConstIterator(end);
1781 }
else if (outputIsQVariant) {
1782 outputSequence.addValue(outputList.data(), &value);
1783 }
else if (outputElementMetaType == value.metaType()){
1784 outputSequence.addValue(outputList.data(), value.constData());
1786 QVariant output(outputElementMetaType);
1788 value.metaType(), value.constData(), outputElementMetaType, output.data());
1789 outputSequence.addValue(outputList.data(), output.constData());
1792 return property.writeProperty(object, outputList.data(), flags);
1796QVariant QQmlPropertyPrivate::convertToWriteTargetType(
const QVariant &value, QMetaType targetMetaType){
1797 QMetaType sourceMetaType = value.metaType();
1798 Q_ASSERT(sourceMetaType != targetMetaType);
1801 if (sourceMetaType == QMetaType::fromType<QString>()) {
1803 QVariant converted = QQmlStringConverters::variantFromString(value.toString(), targetMetaType, &ok);
1810 if (QMetaType::canConvert(sourceMetaType, targetMetaType))
1811 if (QVariant copy = value; copy.convert(targetMetaType))
1819
1820
1821 QMetaSequence::Iterable iterable;
1822 QVariant sequenceVariant = QVariant(targetMetaType);
1823 if (QMetaType::view(
1824 targetMetaType, sequenceVariant.data(),
1825 QMetaType::fromType<QMetaSequence::Iterable>(),
1827 const QMetaSequence propertyMetaSequence = iterable.metaContainer();
1828 if (propertyMetaSequence.canAddValueAtEnd()) {
1829 const QMetaType elementMetaType = propertyMetaSequence.valueMetaType();
1830 void *propertyContainer = iterable.mutableIterable();
1832 if (sourceMetaType == elementMetaType) {
1833 propertyMetaSequence.addValueAtEnd(propertyContainer, value.constData());
1834 return sequenceVariant;
1835 }
else if (sourceMetaType == QMetaType::fromType<QVariantList>()) {
1836 const QVariantList list = value.value<QVariantList>();
1837 for (
const QVariant &valueElement : list) {
1838 if (valueElement.metaType() == elementMetaType) {
1839 propertyMetaSequence.addValueAtEnd(
1840 propertyContainer, valueElement.constData());
1842 QVariant converted(elementMetaType);
1844 valueElement.metaType(), valueElement.constData(),
1845 elementMetaType, converted.data());
1846 propertyMetaSequence.addValueAtEnd(
1847 propertyContainer, converted.constData());
1850 return sequenceVariant;
1851 }
else if (elementMetaType.flags().testFlag(QMetaType::PointerToQObject)) {
1852 const QMetaObject *elementMetaObject = elementMetaType.metaObject();
1853 Q_ASSERT(elementMetaObject);
1855 const auto doAppend = [&](QObject *o) {
1856 QObject *casted = elementMetaObject->cast(o);
1857 propertyMetaSequence.addValueAtEnd(propertyContainer, &casted);
1860 if (sourceMetaType.flags().testFlag(QMetaType::PointerToQObject)) {
1861 doAppend(*
static_cast<QObject *
const *>(value.data()));
1862 return sequenceVariant;
1863 }
else if (sourceMetaType == QMetaType::fromType<QQmlListReference>()) {
1864 const QQmlListReference *reference
1865 =
static_cast<
const QQmlListReference *>(value.constData());
1866 Q_ASSERT(elementMetaObject);
1867 for (
int i = 0, end = reference->size(); i < end; ++i)
1868 doAppend(reference->at(i));
1869 return sequenceVariant;
1870 }
else if (!iterateQObjectContainer(
1871 sourceMetaType, value.data(), doAppend)) {
1872 doAppend(QQmlMetaType::toQObject(value));
1875 QVariant converted = value;
1876 if (converted.convert(elementMetaType)) {
1877 propertyMetaSequence.addValueAtEnd(propertyContainer, converted.constData());
1878 return sequenceVariant;
1885 if (QQmlMetaType::isInterface(targetMetaType)) {
1886 auto valueAsQObject = qvariant_cast<QObject *>(value);
1888 if (
void *iface = valueAsQObject
1889 ? valueAsQObject->qt_metacast(QQmlMetaType::interfaceIId(targetMetaType))
1894 return QVariant(targetMetaType, &iface);
1900bool QQmlPropertyPrivate::write(
1901 QObject *object,
const QQmlPropertyData &property,
const QVariant &value,
1902 const QQmlRefPointer<QQmlContextData> &context, QQmlPropertyData::WriteFlags flags)
1904 const QMetaType propertyMetaType = property.propType();
1905 const QMetaType variantMetaType = value.metaType();
1907 const BindingFixer bindingFixer(object, property, flags);
1910 if (property.isResettable() && !value.isValid()) {
1911 property.resetProperty(object, flags);
1915 if (property.isEnum()) {
1916 QMetaProperty prop = object->metaObject()->property(property.coreIndex());
1919 if (variantMetaType == QMetaType::fromType<
double>()) {
1921 double fractional = std::modf(value.toDouble(), &integral);
1922 if (qFuzzyIsNull(fractional))
1923 v.convert(QMetaType::fromType<qint32>());
1925 return writeEnumProperty(prop, property.coreIndex(), object, v, flags);
1928 QQmlEnginePrivate *enginePriv = QQmlEnginePrivate::get(context);
1929 const bool isUrl = propertyMetaType == QMetaType::fromType<QUrl>();
1932 if (tryAssignBinding(object, property, value, variantMetaType))
1936 if (propertyMetaType == variantMetaType && !isUrl
1937 && propertyMetaType != QMetaType::fromType<QList<QUrl>>() && !property.isQList()) {
1938 return property.writeProperty(object,
const_cast<
void *>(value.constData()), flags);
1939 }
else if (property.isQObject()) {
1940 QVariant val = value;
1942 if (variantMetaType == QMetaType::fromType<std::nullptr_t>()) {
1945 varType = QMetaType::fromType<QObject*>();
1946 val = QVariant(varType,
nullptr);
1948 varType = variantMetaType;
1951 if (!varType.flags().testFlag(QMetaType::PointerToQObject))
1954 QObject *o = *
static_cast<QObject *
const *>(val.constData());
1958 return QQmlMetaType::canConvert(o, propertyMetaType)
1959 && property.writeProperty(object, &o, flags);
1965 const QQmlMetaObject valMo = rawMetaObjectForType(varType);
1966 const QQmlMetaObject propMo = rawMetaObjectForType(propertyMetaType);
1967 if (QQmlMetaObject::canConvert(valMo, propMo) || QQmlMetaObject::canConvert(propMo, valMo))
1968 return property.writeProperty(object, &o, flags);
1970 }
else if (ConvertAndAssignResult result = tryConvertAndAssign(
1971 object, property, value, flags, propertyMetaType, variantMetaType, isUrl,
1973 return result.couldWrite;
1974 }
else if (propertyMetaType == QMetaType::fromType<QVariant>()) {
1975 return property.writeProperty(object,
const_cast<QVariant *>(&value), flags);
1978 if (variantMetaType == QMetaType::fromType<QUrl>()) {
1980 if (compatResolveUrlsOnAssigment() && context && u.isRelative() && !u.isEmpty())
1981 u = context->resolvedUrl(u);
1983 else if (variantMetaType == QMetaType::fromType<QByteArray>())
1984 u = QUrl(QString::fromUtf8(value.toByteArray()));
1985 else if (variantMetaType == QMetaType::fromType<QString>())
1986 u = QUrl(value.toString());
1990 return property.writeProperty(object, &u, flags);
1991 }
else if (propertyMetaType == QMetaType::fromType<QList<QUrl>>()) {
1992 QList<QUrl> urlSeq = compatResolveUrlsOnAssigment()
1993 ? urlSequence(value, context)
1994 : urlSequence(value);
1995 return property.writeProperty(object, &urlSeq, flags);
1996 }
else if (property.isQList()) {
1997 if (propertyMetaType.flags() & QMetaType::IsQmlList) {
1998 return assignToQQmlListProperty(object, property, propertyMetaType, value);
2000 return assignToListProperty(property, flags, propertyMetaType, variantMetaType, value, object);
2002 }
else if (enginePriv && propertyMetaType == QMetaType::fromType<QJSValue>()) {
2004 QJSValue jsValue = QJSValuePrivate::fromReturnedValue(
2005 enginePriv->v4Engine->metaTypeToJS(variantMetaType, value.constData()));
2006 return property.writeProperty(object, &jsValue, flags);
2008 Q_ASSERT(variantMetaType != propertyMetaType);
2010 QVariant converted = convertToWriteTargetType(value, propertyMetaType);
2011 if (converted.isValid()) {
2012 return property.writeProperty(object,
const_cast<
void *>(converted.constData()), flags);