Rework indentation system. Fix #91. Fix #86. Fix #54.

This commit is contained in:
Hackerpilot 2015-03-11 05:13:13 -07:00
parent 52eabd4f47
commit 1ef17c5d78
13 changed files with 401 additions and 206 deletions

View File

@ -162,11 +162,6 @@ struct TokenFormatter(OutputRange)
formatStep(); formatStep();
} }
invariant
{
assert (indentLevel >= 0);
}
private: private:
void formatStep() void formatStep()
@ -228,7 +223,6 @@ private:
if (currentIs(tok!";")) if (currentIs(tok!";"))
{ {
writeToken(); writeToken();
tempIndent = 0;
if (index >= tokens.length) if (index >= tokens.length)
{ {
newline(); newline();
@ -265,7 +259,8 @@ private:
writeToken(); writeToken();
if (currentLineLength + 1 + length_of_next_chunk >= config.columnSoftLimit) if (currentLineLength + 1 + length_of_next_chunk >= config.columnSoftLimit)
{ {
pushIndent(); if (indents.tempIndents < 2)
indents.push(tok!",");
newline(); newline();
} }
else else
@ -281,8 +276,29 @@ private:
if (!currentIs(tok!";") && !currentIs(tok!")")) if (!currentIs(tok!";") && !currentIs(tok!")"))
write(" "); write(" ");
} }
else if (currentIs(tok!"with"))
{
if (indents.length == 0 || indents.top != tok!"switch")
indents.push(tok!"with");
writeToken();
write(" ");
if (currentIs(tok!"("))
writeParens(false);
if (!currentIs(tok!"switch") && !currentIs(tok!"with")
&& !currentIs(tok!"{"))
{
newline();
}
else if (!currentIs(tok!"{"))
write(" ");
}
else if (currentIs(tok!"switch")) else if (currentIs(tok!"switch"))
formatSwitch(); {
if (indents.length == 0 || indents.top != tok!"with")
indents.push(tok!"switch");
writeToken(); // switch
write(" ");
}
else if ((currentIs(tok!"version") || currentIs(tok!"extern")) else if ((currentIs(tok!"version") || currentIs(tok!"extern"))
&& peekIs(tok!"(")) && peekIs(tok!"("))
{ {
@ -292,38 +308,32 @@ private:
} }
else if (isBlockHeader() && peekIs(tok!"(", false)) else if (isBlockHeader() && peekIs(tok!"(", false))
{ {
if (currentIs(tok!"if") && !peekBackIs(tok!"else")) indents.push(current.type);
ifIndents.push(tempIndent);
writeToken(); writeToken();
write(" "); write(" ");
writeParens(false); writeParens(false);
if (currentIs(tok!"switch") || (currentIs(tok!"final") && peekIs(tok!"switch"))) if (currentIs(tok!"switch") || (currentIs(tok!"final") && peekIs(tok!"switch")))
write(" "); write(" ");
else if (currentIs(tok!"comment")) else if (currentIs(tok!"comment"))
{
if (!peekIs(tok!"{") && !peekIs(tok!";"))
pushIndent();
formatStep(); formatStep();
}
else if (!currentIs(tok!"{") && !currentIs(tok!";")) else if (!currentIs(tok!"{") && !currentIs(tok!";"))
{
pushIndent();
newline(); newline();
} }
}
else if (currentIs(tok!"else")) else if (currentIs(tok!"else"))
{ {
writeToken(); writeToken();
if (currentIs(tok!"if") || (currentIs(tok!"static") && peekIs(tok!"if")) if (currentIs(tok!"if") || (currentIs(tok!"static") && peekIs(tok!"if"))
|| currentIs(tok!"version")) || currentIs(tok!"version"))
{ {
if (indents.top() == tok!"if")
indents.pop();
write(" "); write(" ");
} }
else if (!currentIs(tok!"{") && !currentIs(tok!"comment")) else if (!currentIs(tok!"{") && !currentIs(tok!"comment"))
{ {
if (ifIndents.length) if (indents.top() == tok!"if")
ifIndents.pop(); indents.pop();
pushIndent(); indents.push(tok!"else");
newline(); newline();
} }
} }
@ -435,29 +445,23 @@ private:
astInformation.attributeDeclarationLines).equalRange( astInformation.attributeDeclarationLines).equalRange(
current.line).empty) current.line).empty)
{ {
indentLevel++;
writeToken(); writeToken();
if (!currentIs(tok!"{")) if (!currentIs(tok!"{"))
newline(); newline();
} }
else if (peekBackIs(tok!"identifier") && (index <= 1 else if (peekBackIs(tok!"identifier") && (peekBack2Is(tok!"{", true)
|| peekBack2Is(tok!"{", true) || peekBack2Is(tok!"}", true) || peekBack2Is(tok!"}", true) || peekBack2Is(tok!";", true)
|| peekBack2Is(tok!";", true) || peekBack2Is(tok!":", true)) || peekBack2Is(tok!":", true)) && !(isBlockHeader(1)
&& !peekIs(tok!"{") && !(isBlockHeader(1) && !peekIs(tok!"if"))) && !peekIs(tok!"if")))
{ {
indentLevel++;
writeToken(); writeToken();
if (!currentIs(tok!"{"))
newline(); newline();
} }
else else
{ {
if (peekIs(tok!"..")) if (peekIs(tok!".."))
writeToken(); writeToken();
else if (peekIs(tok!"{"))
{
writeToken();
pushIndent();
}
else if (isBlockHeader(1) && !peekIs(tok!"if")) else if (isBlockHeader(1) && !peekIs(tok!"if"))
{ {
writeToken(); writeToken();
@ -476,31 +480,64 @@ private:
write(" "); write(" ");
break; break;
case tok!";": case tok!";":
if (peekIs(tok!"else") && ifIndents.length)
tempIndent = ifIndents.top();
else if (!peekIs(tok!"}") || peekIs(tok!"comment", false))
{
if (ifIndents.length)
{
tempIndent = ifIndents.top();
ifIndents.pop();
}
else
tempIndent = 0;
}
writeToken(); writeToken();
linebreakHints = []; linebreakHints = [];
newline(); newline();
break; break;
case tok!"{": case tok!"{":
writeBraces(); if (assumeSorted(astInformation.structInitStartLocations)
.equalRange(tokens[index].index).length)
{
writeToken();
}
else
{
if (!justAddedExtraNewline && !peekBackIs(tok!"{")
&& !peekBackIs(tok!"}") && !peekBackIs(tok!";")
&& !peekBackIs(tok!";"))
{
if (config.braceStyle == BraceStyle.otbs)
{
write(" ");
}
else if (index > 0 && (!peekBackIs(tok!"comment") || tokens[index - 1].text[0 .. 2] != "//"))
newline();
}
writeToken();
newline();
}
break;
case tok!"}":
if (assumeSorted(astInformation.structInitEndLocations)
.equalRange(tokens[index].index).length)
{
writeToken();
}
else
{
// Silly hack to format enums better.
if (peekBackIsLiteralOrIdent() || peekBackIs(tok!","))
newline();
write("}");
if (index < tokens.length - 1 &&
assumeSorted(astInformation.doubleNewlineLocations)
.equalRange(tokens[index].index).length && !peekIs(tok!"}"))
{
output.put("\n");
justAddedExtraNewline = true;
}
if (config.braceStyle == BraceStyle.otbs && currentIs(tok!"else"))
write(" ");
index++;
newline();
}
break; break;
case tok!".": case tok!".":
if (linebreakHints.canFind(index) || (linebreakHints.length == 0 if (linebreakHints.canFind(index) || (linebreakHints.length == 0
&& currentLineLength + nextTokenLength() > config.columnHardLimit)) && currentLineLength + nextTokenLength() > config.columnHardLimit))
{ {
if (tempIndent < 2) if (indents.tempIndents < 2)
pushIndent(); indents.push(tok!".");
newline(); newline();
} }
writeToken(); writeToken();
@ -511,17 +548,15 @@ private:
&& currentLineLength > config.columnSoftLimit))) && currentLineLength > config.columnSoftLimit)))
{ {
writeToken(); writeToken();
if (tempIndent < 2) if (indents.tempIndents < 2)
pushIndent(); indents.push(tok!",");
newline(); newline();
} }
else else
{ {
writeToken(); writeToken();
if (currentIs(tok!"}", false)) if (!currentIs(tok!")", false) && !currentIs(tok!"]", false)
tempIndent = 0; && !currentIs(tok!"}", false) && !currentIs(tok!"comment", false))
else if (!currentIs(tok!")", false) && !currentIs(tok!"]", false)
&& !currentIs(tok!"comment", false))
{ {
write(" "); write(" ");
} }
@ -578,8 +613,8 @@ private:
binary: binary:
if (linebreakHints.canFind(index)) if (linebreakHints.canFind(index))
{ {
if (tempIndent < 2) if (indents.tempIndents < 2)
pushIndent(); indents.push(current.type);
newline(); newline();
} }
else else
@ -616,24 +651,6 @@ private:
} }
} }
/// Pushes a temporary indent level
void pushIndent()
{
// stderr.writeln("pushIndent: ", current.line, ",", current.column);
tempIndent++;
}
/// Pops a temporary indent level
void popIndent()
{
// if (index < tokens.length)
// stderr.writeln("popIndent: ", current.line, ",", current.column);
// else
// stderr.writeln("popIndent: EOF");
if (tempIndent > 0)
tempIndent--;
}
size_t expressionEndIndex(size_t i) const pure @safe @nogc size_t expressionEndIndex(size_t i) const pure @safe @nogc
{ {
int parenDepth = 0; int parenDepth = 0;
@ -680,79 +697,6 @@ private:
return i; return i;
} }
/// Writes balanced braces
void writeBraces()
{
import std.range : assumeSorted;
int depth = 0;
do
{
if (currentIs(tok!"{"))
{
braceIndents.push(indentLevel);
braceTempIndents.push(tempIndent);
depth++;
if (assumeSorted(astInformation.structInitStartLocations)
.equalRange(tokens[index].index).length)
{
writeToken();
}
else
{
if (index > 0 && !justAddedExtraNewline && !peekBackIs(tok!"{")
&& !peekBackIs(tok!"}") && !peekBackIs(tok!";"))
{
if (config.braceStyle == BraceStyle.otbs)
{
write(" ");
}
else if (!peekBackIs(tok!"comment") || tokens[index - 1].text[0 .. 2] != "//")
newline();
}
write("{");
indentLevel++;
index++;
newline();
}
}
else if (currentIs(tok!"}"))
{
depth--;
if (assumeSorted(astInformation.structInitEndLocations)
.equalRange(tokens[index].index).length)
{
writeToken();
}
else
{
// Silly hack to format enums better.
if (peekBackIsLiteralOrIdent() || peekBackIs(tok!","))
newline();
write("}");
if (index < tokens.length - 1 &&
assumeSorted(astInformation.doubleNewlineLocations)
.equalRange(tokens[index].index).length && !peekIs(tok!"}"))
{
output.put("\n");
justAddedExtraNewline = true;
}
if (config.braceStyle == BraceStyle.otbs && currentIs(tok!"else"))
write(" ");
index++;
if (ifIndents.length >= 2 && ifIndents.top == tempIndent && !currentIs(tok!"else"))
{
ifIndents.pop();
tempIndent = ifIndents.top();
}
newline();
}
}
else
formatStep();
}
while (index < tokens.length && depth > 0);
}
void writeParens(bool space_afterwards) void writeParens(bool space_afterwards)
in in
{ {
@ -762,7 +706,6 @@ private:
{ {
import std.range : assumeSorted; import std.range : assumeSorted;
immutable t = tempIndent;
int depth = 0; int depth = 0;
do do
{ {
@ -784,8 +727,7 @@ private:
&& currentLineLength > config.columnSoftLimit && currentLineLength > config.columnSoftLimit
&& !currentIs(tok!")"))) && !currentIs(tok!")")))
{ {
if (tempIndent < 2) indents.push(tok!"(");
pushIndent();
newline(); newline();
} }
regenLineBreakHintsIfNecessary(index - 1); regenLineBreakHintsIfNecessary(index - 1);
@ -819,7 +761,6 @@ private:
formatStep(); formatStep();
} }
while (index < tokens.length && depth > 0); while (index < tokens.length && depth > 0);
tempIndent = t;
linebreakHints = []; linebreakHints = [];
} }
@ -838,20 +779,6 @@ private:
return peekIs(tok!"identifier") && peek2Is(tok!":"); return peekIs(tok!"identifier") && peek2Is(tok!":");
} }
void formatSwitch()
{
switchIndents.push(indentLevel);
writeToken(); // switch
write(" ");
writeParens(true);
while (currentIs(tok!"with"))
{
writeToken();
write(" ");
writeParens(true);
}
}
int currentTokenLength() pure @safe @nogc int currentTokenLength() pure @safe @nogc
{ {
return tokenLength(tokens[index]); return tokenLength(tokens[index]);
@ -1015,36 +942,81 @@ private:
} }
justAddedExtraNewline = false; justAddedExtraNewline = false;
currentLineLength = 0; currentLineLength = 0;
if (hasCurrent) if (hasCurrent)
{ {
bool switchLabel = false; bool switchLabel = false;
if (switchIndents.length && (currentIs(tok!"case") || currentIs(tok!"default") if (currentIs(tok!"else"))
|| (currentIs(tok!"identifier") && peekIs(tok!":"))))
{ {
indentLevel = switchIndents.top(); auto l = indents.indentToMostRecent(tok!"if");
if (l != -1)
indentLevel = l;
}
else if (currentIs(tok!"identifier") && peekIs(tok!":"))
{
auto l = indents.indentToMostRecent(tok!"switch");
if (l != -1)
{
indentLevel = l;
switchLabel = true; switchLabel = true;
} }
if (currentIs(tok!"}")) else if (!isBlockHeader(2) || peek2Is(tok!"if"))
{ {
if (braceIndents.length) auto l2 = indents.indentToMostRecent(tok!"{");
{ indentLevel = l2 == -1 ? indentLevel : l2;
assert (braceTempIndents.length); }
indentLevel = braceIndents.top(); else
tempIndent = braceTempIndents.top(); indentLevel = indents.indentSize;
braceIndents.pop(); }
braceTempIndents.pop(); else if (currentIs(tok!"case") || currentIs(tok!"default"))
{
auto l = indents.indentToMostRecent(tok!"switch");
if (l != -1)
indentLevel = l;
}
else if (currentIs(tok!"{") && assumeSorted(
astInformation.structInitStartLocations).equalRange(
tokens[index].index).empty)
{
while (indents.length && isWrapIndent(indents.top))
indents.pop();
indents.push(tok!"{");
if (index == 1 || peekBackIs(tok!":") || peekBackIs(tok!"{")
|| peekBackIs(tok!"}") || peekBackIs(tok!")"))
{
indentLevel = indents.indentSize - 1;
}
}
else if (currentIs(tok!"}"))
{
while (indents.length && isTempIndent(indents.top()))
indents.pop();
if (indents.top == tok!"{")
{
indentLevel = indents.indentToMostRecent(tok!"{");
indents.pop();
}
while (indents.length && isTempIndent(indents.top)
&& (indents.top != tok!"if" || !peekIs(tok!"else")))
{
indents.pop();
} }
if (switchIndents.length && indentLevel == switchIndents.top)
switchIndents.pop();
} }
else if ((!assumeSorted(astInformation.attributeDeclarationLines) else if ((!assumeSorted(astInformation.attributeDeclarationLines)
.equalRange(current.line).empty) || (!switchLabel .equalRange(current.line).empty))
&& currentIs(tok!"identifier") && peekIs(tok!":")
&& (!isBlockHeader(2) || peek2Is(tok!"if"))))
{ {
popIndent(); auto l = indents.indentToMostRecent(tok!"{");
if (braceIndents.length) if (l != -1)
indentLevel = braceIndents.top(); indentLevel = l;
}
else
{
while ((peekBackIs(tok!"}", true) || peekBackIs(tok!";", true))
&& indents.length && isTempIndent(indents.top()))
{
indents.pop();
}
indentLevel = indents.indentSize;
} }
indent(); indent();
} }
@ -1068,15 +1040,14 @@ private:
void indent() void indent()
{ {
import std.range : repeat, take;
if (config.useTabs) if (config.useTabs)
foreach (i; 0 .. indentLevel + tempIndent) foreach (i; 0 .. indentLevel)
{ {
currentLineLength += config.tabSize; currentLineLength += config.tabSize;
output.put("\t"); output.put("\t");
} }
else else
foreach (i; 0 .. indentLevel + tempIndent) foreach (i; 0 .. indentLevel)
foreach (j; 0 .. config.indentSize) foreach (j; 0 .. config.indentSize)
{ {
output.put(" "); output.put(" ");
@ -1084,14 +1055,10 @@ private:
} }
} }
/// Current index into the tokens array
size_t index;
/// Current indent level
int indentLevel; int indentLevel;
/// Current temproray indententation level; /// Current index into the tokens array
int tempIndent; size_t index;
/// Length of the current line (so far) /// Length of the current line (so far)
uint currentLineLength = 0; uint currentLineLength = 0;
@ -1107,10 +1074,7 @@ private:
size_t[] linebreakHints; size_t[] linebreakHints;
FixedStack ifIndents; IndentStack indents;
FixedStack braceTempIndents;
FixedStack braceIndents;
FixedStack switchIndents;
/// Configuration /// Configuration
FormatterConfig* config; FormatterConfig* config;
@ -1120,6 +1084,16 @@ private:
bool justAddedExtraNewline; bool justAddedExtraNewline;
} }
bool isWrapIndent(IdType type) pure nothrow @nogc @safe
{
return type != tok!"{" && isOperator(type);
}
bool isTempIndent(IdType type) pure nothrow @nogc @safe
{
return type != tok!"{";
}
/// The only good brace styles /// The only good brace styles
enum BraceStyle enum BraceStyle
{ {
@ -1623,12 +1597,40 @@ State[] validMoves(const Token[] tokens, ref const State current,
return states; return states;
} }
struct FixedStack struct IndentStack
{ {
void push(int i) pure nothrow int indentToMostRecent(IdType item)
{
size_t i = index;
while (true)
{
if (arr[i] == item)
return indentSize(i);
if (i > 0)
i--;
else
return -1;
}
}
int tempIndents() const pure nothrow @property
{
if (index == 0)
return 0;
int tempIndentCount = 0;
for(size_t i = index; i > 0; i--)
{
if (!isTempIndent(arr[i]))
break;
tempIndentCount++;
}
return tempIndentCount;
}
void push(IdType item) pure nothrow
{ {
index = index == 255 ? index : index + 1; index = index == 255 ? index : index + 1;
arr[index] = i; arr[index] = item;
} }
void pop() pure nothrow void pop() pure nothrow
@ -1636,19 +1638,37 @@ struct FixedStack
index = index == 0 ? index : index - 1; index = index == 0 ? index : index - 1;
} }
int top() const pure nothrow @property IdType top() const pure nothrow @property
{ {
return arr[index]; return arr[index];
} }
size_t length() const pure nothrow @property int indentSize(size_t k = size_t.max) const /+pure nothrow+/
{ {
return index; if (index == 0)
return 0;
immutable size_t j = k == size_t.max ? index : k - 1;
int size = 0;
foreach (i; 1 .. j + 1)
{
if ((i + 1 <= index && !isWrapIndent(arr[i]) && isTempIndent(arr[i])
&& (!isTempIndent(arr[i + 1]))))
{
continue;
}
size++;
}
return size;
}
int length() const pure nothrow @property
{
return cast(int) index;
} }
private: private:
size_t index; size_t index;
int[256] arr; IdType[256] arr;
} }
unittest unittest

29
tests/issue0054.d Normal file
View File

@ -0,0 +1,29 @@
struct ClassFlags
{
alias Type = uint;
enum Enum : int
{
isCOMclass = 0x1,
noPointers = 0x2,
hasOffTi = 0x4,
hasCtor = 0x8,
hasGetMembers = 0x10,
hasTypeInfo = 0x20,
isAbstract = 0x40,
isCPPclass = 0x80,
hasDtor = 0x100,
}
alias isCOMclass = Enum.isCOMclass;
alias noPointers = Enum.noPointers;
alias hasOffTi = Enum.hasOffTi;
alias hasCtor = Enum.hasCtor;
alias hasGetMembers = Enum.hasGetMembers;
alias hasTypeInfo = Enum.hasTypeInfo;
alias isAbstract = Enum.isAbstract;
alias isCPPclass = Enum.isCPPclass;
alias hasDtor = Enum.hasDtor;
}

21
tests/issue0054.d.ref Normal file
View File

@ -0,0 +1,21 @@
struct ClassFlags
{
alias Type = uint;
enum Enum : int
{
isCOMclass = 0x1, noPointers = 0x2, hasOffTi = 0x4, hasCtor = 0x8,
hasGetMembers = 0x10, hasTypeInfo = 0x20, isAbstract = 0x40,
isCPPclass = 0x80, hasDtor = 0x100,
}
alias isCOMclass = Enum.isCOMclass;
alias noPointers = Enum.noPointers;
alias hasOffTi = Enum.hasOffTi;
alias hasCtor = Enum.hasCtor;
alias hasGetMembers = Enum.hasGetMembers;
alias hasTypeInfo = Enum.hasTypeInfo;
alias isAbstract = Enum.isAbstract;
alias isCPPclass = Enum.isCPPclass;
alias hasDtor = Enum.hasDtor;
}

View File

@ -1 +1 @@
{{{}}{}} unittest{{{}}{}{{{{}}}}}

View File

@ -1,3 +1,4 @@
unittest
{ {
{ {
{ {
@ -5,4 +6,12 @@
} }
{ {
} }
{
{
{
{
}
}
}
}
} }

26
tests/issue0086.d Normal file
View File

@ -0,0 +1,26 @@
unittest
{
if (a)
if (b)
doSomething();
doSomethingElse();
}
void indent()
{
import std.range : repeat, take;
if (config.useTabs)
foreach (i; 0 .. indentLevel + tempIndent)
{
currentLineLength += config.tabSize;
output.put("\t");
}
else
foreach (i; 0 .. indentLevel + tempIndent)
foreach (j; 0 .. config.indentSize)
{
output.put(" ");
currentLineLength++;
}
}

26
tests/issue0086.d.ref Normal file
View File

@ -0,0 +1,26 @@
unittest
{
if (a)
if (b)
doSomething();
doSomethingElse();
}
void indent()
{
import std.range : repeat, take;
if (config.useTabs)
foreach (i; 0 .. indentLevel + tempIndent)
{
currentLineLength += config.tabSize;
output.put("\t");
}
else
foreach (i; 0 .. indentLevel + tempIndent)
foreach (j; 0 .. config.indentSize)
{
output.put(" ");
currentLineLength++;
}
}

13
tests/issue0091.d Normal file
View File

@ -0,0 +1,13 @@
unittest
{
switch (x)
{
version (none)
{
x();
case Case:
doSomething();
doSomethingElse();
}
}
}

13
tests/issue0091.d.ref Normal file
View File

@ -0,0 +1,13 @@
unittest
{
switch (x)
{
version (none)
{
x();
case Case:
doSomething();
doSomethingElse();
}
}
}

14
tests/longParamList.d Normal file
View File

@ -0,0 +1,14 @@
version (AArch64)
{
class SomeLongClassName
{
public:
double javaStyleFunctionName(double alpha, double bravo, double charlie, double delta, double echo, double foxtrot, double golf, double hotel)
{
if (alpha < beta && alpha > golf && hotel < alpha && bravo >= charlie && echo < delta)
{
if (alpha < beta && alpha > golf && hotel < alpha && bravo >= charlie && echo < delta)
{
if (alpha < beta && alpha > golf && hotel < alpha && bravo >= charlie && echo < delta)
{}}}}}}

24
tests/longParamList.d.ref Normal file
View File

@ -0,0 +1,24 @@
version (AArch64)
{
class SomeLongClassName
{
public:
double javaStyleFunctionName(double alpha, double bravo, double charlie,
double delta, double echo, double foxtrot, double golf, double hotel)
{
if (alpha < beta && alpha > golf && hotel < alpha && bravo >= charlie
&& echo < delta)
{
if (alpha < beta && alpha > golf && hotel < alpha && bravo >= charlie
&& echo < delta)
{
if (alpha < beta && alpha > golf && hotel < alpha && bravo >= charlie
&& echo < delta)
{
}
}
}
}
}
}