1 | /**************************************************************************** |
2 | ** |
3 | ** Copyright (C) 2019 The Qt Company Ltd. |
4 | ** Contact: https://www.qt.io/licensing/ |
5 | ** |
6 | ** This file is part of the tools applications of the Qt Toolkit. |
7 | ** |
8 | ** $QT_BEGIN_LICENSE:GPL-EXCEPT$ |
9 | ** Commercial License Usage |
10 | ** Licensees holding valid commercial Qt licenses may use this file in |
11 | ** accordance with the commercial license agreement provided with the |
12 | ** Software or, alternatively, in accordance with the terms contained in |
13 | ** a written agreement between you and The Qt Company. For licensing terms |
14 | ** and conditions see https://www.qt.io/terms-conditions. For further |
15 | ** information use the contact form at https://www.qt.io/contact-us. |
16 | ** |
17 | ** GNU General Public License Usage |
18 | ** Alternatively, this file may be used under the terms of the GNU |
19 | ** General Public License version 3 as published by the Free Software |
20 | ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT |
21 | ** included in the packaging of this file. Please review the following |
22 | ** information to ensure the GNU General Public License requirements will |
23 | ** be met: https://www.gnu.org/licenses/gpl-3.0.html. |
24 | ** |
25 | ** $QT_END_LICENSE$ |
26 | ** |
27 | ****************************************************************************/ |
28 | |
29 | #ifndef SCOPETREE_H |
30 | #define SCOPETREE_H |
31 | |
32 | // |
33 | // W A R N I N G |
34 | // ------------- |
35 | // |
36 | // This file is not part of the Qt API. It exists purely as an |
37 | // implementation detail. This header file may change from version to |
38 | // version without notice, or even be removed. |
39 | // |
40 | // We mean it. |
41 | |
42 | #include "metatypes.h" |
43 | #include "componentversion.h" |
44 | |
45 | #include <QtQml/private/qqmljsast_p.h> |
46 | #include <QtQml/private/qqmljssourcelocation_p.h> |
47 | |
48 | #include <QtCore/qset.h> |
49 | #include <QtCore/qhash.h> |
50 | #include <QtCore/qstring.h> |
51 | |
52 | enum MessageColors |
53 | { |
54 | Error, |
55 | Warning, |
56 | Info, |
57 | Normal, |
58 | Hint |
59 | }; |
60 | |
61 | enum class ScopeType |
62 | { |
63 | JSFunctionScope, |
64 | JSLexicalScope, |
65 | QMLScope |
66 | }; |
67 | |
68 | struct MethodUsage |
69 | { |
70 | MetaMethod method; |
71 | QQmlJS::SourceLocation loc; |
72 | bool hasMultilineHandlerBody; |
73 | }; |
74 | |
75 | class ColorOutput; |
76 | class ScopeTree |
77 | { |
78 | Q_DISABLE_COPY_MOVE(ScopeTree) |
79 | public: |
80 | using Ptr = QSharedPointer<ScopeTree>; |
81 | using ConstPtr = QSharedPointer<const ScopeTree>; |
82 | |
83 | class Export { |
84 | public: |
85 | Export() = default; |
86 | Export(QString package, QString type, const ComponentVersion &version, |
87 | int metaObjectRevision); |
88 | |
89 | bool isValid() const; |
90 | |
91 | int metaObjectRevision() const { return m_metaObjectRevision; } |
92 | void setMetaObjectRevision(int metaObjectRevision) |
93 | { |
94 | m_metaObjectRevision = metaObjectRevision; |
95 | } |
96 | |
97 | QString package() const { return m_package; } |
98 | QString type() const { return m_type; } |
99 | |
100 | private: |
101 | QString m_package; |
102 | QString m_type; |
103 | ComponentVersion m_version; |
104 | int m_metaObjectRevision = 0; |
105 | }; |
106 | |
107 | ScopeTree(ScopeType type, QString name = QString(), |
108 | ScopeTree *parentScope = nullptr); |
109 | |
110 | ScopeTree::Ptr createNewChildScope(ScopeType type, const QString &name); |
111 | ScopeTree *parentScope() const { return m_parentScope; } |
112 | |
113 | void insertJSIdentifier(const QString &id, QQmlJS::AST::VariableScope scope); |
114 | void insertSignalIdentifier(const QString &id, const MetaMethod &method, |
115 | const QQmlJS::SourceLocation &loc, bool hasMultilineHandlerBody); |
116 | // inserts property as qml identifier as well as the corresponding |
117 | void insertPropertyIdentifier(const MetaProperty &prop); |
118 | void addUnmatchedSignalHandler(const QString &handler, |
119 | const QQmlJS::SourceLocation &location); |
120 | |
121 | bool isIdInCurrentScope(const QString &id) const; |
122 | void addIdToAccessed(const QString &id, const QQmlJS::SourceLocation &location); |
123 | void accessMember(const QString &name, const QString &parentType, |
124 | const QQmlJS::SourceLocation &location); |
125 | void resetMemberScope(); |
126 | |
127 | bool isVisualRootScope() const; |
128 | QString name() const { return m_name; } |
129 | |
130 | bool recheckIdentifiers( |
131 | const QString &code, |
132 | const QHash<QString, const ScopeTree *> &qmlIDs, |
133 | const QHash<QString, ScopeTree::ConstPtr> &types, |
134 | const ScopeTree *root, const QString& rootId, ColorOutput &colorOut) const; |
135 | |
136 | ScopeType scopeType() const { return m_scopeType; } |
137 | |
138 | void addMethods(const QHash<QString, MetaMethod> &methods) { m_methods.insert(hash: methods); } |
139 | void addMethod(const MetaMethod &method) { m_methods.insert(akey: method.methodName(), avalue: method); } |
140 | QHash<QString, MetaMethod> methods() const { return m_methods; } |
141 | |
142 | void addEnum(const MetaEnum &fakeEnum) { m_enums.insert(akey: fakeEnum.name(), avalue: fakeEnum); } |
143 | QHash<QString, MetaEnum> enums() const { return m_enums; } |
144 | |
145 | QString className() const { return m_className; } |
146 | void setClassName(const QString &name) { m_className = name; } |
147 | |
148 | void addExport(const QString &name, const QString &package, const ComponentVersion &version); |
149 | void setExportMetaObjectRevision(int exportIndex, int metaObjectRevision); |
150 | QList<Export> exports() const { return m_exports; } |
151 | |
152 | void setSuperclassName(const QString &superclass) { m_superName = superclass; } |
153 | QString superclassName() const { return m_superName; } |
154 | |
155 | void addProperty(const MetaProperty &prop) { m_properties.insert(akey: prop.propertyName(), avalue: prop); } |
156 | QHash<QString, MetaProperty> properties() const { return m_properties; } |
157 | void updateParentProperty(const ScopeTree *scope); |
158 | |
159 | QString defaultPropertyName() const { return m_defaultPropertyName; } |
160 | void setDefaultPropertyName(const QString &name) { m_defaultPropertyName = name; } |
161 | |
162 | QString attachedTypeName() const { return m_attachedTypeName; } |
163 | void setAttachedTypeName(const QString &name) { m_attachedTypeName = name; } |
164 | |
165 | bool isSingleton() const { return m_isSingleton; } |
166 | bool isCreatable() const { return m_isCreatable; } |
167 | bool isComposite() const { return m_isComposite; } |
168 | void setIsSingleton(bool value) { m_isSingleton = value; } |
169 | void setIsCreatable(bool value) { m_isCreatable = value; } |
170 | void setIsComposite(bool value) { m_isSingleton = value; } |
171 | |
172 | private: |
173 | struct FieldMemberList |
174 | { |
175 | QString m_name; |
176 | QString m_parentType; |
177 | QQmlJS::SourceLocation m_location; |
178 | std::unique_ptr<FieldMemberList> m_child; |
179 | }; |
180 | |
181 | QSet<QString> m_jsIdentifiers; |
182 | QMultiHash<QString, MethodUsage> m_injectedSignalIdentifiers; |
183 | |
184 | QHash<QString, MetaMethod> m_methods; |
185 | QHash<QString, MetaProperty> m_properties; |
186 | QHash<QString, MetaEnum> m_enums; |
187 | |
188 | std::vector<std::unique_ptr<FieldMemberList>> m_accessedIdentifiers; |
189 | FieldMemberList *m_currentFieldMember = nullptr; |
190 | |
191 | QVector<QPair<QString, QQmlJS::SourceLocation>> m_unmatchedSignalHandlers; |
192 | |
193 | QVector<ScopeTree::Ptr> m_childScopes; |
194 | ScopeTree *m_parentScope; |
195 | |
196 | QString m_name; |
197 | QString m_className; |
198 | QString m_superName; |
199 | |
200 | ScopeType m_scopeType = ScopeType::QMLScope; |
201 | QList<Export> m_exports; |
202 | |
203 | QString m_defaultPropertyName; |
204 | QString m_attachedTypeName; |
205 | bool m_isSingleton = false; |
206 | bool m_isCreatable = true; |
207 | bool m_isComposite = false; |
208 | |
209 | bool isIdInCurrentQMlScopes(const QString &id) const; |
210 | bool isIdInCurrentJSScopes(const QString &id) const; |
211 | bool isIdInjectedFromSignal(const QString &id) const; |
212 | const ScopeTree *currentQMLScope() const; |
213 | void printContext(ColorOutput &colorOut, const QString &code, |
214 | const QQmlJS::SourceLocation &location) const; |
215 | bool checkMemberAccess( |
216 | const QString &code, |
217 | FieldMemberList *members, |
218 | const ScopeTree *scope, |
219 | const QHash<QString, ScopeTree::ConstPtr> &types, |
220 | ColorOutput& colorOut) const; |
221 | }; |
222 | |
223 | #endif // SCOPETREE_H |
224 | |