/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtQml module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or (at your option) the GNU General ** Public license version 3 or any later version approved by the KDE Free ** Qt Foundation. The licenses are as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-2.0.html and ** https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "qqmlopenmetaobject_p.h" #include #include #include #include #include #include QT_BEGIN_NAMESPACE class QQmlOpenMetaObjectTypePrivate { public: QQmlOpenMetaObjectTypePrivate() : mem(0), cache(0), engine(0) {} void init(const QMetaObject *metaObj); int propertyOffset; int signalOffset; QHash names; QMetaObjectBuilder mob; QMetaObject *mem; QQmlPropertyCache *cache; QQmlEngine *engine; QSet referers; }; QQmlOpenMetaObjectType::QQmlOpenMetaObjectType(const QMetaObject *base, QQmlEngine *engine) : QQmlCleanup(engine), d(new QQmlOpenMetaObjectTypePrivate) { d->engine = engine; d->init(base); } QQmlOpenMetaObjectType::~QQmlOpenMetaObjectType() { if (d->mem) free(d->mem); if (d->cache) d->cache->release(); delete d; } void QQmlOpenMetaObjectType::clear() { d->engine = 0; } int QQmlOpenMetaObjectType::propertyOffset() const { return d->propertyOffset; } int QQmlOpenMetaObjectType::signalOffset() const { return d->signalOffset; } int QQmlOpenMetaObjectType::propertyCount() const { return d->names.count(); } QByteArray QQmlOpenMetaObjectType::propertyName(int idx) const { Q_ASSERT(idx >= 0 && idx < d->names.count()); return d->mob.property(idx).name(); } QMetaObject *QQmlOpenMetaObjectType::metaObject() const { return d->mem; } void QQmlOpenMetaObjectType::createProperties(const QVector &names) { for (int i = 0; i < names.count(); ++i) { const QByteArray &name = names.at(i); const int id = d->mob.propertyCount(); d->mob.addSignal("__" + QByteArray::number(id) + "()"); QMetaPropertyBuilder build = d->mob.addProperty(name, "QVariant", id); propertyCreated(id, build); d->names.insert(name, id); } free(d->mem); d->mem = d->mob.toMetaObject(); QSet::iterator it = d->referers.begin(); while (it != d->referers.end()) { QQmlOpenMetaObject *omo = *it; *static_cast(omo) = *d->mem; if (d->cache) d->cache->update(omo); ++it; } } int QQmlOpenMetaObjectType::createProperty(const QByteArray &name) { int id = d->mob.propertyCount(); d->mob.addSignal("__" + QByteArray::number(id) + "()"); QMetaPropertyBuilder build = d->mob.addProperty(name, "QVariant", id); propertyCreated(id, build); free(d->mem); d->mem = d->mob.toMetaObject(); d->names.insert(name, id); QSet::iterator it = d->referers.begin(); while (it != d->referers.end()) { QQmlOpenMetaObject *omo = *it; *static_cast(omo) = *d->mem; if (d->cache) d->cache->update(omo); ++it; } return d->propertyOffset + id; } void QQmlOpenMetaObjectType::propertyCreated(int id, QMetaPropertyBuilder &builder) { if (d->referers.count()) (*d->referers.begin())->propertyCreated(id, builder); } void QQmlOpenMetaObjectTypePrivate::init(const QMetaObject *metaObj) { if (!mem) { mob.setSuperClass(metaObj); mob.setClassName(metaObj->className()); mob.setFlags(QMetaObjectBuilder::DynamicMetaObject); mem = mob.toMetaObject(); propertyOffset = mem->propertyOffset(); signalOffset = mem->methodOffset(); } } //---------------------------------------------------------------------------- class QQmlOpenMetaObjectPrivate { public: QQmlOpenMetaObjectPrivate(QQmlOpenMetaObject *_q) : q(_q), parent(0), type(0), cacheProperties(false) {} inline QPair &getDataRef(int idx) { while (data.count() <= idx) data << QPair(QVariant(), false); return data[idx]; } inline QVariant &getData(int idx) { QPair &prop = getDataRef(idx); if (!prop.second) { prop.first = q->initialValue(idx); prop.second = true; } return prop.first; } inline bool hasData(int idx) const { if (idx >= data.count()) return false; return data[idx].second; } bool autoCreate; QQmlOpenMetaObject *q; QAbstractDynamicMetaObject *parent; QList > data; QObject *object; QQmlOpenMetaObjectType *type; bool cacheProperties; }; QQmlOpenMetaObject::QQmlOpenMetaObject(QObject *obj, const QMetaObject *base, bool automatic) : d(new QQmlOpenMetaObjectPrivate(this)) { d->autoCreate = automatic; d->object = obj; d->type = new QQmlOpenMetaObjectType(base ? base : obj->metaObject(), 0); d->type->d->referers.insert(this); QObjectPrivate *op = QObjectPrivate::get(obj); d->parent = static_cast(op->metaObject); *static_cast(this) = *d->type->d->mem; op->metaObject = this; } QQmlOpenMetaObject::QQmlOpenMetaObject(QObject *obj, QQmlOpenMetaObjectType *type, bool automatic) : d(new QQmlOpenMetaObjectPrivate(this)) { d->autoCreate = automatic; d->object = obj; d->type = type; d->type->addref(); d->type->d->referers.insert(this); QObjectPrivate *op = QObjectPrivate::get(obj); d->parent = static_cast(op->metaObject); *static_cast(this) = *d->type->d->mem; op->metaObject = this; } QQmlOpenMetaObject::~QQmlOpenMetaObject() { if (d->parent) delete d->parent; d->type->d->referers.remove(this); d->type->release(); delete d; } QQmlOpenMetaObjectType *QQmlOpenMetaObject::type() const { return d->type; } void QQmlOpenMetaObject::emitPropertyNotification(const QByteArray &propertyName) { QHash::ConstIterator iter = d->type->d->names.constFind(propertyName); if (iter == d->type->d->names.constEnd()) return; activate(d->object, *iter + d->type->d->signalOffset, 0); } int QQmlOpenMetaObject::metaCall(QObject *o, QMetaObject::Call c, int id, void **a) { Q_ASSERT(d->object == o); if (( c == QMetaObject::ReadProperty || c == QMetaObject::WriteProperty) && id >= d->type->d->propertyOffset) { int propId = id - d->type->d->propertyOffset; if (c == QMetaObject::ReadProperty) { propertyRead(propId); *reinterpret_cast(a[0]) = d->getData(propId); } else if (c == QMetaObject::WriteProperty) { if (propId >= d->data.count() || d->data.at(propId).first != *reinterpret_cast(a[0])) { propertyWrite(propId); QPair &prop = d->getDataRef(propId); prop.first = propertyWriteValue(propId, *reinterpret_cast(a[0])); prop.second = true; propertyWritten(propId); activate(o, d->type->d->signalOffset + propId, 0); } } return -1; } else { if (d->parent) return d->parent->metaCall(o, c, id, a); else return o->qt_metacall(c, id, a); } } QAbstractDynamicMetaObject *QQmlOpenMetaObject::parent() const { return d->parent; } QVariant QQmlOpenMetaObject::value(int id) const { return d->getData(id); } void QQmlOpenMetaObject::setValue(int id, const QVariant &value) { QPair &prop = d->getDataRef(id); prop.first = propertyWriteValue(id, value); prop.second = true; activate(d->object, id + d->type->d->signalOffset, 0); } QVariant QQmlOpenMetaObject::value(const QByteArray &name) const { QHash::ConstIterator iter = d->type->d->names.constFind(name); if (iter == d->type->d->names.cend()) return QVariant(); return d->getData(*iter); } QVariant &QQmlOpenMetaObject::operator[](const QByteArray &name) { QHash::ConstIterator iter = d->type->d->names.constFind(name); Q_ASSERT(iter != d->type->d->names.cend()); return d->getData(*iter); } QVariant &QQmlOpenMetaObject::operator[](int id) { return d->getData(id); } bool QQmlOpenMetaObject::setValue(const QByteArray &name, const QVariant &val) { QHash::ConstIterator iter = d->type->d->names.constFind(name); int id = -1; if (iter == d->type->d->names.cend()) { id = createProperty(name.constData(), "") - d->type->d->propertyOffset; } else { id = *iter; } if (id >= 0) { QVariant &dataVal = d->getData(id); if (dataVal == val) return false; dataVal = val; activate(d->object, id + d->type->d->signalOffset, 0); return true; } return false; } // returns true if this value has been initialized by a call to either value() or setValue() bool QQmlOpenMetaObject::hasValue(int id) const { return d->hasData(id); } void QQmlOpenMetaObject::setCached(bool c) { if (c == d->cacheProperties || !d->type->d->engine) return; d->cacheProperties = c; QQmlData *qmldata = QQmlData::get(d->object, true); if (d->cacheProperties) { if (!d->type->d->cache) d->type->d->cache = new QQmlPropertyCache(QV8Engine::getV4(d->type->d->engine), this); qmldata->propertyCache = d->type->d->cache; d->type->d->cache->addref(); } else { if (d->type->d->cache) d->type->d->cache->release(); qmldata->propertyCache = 0; } } int QQmlOpenMetaObject::createProperty(const char *name, const char *) { if (d->autoCreate) { int result = d->type->createProperty(name); if (QQmlData *ddata = QQmlData::get(d->object, /*create*/false)) { if (ddata->propertyCache) { ddata->propertyCache->release(); ddata->propertyCache = 0; } } return result; } else return -1; } void QQmlOpenMetaObject::propertyRead(int) { } void QQmlOpenMetaObject::propertyWrite(int) { } QVariant QQmlOpenMetaObject::propertyWriteValue(int, const QVariant &value) { return value; } void QQmlOpenMetaObject::propertyWritten(int) { } void QQmlOpenMetaObject::propertyCreated(int, QMetaPropertyBuilder &) { } QVariant QQmlOpenMetaObject::initialValue(int) { return QVariant(); } int QQmlOpenMetaObject::count() const { return d->type->d->names.count(); } QByteArray QQmlOpenMetaObject::name(int idx) const { Q_ASSERT(idx >= 0 && idx < d->type->d->names.count()); return d->type->d->mob.property(idx).name(); } QObject *QQmlOpenMetaObject::object() const { return d->object; } QT_END_NAMESPACE