/**************************************************************************** ** ** 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 "qmldesignercorelib_global.h" #include "import.h" #include "nodelistproperty.h" #include "modelnode.h" #include #include #include #include namespace QmlDesigner { class RewriterView; class DocumentMessage; namespace Internal { class DifferenceHandler; class ReadingContext; class TextToModelMerger { TextToModelMerger(const TextToModelMerger&); TextToModelMerger &operator=(const TextToModelMerger&); public: static QmlJS::Document::MutablePtr createParsedDocument(const QUrl &url, const QString &data, QList *errors); TextToModelMerger(RewriterView *reWriterView); bool isActive() const; void setupImports(const QmlJS::Document::Ptr &doc, DifferenceHandler &differenceHandler); void setupPossibleImports(const QmlJS::Snapshot &snapshot, const QmlJS::ViewerContext &viewContext); void setupUsedImports(); bool load(const QString &data, DifferenceHandler &differenceHandler); RewriterView *view() const { return m_rewriterView; } const QmlJS::ScopeChain *scopeChain() const { return m_scopeChain.data(); } const QmlJS::Document *document() const { return m_document.data(); } const QmlJS::ViewerContext &vContext() const { return m_vContext; } protected: void setActive(bool active); public: void syncNode(ModelNode &modelNode, QmlJS::AST::UiObjectMember *astNode, ReadingContext *context, DifferenceHandler &differenceHandler); QmlDesigner::PropertyName syncScriptBinding(ModelNode &modelNode, const QString &prefix, QmlJS::AST::UiScriptBinding *script, ReadingContext *context, DifferenceHandler &differenceHandler); void syncNodeId(ModelNode &modelNode, const QString &astObjectId, DifferenceHandler &differenceHandler); void syncNodeProperty(AbstractProperty &modelProperty, QmlJS::AST::UiObjectBinding *binding, ReadingContext *context, const TypeName &astType, DifferenceHandler &differenceHandler); void syncExpressionProperty(AbstractProperty &modelProperty, const QString &javascript, const TypeName &astType, DifferenceHandler &differenceHandler); void syncSignalHandler(AbstractProperty &modelProperty, const QString &javascript, DifferenceHandler &differenceHandler); void syncArrayProperty(AbstractProperty &modelProperty, const QList &arrayMembers, ReadingContext *context, DifferenceHandler &differenceHandler); void syncVariantProperty(AbstractProperty &modelProperty, const QVariant &astValue, const TypeName &astType, DifferenceHandler &differenceHandler); void syncNodeListProperty(NodeListProperty &modelListProperty, const QList arrayMembers, ReadingContext *context, DifferenceHandler &differenceHandler); ModelNode createModelNode(const TypeName &typeName, int majorVersion, int minorVersion, bool isImplicitComponent, QmlJS::AST::UiObjectMember *astNode, ReadingContext *context, DifferenceHandler &differenceHandler); QStringList syncGroupedProperties(ModelNode &modelNode, const QString &name, QmlJS::AST::UiObjectMemberList *members, ReadingContext *context, DifferenceHandler &differenceHandler); void setupComponentDelayed(const ModelNode &node, bool synchron); void setupCustomParserNodeDelayed(const ModelNode &node, bool synchron); void delayedSetup(); QSet > qrcMapping() const; private: void setupCustomParserNode(const ModelNode &node); void setupComponent(const ModelNode &node); void collectLinkErrors(QList *errors, const ReadingContext &ctxt); void collectImportErrors(QList *errors); void collectSemanticErrorsAndWarnings(QList *errors, QList *warnings); void populateQrcMapping(const QString &filePath); void addIsoIconQrcMapping(const QUrl &fileUrl); static QString textAt(const QmlJS::Document::Ptr &doc, const QmlJS::AST::SourceLocation &location); static QString textAt(const QmlJS::Document::Ptr &doc, const QmlJS::AST::SourceLocation &from, const QmlJS::AST::SourceLocation &to); private: RewriterView *m_rewriterView; bool m_isActive; QSharedPointer m_scopeChain; QmlJS::Document::Ptr m_document; QTimer m_setupTimer; QSet m_setupComponentList; QSet m_setupCustomParserList; QmlJS::ViewerContext m_vContext; QSet > m_qrcMapping; }; class DifferenceHandler { public: DifferenceHandler(TextToModelMerger *textToModelMerger): m_merger(textToModelMerger) {} virtual ~DifferenceHandler() = default; virtual void modelMissesImport(const QmlDesigner::Import &import) = 0; virtual void importAbsentInQMl(const QmlDesigner::Import &import) = 0; virtual void bindingExpressionsDiffer(BindingProperty &modelProperty, const QString &javascript, const TypeName &astType) = 0; virtual void signalHandlerSourceDiffer(SignalHandlerProperty &modelProperty, const QString &javascript) = 0; virtual void shouldBeBindingProperty(AbstractProperty &modelProperty, const QString &javascript, const TypeName &astType) = 0; virtual void shouldBeSignalHandlerProperty(AbstractProperty &modelProperty, const QString &javascript) = 0; virtual void shouldBeNodeListProperty(AbstractProperty &modelProperty, const QList arrayMembers, ReadingContext *context) = 0; virtual void variantValuesDiffer(VariantProperty &modelProperty, const QVariant &qmlVariantValue, const TypeName &dynamicTypeName) = 0; virtual void shouldBeVariantProperty(AbstractProperty &modelProperty, const QVariant &qmlVariantValue, const TypeName &dynamicTypeName) = 0; virtual void shouldBeNodeProperty(AbstractProperty &modelProperty, const TypeName &typeName, int majorVersion, int minorVersion, QmlJS::AST::UiObjectMember *astNode, const TypeName &dynamicPropertyType, ReadingContext *context) = 0; virtual void modelNodeAbsentFromQml(ModelNode &modelNode) = 0; virtual ModelNode listPropertyMissingModelNode(NodeListProperty &modelProperty, ReadingContext *context, QmlJS::AST::UiObjectMember *arrayMember) = 0; virtual void typeDiffers(bool isRootNode, ModelNode &modelNode, const TypeName &typeName, int majorVersion, int minorVersion, QmlJS::AST::UiObjectMember *astNode, ReadingContext *context) = 0; virtual void propertyAbsentFromQml(AbstractProperty &modelProperty) = 0; virtual void idsDiffer(ModelNode &modelNode, const QString &qmlId) = 0; virtual bool isValidator() const = 0; protected: TextToModelMerger *m_merger; }; class ModelValidator: public DifferenceHandler { public: ModelValidator(TextToModelMerger *textToModelMerger): DifferenceHandler(textToModelMerger) {} ~ModelValidator() override = default; void modelMissesImport(const QmlDesigner::Import &import) override; void importAbsentInQMl(const QmlDesigner::Import &import) override; void bindingExpressionsDiffer(BindingProperty &modelProperty, const QString &javascript, const TypeName &astType) override; void shouldBeBindingProperty(AbstractProperty &modelProperty, const QString &javascript, const TypeName &astType) override; void signalHandlerSourceDiffer(SignalHandlerProperty &modelProperty, const QString &javascript) override; void shouldBeSignalHandlerProperty(AbstractProperty &modelProperty, const QString &javascript) override; void shouldBeNodeListProperty(AbstractProperty &modelProperty, const QList arrayMembers, ReadingContext *context) override; void variantValuesDiffer(VariantProperty &modelProperty, const QVariant &qmlVariantValue, const TypeName &dynamicTypeName) override; void shouldBeVariantProperty(AbstractProperty &modelProperty, const QVariant &qmlVariantValue, const TypeName &dynamicTypeName) override; void shouldBeNodeProperty(AbstractProperty &modelProperty, const TypeName &typeName, int majorVersion, int minorVersion, QmlJS::AST::UiObjectMember *astNode, const TypeName &dynamicPropertyType, ReadingContext *context) override; void modelNodeAbsentFromQml(ModelNode &modelNode) override; ModelNode listPropertyMissingModelNode(NodeListProperty &modelProperty, ReadingContext *context, QmlJS::AST::UiObjectMember *arrayMember) override; void typeDiffers(bool isRootNode, ModelNode &modelNode, const TypeName &typeName, int majorVersion, int minorVersion, QmlJS::AST::UiObjectMember *astNode, ReadingContext *context) override; void propertyAbsentFromQml(AbstractProperty &modelProperty) override; void idsDiffer(ModelNode &modelNode, const QString &qmlId) override; bool isValidator() const override {return false; } }; class ModelAmender: public DifferenceHandler { public: ModelAmender(TextToModelMerger *textToModelMerger): DifferenceHandler(textToModelMerger) {} ~ModelAmender() override = default; void modelMissesImport(const QmlDesigner::Import &import) override; void importAbsentInQMl(const QmlDesigner::Import &import) override; void bindingExpressionsDiffer(BindingProperty &modelProperty, const QString &javascript, const TypeName &astType) override; void shouldBeBindingProperty(AbstractProperty &modelProperty, const QString &javascript, const TypeName &astType) override; void signalHandlerSourceDiffer(SignalHandlerProperty &modelProperty, const QString &javascript) override; void shouldBeSignalHandlerProperty(AbstractProperty &modelProperty, const QString &javascript) override; void shouldBeNodeListProperty(AbstractProperty &modelProperty, const QList arrayMembers, ReadingContext *context) override; void variantValuesDiffer(VariantProperty &modelProperty, const QVariant &qmlVariantValue, const TypeName &dynamicType) override; void shouldBeVariantProperty(AbstractProperty &modelProperty, const QVariant &qmlVariantValue, const TypeName &dynamicTypeName) override; void shouldBeNodeProperty(AbstractProperty &modelProperty, const TypeName &typeName, int majorVersion, int minorVersion, QmlJS::AST::UiObjectMember *astNode, const TypeName &dynamicPropertyType, ReadingContext *context) override; void modelNodeAbsentFromQml(ModelNode &modelNode) override; ModelNode listPropertyMissingModelNode(NodeListProperty &modelProperty, ReadingContext *context, QmlJS::AST::UiObjectMember *arrayMember) override; void typeDiffers(bool isRootNode, ModelNode &modelNode, const TypeName &typeName, int majorVersion, int minorVersion, QmlJS::AST::UiObjectMember *astNode, ReadingContext *context) override; void propertyAbsentFromQml(AbstractProperty &modelProperty) override; void idsDiffer(ModelNode &modelNode, const QString &qmlId) override; bool isValidator() const override {return true; } }; } //Internal } //QmlDesigner