/**************************************************************************** ** ** 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$ ** ****************************************************************************/ #ifndef QJSENGINE_H #define QJSENGINE_H #include #include #include #include #include #include #include QT_BEGIN_NAMESPACE template inline T qjsvalue_cast(const QJSValue &); class QJSEnginePrivate; class Q_QML_EXPORT QJSEngine : public QObject { Q_OBJECT Q_PROPERTY(QString uiLanguage READ uiLanguage WRITE setUiLanguage NOTIFY uiLanguageChanged) public: QJSEngine(); explicit QJSEngine(QObject *parent); ~QJSEngine() override; QJSValue globalObject() const; QJSValue evaluate(const QString &program, const QString &fileName = QString(), int lineNumber = 1, QStringList *exceptionStackTrace = nullptr); QJSValue importModule(const QString &fileName); bool registerModule(const QString &moduleName, const QJSValue &value); QJSValue newObject(); QJSValue newSymbol(const QString &name); QJSValue newArray(uint length = 0); QJSValue newQObject(QObject *object); QJSValue newQMetaObject(const QMetaObject* metaObject); template QJSValue newQMetaObject() { return newQMetaObject(&T::staticMetaObject); } QJSValue newErrorObject(QJSValue::ErrorType errorType, const QString &message = QString()); template inline QJSValue toScriptValue(const T &value) { return create(QMetaType::fromType(), &value); } template inline QJSManagedValue toManagedValue(const T &value) { return createManaged(QMetaType::fromType(), &value); } template inline T fromScriptValue(const QJSValue &value) { return qjsvalue_cast(value); } template inline T fromManagedValue(const QJSManagedValue &value) { return qjsvalue_cast(value); } template inline T fromVariant(const QVariant &value) { if constexpr (std::is_same_v) return value; const QMetaType targetType = QMetaType::fromType(); if (value.metaType() == targetType) return *reinterpret_cast(value.constData()); if constexpr (std::is_same_v> const *>) { using nonConstT = std::remove_const_t> *; const QMetaType nonConstTargetType = QMetaType::fromType(); if (value.metaType() == nonConstTargetType) return *reinterpret_cast(value.constData()); } { T t{}; if (convertVariant(value, QMetaType::fromType(), &t)) return t; QMetaType::convert(value.metaType(), value.constData(), targetType, &t); return t; } } void collectGarbage(); enum ObjectOwnership { CppOwnership, JavaScriptOwnership }; static void setObjectOwnership(QObject *, ObjectOwnership); static ObjectOwnership objectOwnership(QObject *); enum Extension { TranslationExtension = 0x1, ConsoleExtension = 0x2, GarbageCollectionExtension = 0x4, AllExtensions = 0xffffffff }; Q_DECLARE_FLAGS(Extensions, Extension) void installExtensions(Extensions extensions, const QJSValue &object = QJSValue()); void setInterrupted(bool interrupted); bool isInterrupted() const; QV4::ExecutionEngine *handle() const { return m_v4Engine; } void throwError(const QString &message); void throwError(QJSValue::ErrorType errorType, const QString &message = QString()); void throwError(const QJSValue &error); bool hasError() const; QJSValue catchError(); QString uiLanguage() const; void setUiLanguage(const QString &language); Q_SIGNALS: void uiLanguageChanged(); private: QJSManagedValue createManaged(QMetaType type, const void *ptr); QJSValue create(QMetaType type, const void *ptr); #if QT_VERSION < QT_VERSION_CHECK(7,0,0) QJSValue create(int id, const void *ptr); // only there for BC reasons #endif static bool convertManaged(const QJSManagedValue &value, int type, void *ptr); static bool convertManaged(const QJSManagedValue &value, QMetaType type, void *ptr); static bool convertV2(const QJSValue &value, int type, void *ptr); static bool convertV2(const QJSValue &value, QMetaType metaType, void *ptr); bool convertVariant(const QVariant &value, QMetaType metaType, void *ptr); template friend inline T qjsvalue_cast(const QJSValue &); template friend inline T qjsvalue_cast(const QJSManagedValue &); protected: QJSEngine(QJSEnginePrivate &dd, QObject *parent = nullptr); private: QV4::ExecutionEngine *m_v4Engine; Q_DISABLE_COPY(QJSEngine) Q_DECLARE_PRIVATE(QJSEngine) }; Q_DECLARE_OPERATORS_FOR_FLAGS(QJSEngine::Extensions) template T qjsvalue_cast(const QJSValue &value) { T t; if (QJSEngine::convertV2(value, QMetaType::fromType(), &t)) return t; else if (value.isVariant()) return qvariant_cast(value.toVariant()); return T(); } template T qjsvalue_cast(const QJSManagedValue &value) { { T t; if (QJSEngine::convertManaged(value, QMetaType::fromType(), &t)) return t; } return qvariant_cast(value.toVariant()); } template <> inline QVariant qjsvalue_cast(const QJSValue &value) { return value.toVariant(); } template <> inline QVariant qjsvalue_cast(const QJSManagedValue &value) { return value.toVariant(); } Q_QML_EXPORT QJSEngine *qjsEngine(const QObject *); QT_END_NAMESPACE #endif // QJSENGINE_H