2019-12-13 15:10:46 +00:00
|
|
|
/****************************************************************************
|
|
|
|
**
|
|
|
|
** 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 <QCoreApplication>
|
|
|
|
#include <QFile>
|
2020-06-18 08:25:30 +00:00
|
|
|
#include <QTextStream>
|
2019-12-13 15:10:46 +00:00
|
|
|
|
|
|
|
#include <QtQml/private/qqmljslexer_p.h>
|
|
|
|
#include <QtQml/private/qqmljsparser_p.h>
|
|
|
|
#include <QtQml/private/qqmljsengine_p.h>
|
|
|
|
#include <QtQml/private/qqmljsastvisitor_p.h>
|
|
|
|
#include <QtQml/private/qqmljsast_p.h>
|
|
|
|
|
|
|
|
#if QT_CONFIG(commandlineparser)
|
|
|
|
#include <QCommandLineParser>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "commentastvisitor.h"
|
|
|
|
#include "dumpastvisitor.h"
|
|
|
|
#include "restructureastvisitor.h"
|
|
|
|
|
2020-03-18 08:19:31 +00:00
|
|
|
bool parseFile(const QString& filename, bool inplace, bool verbose, bool sortImports, bool force, const QString& newline)
|
2019-12-13 15:10:46 +00:00
|
|
|
{
|
|
|
|
QFile file(filename);
|
|
|
|
|
|
|
|
if (!file.open(QIODevice::Text | QIODevice::ReadOnly)) {
|
|
|
|
qWarning().noquote() << "Failed to open" << filename << "for reading.";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString code = QString::fromUtf8(file.readAll());
|
|
|
|
file.close();
|
|
|
|
|
|
|
|
QQmlJS::Engine engine;
|
|
|
|
QQmlJS::Lexer lexer(&engine);
|
|
|
|
|
|
|
|
lexer.setCode(code, 1, true);
|
|
|
|
QQmlJS::Parser parser(&engine);
|
|
|
|
|
|
|
|
bool success = parser.parse();
|
|
|
|
|
|
|
|
if (!success) {
|
|
|
|
const auto diagnosticMessages = parser.diagnosticMessages();
|
|
|
|
for (const QQmlJS::DiagnosticMessage &m : diagnosticMessages) {
|
|
|
|
qWarning().noquote() << QString::fromLatin1("%1:%2 : %3")
|
2020-02-27 09:49:14 +00:00
|
|
|
.arg(filename).arg(m.loc.startLine).arg(m.message);
|
2019-12-13 15:10:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
qWarning().noquote() << "Failed to parse" << filename;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to attach comments to AST nodes
|
|
|
|
CommentAstVisitor comment(&engine, parser.rootNode());
|
|
|
|
|
|
|
|
if (verbose)
|
|
|
|
qWarning().noquote() << comment.attachedComments().size() << "comment(s) attached.";
|
|
|
|
|
|
|
|
if (verbose) {
|
|
|
|
int orphaned = 0;
|
|
|
|
|
|
|
|
for (const auto& orphanList : comment.orphanComments().values())
|
|
|
|
orphaned += orphanList.size();
|
|
|
|
|
|
|
|
qWarning().noquote() << orphaned << "comments are orphans.";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (verbose && sortImports)
|
|
|
|
qWarning().noquote() << "Sorting imports";
|
|
|
|
|
|
|
|
// Do the actual restructuring
|
|
|
|
RestructureAstVisitor restructure(parser.rootNode(), sortImports);
|
|
|
|
|
|
|
|
// Turn AST back into source code
|
|
|
|
if (verbose)
|
|
|
|
qWarning().noquote() << "Dumping" << filename;
|
|
|
|
|
2020-06-12 11:52:38 +00:00
|
|
|
DumpAstVisitor dump(&engine, parser.rootNode(), &comment);
|
2019-12-13 15:10:46 +00:00
|
|
|
|
2020-01-17 13:28:09 +00:00
|
|
|
QString dumpCode = dump.toString();
|
|
|
|
|
|
|
|
lexer.setCode(dumpCode, 1, true);
|
|
|
|
|
|
|
|
bool dumpSuccess = parser.parse();
|
|
|
|
|
|
|
|
if (!dumpSuccess) {
|
|
|
|
if (verbose) {
|
|
|
|
const auto diagnosticMessages = parser.diagnosticMessages();
|
|
|
|
for (const QQmlJS::DiagnosticMessage &m : diagnosticMessages) {
|
|
|
|
qWarning().noquote() << QString::fromLatin1("<formatted>:%2 : %3")
|
2020-02-27 09:49:14 +00:00
|
|
|
.arg(m.loc.startLine).arg(m.message);
|
2020-01-17 13:28:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
qWarning().noquote() << "Failed to parse formatted code.";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dump.error() || !dumpSuccess) {
|
2020-01-15 10:25:18 +00:00
|
|
|
if (force) {
|
|
|
|
qWarning().noquote() << "An error has occurred. The output may not be reliable.";
|
|
|
|
} else {
|
2020-01-17 13:28:09 +00:00
|
|
|
qWarning().noquote() << "An error has occurred. Aborting.";
|
2020-01-15 10:25:18 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2019-12-13 15:10:46 +00:00
|
|
|
|
2020-03-18 08:19:31 +00:00
|
|
|
|
|
|
|
const bool native = newline == "native";
|
|
|
|
|
|
|
|
if (!native) {
|
|
|
|
if (newline == "macos") {
|
|
|
|
dumpCode = dumpCode.replace("\n","\r");
|
|
|
|
} else if (newline == "windows") {
|
|
|
|
dumpCode = dumpCode.replace("\n", "\r\n");
|
|
|
|
} else if (newline == "unix") {
|
|
|
|
// Nothing needs to be done for unix line-endings
|
|
|
|
} else {
|
|
|
|
qWarning().noquote() << "Unknown line ending type" << newline;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-13 15:10:46 +00:00
|
|
|
if (inplace) {
|
|
|
|
if (verbose)
|
|
|
|
qWarning().noquote() << "Writing to file" << filename;
|
|
|
|
|
2020-03-18 08:19:31 +00:00
|
|
|
if (!file.open(native ? QIODevice::WriteOnly | QIODevice::Text : QIODevice::WriteOnly))
|
2019-12-13 15:10:46 +00:00
|
|
|
{
|
|
|
|
qWarning().noquote() << "Failed to open" << filename << "for writing";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-01-17 13:28:09 +00:00
|
|
|
file.write(dumpCode.toUtf8());
|
2019-12-13 15:10:46 +00:00
|
|
|
file.close();
|
|
|
|
} else {
|
2020-03-18 08:19:31 +00:00
|
|
|
QFile out;
|
|
|
|
out.open(stdout, QIODevice::WriteOnly);
|
|
|
|
out.write(dumpCode.toUtf8());
|
2019-12-13 15:10:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
QCoreApplication app(argc, argv);
|
|
|
|
QCoreApplication::setApplicationName("qmlformat");
|
|
|
|
QCoreApplication::setApplicationVersion("1.0");
|
|
|
|
|
|
|
|
bool success = true;
|
|
|
|
#if QT_CONFIG(commandlineparser)
|
|
|
|
QCommandLineParser parser;
|
|
|
|
parser.setApplicationDescription("Formats QML files according to the QML Coding Conventions.");
|
|
|
|
parser.addHelpOption();
|
|
|
|
parser.addVersionOption();
|
|
|
|
|
|
|
|
parser.addOption(QCommandLineOption({"V", "verbose"},
|
|
|
|
QStringLiteral("Verbose mode. Outputs more detailed information.")));
|
|
|
|
|
|
|
|
parser.addOption(QCommandLineOption({"n", "no-sort"},
|
|
|
|
QStringLiteral("Do not sort imports.")));
|
|
|
|
|
|
|
|
parser.addOption(QCommandLineOption({"i", "inplace"},
|
|
|
|
QStringLiteral("Edit file in-place instead of outputting to stdout.")));
|
|
|
|
|
2020-01-15 10:25:18 +00:00
|
|
|
parser.addOption(QCommandLineOption({"f", "force"},
|
|
|
|
QStringLiteral("Continue even if an error has occurred.")));
|
|
|
|
|
2020-06-18 08:25:30 +00:00
|
|
|
parser.addOption(QCommandLineOption(
|
|
|
|
{ "F", "files" }, QStringLiteral("Format all files listed in file, in-place"), "file"));
|
|
|
|
|
2020-03-18 08:19:31 +00:00
|
|
|
parser.addOption(QCommandLineOption({"l", "newline"},
|
|
|
|
QStringLiteral("Override the new line format to use (native macos unix windows)."),
|
|
|
|
"newline", "native"));
|
|
|
|
|
2019-12-13 15:10:46 +00:00
|
|
|
parser.addPositionalArgument("filenames", "files to be processed by qmlformat");
|
|
|
|
|
|
|
|
parser.process(app);
|
|
|
|
|
|
|
|
const auto positionalArguments = parser.positionalArguments();
|
|
|
|
|
2020-06-18 08:25:30 +00:00
|
|
|
if (positionalArguments.isEmpty() && !parser.isSet("files"))
|
2019-12-13 15:10:46 +00:00
|
|
|
parser.showHelp(-1);
|
|
|
|
|
2020-03-18 08:19:31 +00:00
|
|
|
if (!parser.isSet("inplace") && parser.value("newline") != "native") {
|
|
|
|
qWarning() << "Error: The -l option can only be used with -i";
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2020-06-18 08:25:30 +00:00
|
|
|
if (parser.isSet("files")) {
|
|
|
|
if (!positionalArguments.isEmpty())
|
|
|
|
qWarning() << "Warning: Positional arguments are ignored when -F is used";
|
|
|
|
|
|
|
|
QFile file(parser.value("files"));
|
|
|
|
file.open(QIODevice::Text | QIODevice::ReadOnly);
|
|
|
|
QTextStream in(&file);
|
|
|
|
while (!in.atEnd()) {
|
|
|
|
QString file = in.readLine();
|
|
|
|
|
|
|
|
if (file.isEmpty())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!parseFile(file, true, parser.isSet("verbose"), !parser.isSet("no-sort"),
|
|
|
|
parser.isSet("force"), parser.value("newline")))
|
|
|
|
success = false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (const QString &file : parser.positionalArguments()) {
|
|
|
|
if (!parseFile(file, parser.isSet("inplace"), parser.isSet("verbose"),
|
|
|
|
!parser.isSet("no-sort"), parser.isSet("force"),
|
|
|
|
parser.value("newline")))
|
|
|
|
success = false;
|
|
|
|
}
|
2019-12-13 15:10:46 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return success ? 0 : 1;
|
|
|
|
}
|