2022-08-19 15:59:36 +02:00
|
|
|
// Copyright (C) 2016 The Qt Company Ltd.
|
2022-12-21 10:12:09 +01:00
|
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
2012-03-02 16:26:22 +01:00
|
|
|
|
2011-10-07 14:04:06 +02:00
|
|
|
#include "qmljsutils.h"
|
|
|
|
|
|
|
|
|
|
#include "parser/qmljsast_p.h"
|
|
|
|
|
|
2022-06-20 12:35:13 +02:00
|
|
|
#include <utils/filepath.h>
|
2020-06-18 19:04:58 +02:00
|
|
|
#include <utils/stringutils.h>
|
|
|
|
|
|
2015-03-04 16:46:23 +01:00
|
|
|
#include <QColor>
|
2016-04-29 16:25:17 +02:00
|
|
|
#include <QDir>
|
|
|
|
|
#include <QRegularExpression>
|
2015-03-04 16:46:23 +01:00
|
|
|
|
2011-10-07 14:04:06 +02:00
|
|
|
using namespace QmlJS;
|
|
|
|
|
using namespace QmlJS::AST;
|
|
|
|
|
|
2011-11-01 14:01:07 +01:00
|
|
|
/*!
|
|
|
|
|
\namespace QmlJS
|
|
|
|
|
QML and JavaScript language support library
|
|
|
|
|
*/
|
|
|
|
|
|
2011-10-07 14:04:06 +02:00
|
|
|
namespace {
|
|
|
|
|
class SharedData
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
SharedData()
|
|
|
|
|
{
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("action"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("bool"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("color"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("date"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("double"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("enumeration"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("font"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("int"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("list"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("point"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("real"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("rect"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("size"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("string"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("time"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("url"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("var"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("variant")); // obsolete in Qt 5
|
2013-08-20 18:15:05 +02:00
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("vector2d"));
|
2011-10-07 14:04:06 +02:00
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("vector3d"));
|
2013-08-20 18:15:05 +02:00
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("vector4d"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("quaternion"));
|
|
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("matrix4x4"));
|
2011-10-07 14:04:06 +02:00
|
|
|
validBuiltinPropertyNames.insert(QLatin1String("alias"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QSet<QString> validBuiltinPropertyNames;
|
|
|
|
|
};
|
|
|
|
|
} // anonymous namespace
|
|
|
|
|
Q_GLOBAL_STATIC(SharedData, sharedData)
|
|
|
|
|
|
|
|
|
|
QColor QmlJS::toQColor(const QString &qmlColorString)
|
|
|
|
|
{
|
|
|
|
|
QColor color;
|
|
|
|
|
if (qmlColorString.size() == 9 && qmlColorString.at(0) == QLatin1Char('#')) {
|
|
|
|
|
bool ok;
|
2020-09-16 15:08:57 +02:00
|
|
|
const int alpha = qmlColorString.mid(1, 2).toInt(&ok, 16);
|
2011-10-07 14:04:06 +02:00
|
|
|
if (ok) {
|
2017-04-24 06:43:24 +02:00
|
|
|
const QString name = qmlColorString.at(0) + qmlColorString.right(6);
|
2011-10-07 14:04:06 +02:00
|
|
|
if (QColor::isValidColor(name)) {
|
|
|
|
|
color.setNamedColor(name);
|
|
|
|
|
color.setAlpha(alpha);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if (QColor::isValidColor(qmlColorString))
|
|
|
|
|
color.setNamedColor(qmlColorString);
|
|
|
|
|
}
|
|
|
|
|
return color;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString QmlJS::toString(UiQualifiedId *qualifiedId, QChar delimiter)
|
|
|
|
|
{
|
|
|
|
|
QString result;
|
|
|
|
|
|
|
|
|
|
for (UiQualifiedId *iter = qualifiedId; iter; iter = iter->next) {
|
|
|
|
|
if (iter != qualifiedId)
|
|
|
|
|
result += delimiter;
|
|
|
|
|
|
2020-09-16 15:08:57 +02:00
|
|
|
result += iter->name.toString();
|
2011-10-07 14:04:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
SourceLocation QmlJS::locationFromRange(const SourceLocation &start,
|
|
|
|
|
const SourceLocation &end)
|
|
|
|
|
{
|
|
|
|
|
return SourceLocation(start.offset,
|
|
|
|
|
end.end() - start.begin(),
|
|
|
|
|
start.startLine,
|
|
|
|
|
start.startColumn);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SourceLocation QmlJS::fullLocationForQualifiedId(AST::UiQualifiedId *qualifiedId)
|
|
|
|
|
{
|
|
|
|
|
SourceLocation start = qualifiedId->identifierToken;
|
|
|
|
|
SourceLocation end = qualifiedId->identifierToken;
|
|
|
|
|
|
|
|
|
|
for (UiQualifiedId *iter = qualifiedId; iter; iter = iter->next) {
|
|
|
|
|
if (iter->identifierToken.isValid())
|
|
|
|
|
end = iter->identifierToken;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return locationFromRange(start, end);
|
|
|
|
|
}
|
|
|
|
|
|
2011-10-24 09:28:02 +02:00
|
|
|
/*!
|
2023-03-24 12:07:30 +01:00
|
|
|
Returns the value of the 'id:' binding in \a object.
|
|
|
|
|
|
|
|
|
|
\a idBinding is optional out parameter to get the UiScriptBinding for the id binding.
|
2011-10-24 09:28:02 +02:00
|
|
|
*/
|
|
|
|
|
QString QmlJS::idOfObject(Node *object, UiScriptBinding **idBinding)
|
2011-10-07 14:04:06 +02:00
|
|
|
{
|
2011-10-19 13:48:10 +02:00
|
|
|
if (idBinding)
|
2019-07-31 17:21:41 +02:00
|
|
|
*idBinding = nullptr;
|
2011-10-19 13:48:10 +02:00
|
|
|
|
2011-10-24 09:28:02 +02:00
|
|
|
UiObjectInitializer *initializer = initializerOfObject(object);
|
2011-11-04 13:59:23 +01:00
|
|
|
if (!initializer) {
|
|
|
|
|
initializer = cast<UiObjectInitializer *>(object);
|
|
|
|
|
if (!initializer)
|
|
|
|
|
return QString();
|
|
|
|
|
}
|
2011-10-07 14:04:06 +02:00
|
|
|
|
|
|
|
|
for (UiObjectMemberList *iter = initializer->members; iter; iter = iter->next) {
|
|
|
|
|
if (UiScriptBinding *script = cast<UiScriptBinding*>(iter->member)) {
|
|
|
|
|
if (!script->qualifiedId)
|
|
|
|
|
continue;
|
|
|
|
|
if (script->qualifiedId->next)
|
|
|
|
|
continue;
|
|
|
|
|
if (script->qualifiedId->name != QLatin1String("id"))
|
|
|
|
|
continue;
|
|
|
|
|
if (ExpressionStatement *expstmt = cast<ExpressionStatement *>(script->statement)) {
|
|
|
|
|
if (IdentifierExpression *idexp = cast<IdentifierExpression *>(expstmt->expression)) {
|
2011-10-19 13:48:10 +02:00
|
|
|
if (idBinding)
|
|
|
|
|
*idBinding = script;
|
2011-10-07 14:04:06 +02:00
|
|
|
return idexp->name.toString();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return QString();
|
|
|
|
|
}
|
|
|
|
|
|
2011-10-24 09:28:02 +02:00
|
|
|
/*!
|
|
|
|
|
\returns the UiObjectInitializer if \a object is a UiObjectDefinition or UiObjectBinding, otherwise 0
|
|
|
|
|
*/
|
|
|
|
|
UiObjectInitializer *QmlJS::initializerOfObject(Node *object)
|
2011-10-07 14:04:06 +02:00
|
|
|
{
|
2011-10-24 09:28:02 +02:00
|
|
|
if (UiObjectDefinition *definition = cast<UiObjectDefinition *>(object))
|
2011-10-07 14:04:06 +02:00
|
|
|
return definition->initializer;
|
2011-10-24 09:28:02 +02:00
|
|
|
if (UiObjectBinding *binding = cast<UiObjectBinding *>(object))
|
2011-10-07 14:04:06 +02:00
|
|
|
return binding->initializer;
|
2019-07-31 17:21:41 +02:00
|
|
|
return nullptr;
|
2011-10-07 14:04:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
UiQualifiedId *QmlJS::qualifiedTypeNameId(Node *node)
|
|
|
|
|
{
|
|
|
|
|
if (UiObjectBinding *binding = AST::cast<UiObjectBinding *>(node))
|
|
|
|
|
return binding->qualifiedTypeNameId;
|
|
|
|
|
else if (UiObjectDefinition *binding = AST::cast<UiObjectDefinition *>(node))
|
|
|
|
|
return binding->qualifiedTypeNameId;
|
2019-07-31 17:21:41 +02:00
|
|
|
return nullptr;
|
2011-10-07 14:04:06 +02:00
|
|
|
}
|
|
|
|
|
|
2020-02-28 17:51:32 +01:00
|
|
|
DiagnosticMessage QmlJS::errorMessage(const SourceLocation &loc, const QString &message)
|
2011-10-07 14:04:06 +02:00
|
|
|
{
|
2013-10-16 14:59:28 +02:00
|
|
|
return DiagnosticMessage(Severity::Error, loc, message);
|
2011-10-07 14:04:06 +02:00
|
|
|
}
|
|
|
|
|
|
2016-05-10 09:48:36 +02:00
|
|
|
namespace {
|
|
|
|
|
const QString undefinedVersion = QLatin1String("-1.-1");
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-29 16:25:17 +02:00
|
|
|
/*!
|
|
|
|
|
* \brief Permissive validation of a string representing a module version.
|
|
|
|
|
* \param version
|
2016-05-10 09:48:36 +02:00
|
|
|
* \return True if \p version is a valid version format (<digit(s)>.<digit(s)>), if it is the
|
|
|
|
|
* undefined version (-1.-1) or if it is empty. False otherwise.
|
2016-04-29 16:25:17 +02:00
|
|
|
*/
|
|
|
|
|
bool QmlJS::maybeModuleVersion(const QString &version) {
|
2021-07-06 23:50:30 +02:00
|
|
|
QRegularExpression re(QLatin1String("^\\d+\\.-?\\d+$"));
|
2016-05-10 09:48:36 +02:00
|
|
|
return version.isEmpty() || version == undefinedVersion || re.match(version).hasMatch();
|
2016-04-29 16:25:17 +02:00
|
|
|
}
|
|
|
|
|
|
2021-06-23 10:27:08 +02:00
|
|
|
const QStringList QmlJS::splitVersion(const QString &version)
|
|
|
|
|
{
|
|
|
|
|
// Successively removing minor and major version numbers.
|
|
|
|
|
QStringList result;
|
|
|
|
|
int versionEnd = version.length();
|
|
|
|
|
while (versionEnd > 0) {
|
|
|
|
|
result.append(version.left(versionEnd));
|
|
|
|
|
// remove numbers and then potential . at the end
|
|
|
|
|
const int oldVersionEnd = versionEnd;
|
|
|
|
|
while (versionEnd > 0 && version.at(versionEnd - 1).isDigit())
|
|
|
|
|
--versionEnd;
|
|
|
|
|
// handle e.g. -1, because an import "QtQuick 2" results in version "2.-1"
|
|
|
|
|
if (versionEnd > 0 && version.at(versionEnd - 1) == '-')
|
|
|
|
|
--versionEnd;
|
|
|
|
|
if (versionEnd > 0 && version.at(versionEnd - 1) == '.')
|
|
|
|
|
--versionEnd;
|
|
|
|
|
// bail out if we didn't proceed because version string contains invalid characters
|
|
|
|
|
if (versionEnd == oldVersionEnd)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-29 16:25:17 +02:00
|
|
|
/*!
|
|
|
|
|
* \brief Get the path of a module
|
|
|
|
|
* \param name
|
|
|
|
|
* \param version
|
|
|
|
|
* \param importPaths
|
|
|
|
|
*
|
|
|
|
|
* Given the qualified \p name and \p version of a module, look for a valid path in \p importPaths.
|
|
|
|
|
* Most specific version are searched first, the version is searched also in parent modules.
|
|
|
|
|
* For example, given the \p name QtQml.Models and \p version 2.0, the following directories are
|
|
|
|
|
* searched in every element of \p importPath:
|
|
|
|
|
*
|
|
|
|
|
* - QtQml/Models.2.0
|
|
|
|
|
* - QtQml.2.0/Models
|
|
|
|
|
* - QtQml/Models.2
|
|
|
|
|
* - QtQml.2/Models
|
|
|
|
|
* - QtQml/Models
|
|
|
|
|
*
|
|
|
|
|
* \return The module paths if found, an empty string otherwise
|
|
|
|
|
* \see qmlimportscanner in qtdeclarative/tools
|
|
|
|
|
*/
|
2022-06-20 12:35:13 +02:00
|
|
|
QList<Utils::FilePath> QmlJS::modulePaths(const QString &name,
|
|
|
|
|
const QString &version,
|
|
|
|
|
const QList<Utils::FilePath> &importPaths)
|
2016-04-29 16:25:17 +02:00
|
|
|
{
|
|
|
|
|
Q_ASSERT(maybeModuleVersion(version));
|
2016-05-04 17:39:41 +02:00
|
|
|
if (importPaths.isEmpty())
|
2021-01-11 14:09:51 +01:00
|
|
|
return {};
|
2016-04-29 16:25:17 +02:00
|
|
|
|
2016-05-10 09:48:36 +02:00
|
|
|
const QString sanitizedVersion = version == undefinedVersion ? QString() : version;
|
2020-07-21 10:19:36 +02:00
|
|
|
const QStringList parts = name.split('.', Qt::SkipEmptyParts);
|
2021-06-23 10:27:08 +02:00
|
|
|
auto mkpath = [](const QStringList &xs) -> QString { return xs.join(QLatin1Char('/')); };
|
2016-05-10 09:48:36 +02:00
|
|
|
|
2022-06-20 12:35:13 +02:00
|
|
|
QList<Utils::FilePath> result;
|
|
|
|
|
Utils::FilePath candidate;
|
2016-04-29 16:25:17 +02:00
|
|
|
|
2021-06-23 10:27:08 +02:00
|
|
|
for (const QString &versionPart : splitVersion(sanitizedVersion)) {
|
2022-06-20 12:35:13 +02:00
|
|
|
for (const Utils::FilePath &path : importPaths) {
|
2016-06-23 10:48:54 +02:00
|
|
|
for (int i = parts.count() - 1; i >= 0; --i) {
|
2022-06-20 12:35:13 +02:00
|
|
|
candidate = path.pathAppended(QString::fromLatin1("%2.%3/%4")
|
|
|
|
|
.arg(mkpath(parts.mid(0, i + 1)),
|
|
|
|
|
versionPart,
|
|
|
|
|
mkpath(parts.mid(i + 1))))
|
|
|
|
|
.cleanPath();
|
|
|
|
|
if (candidate.exists())
|
2021-01-11 14:09:51 +01:00
|
|
|
result << candidate;
|
2016-04-29 16:25:17 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-06-23 10:48:54 +02:00
|
|
|
|
|
|
|
|
// Version is empty
|
2022-06-20 12:35:13 +02:00
|
|
|
for (const Utils::FilePath &path : importPaths) {
|
|
|
|
|
candidate = path.pathAppended(mkpath(parts)).cleanPath();
|
|
|
|
|
if (candidate.exists())
|
2021-01-11 14:09:51 +01:00
|
|
|
result << candidate;
|
2016-06-23 10:48:54 +02:00
|
|
|
}
|
|
|
|
|
|
2021-01-11 14:09:51 +01:00
|
|
|
return result;
|
2016-04-29 16:25:17 +02:00
|
|
|
}
|
|
|
|
|
|
2011-10-07 14:04:06 +02:00
|
|
|
bool QmlJS::isValidBuiltinPropertyType(const QString &name)
|
|
|
|
|
{
|
|
|
|
|
return sharedData()->validBuiltinPropertyNames.contains(name);
|
|
|
|
|
}
|
|
|
|
|
|