More fixes to the A* line splitting

This commit is contained in:
Hackerpilot 2015-02-18 23:21:12 -08:00
parent 7dea7ea962
commit 32759fafae
1 changed files with 42 additions and 42 deletions

View File

@ -171,7 +171,8 @@ private:
void formatStep() void formatStep()
{ {
import std.range:assumeSorted; import std.range : assumeSorted;
import std.algorithm : canFind;
assert (index < tokens.length); assert (index < tokens.length);
if (current.type == tok!"comment") if (current.type == tok!"comment")
@ -377,6 +378,7 @@ private:
case tok!";": case tok!";":
tempIndent = 0; tempIndent = 0;
writeToken(); writeToken();
linebreakHints = [];
if (index >= tokens.length || current.type != tok!"comment") if (index >= tokens.length || current.type != tok!"comment")
newline(); newline();
if (peekImplementation(tok!"class",0)) if (peekImplementation(tok!"class",0))
@ -386,24 +388,20 @@ private:
writeBraces(); writeBraces();
break; break;
case tok!".": case tok!".":
if (currentLineLength + nextTokenLength() >= config.columnHardLimit) writeToken();
{
pushIndent();
newline();
writeToken();
}
else
writeToken();
break; break;
case tok!",": case tok!",":
writeToken(); if (linebreakHints.canFind(index))
if (currentLineLength + distanceToNextPreferredBreak() >= config.columnSoftLimit)
{ {
writeToken();
pushIndent(); pushIndent();
newline(); newline();
} }
else else
{
writeToken();
write(" "); write(" ");
}
break; break;
case tok!"=": case tok!"=":
case tok!">=": case tok!">=":
@ -416,7 +414,13 @@ private:
case tok!"&=": case tok!"&=":
case tok!"%=": case tok!"%=":
case tok!"+=": case tok!"+=":
goto case; write(" ");
writeToken();
write(" ");
immutable size_t i = expressionEndIndex();
linebreakHints = chooseLineBreakTokens(index, tokens[index .. i],
config, currentLineLength, indentLevel);
break;
case tok!"^^": case tok!"^^":
case tok!"^=": case tok!"^=":
case tok!"^": case tok!"^":
@ -447,7 +451,7 @@ private:
case tok!"%": case tok!"%":
case tok!"&&": case tok!"&&":
binary: binary:
if (currentLineLength + distanceToNextPreferredBreak() >= config.columnSoftLimit) if (linebreakHints.canFind(index))
{ {
pushIndent(); pushIndent();
newline(); newline();
@ -600,6 +604,9 @@ private:
{ {
writeToken(); writeToken();
depth++; depth++;
immutable size_t i = expressionEndIndex();
linebreakHints = chooseLineBreakTokens(index, tokens[index .. i],
config, currentLineLength, indentLevel);
continue; continue;
} }
else if (current.type == tok!")") else if (current.type == tok!")")
@ -639,6 +646,7 @@ private:
while (index < tokens.length && depth > 0); while (index < tokens.length && depth > 0);
popIndent(); popIndent();
tempIndent = t; tempIndent = t;
linebreakHints = [];
} }
bool peekIsLabel() bool peekIsLabel()
@ -719,27 +727,6 @@ private:
return tokenLength(tokens[i]); return tokenLength(tokens[i]);
} }
int distanceToNextPreferredBreak() pure @safe @nogc
{
size_t i = index + 1;
int l;
loop: while (i < tokens.length) switch (tokens[i].type)
{
case tok!"||":
case tok!"&&":
case tok!";":
case tok!")":
case tok!",":
case tok!"(":
break loop;
default:
l += tokenLength(tokens[i]);
i++;
break;
}
return l;
}
ref current() const @property ref current() const @property
in in
{ {
@ -846,6 +833,8 @@ private:
/// Information about the AST /// Information about the AST
ASTInformation* astInformation; ASTInformation* astInformation;
size_t[] linebreakHints;
/// Configuration /// Configuration
FormatterConfig* config; FormatterConfig* config;
} }
@ -1157,7 +1146,7 @@ struct State
this._cost = breaks.map!(b => breakCost(tokens[b].type)).sum() + ((depth - 1) * 50); this._cost = breaks.map!(b => breakCost(tokens[b].type)).sum() + ((depth - 1) * 50);
int ll = currentLineLength; int ll = currentLineLength;
size_t breakIndex = 0; size_t breakIndex = 0;
size_t i; size_t i = 0;
bool s = true; bool s = true;
if (breaks.length == 0) if (breaks.length == 0)
{ {
@ -1190,8 +1179,12 @@ struct State
int opCmp(ref const State other) const pure nothrow @safe int opCmp(ref const State other) const pure nothrow @safe
{ {
if (cost < other.cost || (_solved && !other.solved)) if (cost < other.cost
|| (cost == other.cost && breaks.length && other.breaks.length && breaks[0] > other.breaks[0])
|| (cost == other.cost && _solved && !other.solved))
{
return -1; return -1;
}
return other.cost > _cost; return other.cost > _cost;
} }
@ -1212,7 +1205,7 @@ private:
bool _solved; bool _solved;
} }
size_t[] chooseLineBreakTokens(const Token[] tokens, size_t[] chooseLineBreakTokens(size_t index, const Token[] tokens,
const FormatterConfig* formatterConfig, int currentLineLength, int indentLevel) const FormatterConfig* formatterConfig, int currentLineLength, int indentLevel)
{ {
import std.container.rbtree : RedBlackTree; import std.container.rbtree : RedBlackTree;
@ -1226,19 +1219,26 @@ size_t[] chooseLineBreakTokens(const Token[] tokens,
State current = open.front(); State current = open.front();
open.removeFront(); open.removeFront();
if (current.solved) if (current.solved)
return current.breaks; {
foreach (ref b; current.breaks)
b += index;
return current.breaks;
}
foreach (next; validMoves(tokens, current, formatterConfig, foreach (next; validMoves(tokens, current, formatterConfig,
currentLineLength, indentLevel, depth)) currentLineLength, indentLevel, depth))
{ {
open.insert(next); open.insert(next);
} }
} }
return open.empty ? [] : open.front().breaks; size_t[] retVal = open.empty ? [] : open.front().breaks;
foreach (ref b; retVal)
b += index;
return retVal;
} }
State[] validMoves(const Token[] tokens, ref const State current, State[] validMoves(const Token[] tokens, ref const State current,
const FormatterConfig* formatterConfig, int currentLineLength, const FormatterConfig* formatterConfig, int currentLineLength,
int indentLevel, int depth) int indentLevel, int depth)
{ {
import std.algorithm : sort, canFind; import std.algorithm : sort, canFind;
import std.array:insertInPlace; import std.array:insertInPlace;
@ -1267,5 +1267,5 @@ unittest
StringCache cache = StringCache(StringCache.defaultBucketCount); StringCache cache = StringCache(StringCache.defaultBucketCount);
auto tokens = byToken(cast(ubyte[]) sourceCode, config, &cache).array(); auto tokens = byToken(cast(ubyte[]) sourceCode, config, &cache).array();
FormatterConfig formatterConfig; FormatterConfig formatterConfig;
assert ([15] == chooseLineBreakTokens(tokens, &formatterConfig, 0, 0)); assert ([15] == chooseLineBreakTokens(0, tokens, &formatterConfig, 0, 0));
} }