qtdeclarative/tools/shared/scopetree.cpp

195 lines
6.9 KiB
C++

/****************************************************************************
**
** Copyright (C) 2019 The Qt Company Ltd.
** Contact: https://www.qt.io/licensing/
**
** This file is part of the tools applications of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:GPL-EXCEPT$
** 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.
**
** $QT_END_LICENSE$
**
****************************************************************************/
#include "scopetree.h"
#include <QtCore/qqueue.h>
#include <QtCore/qsharedpointer.h>
#include <algorithm>
ScopeTree::ScopeTree(ScopeType type, const ScopeTree::Ptr &parentScope)
: m_parentScope(parentScope), m_scopeType(type) {}
ScopeTree::Ptr ScopeTree::create(ScopeType type, const ScopeTree::Ptr &parentScope)
{
ScopeTree::Ptr childScope(new ScopeTree{type, parentScope});
if (parentScope) {
Q_ASSERT(type != ScopeType::QMLScope
|| !parentScope->m_parentScope
|| parentScope->parentScope()->m_scopeType == ScopeType::QMLScope
|| parentScope->parentScope()->m_internalName == QLatin1String("global"));
parentScope->m_childScopes.push_back(childScope);
}
return childScope;
}
void ScopeTree::insertJSIdentifier(const QString &id, ScopeType scope)
{
Q_ASSERT(m_scopeType != ScopeType::QMLScope);
Q_ASSERT(scope != ScopeType::QMLScope);
if (scope != ScopeType::JSFunctionScope || m_scopeType == ScopeType::JSFunctionScope) {
m_jsIdentifiers.insert(id);
} else {
auto targetScope = parentScope();
while (targetScope->m_scopeType != ScopeType::JSFunctionScope)
targetScope = targetScope->parentScope();
targetScope->m_jsIdentifiers.insert(id);
}
}
void ScopeTree::insertSignalIdentifier(const QString &id, const MetaMethod &method,
const QQmlJS::SourceLocation &loc,
bool hasMultilineHandlerBody)
{
Q_ASSERT(m_scopeType == ScopeType::QMLScope);
m_injectedSignalIdentifiers.insert(id, {method, loc, hasMultilineHandlerBody});
}
void ScopeTree::insertPropertyIdentifier(const MetaProperty &property)
{
addProperty(property);
MetaMethod method(property.propertyName() + QLatin1String("Changed"), QLatin1String("void"));
addMethod(method);
}
void ScopeTree::addUnmatchedSignalHandler(const QString &handler,
const QQmlJS::SourceLocation &location)
{
m_unmatchedSignalHandlers.append(qMakePair(handler, location));
}
bool ScopeTree::isIdInCurrentScope(const QString &id) const
{
return isIdInCurrentQMlScopes(id) || isIdInCurrentJSScopes(id);
}
void ScopeTree::addIdToAccessed(const QString &id, const QQmlJS::SourceLocation &location) {
m_memberAccessChains.append(QVector<FieldMember>());
m_memberAccessChains.last().append(FieldMember {id, QString(), location});
}
void ScopeTree::accessMember(const QString &name, const QString &parentType,
const QQmlJS::SourceLocation &location)
{
Q_ASSERT(!m_memberAccessChains.last().isEmpty());
m_memberAccessChains.last().append(FieldMember {name, parentType, location });
}
bool ScopeTree::isIdInCurrentQMlScopes(const QString &id) const
{
if (m_scopeType == ScopeType::QMLScope)
return m_properties.contains(id) || m_methods.contains(id) || m_enums.contains(id);
const auto qmlScope = findCurrentQMLScope(parentScope());
return qmlScope->m_properties.contains(id)
|| qmlScope->m_methods.contains(id)
|| qmlScope->m_enums.contains(id);
}
bool ScopeTree::isIdInCurrentJSScopes(const QString &id) const
{
if (m_scopeType != ScopeType::QMLScope && m_jsIdentifiers.contains(id))
return true;
for (auto jsScope = parentScope(); jsScope; jsScope = jsScope->parentScope()) {
if (jsScope->m_scopeType != ScopeType::QMLScope && jsScope->m_jsIdentifiers.contains(id))
return true;
}
return false;
}
bool ScopeTree::isIdInjectedFromSignal(const QString &id) const
{
if (m_scopeType == ScopeType::QMLScope)
return m_injectedSignalIdentifiers.contains(id);
return findCurrentQMLScope(parentScope())->m_injectedSignalIdentifiers.contains(id);
}
void ScopeTree::resolveTypes(const QHash<QString, ScopeTree::ConstPtr> &contextualTypes)
{
auto findType = [&](const QString &name) {
auto type = contextualTypes.constFind(name);
if (type != contextualTypes.constEnd())
return *type;
return ScopeTree::ConstPtr();
};
m_baseType = findType(m_baseTypeName);
m_attachedType = findType(m_attachedTypeName);
for (auto it = m_properties.begin(), end = m_properties.end(); it != end; ++it)
it->setType(findType(it->typeName()));
for (auto it = m_methods.begin(), end = m_methods.end(); it != end; ++it) {
it->setReturnType(findType(it->returnTypeName()));
const auto paramNames = it->parameterTypeNames();
QList<ScopeTree::ConstPtr> paramTypes;
for (const QString &paramName: paramNames)
paramTypes.append(findType(paramName));
it->setParameterTypes(paramTypes);
}
}
ScopeTree::ConstPtr ScopeTree::findCurrentQMLScope(const ScopeTree::ConstPtr &scope)
{
auto qmlScope = scope;
while (qmlScope && qmlScope->m_scopeType != ScopeType::QMLScope)
qmlScope = qmlScope->parentScope();
return qmlScope;
}
void ScopeTree::addExport(const QString &name, const QString &package,
const ComponentVersion &version)
{
m_exports.append(Export(package, name, version, 0));
}
void ScopeTree::setExportMetaObjectRevision(int exportIndex, int metaObjectRevision)
{
m_exports[exportIndex].setMetaObjectRevision(metaObjectRevision);
}
ScopeTree::Export::Export(QString package, QString type, const ComponentVersion &version,
int metaObjectRevision) :
m_package(std::move(package)),
m_type(std::move(type)),
m_version(version),
m_metaObjectRevision(metaObjectRevision)
{
}
bool ScopeTree::Export::isValid() const
{
return m_version.isValid() || !m_package.isEmpty() || !m_type.isEmpty();
}