Files
qt-creator/src/libs/qmljs/qmljsdocument.h
Christian Kamm f64146741d QmlJS: Use "import" commands from qmldir files
qmldir files can have lines like "import QtQml". These were already
parsed, but not stored in LibraryInfo. Store them.

When imports are resolved in Link and a library has such an import, also
load the module it refers to, with the same version and "as" scope.

Add a test to verify the behavior works.

Change-Id: I80b260bfaa36a9e5de0849fa5632b3361077ef01
Task-number: QTCREATORBUG-23986
Reviewed-by: Ulf Hermann <ulf.hermann@qt.io>
2020-05-21 07:53:44 +00:00

267 lines
7.2 KiB
C++

/****************************************************************************
**
** Copyright (C) 2016 The Qt Company Ltd.
** Contact: https://www.qt.io/licensing/
**
** This file is part of Qt Creator.
**
** 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 General Public License Usage
** Alternatively, this file may be used under the terms of the GNU
** General Public License version 3 as published by the Free Software
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
** 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-3.0.html.
**
****************************************************************************/
#pragma once
#include <QList>
#include <QSharedPointer>
#include <QString>
#include <languageutils/fakemetaobject.h>
#include "qmljsdialect.h"
#include "parser/qmldirparser_p.h"
#include "parser/qmljsastfwd_p.h"
#include "qmljs_global.h"
#include "qmljsconstants.h"
#include "qmljsimportdependencies.h"
namespace QmlJS {
class Bind;
class DiagnosticMessage;
class Engine;
class Snapshot;
class ImportDependencies;
class QMLJS_EXPORT Document
{
Q_DISABLE_COPY(Document)
public:
typedef QSharedPointer<const Document> Ptr;
typedef QSharedPointer<Document> MutablePtr;
protected:
Document(const QString &fileName, Dialect language);
public:
~Document();
static MutablePtr create(const QString &fileName, Dialect language);
Document::Ptr ptr() const;
bool isQmlDocument() const;
Dialect language() const;
void setLanguage(Dialect l);
QString importId() const;
QByteArray fingerprint() const;
AST::UiProgram *qmlProgram() const;
AST::Program *jsProgram() const;
AST::ExpressionNode *expression() const;
AST::Node *ast() const;
const QmlJS::Engine *engine() const;
QList<DiagnosticMessage> diagnosticMessages() const;
QString source() const;
void setSource(const QString &source);
bool parse();
bool parseQml();
bool parseJavaScript();
bool parseExpression();
bool isParsedCorrectly() const
{ return _parsedCorrectly; }
Bind *bind() const;
int editorRevision() const;
void setEditorRevision(int revision);
QString fileName() const;
QString path() const;
QString componentName() const;
QList<SourceLocation> jsDirectives() const;
private:
bool parse_helper(int kind);
private:
QmlJS::Engine *_engine;
AST::Node *_ast;
Bind *_bind;
QList<QmlJS::DiagnosticMessage> _diagnosticMessages;
QString _fileName;
QString _path;
QString _componentName;
QString _source;
QList<SourceLocation> _jsdirectives;
QWeakPointer<Document> _ptr;
QByteArray _fingerprint;
int _editorRevision;
Dialect _language;
bool _parsedCorrectly;
// for documentFromSource
friend class Snapshot;
};
class QMLJS_EXPORT ModuleApiInfo
{
public:
QString uri;
LanguageUtils::ComponentVersion version;
QString cppName;
void addToHash(QCryptographicHash &hash) const;
};
class QMLJS_EXPORT LibraryInfo
{
public:
enum PluginTypeInfoStatus {
NoTypeInfo,
DumpDone,
DumpError,
TypeInfoFileDone,
TypeInfoFileError
};
enum Status {
NotScanned,
NotFound,
Found
};
private:
Status _status = NotScanned;
QList<QmlDirParser::Component> _components;
QList<QmlDirParser::Plugin> _plugins;
QList<QmlDirParser::TypeInfo> _typeinfos;
typedef QList<LanguageUtils::FakeMetaObject::ConstPtr> FakeMetaObjectList;
FakeMetaObjectList _metaObjects;
QList<ModuleApiInfo> _moduleApis;
QStringList _dependencies; // from qmltypes "dependencies: [...]"
QStringList _imports; // from qmldir "import" commands
QByteArray _fingerprint;
PluginTypeInfoStatus _dumpStatus = NoTypeInfo;
QString _dumpError;
public:
LibraryInfo();
explicit LibraryInfo(Status status);
explicit LibraryInfo(const QmlDirParser::TypeInfo &typeInfo);
explicit LibraryInfo(const QmlDirParser &parser, const QByteArray &fingerprint = QByteArray());
~LibraryInfo() = default;
LibraryInfo(const LibraryInfo &other) = default;
QByteArray calculateFingerprint() const;
void updateFingerprint();
QByteArray fingerprint() const
{ return _fingerprint; }
QList<QmlDirParser::Component> components() const
{ return _components; }
QList<QmlDirParser::Plugin> plugins() const
{ return _plugins; }
QList<QmlDirParser::TypeInfo> typeInfos() const
{ return _typeinfos; }
FakeMetaObjectList metaObjects() const
{ return _metaObjects; }
void setMetaObjects(const FakeMetaObjectList &objects)
{ _metaObjects = objects; }
QList<ModuleApiInfo> moduleApis() const
{ return _moduleApis; }
void setModuleApis(const QList<ModuleApiInfo> &apis)
{ _moduleApis = apis; }
QStringList dependencies() const
{ return _dependencies; }
void setDependencies(const QStringList &deps)
{ _dependencies = deps; }
QStringList imports() const
{ return _imports; }
void setImports(const QStringList &imports)
{ _imports = imports; }
bool isValid() const
{ return _status == Found; }
bool wasScanned() const
{ return _status != NotScanned; }
bool wasFound() const
{ return _status != NotFound; }
PluginTypeInfoStatus pluginTypeInfoStatus() const
{ return _dumpStatus; }
QString pluginTypeInfoError() const
{ return _dumpError; }
void setPluginTypeInfoStatus(PluginTypeInfoStatus dumped, const QString &error = QString())
{ _dumpStatus = dumped; _dumpError = error; }
};
class QMLJS_EXPORT Snapshot
{
typedef QHash<QString, Document::Ptr> Base;
QHash<QString, Document::Ptr> _documents;
QHash<QString, QList<Document::Ptr> > _documentsByPath;
QHash<QString, LibraryInfo> _libraries;
ImportDependencies _dependencies;
public:
Snapshot();
~Snapshot();
typedef Base::iterator iterator;
typedef Base::const_iterator const_iterator;
const_iterator begin() const { return _documents.begin(); }
const_iterator end() const { return _documents.end(); }
void insert(const Document::Ptr &document, bool allowInvalid = false);
void insertLibraryInfo(const QString &path, const LibraryInfo &info);
void remove(const QString &fileName);
const ImportDependencies *importDependencies() const;
ImportDependencies *importDependencies();
Document::Ptr document(const QString &fileName) const;
QList<Document::Ptr> documentsInDirectory(const QString &path) const;
LibraryInfo libraryInfo(const QString &path) const;
Document::MutablePtr documentFromSource(const QString &code,
const QString &fileName,
Dialect language) const;
};
} // namespace QmlJS