/**************************************************************************** ** ** 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 QQMLIMPORT_P_H #define QQMLIMPORT_P_H #include #include #include #include #include #include #include #include #include #include #include // // 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. // QT_BEGIN_NAMESPACE class QQmlTypeNameCache; class QQmlEngine; class QDir; class QQmlImportNamespace; class QQmlImportsPrivate; class QQmlImportDatabase; class QQmlTypeLoader; class QQmlTypeLoaderQmldirContent; const QLoggingCategory &lcQmlImport(); namespace QQmlImport { enum RecursionRestriction { PreventRecursion, AllowRecursion }; } struct QQmlImportInstance { QString uri; // e.g. QtQuick QString url; // the base path of the import QQmlType containingType; // points to the containing type for inline components QTypeRevision version; // the version imported bool isLibrary; // true means that this is not a file import bool implicitlyImported = false; bool isInlineComponent = false; QQmlDirComponents qmlDirComponents; // a copy of the components listed in the qmldir QQmlDirScripts qmlDirScripts; // a copy of the scripts in the qmldir bool setQmldirContent(const QString &resolvedUrl, const QQmlTypeLoaderQmldirContent &qmldir, QQmlImportNamespace *nameSpace, QList *errors); static QQmlDirScripts getVersionedScripts(const QQmlDirScripts &qmldirscripts, QTypeRevision version); bool resolveType(QQmlTypeLoader *typeLoader, const QHashedStringRef &type, QTypeRevision *version_return, QQmlType* type_return, QString *base = nullptr, bool *typeRecursionDetected = nullptr, QQmlType::RegistrationType = QQmlType::AnyRegistrationType, QQmlImport::RecursionRestriction recursionRestriction = QQmlImport::PreventRecursion, QList *errors = nullptr) const; }; class QQmlImportNamespace { public: QQmlImportNamespace() : nextNamespace(nullptr) {} ~QQmlImportNamespace() { qDeleteAll(imports); } QList imports; QQmlImportInstance *findImport(const QString &uri) const; bool resolveType(QQmlTypeLoader *typeLoader, const QHashedStringRef& type, QTypeRevision *version_return, QQmlType* type_return, QString *base = nullptr, QList *errors = nullptr, QQmlType::RegistrationType registrationType = QQmlType::AnyRegistrationType, bool *typeRecursionDeteced = nullptr); // Prefix when used as a qualified import. Otherwise empty. QHashedString prefix; // Used by QQmlImportsPrivate::qualifiedSets // set to this in unqualifiedSet to indicate that the lists of imports needs // to be sorted when an inline component import was added // We can't use flag pointer, as that does not work with QFieldList QQmlImportNamespace *nextNamespace; bool needsSorting() const; void setNeedsSorting(bool needsSorting); }; class Q_QML_PRIVATE_EXPORT QQmlImports { public: enum ImportVersion { FullyVersioned, PartiallyVersioned, Unversioned }; enum ImportFlag { ImportIncomplete = 0x1, ImportLowPrecedence = 0x2, ImportImplicit = 0x4 }; QQmlImports(QQmlTypeLoader *); QQmlImports(const QQmlImports &); ~QQmlImports(); QQmlImports &operator=(const QQmlImports &); void setBaseUrl(const QUrl &url, const QString &urlString = QString()); QUrl baseUrl() const; bool resolveType(const QHashedStringRef &type, QQmlType *type_return, QTypeRevision *version_return, QQmlImportNamespace **ns_return, QList *errors = nullptr, QQmlType::RegistrationType registrationType = QQmlType::AnyRegistrationType, bool *typeRecursionDetected = nullptr) const; bool resolveType(QQmlImportNamespace *, const QHashedStringRef& type, QQmlType *type_return, QTypeRevision *version_return, QQmlType::RegistrationType registrationType = QQmlType::AnyRegistrationType) const; QTypeRevision addImplicitImport( QQmlImportDatabase *importDb, QList *errors); bool addInlineComponentImport(QQmlImportInstance *const importInstance, const QString &name, const QUrl importUrl, QQmlType containingType); QTypeRevision addFileImport( QQmlImportDatabase *, const QString& uri, const QString& prefix, QTypeRevision version, uint flags, QList *errors); QTypeRevision addLibraryImport( QQmlImportDatabase *importDb, const QString &uri, const QString &prefix, QTypeRevision version, const QString &qmldirIdentifier, const QString &qmldirUrl, uint flags, QList *errors); QTypeRevision updateQmldirContent( QQmlImportDatabase *importDb, const QString &uri, const QString &prefix, const QString &qmldirIdentifier, const QString &qmldirUrl, QList *errors); void populateCache(QQmlTypeNameCache *cache) const; struct ScriptReference { QString nameSpace; QString qualifier; QUrl location; }; QList resolvedScripts() const; struct CompositeSingletonReference { QString typeName; QString prefix; QTypeRevision version; }; QList resolvedCompositeSingletons() const; static QStringList completeQmldirPaths(const QString &uri, const QStringList &basePaths, QTypeRevision version); static QString versionString(QTypeRevision version, ImportVersion importVersion); static bool isLocal(const QString &url); static bool isLocal(const QUrl &url); static QUrl urlFromLocalFileOrQrcOrUrl(const QString &); static void setDesignerSupportRequired(bool b); private: friend class QQmlImportDatabase; QQmlImportsPrivate *d; }; class Q_QML_PRIVATE_EXPORT QQmlImportDatabase { Q_DECLARE_TR_FUNCTIONS(QQmlImportDatabase) public: enum PathType { Local, Remote, LocalOrRemote }; enum LocalQmldirSearchLocation { QmldirFileAndCache, QmldirCacheOnly, }; enum LocalQmldirResult { QmldirFound, QmldirNotFound, QmldirInterceptedToRemote, QmldirRejected }; QQmlImportDatabase(QQmlEngine *); ~QQmlImportDatabase(); bool removeDynamicPlugin(const QString &pluginId); QStringList dynamicPlugins() const; QStringList importPathList(PathType type = LocalOrRemote) const; void setImportPathList(const QStringList &paths); void addImportPath(const QString& dir); QStringList pluginPathList() const; void setPluginPathList(const QStringList &paths); void addPluginPath(const QString& path); template LocalQmldirResult locateLocalQmldir( const QString &uri, QTypeRevision version, LocalQmldirSearchLocation location, const Callback &callback); static QTypeRevision lockModule(const QString &uri, const QString &typeNamespace, QTypeRevision version, QList *errors); private: friend class QQmlImportsPrivate; friend class QQmlPluginImporter; QString absoluteFilePath(const QString &path) const; void clearDirCache(); struct QmldirCache { QTypeRevision version; QString qmldirFilePath; QString qmldirPathUrl; QmldirCache *next; }; // Maps from an import to a linked list of qmldir info. // Used in QQmlImportsPrivate::locateQmldir() QStringHash qmldirCache; // XXX thread QStringList filePluginPath; QStringList fileImportPath; QSet modulesForWhichPluginsHaveBeenLoaded; QSet initializedPlugins; QQmlEngine *engine; }; template QQmlImportDatabase::LocalQmldirResult QQmlImportDatabase::locateLocalQmldir( const QString &uri, QTypeRevision version, QQmlImportDatabase::LocalQmldirSearchLocation location, const Callback &callback) { // Check cache first LocalQmldirResult result = QmldirNotFound; QmldirCache *cacheTail = nullptr; QmldirCache **cachePtr = qmldirCache.value(uri); QmldirCache *cacheHead = cachePtr ? *cachePtr : nullptr; if (cacheHead) { cacheTail = cacheHead; do { if (cacheTail->version == version) { if (cacheTail->qmldirFilePath.isEmpty()) { return cacheTail->qmldirPathUrl.isEmpty() ? QmldirNotFound : QmldirInterceptedToRemote; } if (callback(cacheTail->qmldirFilePath, cacheTail->qmldirPathUrl)) return QmldirFound; result = QmldirRejected; } } while (cacheTail->next && (cacheTail = cacheTail->next)); } // Do not try to construct the cache if it already had any entries for the URI. // Otherwise we might duplicate cache entries. if (location == QmldirCacheOnly || result != QmldirNotFound) return result; const bool hasInterceptors = !engine->urlInterceptors().isEmpty(); // Interceptor might redirect remote files to local ones. QStringList localImportPaths = importPathList(hasInterceptors ? LocalOrRemote : Local); // Search local import paths for a matching version const QStringList qmlDirPaths = QQmlImports::completeQmldirPaths( uri, localImportPaths, version); QString qmldirAbsoluteFilePath; for (QString qmldirPath : qmlDirPaths) { if (hasInterceptors) { const QUrl intercepted = engine->interceptUrl( QQmlImports::urlFromLocalFileOrQrcOrUrl(qmldirPath), QQmlAbstractUrlInterceptor::QmldirFile); qmldirPath = QQmlFile::urlToLocalFileOrQrc(intercepted); if (result != QmldirInterceptedToRemote && qmldirPath.isEmpty() && !QQmlFile::isLocalFile(intercepted)) { result = QmldirInterceptedToRemote; } } qmldirAbsoluteFilePath = absoluteFilePath(qmldirPath); if (!qmldirAbsoluteFilePath.isEmpty()) { QString url; const QString absolutePath = qmldirAbsoluteFilePath.left( qmldirAbsoluteFilePath.lastIndexOf(u'/') + 1); if (absolutePath.at(0) == u':') { url = QStringLiteral("qrc") + absolutePath; } else { url = QUrl::fromLocalFile(absolutePath).toString(); // This handles the UNC path case as when the path is retrieved from the QUrl it // will convert the host name from upper case to lower case. So the absoluteFilePath // is changed at this point to make sure it will match later on in that case. if (qmldirAbsoluteFilePath.startsWith(QStringLiteral("//"))) { qmldirAbsoluteFilePath = QUrl::fromLocalFile(qmldirAbsoluteFilePath) .toString(QUrl::RemoveScheme); } } QmldirCache *cache = new QmldirCache; cache->version = version; cache->qmldirFilePath = qmldirAbsoluteFilePath; cache->qmldirPathUrl = url; cache->next = nullptr; if (cacheTail) cacheTail->next = cache; else qmldirCache.insert(uri, cache); cacheTail = cache; if (result != QmldirFound) result = callback(qmldirAbsoluteFilePath, url) ? QmldirFound : QmldirRejected; // Do not return here. Rather, construct the complete cache for this URI. } } // Nothing found? Add an empty cache entry to signal that for further requests. if (result == QmldirNotFound || result == QmldirInterceptedToRemote) { QmldirCache *cache = new QmldirCache; cache->version = version; cache->next = cacheHead; if (result == QmldirInterceptedToRemote) { // The actual value doesn't matter as long as it's not empty. // We only use it to discern QmldirInterceptedToRemote from QmldirNotFound above. cache->qmldirPathUrl = QStringLiteral("intercepted"); } qmldirCache.insert(uri, cache); if (result == QmldirNotFound) { qCDebug(lcQmlImport) << "locateLocalQmldir:" << qPrintable(uri) << "module's qmldir file not found"; } } else { qCDebug(lcQmlImport) << "locateLocalQmldir:" << qPrintable(uri) << "module's qmldir found at" << qmldirAbsoluteFilePath; } return result; } void qmlClearEnginePlugins();// For internal use by qmlClearRegisteredProperties QT_END_NAMESPACE #endif // QQMLIMPORT_P_H