242void QQmlPropertyPrivate::initProperty(QObject *obj,
const QString &name,
243 QQmlPropertyPrivate::InitFlags flags)
245 QQmlRefPointer<QQmlTypeNameCache> typeNameCache = context ? context->imports() :
nullptr;
247 QObject *currentObject = obj;
248 QList<QStringView> path;
249 QStringView terminal(name);
251 if (name.contains(QLatin1Char(
'.'))) {
252 path = QStringView{name}.split(QLatin1Char(
'.'));
253 if (path.isEmpty())
return;
256 for (
int ii = 0; ii < path.size() - 1; ++ii) {
257 const QStringView &pathName = path.at(ii);
261 if (typeNameCache && !pathName.isEmpty() && pathName.at(0).isUpper()) {
262 QQmlTypeLoader *typeLoader = QQmlTypeLoader::get(engine.data());
263 QQmlTypeNameCache::Result r = typeNameCache->query(pathName, typeLoader);
265 if (r.type.isValid()) {
266 QQmlAttachedPropertiesFunc func = r.type.attachedPropertiesFunction(typeLoader);
269 currentObject = qmlAttachedPropertiesObject(currentObject, func);
270 if (!currentObject)
return;
271 }
else if (r.importNamespace) {
272 if (++ii == path.size())
276 r = typeNameCache->query<QQmlTypeNameCache::QueryNamespaced::No>(
277 path.at(ii), r.importNamespace, typeLoader);
279 if (!r.type.isValid())
282 QQmlAttachedPropertiesFunc func = r.type.attachedPropertiesFunction(typeLoader);
286 currentObject = qmlAttachedPropertiesObject(currentObject, func);
290 }
else if (r.scriptIndex != -1) {
293 Q_ASSERT(!
"Unreachable");
300 QQmlPropertyData local;
301 const QQmlPropertyData *property = currentObject
302 ? QQmlPropertyCache::property(currentObject, pathName, context, &local)
308 if (currentObject || !(flags & InitFlag::AllowId))
311 for (
auto idContext = context; idContext; idContext = idContext->parent()) {
312 const int objectId = idContext->propertyIndex(pathName.toString());
313 if (objectId != -1 && objectId < idContext->numIdValues()) {
314 currentObject = idContext->idValue(objectId);
323 }
else if (property->isFunction()) {
327 if (ii == (path.size() - 2) && QQmlMetaType::isValueType(property->propType())) {
329 const QMetaObject *valueTypeMetaObject = QQmlMetaType::metaObjectForValueType(property->propType());
330 if (!valueTypeMetaObject)
return;
332 int idx = valueTypeMetaObject->indexOfProperty(path.last().toUtf8().constData());
333 if (idx == -1)
return;
335 QMetaProperty vtProp = valueTypeMetaObject->property(idx);
337 Q_ASSERT(idx <= 0x0000FFFF);
339 object = currentObject;
341 valueTypeData.setFlags(QQmlPropertyData::flagsForProperty(vtProp));
342 valueTypeData.setPropType(vtProp.metaType());
343 valueTypeData.setCoreIndex(idx);
347 if (!property->isQObject()) {
348 if (
auto asPropertyMap = qobject_cast<QQmlPropertyMap*>(currentObject))
349 currentObject = asPropertyMap->value(path.at(ii).toString()).value<QObject*>();
353 property->readProperty(currentObject, ¤tObject);
356 if (!currentObject)
return;
362 terminal = path.last();
363 }
else if (!currentObject) {
367 auto findSignalInMetaObject = [&](
const QByteArray &signalName) {
368 const QMetaMethod method = findSignalByName(currentObject->metaObject(), signalName);
369 if (!method.isValid())
372 object = currentObject;
377 QQmlData *ddata = QQmlData::get(currentObject,
false);
378 auto findChangeSignal = [&](QStringView signalName) {
379 if (
auto propName = QQmlSignalNames::changedSignalNameToPropertyName(signalName)) {
380 const QQmlPropertyData *d =
381 ddata->propertyCache->property(*propName, currentObject, context);
382 while (d && d->isFunction())
383 d = ddata->propertyCache->overrideData(d);
385 if (d && d->notifyIndex() != -1) {
386 object = currentObject;
387 core = *ddata->propertyCache->signal(d->notifyIndex());
394 const auto findSignal = [&](
const QString &signalName) {
395 if (ddata && ddata->propertyCache) {
397 const QQmlPropertyData *d
398 = ddata->propertyCache->property(signalName, currentObject, context);
403 while (d && !d->isFunction())
404 d = ddata->propertyCache->overrideData(d);
407 object = currentObject;
412 return findChangeSignal(signalName);
415 return findSignalInMetaObject(signalName.toUtf8());
418 auto signalName = QQmlSignalNames::handlerNameToSignalName(terminal);
420 if (findSignal(*signalName))
423 signalName = QQmlSignalNames::badHandlerNameToSignalName(terminal);
425 if (findSignal(*signalName)) {
428 <<
"is not a properly capitalized signal handler name."
429 << QQmlSignalNames::signalNameToHandlerName(*signalName)
430 <<
"would be correct.";
436 if (ddata && ddata->propertyCache) {
437 const QQmlPropertyData *property = ddata->propertyCache->property(
438 terminal, currentObject, context);
441 while (property && !property->isSignal()) {
442 if (!property->isFunction()) {
443 object = currentObject;
445 nameCache = terminal.toString();
448 property = ddata->propertyCache->overrideData(property);
451 if (!(flags & InitFlag::AllowSignal))
455 Q_ASSERT(property->isSignal());
456 object = currentObject;
462 findChangeSignal(terminal);
465 const QByteArray propertyName = terminal.toUtf8();
466 const QMetaProperty prop = findPropertyByName(currentObject->metaObject(), propertyName);
468 if (prop.isValid()) {
469 object = currentObject;
474 if (flags & InitFlag::AllowSignal)
475 findSignalInMetaObject(terminal.toUtf8());
856 QObject *object, QQmlPropertyIndex index,
859 int coreIndex = index.coreIndex();
860 int valueTypeIndex = index.valueTypeIndex();
862 QQmlData *data = QQmlData::get(object,
false);
864 if (!data || !data->hasBindingBit(coreIndex))
867 QQmlAbstractBinding::Ptr oldBinding;
868 oldBinding = data->bindings;
870 while (oldBinding && (oldBinding->targetPropertyIndex().coreIndex() != coreIndex ||
871 oldBinding->targetPropertyIndex().hasValueTypeIndex())) {
872 oldBinding = oldBinding->nextBinding();
877 data->clearBindingBit(coreIndex);
881 if (valueTypeIndex != -1 && oldBinding->kind() == QQmlAbstractBinding::ValueTypeProxy) {
882 oldBinding =
static_cast<QQmlValueTypeProxyBinding *>(oldBinding.data())->binding(index);
887 if (oldBinding->isSticky() && !(flags & QQmlPropertyPrivate::OverrideSticky))
890 if (!(flags & QQmlPropertyPrivate::DontEnable))
891 oldBinding->setEnabled(
false, {});
892 oldBinding->removeFromObject();
924QQmlPropertyPrivate::binding(QObject *object, QQmlPropertyIndex index)
926 auto aliasTarget = findAliasTarget(object, index);
927 object = aliasTarget.targetObject;
928 index = aliasTarget.targetIndex;
930 QQmlData *data = QQmlData::get(object);
934 const int coreIndex = index.coreIndex();
935 const int valueTypeIndex = index.valueTypeIndex();
937 if (coreIndex < 0 || !data->hasBindingBit(coreIndex))
940 QQmlAbstractBinding *binding = data->bindings;
941 while (binding && (binding->targetPropertyIndex().coreIndex() != coreIndex ||
942 binding->targetPropertyIndex().hasValueTypeIndex()))
943 binding = binding->nextBinding();
945 if (binding && valueTypeIndex != -1) {
946 if (binding->kind() == QQmlAbstractBinding::ValueTypeProxy)
947 binding =
static_cast<QQmlValueTypeProxyBinding *>(binding)->binding(index);
953void QQmlPropertyPrivate::findAliasTarget(QObject *object, QQmlPropertyIndex bindingIndex,
954 QObject **targetObject,
955 QQmlPropertyIndex *targetBindingIndex)
957 QQmlData *data = QQmlData::get(object,
false);
959 int coreIndex = bindingIndex.coreIndex();
960 int valueTypeIndex = bindingIndex.valueTypeIndex();
962 const QQmlPropertyData *propertyData =
963 data->propertyCache?data->propertyCache->property(coreIndex):
nullptr;
964 if (propertyData && propertyData->isAlias()) {
965 QQmlVMEMetaObject *vme = QQmlVMEMetaObject::getForProperty(object, coreIndex);
967 QObject *aObject =
nullptr;
int aCoreIndex = -1;
int aValueTypeIndex = -1;
968 if (vme->aliasTarget(coreIndex, &aObject, &aCoreIndex, &aValueTypeIndex)) {
970 Q_ASSERT(valueTypeIndex == -1 || aValueTypeIndex == -1);
972 QQmlPropertyIndex aBindingIndex(aCoreIndex);
973 if (aValueTypeIndex != -1) {
974 aBindingIndex = QQmlPropertyIndex(aCoreIndex, aValueTypeIndex);
975 }
else if (valueTypeIndex != -1) {
976 aBindingIndex = QQmlPropertyIndex(aCoreIndex, valueTypeIndex);
979 findAliasTarget(aObject, aBindingIndex, targetObject, targetBindingIndex);
985 *targetObject = object;
986 *targetBindingIndex = bindingIndex;
1064void QQmlPropertyPrivate::takeSignalExpression(
const QQmlProperty &that,
1065 QQmlBoundSignalExpression *expr)
1067 if (!(that.type() & QQmlProperty::SignalProperty)) {
1073 if (!that.d->object)
1075 QQmlData *data = QQmlData::get(that.d->object,
nullptr != expr);
1079 QQmlBoundSignal *signalHandler = data->signalHandlers;
1081 while (signalHandler && signalHandler->signalIndex() != QQmlPropertyPrivate::get(that)->signalIndex())
1082 signalHandler = signalHandler->m_nextSignal;
1084 if (signalHandler) {
1085 signalHandler->takeExpression(expr);
1090 int signalIndex = QQmlPropertyPrivate::get(that)->signalIndex();
1091 QQmlBoundSignal *signal =
new QQmlBoundSignal(that.d->object, signalIndex, that.d->object,
1093 signal->takeExpression(expr);
1165QVariant QQmlPropertyPrivate::readValueProperty()
1167 auto doRead = [&](QQmlGadgetPtrWrapper *wrapper) {
1168 wrapper->read(object, core.coreIndex());
1169 return wrapper->readOnGadget(wrapper->property(valueTypeData.coreIndex()));
1172 if (isValueType()) {
1173 if (QQmlGadgetPtrWrapper *wrapper = QQmlGadgetPtrWrapper::instance(engine, core.propType()))
1174 return doRead(wrapper);
1175 if (QQmlValueType *valueType = QQmlMetaType::valueType(core.propType())) {
1176 QQmlGadgetPtrWrapper wrapper(valueType,
nullptr);
1177 return doRead(&wrapper);
1180 }
else if (core.isQList()) {
1181 auto coreMetaType = core.propType();
1184 if (coreMetaType.flags() & QMetaType::IsQmlList) {
1185 QQmlListProperty<QObject> prop;
1186 core.readProperty(object, &prop);
1187 return QVariant::fromValue(QQmlListReferencePrivate::init(prop, coreMetaType));
1190 QVariant result(coreMetaType);
1193 coreMetaType.construct(result.data());
1194 core.readProperty(object, result.data());
1198 }
else if (core.isQObject()) {
1200 QObject *rv =
nullptr;
1201 core.readProperty(object, &rv);
1202 return QVariant::fromValue(rv);
1206 if (!core.propType().isValid())
1207 return object->metaObject()->property(core.coreIndex()).read(object);
1211 void *args[] = {
nullptr, &value, &status };
1212 if (core.propType() == QMetaType::fromType<QVariant>()) {
1215 value = QVariant(core.propType(), (
void*)
nullptr);
1216 args[0] = value.data();
1218 core.readPropertyWithArgs(object, args);
1219 if (core.propType() != QMetaType::fromType<QVariant>() && args[0] != value.data())
1220 return QVariant(QMetaType(core.propType()), args[0]);
1227QList<QUrl> QQmlPropertyPrivate::urlSequence(
const QVariant &value)
1229 if (value.metaType() == QMetaType::fromType<QList<QUrl>>())
1230 return value.value<QList<QUrl> >();
1233 if (value.metaType() == QMetaType::fromType<QUrl>()) {
1234 urls.append(value.toUrl());
1235 }
else if (value.metaType() == QMetaType::fromType<QString>()) {
1236 urls.append(QUrl(value.toString()));
1237 }
else if (value.metaType() == QMetaType::fromType<QByteArray>()) {
1238 urls.append(QUrl(QString::fromUtf8(value.toByteArray())));
1239 }
else if (value.metaType() == QMetaType::fromType<QStringList>()) {
1240 QStringList urlStrings = value.value<QStringList>();
1241 const int urlStringsSize = urlStrings.size();
1242 urls.reserve(urlStringsSize);
1243 for (
int i = 0; i < urlStringsSize; ++i)
1244 urls.append(QUrl(urlStrings.at(i)));
1262bool QQmlPropertyPrivate::writeEnumProperty(
const QMetaProperty &prop,
int idx, QObject *object,
const QVariant &value,
int flags)
1264 if (!object || !prop.isWritable())
1268 if (prop.isEnumType() && v.metaType() != prop.metaType()) {
1269 QMetaEnum menum = prop.enumerator();
1270 if (v.userType() == QMetaType::QString) {
1272 if (prop.isFlagType())
1273 v = QVariant(menum.keysToValue(value.toByteArray(), &ok));
1275 v = QVariant(menum.keyToValue(value.toByteArray(), &ok));
1279 if (!v.convert(prop.metaType()))
1289 void *argv[] = { v.data(), &v, &status, &flags };
1290 QMetaObject::metacall(object, QMetaObject::WriteProperty, idx, argv);
1324 QObject *object,
const QQmlPropertyData &core,
1325 const QQmlRefPointer<QQmlContextData> &context, QQmlPropertyData::WriteFlags flags,
1326 int internalIndex, Op op)
1328 if (QQmlGadgetPtrWrapper *wrapper = context
1329 ? QQmlGadgetPtrWrapper::instance(context->engine(), core.propType())
1331 return changePropertyAndWriteBack(
1332 object, core.coreIndex(), wrapper, flags, internalIndex, op);
1335 if (QQmlValueType *valueType = QQmlMetaType::valueType(core.propType())) {
1336 QQmlGadgetPtrWrapper wrapper(valueType,
nullptr);
1337 return changePropertyAndWriteBack(
1338 object, core.coreIndex(), &wrapper, flags, internalIndex, op);
1344bool QQmlPropertyPrivate::writeValueProperty(
1345 QObject *object,
const QQmlPropertyData &core,
const QQmlPropertyData &valueTypeData,
1346 const QVariant &value,
const QQmlRefPointer<QQmlContextData> &context,
1347 QQmlPropertyData::WriteFlags flags)
1349 removeValuePropertyBinding(object, core, valueTypeData, flags);
1351 if (!valueTypeData.isValid())
1352 return write(object, core, value, context, flags);
1354 return changeThroughGadgetPtrWrapper(
1355 object, core, context, flags | QQmlPropertyData::HasInternalIndex,
1356 valueTypeData.coreIndex(), [&](QQmlGadgetPtrWrapper *wrapper) {
1357 return write(wrapper, valueTypeData, value, context, flags);
1361bool QQmlPropertyPrivate::resetValueProperty(
1362 QObject *object,
const QQmlPropertyData &core,
const QQmlPropertyData &valueTypeData,
1363 const QQmlRefPointer<QQmlContextData> &context, QQmlPropertyData::WriteFlags flags)
1365 removeValuePropertyBinding(object, core, valueTypeData, flags);
1367 if (!valueTypeData.isValid())
1368 return reset(object, core, flags);
1370 return changeThroughGadgetPtrWrapper(
1371 object, core, context, flags | QQmlPropertyData::HasInternalIndex,
1372 valueTypeData.coreIndex(), [&](QQmlGadgetPtrWrapper *wrapper) {
1373 return reset(wrapper, valueTypeData, flags);
1421 QObject *object,
const QQmlPropertyData &property,
const QVariant &value,
1422 QQmlPropertyData::WriteFlags flags, QMetaType propertyMetaType, QMetaType variantMetaType,
1423 bool isUrl, QQmlEnginePrivate *enginePriv) {
1426 || variantMetaType == QMetaType::fromType<QString>()
1427 || propertyMetaType == QMetaType::fromType<QList<QUrl>>()
1428 || property.isQList()) {
1429 return {
false,
false};
1433 switch (propertyMetaType.id()) {
1434 case QMetaType::Bool:
1435 if (value.canConvert(propertyMetaType)) {
1436 bool b = value.toBool();
1437 return {
true, property.writeProperty(object, &b, flags)};
1439 return {
false,
false};
1440 case QMetaType::Int: {
1442 int i = value.toInt(&ok);
1443 return {ok, ok && property.writeProperty(object, &i, flags)};
1445 case QMetaType::UInt: {
1447 uint u = value.toUInt(&ok);
1448 return {ok, ok && property.writeProperty(object, &u, flags)};
1450 case QMetaType::Double: {
1452 double d = value.toDouble(&ok);
1453 return {ok, ok && property.writeProperty(object, &d, flags)};
1455 case QMetaType::Float: {
1457 float f = value.toFloat(&ok);
1458 return {ok, ok && property.writeProperty(object, &f, flags)};
1460 case QMetaType::QString:
1461 if (value.canConvert(propertyMetaType)) {
1462 QString s = value.toString();
1463 return {
true, property.writeProperty(object, &s, flags)};
1465 return {
false,
false};
1466 case QMetaType::QVariantMap:
1467 if (value.canConvert(propertyMetaType)) {
1468 QVariantMap m = value.toMap();
1469 return {
true, property.writeProperty(object, &m, flags)};
1471 return {
false,
false};
1477 QVariant converted = QQmlValueTypeProvider::createValueType(
1478 value, propertyMetaType, enginePriv ? enginePriv->v4Engine.get() :
nullptr);
1479 if (!converted.isValid()) {
1480 converted = QVariant(propertyMetaType);
1481 if (!QMetaType::convert(value.metaType(), value.constData(),
1482 propertyMetaType, converted.data())) {
1483 return {
false,
false};
1486 return {
true, property.writeProperty(object, converted.data(), flags)};
1492 QMetaSequence::Iterable iterable;
1493 if (!QMetaType::convert(metaType, data, QMetaType::fromType<QMetaSequence::Iterable>(), &iterable))
1496 const QMetaSequence metaSequence = iterable.metaContainer();
1498 if (!metaSequence.hasConstIterator()
1499 || !metaSequence.canGetValueAtConstIterator()
1500 || !metaSequence.valueMetaType().flags().testFlag(QMetaType::PointerToQObject)) {
1504 const void *container = iterable.constIterable();
1505 void *it = metaSequence.constBegin(container);
1506 const void *end = metaSequence.constEnd(container);
1507 QObject *o =
nullptr;
1508 while (!metaSequence.compareConstIterator(it, end)) {
1509 metaSequence.valueAtConstIterator(it, &o);
1511 metaSequence.advanceConstIterator(it, 1);
1513 metaSequence.destroyConstIterator(it);
1514 metaSequence.destroyConstIterator(end);
1593 QQmlListProperty<QObject> *prop, QMetaType metaType,
const void *data, DoAppend &&doAppend)
1595 QMetaSequence::Iterable iterable;
1596 if (!QMetaType::convert(metaType, data, QMetaType::fromType<QMetaSequence::Iterable>(), &iterable))
1597 return AssignResult::TypeMismatch;
1599 const QMetaSequence metaSequence = iterable.metaContainer();
1600 if (!metaSequence.hasConstIterator()
1601 || !metaSequence.canGetValueAtConstIterator()
1602 || !metaSequence.valueMetaType().flags().testFlag(QMetaType::PointerToQObject)) {
1603 return AssignResult::TypeMismatch;
1606 const void *container = iterable.constIterable();
1607 const void *end = metaSequence.constEnd(container);
1609 QObject *o =
nullptr;
1611 if (prop->at && prop->count && metaSequence.hasSize()
1612 && metaSequence.size(container) == prop->count(prop)) {
1613 void *it = metaSequence.constBegin(container);
1616 while (!metaSequence.compareConstIterator(it, end)) {
1617 metaSequence.valueAtConstIterator(it, &o);
1618 if (o != prop->at(prop, ++i)) {
1622 metaSequence.advanceConstIterator(it, 1);
1624 metaSequence.destroyConstIterator(it);
1629 void *it = metaSequence.constBegin(container);
1630 while (!metaSequence.compareConstIterator(it, end)) {
1631 metaSequence.valueAtConstIterator(it, &o);
1633 metaSequence.advanceConstIterator(it, 1);
1635 metaSequence.destroyConstIterator(it);
1638 metaSequence.destroyConstIterator(end);
1639 return same ? AssignResult::WasEqual : AssignResult::Copied;
1650QQmlPropertyPrivate::ListCopyResult QQmlPropertyPrivate::convertToQQmlListProperty(QQmlListProperty<QObject> *listProperty, QMetaType actualListType,
const QVariant &value) {
1651 QQmlListProperty<QObject> &prop = *listProperty;
1652 QMetaType listValueType = QQmlMetaType::listValueType(actualListType);
1655 QQmlMetaObject elementMetaObject = QQmlMetaType::rawMetaObjectForType(listValueType);
1657 auto doAppend = [&](QQmlListProperty<QObject> *propPtr, QObject *o) {
1658 if (Q_UNLIKELY(o && (elementMetaObject.isNull()
1659 || !QQmlMetaObject::canConvert(o, elementMetaObject)))) {
1660 qCWarning(lcIncompatibleElement)
1661 <<
"Cannot append" << o <<
"to a QML list of" << listValueType.name();
1664 propPtr->append(propPtr, o);
1668 AssignResult result = AssignResult::TypeMismatch;
1669 QMetaType variantMetaType = value.metaType();
1670 if (variantMetaType == QMetaType::fromType<QQmlListReference>()) {
1671 result = assignListToListProperty(
1672 &prop,
static_cast<
const QQmlListReference *>(value.constData()),
1673 std::move(doAppend));
1674 }
else if (variantMetaType == QMetaType::fromType<QObjectList>()) {
1675 result = assignListToListProperty(
1676 &prop,
static_cast<
const QObjectList *>(value.constData()),
1677 std::move(doAppend));
1678 }
else if (variantMetaType == QMetaType::fromType<QVariantList>()) {
1679 result = assignListToListProperty(
1680 &prop,
static_cast<
const QVariantList *>(value.constData()),
1681 std::move(doAppend));
1683 result = assignMetaContainerToListProperty(
1684 &prop, variantMetaType, value.data(), doAppend);
1685 if (result == AssignResult::TypeMismatch) {
1687 doAppend(&prop, QQmlMetaType::toQObject(value));
1688 result = AssignResult::Copied;
1698 const QQmlPropertyData &property,
1699 const QMetaType propertyMetaType,
const QVariant &value)
1701 Q_ASSERT(propertyMetaType.flags() & QMetaType::IsQmlList);
1702 QQmlListProperty<QObject> prop;
1703 property.readProperty(object, &prop);
1706 if (!prop.clear || !prop.append)
1709 const bool useNonsignalingListOps = prop.clear == &QQmlVMEMetaObject::list_clear
1710 && prop.append == &QQmlVMEMetaObject::list_append;
1711 if (useNonsignalingListOps) {
1712 prop.clear = &QQmlVMEMetaObject::list_clear_nosignal;
1713 prop.append = &QQmlVMEMetaObject::list_append_nosignal;
1716 auto result = QQmlPropertyPrivate::convertToQQmlListProperty(&prop, propertyMetaType, value);
1718 if (useNonsignalingListOps && result == QQmlPropertyPrivate::ListCopyResult::Copied) {
1719 Q_ASSERT(QQmlVMEMetaObject::get(object));
1720 QQmlVMEResolvedList(&prop).activateSignal();
1723 return result != QQmlPropertyPrivate::ListCopyResult::TypeMismatch;
1729 const QQmlPropertyData &property, QQmlPropertyData::WriteFlags flags,
1730 const QMetaType propertyMetaType,
const QMetaType variantMetaType,
const QVariant &value,
1733 Q_ASSERT(!(propertyMetaType.flags() & QMetaType::IsQmlList));
1734 if (variantMetaType == propertyMetaType) {
1736 return property.writeProperty(object, v.data(), flags);
1738 QVariant outputList(propertyMetaType);
1739 const QQmlType type = QQmlMetaType::qmlListType(propertyMetaType);
1740 const QMetaSequence outputSequence = type.listMetaSequence();
1741 if (!outputSequence.canAddValue())
1742 return property.writeProperty(object, outputList.data(), flags);
1744 const QMetaType outputElementMetaType = outputSequence.valueMetaType();
1745 const bool outputIsQVariant = (outputElementMetaType == QMetaType::fromType<QVariant>());
1747 QMetaSequence::Iterable inputIterable;
1748 QVariant inputList = value;
1749 if (QMetaType::view(
1750 inputList.metaType(), inputList.data(),
1751 QMetaType::fromType<QMetaSequence::Iterable>(), &inputIterable)) {
1753 const QMetaSequence inputSequence = inputIterable.metaContainer();
1754 const QMetaType inputElementMetaType = inputSequence.valueMetaType();
1755 const bool inputIsQVariant = (inputElementMetaType == QMetaType::fromType<QVariant>());
1757 QVariant outputElement
1758 = outputIsQVariant ? QVariant() : QVariant(outputElementMetaType);
1759 QVariant inputElement
1760 = inputIsQVariant ? QVariant() : QVariant(inputElementMetaType);
1762 void *it = inputSequence.constBegin(inputList.constData());
1763 void *end = inputSequence.constEnd(inputList.constData());
1765 for (; !inputSequence.compareConstIterator(it, end);
1766 inputSequence.advanceConstIterator(it, 1)) {
1768 if (inputIsQVariant)
1769 inputSequence.valueAtIterator(it, &inputElement);
1771 inputSequence.valueAtIterator(it, inputElement.data());
1773 if (outputIsQVariant) {
1774 outputSequence.addValue(outputList.data(), &inputElement);
1775 }
else if (inputElement.metaType() == outputElement.metaType()) {
1776 outputSequence.addValue(outputList.data(), inputElement.constData());
1779 inputElement.metaType(), inputElement.constData(),
1780 outputElementMetaType, outputElement.data());
1781 outputSequence.addValue(outputList.data(), outputElement.constData());
1785 inputSequence.destroyConstIterator(it);
1786 inputSequence.destroyConstIterator(end);
1787 }
else if (outputIsQVariant) {
1788 outputSequence.addValue(outputList.data(), &value);
1789 }
else if (outputElementMetaType == value.metaType()){
1790 outputSequence.addValue(outputList.data(), value.constData());
1792 QVariant output(outputElementMetaType);
1794 value.metaType(), value.constData(), outputElementMetaType, output.data());
1795 outputSequence.addValue(outputList.data(), output.constData());
1798 return property.writeProperty(object, outputList.data(), flags);
1802QVariant QQmlPropertyPrivate::convertToWriteTargetType(
const QVariant &value, QMetaType targetMetaType){
1803 QMetaType sourceMetaType = value.metaType();
1804 Q_ASSERT(sourceMetaType != targetMetaType);
1807 if (sourceMetaType == QMetaType::fromType<QString>()) {
1809 QVariant converted = QQmlStringConverters::variantFromString(value.toString(), targetMetaType, &ok);
1816 if (QMetaType::canConvert(sourceMetaType, targetMetaType))
1817 if (QVariant copy = value; copy.convert(targetMetaType))
1825
1826
1827 QMetaSequence::Iterable iterable;
1828 QVariant sequenceVariant = QVariant(targetMetaType);
1829 if (QMetaType::view(
1830 targetMetaType, sequenceVariant.data(),
1831 QMetaType::fromType<QMetaSequence::Iterable>(),
1833 const QMetaSequence propertyMetaSequence = iterable.metaContainer();
1834 if (propertyMetaSequence.canAddValueAtEnd()) {
1835 const QMetaType elementMetaType = propertyMetaSequence.valueMetaType();
1836 void *propertyContainer = iterable.mutableIterable();
1838 if (sourceMetaType == elementMetaType) {
1839 propertyMetaSequence.addValueAtEnd(propertyContainer, value.constData());
1840 return sequenceVariant;
1841 }
else if (sourceMetaType == QMetaType::fromType<QVariantList>()) {
1842 const QVariantList list = value.value<QVariantList>();
1843 for (
const QVariant &valueElement : list) {
1844 if (valueElement.metaType() == elementMetaType) {
1845 propertyMetaSequence.addValueAtEnd(
1846 propertyContainer, valueElement.constData());
1848 QVariant converted(elementMetaType);
1850 valueElement.metaType(), valueElement.constData(),
1851 elementMetaType, converted.data());
1852 propertyMetaSequence.addValueAtEnd(
1853 propertyContainer, converted.constData());
1856 return sequenceVariant;
1857 }
else if (elementMetaType.flags().testFlag(QMetaType::PointerToQObject)) {
1858 const QMetaObject *elementMetaObject = elementMetaType.metaObject();
1859 Q_ASSERT(elementMetaObject);
1861 const auto doAppend = [&](QObject *o) {
1862 QObject *casted = elementMetaObject->cast(o);
1863 propertyMetaSequence.addValueAtEnd(propertyContainer, &casted);
1866 if (sourceMetaType.flags().testFlag(QMetaType::PointerToQObject)) {
1867 doAppend(*
static_cast<QObject *
const *>(value.data()));
1868 return sequenceVariant;
1869 }
else if (sourceMetaType == QMetaType::fromType<QQmlListReference>()) {
1870 const QQmlListReference *reference
1871 =
static_cast<
const QQmlListReference *>(value.constData());
1872 Q_ASSERT(elementMetaObject);
1873 for (
int i = 0, end = reference->size(); i < end; ++i)
1874 doAppend(reference->at(i));
1875 return sequenceVariant;
1876 }
else if (!iterateQObjectContainer(
1877 sourceMetaType, value.data(), doAppend)) {
1878 doAppend(QQmlMetaType::toQObject(value));
1881 QVariant converted = value;
1882 if (converted.convert(elementMetaType)) {
1883 propertyMetaSequence.addValueAtEnd(propertyContainer, converted.constData());
1884 return sequenceVariant;
1891 if (QQmlMetaType::isInterface(targetMetaType)) {
1892 auto valueAsQObject = qvariant_cast<QObject *>(value);
1894 if (
void *iface = valueAsQObject
1895 ? valueAsQObject->qt_metacast(QQmlMetaType::interfaceIId(targetMetaType))
1900 return QVariant(targetMetaType, &iface);
1906bool QQmlPropertyPrivate::write(
1907 QObject *object,
const QQmlPropertyData &property,
const QVariant &value,
1908 const QQmlRefPointer<QQmlContextData> &context, QQmlPropertyData::WriteFlags flags)
1910 const QMetaType propertyMetaType = property.propType();
1911 const QMetaType variantMetaType = value.metaType();
1913 const BindingFixer bindingFixer(object, property, flags);
1916 if (property.isResettable() && !value.isValid()) {
1917 property.resetProperty(object, flags);
1921 if (property.isEnum()) {
1922 QMetaProperty prop = object->metaObject()->property(property.coreIndex());
1925 if (variantMetaType == QMetaType::fromType<
double>()) {
1927 double fractional = std::modf(value.toDouble(), &integral);
1928 if (qFuzzyIsNull(fractional))
1929 v.convert(QMetaType::fromType<qint32>());
1931 return writeEnumProperty(prop, property.coreIndex(), object, v, flags);
1934 QQmlEnginePrivate *enginePriv = QQmlEnginePrivate::get(context);
1935 const bool isUrl = propertyMetaType == QMetaType::fromType<QUrl>();
1938 if (tryAssignBinding(object, property, value, variantMetaType))
1942 if (propertyMetaType == variantMetaType && !isUrl
1943 && propertyMetaType != QMetaType::fromType<QList<QUrl>>() && !property.isQList()) {
1944 return property.writeProperty(object,
const_cast<
void *>(value.constData()), flags);
1945 }
else if (property.isQObject()) {
1946 QVariant val = value;
1948 if (variantMetaType == QMetaType::fromType<std::nullptr_t>()) {
1951 varType = QMetaType::fromType<QObject*>();
1952 val = QVariant(varType,
nullptr);
1954 varType = variantMetaType;
1957 if (!varType.flags().testFlag(QMetaType::PointerToQObject))
1960 QObject *o = *
static_cast<QObject *
const *>(val.constData());
1964 return QQmlMetaType::canConvert(o, propertyMetaType)
1965 && property.writeProperty(object, &o, flags);
1971 const QQmlMetaObject valMo = rawMetaObjectForType(varType);
1972 const QQmlMetaObject propMo = rawMetaObjectForType(propertyMetaType);
1973 if (QQmlMetaObject::canConvert(valMo, propMo) || QQmlMetaObject::canConvert(propMo, valMo))
1974 return property.writeProperty(object, &o, flags);
1976 }
else if (ConvertAndAssignResult result = tryConvertAndAssign(
1977 object, property, value, flags, propertyMetaType, variantMetaType, isUrl,
1979 return result.couldWrite;
1980 }
else if (propertyMetaType == QMetaType::fromType<QVariant>()) {
1981 return property.writeProperty(object,
const_cast<QVariant *>(&value), flags);
1984 if (variantMetaType == QMetaType::fromType<QUrl>()) {
1986 if (compatResolveUrlsOnAssigment() && context && u.isRelative() && !u.isEmpty())
1987 u = context->resolvedUrl(u);
1989 else if (variantMetaType == QMetaType::fromType<QByteArray>())
1990 u = QUrl(QString::fromUtf8(value.toByteArray()));
1991 else if (variantMetaType == QMetaType::fromType<QString>())
1992 u = QUrl(value.toString());
1996 return property.writeProperty(object, &u, flags);
1997 }
else if (propertyMetaType == QMetaType::fromType<QList<QUrl>>()) {
1998 QList<QUrl> urlSeq = compatResolveUrlsOnAssigment()
1999 ? urlSequence(value, context)
2000 : urlSequence(value);
2001 return property.writeProperty(object, &urlSeq, flags);
2002 }
else if (property.isQList()) {
2003 if (propertyMetaType.flags() & QMetaType::IsQmlList) {
2004 return assignToQQmlListProperty(object, property, propertyMetaType, value);
2006 return assignToListProperty(property, flags, propertyMetaType, variantMetaType, value, object);
2008 }
else if (enginePriv && propertyMetaType == QMetaType::fromType<QJSValue>()) {
2010 QJSValue jsValue = QJSValuePrivate::fromReturnedValue(
2011 enginePriv->v4Engine->metaTypeToJS(variantMetaType, value.constData()));
2012 return property.writeProperty(object, &jsValue, flags);
2014 Q_ASSERT(variantMetaType != propertyMetaType);
2016 QVariant converted = convertToWriteTargetType(value, propertyMetaType);
2017 if (converted.isValid()) {
2018 return property.writeProperty(object,
const_cast<
void *>(converted.constData()), flags);