/************************************************************************** ** ** This file is part of Qt Creator ** ** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies). ** ** Contact: Nokia Corporation (info@qt.nokia.com) ** ** ** GNU Lesser General Public License Usage ** ** 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, Nokia gives you certain additional ** rights. These rights are described in the Nokia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** Other Usage ** ** Alternatively, this file may be used in accordance with the terms and ** conditions contained in a signed written agreement between you and Nokia. ** ** If you have questions regarding the use of this file, please contact ** Nokia at qt-info@nokia.com. ** **************************************************************************/ #include #include #include #include #include #include using namespace CppTools; class tst_CodeFormatter: public QObject { Q_OBJECT private Q_SLOTS: void ifStatementWithoutBraces1(); void ifStatementWithoutBraces2(); void ifStatementWithBraces1(); void ifStatementWithBraces2(); void ifStatementMixed(); void ifStatementAndComments(); void ifStatementLongCondition(); void strayElse(); void macrosNoSemicolon(); void oneLineIf(); void doWhile(); void closingCurlies(); void ifdefedInsideIf(); void ifdefs(); void preprocessorContinuation(); void cStyleComments(); void cppStyleComments(); void expressionContinuation(); void classAccess(); void ternary(); void objcAtDeclarations(); void objcCall(); void objcCallAndFor(); void braceList(); void bug1(); void bug2(); void bug3(); void switch1(); void memberInitializer(); void templates(); void operatorOverloads(); void gnuStyle(); void whitesmithsStyle(); void singleLineEnum(); void functionReturnType(); void streamOp(); void blockStmtInIf(); void nestedInitializer(); void forStatement(); void templateSingleline(); void macrosNoSemicolon2(); void renamedNamespace(); void cpp0xFor(); void gnuStyleSwitch(); void whitesmithsStyleSwitch(); void indentToNextToken(); void labels(); void functionsWithExtraSpecifier(); }; struct Line { Line(QString l) : line(l) , expectedIndent(-1) , expectedPadding(0) { for (int i = 0; i < l.size(); ++i) { if (!l.at(i).isSpace()) { expectedIndent = i; break; } } if (expectedIndent == -1) expectedIndent = l.size(); if (expectedIndent < l.size() && l.at(expectedIndent) == '~') { line[expectedIndent] = ' '; for (int i = expectedIndent + 1; i < l.size(); ++i) { if (!l.at(i).isSpace()) { expectedPadding = i - expectedIndent; break; } } } } Line(QString l, int expectIndent, int expectPadding = 0) : line(l), expectedIndent(expectIndent), expectedPadding(expectPadding) { for (int i = 0; i < line.size(); ++i) { if (line.at(i) == '~') { line[i] = ' '; break; } if (!line.at(i).isSpace()) break; } } QString line; int expectedIndent; int expectedPadding; }; QString concatLines(QList lines) { QString result; foreach (const Line &l, lines) { result += l.line; result += "\n"; } return result; } void checkIndent(QList data, int style = 0) { QString text = concatLines(data); QTextDocument document(text); QtStyleCodeFormatter formatter; if (style == 1) {// gnu formatter.setIndentSubstatementBraces(true); } else if (style == 2) { // whitesmiths formatter.setIndentSubstatementStatements(false); formatter.setIndentSubstatementBraces(true); formatter.setIndentDeclarationMembers(false); formatter.setIndentDeclarationBraces(true); } int i = 0; foreach (const Line &l, data) { QTextBlock b = document.findBlockByLineNumber(i); if (l.expectedIndent != -1) { int indent, padding; formatter.indentFor(b, &indent, &padding); QVERIFY2(indent == l.expectedIndent, qPrintable(QString("Wrong indent in line %1 with text '%2', expected indent %3, got %4") .arg(QString::number(i+1), l.line, QString::number(l.expectedIndent), QString::number(indent)))); QVERIFY2(padding == l.expectedPadding, qPrintable(QString("Wrong padding in line %1 with text '%2', expected padding %3, got %4") .arg(QString::number(i+1), l.line, QString::number(l.expectedPadding), QString::number(padding)))); } formatter.updateLineStateChange(b); ++i; } } void tst_CodeFormatter::ifStatementWithoutBraces1() { QList data; data << Line("void foo() {") << Line(" if (a)") << Line(" if (b)") << Line(" foo;") << Line(" else if (c)") << Line(" foo;") << Line(" else") << Line(" if (d)") << Line(" foo;") << Line(" else") << Line(" while (e)") << Line(" bar;") << Line(" else") << Line(" foo;") << Line("}") ; checkIndent(data); } void tst_CodeFormatter::ifStatementWithoutBraces2() { QList data; data << Line("void foo() {") << Line(" if (a)") << Line(" if (b)") << Line(" foo;") << Line(" if (a) b();") << Line(" if (a) b(); else") << Line(" foo;") << Line(" if (a)") << Line(" if (b)") << Line(" foo;") << Line(" else if (c)") << Line(" foo;") << Line(" else") << Line(" if (d)") << Line(" foo;") << Line(" else") << Line(" while (e)") << Line(" bar;") << Line(" else") << Line(" foo;") << Line("}") ; checkIndent(data); } void tst_CodeFormatter::ifStatementWithBraces1() { QList data; data << Line("void foo() {") << Line(" if (a) {") << Line(" if (b) {") << Line(" foo;") << Line(" } else if (c) {") << Line(" foo;") << Line(" } else {") << Line(" if (d) {") << Line(" foo;") << Line(" } else {") << Line(" foo;") << Line(" }") << Line(" }") << Line(" } else {") << Line(" foo;") << Line(" }") << Line("}"); checkIndent(data); } void tst_CodeFormatter::ifStatementWithBraces2() { QList data; data << Line("void foo()") << Line("{") << Line(" if (a)") << Line(" {") << Line(" if (b)") << Line(" {") << Line(" foo;") << Line(" }") << Line(" else if (c)") << Line(" {") << Line(" foo;") << Line(" }") << Line(" else") << Line(" {") << Line(" if (d)") << Line(" {") << Line(" foo;") << Line(" }") << Line(" else") << Line(" {") << Line(" foo;") << Line(" }") << Line(" }") << Line(" }") << Line(" else") << Line(" {") << Line(" foo;") << Line(" }") << Line("}"); checkIndent(data); } void tst_CodeFormatter::ifStatementMixed() { QList data; data << Line("void foo()") << Line("{") << Line(" if (foo)") << Line(" if (bar)") << Line(" {") << Line(" foo;") << Line(" }") << Line(" else") << Line(" if (car)") << Line(" {}") << Line(" else doo;") << Line(" else abc;") << Line("}"); checkIndent(data); } void tst_CodeFormatter::ifStatementAndComments() { QList data; data << Line("void foo()") << Line("{") << Line(" if (foo)") << Line(" ; // bla") << Line(" else if (bar)") << Line(" ;") << Line(" if (foo)") << Line(" ; /* bla") << Line(" bla */") << Line(" else if (bar)") << Line(" // foobar") << Line(" ;") << Line(" else if (bar)") << Line(" /* bla") << Line(" bla */") << Line(" ;") << Line("}"); checkIndent(data); } void tst_CodeFormatter::ifStatementLongCondition() { QList data; data << Line("void foo()") << Line("{") << Line(" if (foo &&") << Line(" ~ bar") << Line(" ~ || (a + b > 4") << Line(" ~ && foo(bar)") << Line(" ~ )") << Line(" ~ ) {") << Line(" foo;") << Line(" }") << Line("}"); checkIndent(data); } void tst_CodeFormatter::strayElse() { QList data; data << Line("void foo()") << Line("{") << Line(" while( true ) {}") << Line(" else", -1) << Line(" else {", -1) << Line(" }", -1) << Line("}"); checkIndent(data); } void tst_CodeFormatter::macrosNoSemicolon() { QList data; data << Line("QT_DECLARE_METATYPE(foo)") << Line("int i;") << Line("Q_BLABLA") << Line("int i;") << Line("Q_BLABLA;") << Line("int i;") << Line("Q_BLABLA();") << Line("int i;") << Line("QML_DECLARE_TYPE(a, b, c, d)") << Line("int i;") << Line("Q_PROPERTY(abc)") << Line("QDOC_PROPERTY(abc)") << Line("void foo() {") << Line(" QT_DECLARE_METATYPE(foo)") << Line(" int i;") << Line(" Q_BLABLA") << Line(" int i;") << Line(" Q_BLABLA;") << Line(" int i;") << Line(" Q_BLABLA();") << Line(" Q_PROPERTY(abc)") << Line(" QDOC_PROPERTY(abc)") << Line(" int i;") << Line("}") ; checkIndent(data); } void tst_CodeFormatter::oneLineIf() { QList data; data << Line("class F {") << Line(" void foo()") << Line(" { if (showIt) show(); }") << Line("};") ; checkIndent(data); } void tst_CodeFormatter::doWhile() { QList data; data << Line("void foo() {") << Line(" do { if (c) foo; } while(a);") << Line(" do {") << Line(" if(a);") << Line(" } while(a);") << Line(" do") << Line(" foo;") << Line(" while(a);") << Line(" do foo; while(a);") << Line("};") ; checkIndent(data); } void tst_CodeFormatter::closingCurlies() { QList data; data << Line("void foo() {") << Line(" if (a)") << Line(" if (b) {") << Line(" foo;") << Line(" }") << Line(" {") << Line(" foo();") << Line(" }") << Line(" foo();") << Line(" {") << Line(" foo();") << Line(" }") << Line(" while (a) {") << Line(" if (a);") << Line(" }") << Line("};") ; checkIndent(data); } void tst_CodeFormatter::ifdefedInsideIf() { QList data; data << Line("void foo() {") << Line(" if (a) {") << Line("#ifndef Q_WS_WINCE") << Line(" if (b) {") << Line("#else") << Line(" if (c) {") << Line("#endif") << Line(" }") << Line(" } else if (d) {") << Line(" }") << Line(" if (a)") << Line(" ;") << Line(" else if (type == Qt::Dialog || type == Qt::Sheet)") << Line("#ifndef Q_WS_WINCE") << Line(" flags |= Qt::WindowTitleHint | Qt::WindowSystemMenuHint | Qt::WindowContextHelpButtonHint | Qt::WindowCloseButtonHint;") << Line("#else") << Line(" bar;") << Line("#endif") << Line("};") ; checkIndent(data); } void tst_CodeFormatter::ifdefs() { QList data; data << Line("#ifdef FOO") << Line("#include ") << Line("void foo()") << Line("{") << Line(" if (bar)") << Line("#if 1") << Line(" foo;") << Line(" else") << Line("#endif") << Line(" foo;") << Line("}") << Line("#endif") ; checkIndent(data); } void tst_CodeFormatter::preprocessorContinuation() { QList data; data << Line("#define x \\") << Line(" foo(x) + \\") << Line(" bar(x)") << Line("int i;") << Line("void foo() {") << Line("#define y y") << Line("#define x \\") << Line(" foo(x) + \\") << Line(" bar(x)") << Line(" int j;") << Line("};") ; checkIndent(data); } void tst_CodeFormatter::cStyleComments() { QList data; data << Line("/*") << Line(" ") << Line(" foo") << Line(" ") << Line(" foo") << Line(" ") << Line("*/") << Line("void foo() {") << Line(" /*") << Line(" ") << Line(" foo") << Line(" ") << Line(" */") << Line(" /* bar */") << Line("}") << Line("struct s {") << Line(" /* foo */") << Line(" /*") << Line(" ") << Line(" foo") << Line(" ") << Line(" */") << Line(" /* bar */") ; checkIndent(data); } void tst_CodeFormatter::cppStyleComments() { QList data; data << Line("// abc") << Line("class C { ") << Line(" // ghij") << Line(" // def") << Line(" // ghij") << Line(" int i; // hik") << Line(" // doo") << Line("} // ba") << Line("// ba") ; checkIndent(data); } void tst_CodeFormatter::expressionContinuation() { QList data; data << Line("void foo() {") << Line(" return (a <= b &&") << Line(" ~ c <= d);") << Line(" return (qMax <= qMin() &&") << Line(" ~ qMax(r1.top(), r2.top()) <= qMin(r1.bottom(), r2.bottom()));") << Line(" return a") << Line(" ~ == b && c == d;") << Line(" return a ==") << Line(" ~ b && c == d;") << Line(" return a == b") << Line(" ~ && c == d;") << Line(" return a == b &&") << Line(" ~ c == d;") << Line(" return a == b && c") << Line(" ~ == d;") << Line(" return a == b && c ==") << Line(" ~ d;") << Line(" return a == b && c == d;") << Line(" qDebug() << foo") << Line(" ~ << bar << moose") << Line(" ~ << bar +") << Line(" ~ foo - blah(1)") << Line(" ~ << '?'") << Line(" ~ << \"\\n\";") << Line(" i += foo(") << Line(" ~ bar,") << Line(" ~ 2);") << Line("}") ; checkIndent(data); } void tst_CodeFormatter::classAccess() { QList data; data << Line("class foo {") << Line(" int i;") << Line("public:") << Line(" class bar {") << Line(" private:") << Line(" int i;") << Line(" public:") << Line(" private slots:") << Line(" void foo();") << Line(" public Q_SLOTS:") << Line(" Q_SIGNALS:") << Line(" };") << Line(" float f;") << Line("private:") << Line(" void bar(){}") << Line("}") ; checkIndent(data); } void tst_CodeFormatter::ternary() { QList data; data << Line("void foo() {") << Line(" int i = a ? b : c;") << Line(" foo += a_bigger_condition ?") << Line(" ~ b") << Line(" ~ : c;") << Line(" int i = a ?") << Line(" ~ b : c;") << Line(" int i = a ? b") << Line(" ~ : c +") << Line(" ~ 2;") << Line(" int i = (a ? b : c) + (foo") << Line(" ~ bar);") << Line("}") ; checkIndent(data); } void tst_CodeFormatter::bug1() { QList data; data << Line("void foo() {") << Line(" if (attribute < int(8*sizeof(uint))) {") << Line(" if (on)") << Line(" data->widget_attributes |= (1<widget_attributes &= ~(1< data; data << Line("void foo() {") << Line(" const int sourceY = foo(") << Line(" ~ bar(") << Line(" ~ car(a,") << Line(" ~ b),") << Line(" ~ b),") << Line(" ~ foo);") << Line(" const int sourceY =") << Line(" ~ foo(") << Line(" ~ bar(a,") << Line(" ~ b),") << Line(" ~ b);") << Line(" int j;") << Line(" const int sourceY =") << Line(" ~ (direction == DirectionEast || direction == DirectionWest) ?") << Line(" ~ (sourceRect.top() + (sourceRect.bottom() - sourceRect.top()) / 2)") << Line(" ~ : (direction == DirectionSouth ? sourceRect.bottom() : sourceRect.top());") << Line("}") ; checkIndent(data); } void tst_CodeFormatter::bug3() { QList data; data << Line("class AutoAttack") << Line("{") << Line("public:") << Line(" AutoAttack(unsigned delay, unsigned warmup)") << Line(" ~ : mWarmup(warmup && warmup < delay ? warmup : delay >> 2)") << Line(" {}") << Line(" unsigned getWarmup() const { return mWarmup; }") << Line("private:") << Line(" unsigned mWarmup;") << Line("}") ; checkIndent(data); } void tst_CodeFormatter::braceList() { QList data; data << Line("enum Foo {") << Line(" a,") << Line(" b,") << Line("};") << Line("enum Foo { a = 2,") << Line(" ~ a = 3,") << Line(" ~ b = 4") << Line("~ };") << Line("void foo () {") << Line(" int a[] = { foo, bar, ") << Line(" ~ car };") << Line(" int a[] = {") << Line(" ~ a, b,") << Line(" ~ c") << Line(" };") << Line(" int k;") ; checkIndent(data); } void tst_CodeFormatter::objcAtDeclarations() { QList data; data << Line("@class Forwarded;") << Line("@protocol Forwarded;") << Line("int i;") ; checkIndent(data); } void tst_CodeFormatter::objcCall() { QList data; data << Line("void foo() {") << Line(" [NSApp windows];") << Line(" [NSObject class];") << Line(" if (a)") << Line(" int a = [window drawers];") << Line("}") << Line("int y;") ; checkIndent(data); } void tst_CodeFormatter::objcCallAndFor() { QList data; data << Line("void foo() {") << Line(" NSArray *windows = [NSApp windows];") << Line(" for (NSWindow *window in windows) {") << Line(" NSArray *drawers = [window drawers];") << Line(" for (NSDrawer *drawer in drawers) {") << Line(" NSArray *views = [[drawer contentView] subviews];") << Line(" int x;") << Line(" }") << Line(" }") << Line("}") << Line("int y;") ; checkIndent(data); } void tst_CodeFormatter::switch1() { QList data; data << Line("void foo() {") << Line(" switch (a) {") << Line(" case 1:") << Line(" foo;") << Line(" if (a);") << Line(" case 2:") << Line(" case 3: {") << Line(" foo;") << Line(" }") << Line(" case 4:") << Line(" {") << Line(" foo;") << Line(" }") << Line(" case bar:") << Line(" break;") << Line(" }") << Line(" case 4:") << Line(" {") << Line(" if (a) {") << Line(" }") << Line(" }") << Line("}") ; checkIndent(data); } void tst_CodeFormatter::memberInitializer() { QList data; data << Line("void foo()") << Line("~ : baR()") << Line("~ , m_member(23)") << Line("{") << Line("}") << Line("class Foo {") << Line(" Foo()") << Line(" ~ : baR(),") << Line(" ~ moodoo(barR + ") << Line(" ~ 42),") << Line(" ~ xyz()") << Line(" {}") << Line("};") << Line("class Foo {") << Line(" Foo() :") << Line(" ~ baR(),") << Line(" ~ moo(barR)") << Line(" ~ , moo(barR)") << Line(" {}") << Line("}") ; checkIndent(data); } void tst_CodeFormatter::templates() { QList data; data << Line("template ") << Line("class Foo {") << Line("private:") << Line("};") << Line("template ") << Line("class Foo {") << Line("private:") << Line("};") << Line("template