diff options
author | Thomas Hartmann <[email protected]> | 2012-11-07 16:38:01 +0100 |
---|---|---|
committer | Thomas Hartmann <[email protected]> | 2012-11-07 16:42:28 +0100 |
commit | 5debabc410ff0b971225fea14a8845db63391904 (patch) | |
tree | db52a1ba21c0e0ef3c984c3343b78ff5556e858e /src/libs/qmljs/qmljssimplereader.cpp | |
parent | cf6298ff328d4314eb2d2fba4f9cf1aad4bdaa37 (diff) |
QmlJS: adding SimpleReader
SimpleReader is a !simple! parser for json like qml files.
SimpleReader only parses literal properties for e.g. configuration
files.
SimpleAbstractStreamReader allows event based parsing and SimpleReader
stores the parsed data in a reference counted tree structure.
Change-Id: I0f6422a97f5c356149c516f227f8bbd7b736a6d0
Reviewed-by: Joerg Bornemann <[email protected]>
Diffstat (limited to 'src/libs/qmljs/qmljssimplereader.cpp')
-rw-r--r-- | src/libs/qmljs/qmljssimplereader.cpp | 327 |
1 files changed, 327 insertions, 0 deletions
diff --git a/src/libs/qmljs/qmljssimplereader.cpp b/src/libs/qmljs/qmljssimplereader.cpp new file mode 100644 index 00000000000..b87b078e351 --- /dev/null +++ b/src/libs/qmljs/qmljssimplereader.cpp @@ -0,0 +1,327 @@ +/**************************************************************************** +** +** Copyright (C) 2012 Digia Plc and/or its subsidiary(-ies). +** Contact: http://www.qt-project.org/legal +** +** 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 Digia. For licensing terms and +** conditions see http://qt.digia.com/licensing. For further information +** use the contact form at http://qt.digia.com/contact-us. +** +** GNU Lesser General Public License Usage +** Alternatively, this file may be used under the terms of the GNU Lesser +** General Public License version 2.1 as published by the Free Software +** Foundation and appearing in the file LICENSE.LGPL included in the +** packaging of this file. Please review the following information to +** ensure the GNU Lesser General Public License version 2.1 requirements +** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. +** +** In addition, as a special exception, Digia gives you certain additional +** rights. These rights are described in the Digia Qt LGPL Exception +** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. +** +****************************************************************************/ + +#include "qmljssimplereader.h" + +#include "parser/qmljsparser_p.h" +#include "parser/qmljslexer_p.h" +#include "parser/qmljsengine_p.h" +#include "parser/qmljsast_p.h" +#include "parser/qmljsastvisitor_p.h" +#include <qmljs/qmljsdocument.h> + +#include "qmljsbind.h" +#include "qmljsinterpreter.h" +#include "qmljsutils.h" + +#include <QFile> + +enum { + debug = false +}; + + +using namespace QmlJS; + +QVariant SimpleReaderNode::property(const QString &name) const +{ + return m_properties.value(name); +} + +QStringList SimpleReaderNode::propertyNames() const +{ + return m_properties.keys(); +} + +SimpleReaderNode::PropertyHash SimpleReaderNode::properties() const +{ + return m_properties; +} + +bool SimpleReaderNode::isRoot() const +{ + return m_parentNode.isNull(); +} + +bool SimpleReaderNode::isValid() const +{ + return !m_name.isEmpty(); +} + +SimpleReaderNode::Ptr SimpleReaderNode::invalidNode() +{ + return Ptr(new SimpleReaderNode); +} + +SimpleReaderNode::WeakPtr SimpleReaderNode::parent() const +{ + return m_parentNode; +} + +QString SimpleReaderNode::name() const +{ + return m_name; +} + +SimpleReaderNode::SimpleReaderNode() +{ +} + +SimpleReaderNode::SimpleReaderNode(const QString &name, WeakPtr parent) + : m_name(name), m_parentNode(parent) +{ +} + +SimpleReaderNode::Ptr SimpleReaderNode::create(const QString &name, WeakPtr parent) +{ + Ptr newNode(new SimpleReaderNode(name, parent)); + newNode->m_weakThis = newNode; + if (parent) + parent.data()->m_children.append(newNode); + return newNode; +} + +const SimpleReaderNode::List SimpleReaderNode::children() const +{ + return m_children; +} + +void SimpleReaderNode::setProperty(const QString &name, const QVariant &value) +{ + m_properties.insert(name, value); +} + +SimpleAbstractStreamReader::SimpleAbstractStreamReader() +{ +} + +bool SimpleAbstractStreamReader::readFile(const QString &fileName) +{ + QFile file(fileName); + if (file.open(QIODevice::ReadOnly)) { + QByteArray source = file.readAll(); + file.close(); + return readFromSource(source); + } + addError(tr("Cannot find file %1").arg(fileName)); + return false; +} + +bool SimpleAbstractStreamReader::readFromSource(const QString &source) +{ + Engine engine; + Lexer lexer(&engine); + Parser parser(&engine); + + lexer.setCode(source, /*line = */ 1, /*qmlMode = */true); + + if (!parser.parse()) { + QString errorMessage = QString("%1:%2: %3").arg( + QString::number(parser.errorLineNumber()), + QString::number(parser.errorColumnNumber()), + parser.errorMessage()); + addError(errorMessage); + return false; + } + return readDocument(parser.ast()); +} + +QStringList SimpleAbstractStreamReader::errors() const +{ + return m_errors; +} + +void SimpleAbstractStreamReader::addError(const QString &error, const AST::SourceLocation &sourceLocation) +{ + m_errors << QString("%1:%2: %3\n").arg( + QString::number(sourceLocation.startLine), + QString::number(sourceLocation.startColumn), + error); +} + +AST::SourceLocation SimpleAbstractStreamReader::currentSourceLocation() const +{ + return m_currentSourceLocation; +} + +bool SimpleAbstractStreamReader::readDocument(AST::UiProgram *ast) +{ + if (!ast) { + addError(tr("Could not parse document")); + false; + } + + AST::UiObjectDefinition *uiObjectDefinition = AST::cast<AST::UiObjectDefinition *>(ast->members->member); + if (!uiObjectDefinition) { + addError(tr("Expected document to contain a single object definition")); + false; + } + readChild(uiObjectDefinition); + + return errors().isEmpty(); +} + +void SimpleAbstractStreamReader::readChildren(AST::UiObjectDefinition *uiObjectDefinition) +{ + Q_ASSERT(uiObjectDefinition); + + for (AST::UiObjectMemberList *it = uiObjectDefinition->initializer->members; it; it = it->next) { + AST::UiObjectMember *member = it->member; + AST::UiObjectDefinition *uiObjectDefinition = AST::cast<AST::UiObjectDefinition *>(member); + if (uiObjectDefinition) + readChild(uiObjectDefinition); + } +} + +void SimpleAbstractStreamReader::readChild(AST::UiObjectDefinition *uiObjectDefinition) +{ + Q_ASSERT(uiObjectDefinition); + + setSourceLocation(uiObjectDefinition->firstSourceLocation()); + + elementStart(toString(uiObjectDefinition->qualifiedTypeNameId)); + + readProperties(uiObjectDefinition); + readChildren(uiObjectDefinition); + + elementEnd(); +} + +void SimpleAbstractStreamReader::readProperties(AST::UiObjectDefinition *uiObjectDefinition) +{ + Q_ASSERT(uiObjectDefinition); + + for (AST::UiObjectMemberList *it = uiObjectDefinition->initializer->members; it; it = it->next) { + AST::UiObjectMember *member = it->member; + AST::UiScriptBinding *scriptBinding = AST::cast<AST::UiScriptBinding *>(member); + if (scriptBinding) + readProperty(scriptBinding); + } +} + +void SimpleAbstractStreamReader::readProperty(AST::UiScriptBinding *uiScriptBinding) +{ + Q_ASSERT(uiScriptBinding); + + setSourceLocation(uiScriptBinding->firstSourceLocation()); + + const QString name = toString(uiScriptBinding->qualifiedId); + const QVariant value = parseProperty(uiScriptBinding); + + propertyDefinition(name, value); +} + +QVariant SimpleAbstractStreamReader::parseProperty(AST::UiScriptBinding *uiScriptBinding) +{ + Q_ASSERT(uiScriptBinding); + + AST::ExpressionStatement *expStmt = AST::cast<AST::ExpressionStatement *>(uiScriptBinding->statement); + if (!expStmt) { + addError(tr("Expected expression statement after colon"), uiScriptBinding->statement->firstSourceLocation()); + return QVariant(); + } + + AST::StringLiteral *stringLiteral = AST::cast<AST::StringLiteral *>(expStmt->expression); + if (stringLiteral) + return stringLiteral->value.toString(); + + AST::TrueLiteral *trueLiteral = AST::cast<AST::TrueLiteral *>(expStmt->expression); + if (trueLiteral) + return true; + + AST::FalseLiteral *falseLiteral = AST::cast<AST::FalseLiteral *>(expStmt->expression); + if (falseLiteral) + return false; + + AST::NumericLiteral *numericLiteral = AST::cast<AST::NumericLiteral *>(expStmt->expression); + if (numericLiteral) + return numericLiteral->value; + + addError(tr("Expected expression statement to be a literal"), uiScriptBinding->statement->firstSourceLocation()); + return QVariant(); +} + +void SimpleAbstractStreamReader::setSourceLocation(const AST::SourceLocation &sourceLocation) +{ + m_currentSourceLocation = sourceLocation; +} + +SimpleReader::SimpleReader() +{ +} + +SimpleReaderNode::Ptr SimpleReader::readFile(const QString &fileName) +{ + SimpleAbstractStreamReader::readFile(fileName); + return m_rootNode; +} + +SimpleReaderNode::Ptr SimpleReader::readFromSource(const QString &source) +{ + SimpleAbstractStreamReader::readFromSource(source); + return m_rootNode; +} + +void SimpleReader::elementStart(const QString &name) +{ + if (debug) + qDebug() << "SimpleReader::elementStart()" << name; + + SimpleReaderNode::Ptr newNode = SimpleReaderNode::create(name, m_currentNode); + + if (newNode->isRoot()) + m_rootNode = newNode; + + Q_ASSERT(newNode->isValid()); + + m_currentNode = newNode; +} + +void SimpleReader::elementEnd() +{ + Q_ASSERT(m_currentNode); + + if (debug) + qDebug() << "SimpleReader::elementEnd()" << m_currentNode.data()->name(); + + m_currentNode = m_currentNode.data()->parent(); +} + +void SimpleReader::propertyDefinition(const QString &name, const QVariant &value) +{ + Q_ASSERT(m_currentNode); + + if (debug) + qDebug() << "SimpleReader::propertyDefinition()" << m_currentNode.data()->name() << name << value; + + if (m_currentNode.data()->propertyNames().contains(name)) + addError(tr("Property is defined twice"), currentSourceLocation()); + + m_currentNode.data()->setProperty(name, value); +} |