// Copyright (C) 2016 The Qt Company Ltd. // Copyright (C) 2016 BasysKom GmbH. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only #ifndef QQMLVMEMETAOBJECT_P_H #define QQMLVMEMETAOBJECT_P_H // // W A R N I N G // ------------- // // This file is not part of the Qt API. It exists purely as an // implementation detail. This header file may change from version to // version without notice, or even be removed. // // We mean it. // #include #include #include #include #include #include #include #if QT_CONFIG(regularexpression) #include #endif #include #include #include #include #include #include QT_BEGIN_NAMESPACE class QQmlVMEMetaObject; class QQmlVMEResolvedList { Q_DISABLE_COPY_MOVE(QQmlVMEResolvedList) public: QQmlVMEResolvedList(QQmlListProperty *prop); ~QQmlVMEResolvedList(); QQmlVMEMetaObject *metaObject() const { return m_metaObject; } QV4::Heap::Object *list() const { return m_list; } quintptr id() const { return m_id; } void append(QObject *o) const; void replace(qsizetype i, QObject *o) const; QObject *at(qsizetype i) const; qsizetype size() const { return m_list->arrayData->length(); } void clear() const { QV4::Scope scope(m_list->internalClass->engine); QV4::ScopedObject object(scope, m_list); m_list->arrayData->vtable()->truncate(object, 0); } void removeLast() const { const uint length = m_list->arrayData->length(); if (length == 0) return; QV4::Scope scope(m_list->internalClass->engine); QV4::ScopedObject object(scope, m_list); m_list->arrayData->vtable()->truncate(object, length - 1); } void activateSignal() const; private: QQmlVMEMetaObject *m_metaObject = nullptr; QV4::Heap::Object *m_list = nullptr; quintptr m_id = 0; }; class QQmlVMEVariantQObjectPtr : public QQmlGuard { public: inline QQmlVMEVariantQObjectPtr(); inline void setGuardedValue(QObject *obj, QQmlVMEMetaObject *target, int index); QQmlVMEMetaObject *m_target; int m_index; private: static void objectDestroyedImpl(QQmlGuardImpl *guard); }; class Q_QML_EXPORT QQmlInterceptorMetaObject : public QDynamicMetaObjectData { public: QQmlInterceptorMetaObject(QObject *obj, const QQmlPropertyCache::ConstPtr &cache); ~QQmlInterceptorMetaObject() override; void registerInterceptor(QQmlPropertyIndex index, QQmlPropertyValueInterceptor *interceptor); static QQmlInterceptorMetaObject *get(QObject *obj); QMetaObject *toDynamicMetaObject(QObject *o) override; // Used by auto-tests for inspection QQmlPropertyCache::ConstPtr propertyCache() const { return cache; } bool intercepts(QQmlPropertyIndex propertyIndex) const { for (auto it = interceptors; it; it = it->m_next) { if (it->m_propertyIndex == propertyIndex) return true; } if (auto parentInterceptor = ((parent.isT1() && parent.flag()) ? static_cast(parent.asT1()) : nullptr)) return parentInterceptor->intercepts(propertyIndex); return false; } void invalidate() { metaObject.setTag(MetaObjectInvalid); } QObject *object = nullptr; QQmlPropertyCache::ConstPtr cache; protected: int metaCall(QObject *o, QMetaObject::Call c, int id, void **a) override; bool intercept(QMetaObject::Call c, int id, void **a) { if (!interceptors) return false; switch (c) { case QMetaObject::WriteProperty: if (*reinterpret_cast(a[3]) & QQmlPropertyData::BypassInterceptor) return false; break; case QMetaObject::BindableProperty: break; default: return false; } return doIntercept(c, id, a); } QBiPointer parent; enum MetaObjectValidity { MetaObjectValid, MetaObjectInvalid }; QTaggedPointer metaObject; private: bool doIntercept(QMetaObject::Call c, int id, void **a); QQmlPropertyValueInterceptor *interceptors = nullptr; }; inline QQmlInterceptorMetaObject *QQmlInterceptorMetaObject::get(QObject *obj) { if (obj) { if (QQmlData *data = QQmlData::get(obj)) { if (data->hasInterceptorMetaObject) return static_cast(QObjectPrivate::get(obj)->metaObject); } } return nullptr; } class QQmlVMEMetaObjectEndpoint; class Q_QML_EXPORT QQmlVMEMetaObject : public QQmlInterceptorMetaObject { public: QQmlVMEMetaObject(QV4::ExecutionEngine *engine, QObject *obj, const QQmlPropertyCache::ConstPtr &cache, const QQmlRefPointer &qmlCompilationUnit, int qmlObjectId); ~QQmlVMEMetaObject() override; bool aliasTarget(int index, QObject **target, int *coreIndex, int *valueTypeIndex) const; QV4::ReturnedValue vmeMethod(int index) const; void setVmeMethod(int index, const QV4::Value &function); QV4::ReturnedValue vmeProperty(int index) const; void setVMEProperty(int index, const QV4::Value &v); void connectAliasSignal(int index, bool indexInSignalRange); static inline QQmlVMEMetaObject *get(QObject *o); static QQmlVMEMetaObject *getForProperty(QObject *o, int coreIndex); static QQmlVMEMetaObject *getForMethod(QObject *o, int coreIndex); static QQmlVMEMetaObject *getForSignal(QObject *o, int coreIndex); static void list_append(QQmlListProperty *prop, QObject *o); static void list_clear(QQmlListProperty *prop); static void list_append_nosignal(QQmlListProperty *prop, QObject *o); static void list_clear_nosignal(QQmlListProperty *prop); protected: int metaCall(QObject *o, QMetaObject::Call _c, int _id, void **_a) override; public: QV4::ExecutionEngine *engine; QQmlGuardedContextData ctxt; inline int propOffset() const; inline int methodOffset() const; inline int signalOffset() const; inline int signalCount() const; QQmlVMEMetaObjectEndpoint *aliasEndpoints; QV4::WeakValue propertyAndMethodStorage; QV4::MemberData *propertyAndMethodStorageAsMemberData() const; int readPropertyAsInt(int id) const; bool readPropertyAsBool(int id) const; double readPropertyAsDouble(int id) const; QString readPropertyAsString(int id) const; QSizeF readPropertyAsSizeF(int id) const; QPointF readPropertyAsPointF(int id) const; QUrl readPropertyAsUrl(int id) const; QDate readPropertyAsDate(int id) const; QTime readPropertyAsTime(int id) const; QDateTime readPropertyAsDateTime(int id) const; #if QT_CONFIG(regularexpression) QRegularExpression readPropertyAsRegularExpression(int id) const; #endif QRectF readPropertyAsRectF(int id) const; QObject *readPropertyAsQObject(int id) const; void initPropertyAsList(int id) const; void writeProperty(int id, int v); void writeProperty(int id, bool v); void writeProperty(int id, double v); void writeProperty(int id, const QString& v); template void writeProperty(int id, const VariantCompatible &v) { QV4::MemberData *md = propertyAndMethodStorageAsMemberData(); if (md) { QV4::Scope scope(engine); QV4::Scoped(scope, md)->set( engine, id, engine->newVariantObject( QMetaType::fromType(), &v)); } } void writeProperty(int id, QObject *v); void ensureQObjectWrapper(); void mark(QV4::MarkStack *markStack); void connectAlias(int aliasId); QV4::ReturnedValue method(int) const; QV4::ReturnedValue readVarProperty(int) const; void writeVarProperty(int, const QV4::Value &); QVariant readPropertyAsVariant(int) const; void writeProperty(int, const QVariant &); inline QQmlVMEMetaObject *parentVMEMetaObject() const; void activate(QObject *, int, void **); QList varObjectGuards; QQmlVMEVariantQObjectPtr *getQObjectGuardForProperty(int) const; // keep a reference to the compilation unit in order to still // do property access when the context has been invalidated. QQmlRefPointer compilationUnit; const QV4::CompiledData::Object *compiledObject; }; QQmlVMEMetaObject *QQmlVMEMetaObject::get(QObject *obj) { if (obj) { if (QQmlData *data = QQmlData::get(obj)) { if (data->hasVMEMetaObject) return static_cast(QObjectPrivate::get(obj)->metaObject); } } return nullptr; } int QQmlVMEMetaObject::propOffset() const { return cache->propertyOffset(); } int QQmlVMEMetaObject::methodOffset() const { return cache->methodOffset(); } int QQmlVMEMetaObject::signalOffset() const { return cache->signalOffset(); } int QQmlVMEMetaObject::signalCount() const { return cache->signalCount(); } QQmlVMEMetaObject *QQmlVMEMetaObject::parentVMEMetaObject() const { if (parent.isT1() && parent.flag()) return static_cast(parent.asT1()); return nullptr; } QT_END_NAMESPACE #endif // QQMLVMEMETAOBJECT_P_H