2014-09-12 14:25:51 +02:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
2014-09-12 14:25:51 +02:00
|
|
|
**
|
|
|
|
|
** 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
|
2016-01-15 14:57:40 +01:00
|
|
|
** 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.
|
2014-09-12 14:25:51 +02:00
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** 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.
|
2014-09-12 14:25:51 +02:00
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "jsexpander.h"
|
|
|
|
|
|
2014-09-16 15:58:32 +02:00
|
|
|
#include "corejsextensions.h"
|
2014-09-12 14:25:51 +02:00
|
|
|
|
2014-10-13 18:49:44 +02:00
|
|
|
#include <utils/macroexpander.h>
|
2014-09-12 14:25:51 +02:00
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
|
|
|
|
|
#include <QCoreApplication>
|
|
|
|
|
#include <QDebug>
|
2015-03-31 17:39:09 +02:00
|
|
|
#include <QJSEngine>
|
2014-09-12 14:25:51 +02:00
|
|
|
|
2019-04-16 16:46:36 +02:00
|
|
|
#include <unordered_map>
|
|
|
|
|
|
|
|
|
|
namespace std {
|
|
|
|
|
template<> struct hash<QString>
|
|
|
|
|
{
|
|
|
|
|
using argument_type = QString;
|
|
|
|
|
using result_type = size_t;
|
|
|
|
|
result_type operator()(const argument_type &v) const
|
|
|
|
|
{
|
|
|
|
|
return hash<string>()(v.toStdString());
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
} // namespace std
|
|
|
|
|
|
|
|
|
|
using ExtensionMap = std::unordered_map<QString, Core::JsExpander::ObjectFactory>;
|
|
|
|
|
Q_GLOBAL_STATIC(ExtensionMap, globalJsExtensions);
|
2014-09-12 14:25:51 +02:00
|
|
|
|
2019-04-16 16:46:36 +02:00
|
|
|
static Core::JsExpander *globalExpander = nullptr;
|
|
|
|
|
|
|
|
|
|
namespace Core {
|
2014-09-12 14:25:51 +02:00
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
class JsExpanderPrivate {
|
|
|
|
|
public:
|
2015-03-31 17:39:09 +02:00
|
|
|
QJSEngine m_engine;
|
2014-09-12 14:25:51 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
} // namespace Internal
|
|
|
|
|
|
2019-04-16 16:46:36 +02:00
|
|
|
void JsExpander::registerGlobalObject(const QString &name, const ObjectFactory &factory)
|
|
|
|
|
{
|
|
|
|
|
globalJsExtensions->insert({name, factory});
|
|
|
|
|
if (globalExpander)
|
|
|
|
|
globalExpander->registerObject(name, factory());
|
|
|
|
|
}
|
2014-09-12 14:25:51 +02:00
|
|
|
|
2019-04-16 16:46:36 +02:00
|
|
|
void JsExpander::registerObject(const QString &name, QObject *obj)
|
2014-09-12 14:25:51 +02:00
|
|
|
{
|
2015-03-31 17:39:09 +02:00
|
|
|
QJSValue jsObj = d->m_engine.newQObject(obj);
|
2014-09-12 14:25:51 +02:00
|
|
|
d->m_engine.globalObject().setProperty(name, jsObj);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString JsExpander::evaluate(const QString &expression, QString *errorMessage)
|
|
|
|
|
{
|
2015-03-31 17:39:09 +02:00
|
|
|
QJSValue value = d->m_engine.evaluate(expression);
|
|
|
|
|
if (value.isError()) {
|
2014-09-12 14:25:51 +02:00
|
|
|
const QString msg = QCoreApplication::translate("Core::JsExpander", "Error in \"%1\": %2")
|
2015-03-31 17:39:09 +02:00
|
|
|
.arg(expression, value.toString());
|
2014-09-12 14:25:51 +02:00
|
|
|
if (errorMessage)
|
|
|
|
|
*errorMessage = msg;
|
|
|
|
|
return QString();
|
|
|
|
|
}
|
|
|
|
|
// Try to convert to bool, be that an int or whatever.
|
|
|
|
|
if (value.isBool())
|
2015-03-31 17:39:09 +02:00
|
|
|
return value.toString();
|
2014-09-12 14:25:51 +02:00
|
|
|
if (value.isNumber())
|
|
|
|
|
return QString::number(value.toNumber());
|
|
|
|
|
if (value.isString())
|
|
|
|
|
return value.toString();
|
|
|
|
|
QString msg = QCoreApplication::translate("Core::JsExpander",
|
|
|
|
|
"Cannot convert result of \"%1\" to string.").arg(expression);
|
|
|
|
|
if (errorMessage)
|
|
|
|
|
*errorMessage = msg;
|
|
|
|
|
return QString();
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-16 16:46:36 +02:00
|
|
|
QJSEngine &JsExpander::engine()
|
2014-09-12 14:25:51 +02:00
|
|
|
{
|
2019-04-16 16:46:36 +02:00
|
|
|
return d->m_engine;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void JsExpander::registerForExpander(Utils::MacroExpander *macroExpander)
|
|
|
|
|
{
|
|
|
|
|
macroExpander->registerPrefix(
|
|
|
|
|
"JS",
|
2014-09-12 14:25:51 +02:00
|
|
|
QCoreApplication::translate("Core::JsExpander",
|
2015-10-22 17:13:38 +02:00
|
|
|
"Evaluate simple JavaScript statements.<br>"
|
2019-05-31 15:29:05 +02:00
|
|
|
"Literal '}' characters must be escaped as \"\\}\", "
|
|
|
|
|
"'\\' characters must be escaped as \"\\\\\", "
|
|
|
|
|
"and \"%{\" must be escaped as \"%\\{\"."),
|
2019-04-16 16:46:36 +02:00
|
|
|
[this](QString in) -> QString {
|
2014-09-12 14:25:51 +02:00
|
|
|
QString errorMessage;
|
2019-04-16 16:46:36 +02:00
|
|
|
QString result = evaluate(in, &errorMessage);
|
2014-09-12 14:25:51 +02:00
|
|
|
if (!errorMessage.isEmpty()) {
|
|
|
|
|
qWarning() << errorMessage;
|
|
|
|
|
return errorMessage;
|
|
|
|
|
} else {
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
});
|
2019-04-16 16:46:36 +02:00
|
|
|
}
|
2014-09-16 15:58:32 +02:00
|
|
|
|
2019-04-16 16:46:36 +02:00
|
|
|
JsExpander *JsExpander::createGlobalJsExpander()
|
|
|
|
|
{
|
|
|
|
|
globalExpander = new JsExpander();
|
|
|
|
|
registerGlobalObject<Internal::UtilsJsExtension>("Util");
|
|
|
|
|
globalExpander->registerForExpander(Utils::globalMacroExpander());
|
|
|
|
|
return globalExpander;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
JsExpander::JsExpander()
|
|
|
|
|
{
|
|
|
|
|
d = new Internal::JsExpanderPrivate;
|
|
|
|
|
for (const std::pair<const QString, ObjectFactory> &obj : *globalJsExtensions)
|
|
|
|
|
registerObject(obj.first, obj.second());
|
2014-09-12 14:25:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
JsExpander::~JsExpander()
|
|
|
|
|
{
|
|
|
|
|
delete d;
|
2018-07-21 21:11:46 +02:00
|
|
|
d = nullptr;
|
2014-09-12 14:25:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace Core
|