always add potential spaces after `@attr`

this uses a new `queueSpace` function, which inserts a space before
writing the next token, unless we already had a newline inserted. This
also fixes cases with double spaces.

This queueSpace function makes it much more resilient to grammar changes
because we can put it in a lot more spaces without worrying about
potentially inserting two spaces. It also catches more cases with
keep_new_lines as every inserted space checks for if it should keep the
existing new line at the current token.
This commit is contained in:
WebFreak001 2022-10-05 23:56:29 +02:00
parent 6744df20f3
commit e9f9f29224
No known key found for this signature in database
GPG Key ID: AEFC88D11109D1AA
7 changed files with 144 additions and 90 deletions

View File

@ -198,6 +198,11 @@ private:
/// True if the next "this" should have a space behind it /// True if the next "this" should have a space behind it
bool thisSpace; bool thisSpace;
/// True if the next write call should first write a space, unless it is a
/// new line or other whitespace, in which case no further space is inserted.
/// Call queueSpace() to set this to support keep_line_breaks.
bool queuedSpace;
/// True if the next "else" should be formatted as a single line /// True if the next "else" should be formatted as a single line
bool inlineElse; bool inlineElse;
@ -217,6 +222,15 @@ private:
return "\n"; return "\n";
} }
/// Returns currentLineLength + things that are yet to be written
uint virtualLineLength() const
{
uint offset = 0;
if (queuedSpace)
offset++;
return currentLineLength + offset;
}
void formatStep() void formatStep()
{ {
import std.range : assumeSorted; import std.range : assumeSorted;
@ -237,7 +251,7 @@ private:
|| isNumberLiteral(t) || t == tok!"characterLiteral" || isNumberLiteral(t) || t == tok!"characterLiteral"
// a!"b" function() // a!"b" function()
|| t == tok!"function" || t == tok!"delegate") || t == tok!"function" || t == tok!"delegate")
write(" "); queueSpace();
} }
} }
else if (currentIs(tok!"module") || currentIs(tok!"import")) else if (currentIs(tok!"module") || currentIs(tok!"import"))
@ -250,14 +264,14 @@ private:
if (!currentIs(tok!";") && !currentIs(tok!")") && !currentIs(tok!"{") if (!currentIs(tok!";") && !currentIs(tok!")") && !currentIs(tok!"{")
&& !currentIs(tok!"in") && !currentIs(tok!"out") && !currentIs(tok!"do") && !currentIs(tok!"in") && !currentIs(tok!"out") && !currentIs(tok!"do")
&& (hasCurrent && tokens[index].text != "body")) && (hasCurrent && tokens[index].text != "body"))
write(" "); queueSpace();
} }
else if (currentIs(tok!"with")) else if (currentIs(tok!"with"))
{ {
if (indents.length == 0 || !indents.topIsOneOf(tok!"switch", tok!"with")) if (indents.length == 0 || !indents.topIsOneOf(tok!"switch", tok!"with"))
indents.push(tok!"with"); indents.push(tok!"with");
writeToken(); writeToken();
write(" "); queueSpace();
if (currentIs(tok!"(")) if (currentIs(tok!"("))
writeParens(false); writeParens(false);
if (!currentIs(tok!"switch") && !currentIs(tok!"with") if (!currentIs(tok!"switch") && !currentIs(tok!"with")
@ -266,7 +280,7 @@ private:
newline(); newline();
} }
else if (!currentIs(tok!"{")) else if (!currentIs(tok!"{"))
write(" "); queueSpace();
} }
else if (currentIs(tok!"switch")) else if (currentIs(tok!"switch"))
{ {
@ -275,7 +289,7 @@ private:
else if (currentIs(tok!"extern") && peekIs(tok!"(")) else if (currentIs(tok!"extern") && peekIs(tok!"("))
{ {
writeToken(); writeToken();
write(" "); queueSpace();
while (hasCurrent) while (hasCurrent)
{ {
if (currentIs(tok!"(")) if (currentIs(tok!"("))
@ -338,7 +352,7 @@ private:
&& (thisSpace || astInformation.constructorDestructorLocations && (thisSpace || astInformation.constructorDestructorLocations
.canFindIndex(thisIndex))) .canFindIndex(thisIndex)))
{ {
write(" "); queueSpace();
thisSpace = false; thisSpace = false;
} }
} }
@ -351,8 +365,8 @@ private:
else if (isBasicType(current.type)) else if (isBasicType(current.type))
{ {
writeToken(); writeToken();
if (currentIs(tok!"identifier") || isKeyword(current.type) || inAsm) if (isLikeIdentifier(current.type) || inAsm)
write(" "); queueSpace();
} }
else if (isOperator(current.type)) else if (isOperator(current.type))
{ {
@ -375,7 +389,7 @@ private:
))) )))
//dfmt on //dfmt on
{ {
write(" "); queueSpace();
} }
} }
else if (currentIs(tok!"scriptLine") || currentIs(tok!"specialTokenSequence")) else if (currentIs(tok!"scriptLine") || currentIs(tok!"specialTokenSequence"))
@ -395,17 +409,17 @@ private:
case _unspecified: case _unspecified:
assert(false, "Config was not validated properly"); assert(false, "Config was not validated properly");
case conditional_newline: case conditional_newline:
immutable l = currentLineLength + betweenParenLength(tokens[index + 1 .. $]); immutable l = virtualLineLength + betweenParenLength(tokens[index + 1 .. $]);
if (l > config.dfmt_soft_max_line_length) if (l > config.dfmt_soft_max_line_length)
newline(); newline();
else if (peekBackIs(tok!")") || peekBackIs(tok!"identifier")) else if (peekBackIs(tok!")") || peekBackIs(tok!"identifier"))
write(" "); queueSpace();
break; break;
case always_newline: case always_newline:
newline(); newline();
break; break;
case conditional_newline_indent: case conditional_newline_indent:
immutable l = currentLineLength + betweenParenLength(tokens[index + 1 .. $]); immutable l = virtualLineLength + betweenParenLength(tokens[index + 1 .. $]);
if (l > config.dfmt_soft_max_line_length) if (l > config.dfmt_soft_max_line_length)
{ {
config.dfmt_single_template_constraint_indent == OB.t ? config.dfmt_single_template_constraint_indent == OB.t ?
@ -413,7 +427,7 @@ private:
newline(); newline();
} }
else if (peekBackIs(tok!")") || peekBackIs(tok!"identifier")) else if (peekBackIs(tok!")") || peekBackIs(tok!"identifier"))
write(" "); queueSpace();
break; break;
case always_newline_indent: case always_newline_indent:
{ {
@ -427,7 +441,7 @@ private:
writeToken(); writeToken();
// assume that the parens are present, otherwise the parser would not // assume that the parens are present, otherwise the parser would not
// have told us there was a constraint here // have told us there was a constraint here
write(" "); queueSpace();
writeParens(false); writeParens(false);
} }
@ -489,11 +503,11 @@ private:
if (peekBackIsOperator() && !peekBackIsOneOf(false, tok!"comment", if (peekBackIsOperator() && !peekBackIsOneOf(false, tok!"comment",
tok!"{", tok!"}", tok!":", tok!";", tok!",", tok!"[", tok!"(") tok!"{", tok!"}", tok!":", tok!";", tok!",", tok!"[", tok!"(")
&& !canAddNewline && prevTokenEndLine < currTokenLine) && !canAddNewline && prevTokenEndLine < currTokenLine)
write(" "); queueSpace();
else if (prevTokenEndLine == currTokenLine || (t == tok!")" && peekIs(tok!"{"))) else if (prevTokenEndLine == currTokenLine || (t == tok!")" && peekIs(tok!"{")))
write(" "); queueSpace();
else if (peekBackIsOneOf(false, tok!"else", tok!"identifier")) else if (peekBackIsOneOf(false, tok!"else", tok!"identifier"))
write(" "); queueSpace();
else if (canAddNewline || (peekIs(tok!"{") && t == tok!"}")) else if (canAddNewline || (peekIs(tok!"{") && t == tok!"}"))
newline(); newline();
@ -518,10 +532,10 @@ private:
{ {
if (indents.topIs(tok!"{")) if (indents.topIs(tok!"{"))
indents.pop(); indents.pop();
write(" "); queueSpace();
} }
else if (!currentIs(tok!"{")) else if (!currentIs(tok!"{"))
write(" "); queueSpace();
} }
else if (!currentIs(tok!"{") && !currentIs(tok!"in") && !currentIs(tok!"out")) else if (!currentIs(tok!"{") && !currentIs(tok!"in") && !currentIs(tok!"out"))
{ {
@ -546,7 +560,7 @@ private:
writeParens(false); writeParens(false);
return; return;
} }
write(" "); queueSpace();
while (hasCurrent) while (hasCurrent)
{ {
if (currentIs(tok!";")) if (currentIs(tok!";"))
@ -585,10 +599,10 @@ private:
else if (currentIs(tok!":")) else if (currentIs(tok!":"))
{ {
if (config.dfmt_selective_import_space) if (config.dfmt_selective_import_space)
write(" "); queueSpace();
writeToken(); writeToken();
if (!currentIs(tok!"comment")) if (!currentIs(tok!"comment"))
write(" "); queueSpace();
pushWrapIndent(tok!","); pushWrapIndent(tok!",");
} }
else if (currentIs(tok!"comment")) else if (currentIs(tok!"comment"))
@ -661,7 +675,7 @@ private:
newline(); newline();
immutable size_t j = expressionEndIndex(index); immutable size_t j = expressionEndIndex(index);
linebreakHints = chooseLineBreakTokens(index, tokens[index .. j], linebreakHints = chooseLineBreakTokens(index, tokens[index .. j],
depths[index .. j], config, currentLineLength, indentLevel); depths[index .. j], config, virtualLineLength, indentLevel);
} }
else if (p == tok!"[" && config.dfmt_keep_line_breaks == OptionalBoolean.t) else if (p == tok!"[" && config.dfmt_keep_line_breaks == OptionalBoolean.t)
{ {
@ -708,7 +722,7 @@ private:
} }
else if (!currentIs(tok!")") && !currentIs(tok!"]") else if (!currentIs(tok!")") && !currentIs(tok!"]")
&& (linebreakHints.canFindIndex(index - 1) || (linebreakHints.length == 0 && (linebreakHints.canFindIndex(index - 1) || (linebreakHints.length == 0
&& currentLineLength > config.max_line_length))) && virtualLineLength > config.max_line_length)))
{ {
newline(); newline();
} }
@ -761,13 +775,13 @@ private:
{ {
writeToken(); writeToken();
if (spaceAfterParens || parenDepth > 0) if (spaceAfterParens || parenDepth > 0)
writeSpace(); queueSpace();
} }
else if ((peekIsKeyword() || peekIs(tok!"@")) && spaceAfterParens else if ((peekIsKeyword() || peekIs(tok!"@")) && spaceAfterParens
&& !peekIs(tok!"in") && !peekIs(tok!"is") && !peekIs(tok!"if")) && !peekIs(tok!"in") && !peekIs(tok!"is") && !peekIs(tok!"if"))
{ {
writeToken(); writeToken();
writeSpace(); queueSpace();
} }
else else
writeToken(); writeToken();
@ -790,7 +804,7 @@ private:
} }
writeToken(); writeToken();
if (currentIs(tok!"identifier")) if (currentIs(tok!"identifier"))
write(" "); queueSpace();
} }
void formatAt() void formatAt()
@ -809,16 +823,12 @@ private:
&& astInformation.atAttributeStartLocations.canFindIndex(atIndex)) && astInformation.atAttributeStartLocations.canFindIndex(atIndex))
newline(); newline();
else else
write(" "); queueSpace();
} }
else if (hasCurrent && (currentIs(tok!"@") else if (hasCurrent
|| isBasicType(tokens[index].type) && (currentIs(tok!"@") || isLikeIdentifier(current.type)))
|| currentIs(tok!"invariant")
|| currentIs(tok!"extern")
|| currentIs(tok!"identifier"))
&& !currentIsIndentedTemplateConstraint())
{ {
writeSpace(); queueSpace();
} }
} }
@ -859,7 +869,7 @@ private:
{ {
writeToken(); writeToken();
if (isStructInitializer) if (isStructInitializer)
write(" "); queueSpace();
else if (!currentIs(tok!"{")) else if (!currentIs(tok!"{"))
newline(); newline();
} }
@ -872,7 +882,7 @@ private:
{ {
writeToken(); writeToken();
if (config.dfmt_compact_labeled_statements) if (config.dfmt_compact_labeled_statements)
write(" "); queueSpace();
else else
newline(); newline();
} }
@ -881,7 +891,7 @@ private:
pushWrapIndent(); pushWrapIndent();
newline(); newline();
writeToken(); writeToken();
write(" "); queueSpace();
} }
else else
{ {
@ -898,7 +908,7 @@ private:
if ((parenDepth > 0 && sBraceDepth == 0) || (sBraceDepth > 0 && niBraceDepth > 0)) if ((parenDepth > 0 && sBraceDepth == 0) || (sBraceDepth > 0 && niBraceDepth > 0))
{ {
if (currentLineLength > config.dfmt_soft_max_line_length) if (virtualLineLength > config.dfmt_soft_max_line_length)
{ {
writeToken(); writeToken();
pushWrapIndent(tok!";"); pushWrapIndent(tok!";");
@ -974,7 +984,7 @@ private:
sBraceDepth++; sBraceDepth++;
if (peekBackIsOneOf(true, tok!")", tok!"identifier")) if (peekBackIsOneOf(true, tok!")", tok!"identifier"))
write(" "); queueSpace(true);
immutable bool multiline = isMultilineAt(index); immutable bool multiline = isMultilineAt(index);
writeToken(); writeToken();
if (multiline) if (multiline)
@ -986,7 +996,7 @@ private:
{ {
niBraceDepth++; niBraceDepth++;
if (!currentIs(tok!"}")) if (!currentIs(tok!"}"))
write(" "); queueSpace(true);
} }
} }
else else
@ -1014,7 +1024,7 @@ private:
&& (peekBackIs(tok!")") || (!peekBackIs(tok!"do") && peekBack().text != "body"))) && (peekBackIs(tok!")") || (!peekBackIs(tok!"do") && peekBack().text != "body")))
newline(); newline();
else if (!peekBackIsOneOf(true, tok!"{", tok!"}", tok!";")) else if (!peekBackIsOneOf(true, tok!"{", tok!"}", tok!";"))
write(" "); queueSpace(true);
writeToken(); writeToken();
} }
indents.push(tok!"{"); indents.push(tok!"{");
@ -1058,7 +1068,7 @@ private:
if (niBraceDepth > 0) if (niBraceDepth > 0)
{ {
if (!peekBackIsSlashSlash() && !peekBackIs(tok!"{")) if (!peekBackIsSlashSlash() && !peekBackIs(tok!"{"))
write(" "); queueSpace();
niBraceDepth--; niBraceDepth--;
} }
if (sBraceDepth > 0) if (sBraceDepth > 0)
@ -1088,7 +1098,7 @@ private:
&& !indents.topIs(tok!"while") && !indents.topIs(tok!"do")) && !indents.topIs(tok!"while") && !indents.topIs(tok!"do"))
|| peekIs(tok!"catch") || peekIs(tok!"finally"))) || peekIs(tok!"catch") || peekIs(tok!"finally")))
{ {
write(" "); queueSpace(true);
index++; index++;
} }
else else
@ -1113,7 +1123,7 @@ private:
indents.pop(); indents.pop();
indents.push(tok!"switch"); indents.push(tok!"switch");
writeToken(); // switch writeToken(); // switch
write(" "); queueSpace();
} }
void formatBlockHeader() void formatBlockHeader()
@ -1144,20 +1154,20 @@ private:
if (currentIs(tok!"(")) if (currentIs(tok!"("))
{ {
write(" "); queueSpace();
writeParens(false); writeParens(false);
} }
if (hasCurrent) if (hasCurrent)
{ {
if (currentIs(tok!"switch") || (currentIs(tok!"final") && peekIs(tok!"switch"))) if (currentIs(tok!"switch") || (currentIs(tok!"final") && peekIs(tok!"switch")))
write(" "); queueSpace();
else if (currentIs(tok!"comment")) else if (currentIs(tok!"comment"))
formatStep(); formatStep();
else if (!shouldPushIndent) else if (!shouldPushIndent)
{ {
if (!currentIs(tok!"{") && !currentIs(tok!";")) if (!currentIs(tok!"{") && !currentIs(tok!";"))
write(" "); queueSpace();
} }
else if (hasCurrent && !currentIs(tok!"{") && !currentIs(tok!";") && !currentIs(tok!"in") && else if (hasCurrent && !currentIs(tok!"{") && !currentIs(tok!";") && !currentIs(tok!"in") &&
!currentIs(tok!"out") && !currentIs(tok!"do") && current.text != "body") !currentIs(tok!"out") && !currentIs(tok!"do") && current.text != "body")
@ -1198,7 +1208,7 @@ private:
if (indents.topIs(tok!"if") || indents.topIs(tok!"version")) if (indents.topIs(tok!"if") || indents.topIs(tok!"version"))
indents.pop(); indents.pop();
inlineElse = false; inlineElse = false;
write(" "); queueSpace();
} }
else if (currentIs(tok!":")) else if (currentIs(tok!":"))
{ {
@ -1243,11 +1253,11 @@ private:
&& astInformation.contractLocations.canFindIndex(current.index)) && astInformation.contractLocations.canFindIndex(current.index))
newline(); newline();
else if (peekBackIsKeyword) else if (peekBackIsKeyword)
write(" "); queueSpace();
} }
writeToken(); writeToken();
if (!currentIs(tok!"{") && !currentIs(tok!"comment")) if (!currentIs(tok!"{") && !currentIs(tok!"comment"))
write(" "); queueSpace();
break; break;
case tok!"try": case tok!"try":
case tok!"finally": case tok!"finally":
@ -1275,7 +1285,7 @@ private:
newline(); newline();
} }
else if (!peekBackIsOneOf(false, tok!"(", tok!",", tok!"!")) else if (!peekBackIsOneOf(false, tok!"(", tok!",", tok!"!"))
write(" "); queueSpace();
} }
writeToken(); writeToken();
immutable isFunctionLit = astInformation.funLitStartLocations.canFindIndex( immutable isFunctionLit = astInformation.funLitStartLocations.canFindIndex(
@ -1283,20 +1293,20 @@ private:
if (isFunctionLit && config.dfmt_brace_style == BraceStyle.allman) if (isFunctionLit && config.dfmt_brace_style == BraceStyle.allman)
newline(); newline();
else if (!isContract || currentIs(tok!"(")) else if (!isContract || currentIs(tok!"("))
write(" "); queueSpace();
break; break;
case tok!"is": case tok!"is":
if (!peekBackIsOneOf(false, tok!"!", tok!"(", tok!",", if (!peekBackIsOneOf(false, tok!"!", tok!"(", tok!",",
tok!"}", tok!"=", tok!"&&", tok!"||") && !peekBackIsKeyword()) tok!"}", tok!"=", tok!"&&", tok!"||") && !peekBackIsKeyword())
write(" "); queueSpace();
writeToken(); writeToken();
if (!currentIs(tok!"(") && !currentIs(tok!"{") && !currentIs(tok!"comment")) if (!currentIs(tok!"(") && !currentIs(tok!"{") && !currentIs(tok!"comment"))
write(" "); queueSpace();
break; break;
case tok!"case": case tok!"case":
writeToken(); writeToken();
if (!currentIs(tok!";")) if (!currentIs(tok!";"))
write(" "); queueSpace();
break; break;
case tok!"enum": case tok!"enum":
if (peekIs(tok!")") || peekIs(tok!"==")) if (peekIs(tok!")") || peekIs(tok!"=="))
@ -1306,11 +1316,11 @@ private:
else else
{ {
if (peekBackIs(tok!"identifier")) if (peekBackIs(tok!"identifier"))
write(" "); queueSpace();
indents.push(tok!"enum"); indents.push(tok!"enum");
writeToken(); writeToken();
if (!currentIs(tok!":") && !currentIs(tok!"{")) if (!currentIs(tok!":") && !currentIs(tok!"{"))
write(" "); queueSpace();
} }
break; break;
case tok!"static": case tok!"static":
@ -1334,12 +1344,12 @@ private:
case tok!"invariant": case tok!"invariant":
writeToken(); writeToken();
if (currentIs(tok!"(")) if (currentIs(tok!"("))
write(" "); queueSpace();
break; break;
default: default:
if (peekBackIs(tok!"identifier")) if (peekBackIs(tok!"identifier"))
{ {
writeSpace(); queueSpace();
} }
if (index + 1 < tokens.length) if (index + 1 < tokens.length)
{ {
@ -1353,7 +1363,7 @@ private:
writeToken(); writeToken();
if (!currentIsIndentedTemplateConstraint()) if (!currentIsIndentedTemplateConstraint())
{ {
writeSpace(); queueSpace();
} }
} }
} }
@ -1369,7 +1379,7 @@ private:
&& astInformation.constraintLocations.canFindIndex(current.index) && astInformation.constraintLocations.canFindIndex(current.index)
&& (config.dfmt_template_constraint_style == TemplateConstraintStyle.always_newline && (config.dfmt_template_constraint_style == TemplateConstraintStyle.always_newline
|| config.dfmt_template_constraint_style == TemplateConstraintStyle.always_newline_indent || config.dfmt_template_constraint_style == TemplateConstraintStyle.always_newline_indent
|| currentLineLength >= config.dfmt_soft_max_line_length); || virtualLineLength >= config.dfmt_soft_max_line_length);
} }
void formatOperator() void formatOperator()
@ -1386,7 +1396,7 @@ private:
if (!currentIs(tok!"*") && !currentIs(tok!")") if (!currentIs(tok!"*") && !currentIs(tok!")")
&& !currentIs(tok!"[") && !currentIs(tok!",") && !currentIs(tok!";")) && !currentIs(tok!"[") && !currentIs(tok!",") && !currentIs(tok!";"))
{ {
write(" "); queueSpace();
} }
break; break;
} }
@ -1403,7 +1413,7 @@ private:
if (!(index == 0 || peekBackIs(tok!"{", true) if (!(index == 0 || peekBackIs(tok!"{", true)
|| peekBackIs(tok!"}", true) || peekBackIs(tok!";", true))) || peekBackIs(tok!"}", true) || peekBackIs(tok!";", true)))
{ {
write(" "); queueSpace(true);
} }
writeToken(); writeToken();
break; break;
@ -1432,7 +1442,7 @@ private:
case tok!"!": case tok!"!":
if (((peekIs(tok!"is") || peekIs(tok!"in")) if (((peekIs(tok!"is") || peekIs(tok!"in"))
&& !peekBackIsOperator()) || peekBackIs(tok!")")) && !peekBackIsOperator()) || peekBackIs(tok!")"))
write(" "); queueSpace();
goto case; goto case;
case tok!"...": case tok!"...":
case tok!"++": case tok!"++":
@ -1459,7 +1469,7 @@ private:
regenLineBreakHintsIfNecessary(index); regenLineBreakHintsIfNecessary(index);
immutable bool ufcsWrap = astInformation.ufcsHintLocations.canFindIndex(current.index); immutable bool ufcsWrap = astInformation.ufcsHintLocations.canFindIndex(current.index);
if (ufcsWrap || linebreakHints.canFind(index) || onNextLine if (ufcsWrap || linebreakHints.canFind(index) || onNextLine
|| (linebreakHints.length == 0 && currentLineLength + nextTokenLength() > config.max_line_length)) || (linebreakHints.length == 0 && virtualLineLength + nextTokenLength() > config.max_line_length))
{ {
if (!indents.topIs(tok!".")) if (!indents.topIs(tok!"."))
indents.push(tok!"."); indents.push(tok!".");
@ -1531,7 +1541,7 @@ private:
} }
else else
{ {
write(" "); queueSpace();
} }
if (rightOperandLine > operatorLine if (rightOperandLine > operatorLine
&& !indents.topIs(tok!"enum")) && !indents.topIs(tok!"enum"))
@ -1547,7 +1557,7 @@ private:
} }
else else
{ {
write(" "); queueSpace();
} }
} }
else if (config.dfmt_split_operator_at_line_end) else if (config.dfmt_split_operator_at_line_end)
@ -1556,16 +1566,16 @@ private:
{ {
if (!indents.topIs(tok!"enum")) if (!indents.topIs(tok!"enum"))
pushWrapIndent(); pushWrapIndent();
write(" "); queueSpace();
writeToken(); writeToken();
newline(); newline();
} }
else else
{ {
write(" "); queueSpace();
writeToken(); writeToken();
if (!currentIs(tok!"comment")) if (!currentIs(tok!"comment"))
write(" "); queueSpace();
} }
} }
else else
@ -1579,11 +1589,11 @@ private:
} }
else else
{ {
write(" "); queueSpace();
writeToken(); writeToken();
} }
if (!currentIs(tok!"comment")) if (!currentIs(tok!"comment"))
write(" "); queueSpace();
} }
break; break;
default: default:
@ -1631,7 +1641,7 @@ private:
{ {
if (tokens[index].line == commaLine) if (tokens[index].line == commaLine)
{ {
write(" "); queueSpace();
} }
else else
{ {
@ -1640,7 +1650,7 @@ private:
} }
} }
else if (!peekIs(tok!"}") && (linebreakHints.canFind(index) else if (!peekIs(tok!"}") && (linebreakHints.canFind(index)
|| (linebreakHints.length == 0 && currentLineLength > config.max_line_length))) || (linebreakHints.length == 0 && virtualLineLength > config.max_line_length)))
{ {
pushWrapIndent(); pushWrapIndent();
writeToken(); writeToken();
@ -1652,7 +1662,7 @@ private:
if (!currentIs(tok!")") && !currentIs(tok!"]") if (!currentIs(tok!")") && !currentIs(tok!"]")
&& !currentIs(tok!"}") && !currentIs(tok!"comment")) && !currentIs(tok!"}") && !currentIs(tok!"comment"))
{ {
write(" "); queueSpace();
} }
} }
regenLineBreakHintsIfNecessary(index - 1); regenLineBreakHintsIfNecessary(index - 1);
@ -1681,7 +1691,7 @@ private:
immutable inLvl = (indents.topIsWrap() || indents.topIs(tok!"]")) ? -indentLevel immutable inLvl = (indents.topIsWrap() || indents.topIs(tok!"]")) ? -indentLevel
: indentLevel; : indentLevel;
linebreakHints = chooseLineBreakTokens(i, tokens[i .. j], depths[i .. j], linebreakHints = chooseLineBreakTokens(i, tokens[i .. j], depths[i .. j],
config, currentLineLength, inLvl); config, virtualLineLength, inLvl);
} }
void regenLineBreakHintsIfNecessary(immutable size_t i) void regenLineBreakHintsIfNecessary(immutable size_t i)
@ -1694,7 +1704,9 @@ private:
{ {
import dfmt.editorconfig : EOL; import dfmt.editorconfig : EOL;
queuedSpace = false;
output.put(eolString); output.put(eolString);
currentLineLength = 0;
} }
void newline() void newline()
@ -1855,8 +1867,23 @@ private:
} }
} }
void commitSpace()
{
if (queuedSpace)
{
if (currentLineLength > 0)
{
// don't put single space at start of line
currentLineLength++;
output.put(" ");
}
queuedSpace = false;
}
}
void write(string str) void write(string str)
{ {
commitSpace();
currentLineLength += str.length; currentLineLength += str.length;
output.put(str); output.put(str);
} }
@ -1871,11 +1898,13 @@ private:
if (current.text is null) if (current.text is null)
{ {
immutable s = str(current.type); immutable s = str(current.type);
commitSpace();
currentLineLength += s.length; currentLineLength += s.length;
output.put(str(current.type)); output.put(str(current.type));
} }
else else
{ {
commitSpace();
output.put(current.text.lineSplitter.joiner(eolString)); output.put(current.text.lineSplitter.joiner(eolString));
switch (current.type) switch (current.type)
{ {
@ -1915,7 +1944,7 @@ private:
if (currentIs(tok!";") && niBraceDepth <= startingNiBraceDepth if (currentIs(tok!";") && niBraceDepth <= startingNiBraceDepth
&& sBraceDepth <= startingSBraceDepth) && sBraceDepth <= startingSBraceDepth)
{ {
if (currentLineLength >= config.dfmt_soft_max_line_length) if (virtualLineLength >= config.dfmt_soft_max_line_length)
{ {
pushWrapIndent(); pushWrapIndent();
writeToken(); writeToken();
@ -1925,7 +1954,7 @@ private:
{ {
writeToken(); writeToken();
if (!currentIs(tok!")") && !currentIs(tok!";")) if (!currentIs(tok!")") && !currentIs(tok!";"))
write(" "); queueSpace();
} }
} }
else else
@ -1943,6 +1972,7 @@ private:
{ {
import dfmt.editorconfig : IndentStyle; import dfmt.editorconfig : IndentStyle;
queuedSpace = false;
if (config.indent_style == IndentStyle.tab) if (config.indent_style == IndentStyle.tab)
{ {
foreach (i; 0 .. indentLevel) foreach (i; 0 .. indentLevel)
@ -1982,16 +2012,13 @@ private:
indents.push(type, detail); indents.push(type, detail);
} }
void writeSpace() void queueSpace(bool avoidNewline = false)
{ {
if (onNextLine) if (!avoidNewline && onNextLine && !queuedSpace)
{
newline(); newline();
} // also queue space after new line, to avoid double newline through
else // queueSpace.
{ queuedSpace = true;
write(" ");
}
} }
const pure @safe @nogc: const pure @safe @nogc:
@ -2023,7 +2050,7 @@ const pure @safe @nogc:
import std.algorithm : map, sum, canFind; import std.algorithm : map, sum, canFind;
auto e = expressionEndIndex(i, matchComma); auto e = expressionEndIndex(i, matchComma);
immutable int l = currentLineLength + tokens[i .. e].map!(a => tokenLength(a)).sum(); immutable int l = virtualLineLength + tokens[i .. e].map!(a => tokenLength(a)).sum();
return l > config.dfmt_soft_max_line_length || tokens[i .. e].canFind!( return l > config.dfmt_soft_max_line_length || tokens[i .. e].canFind!(
a => a.type == tok!"comment" || isBlockHeaderToken(a.type))(); a => a.type == tok!"comment" || isBlockHeaderToken(a.type))();
} }

View File

@ -241,3 +241,10 @@ private string generateFixedLengthCases()
a => format(`case tok!"%s": return %d;`, a, a.length)).join("\n\t"); a => format(`case tok!"%s": return %d;`, a, a.length)).join("\n\t");
return spacedOperatorTokenCases ~ identifierTokenCases; return spacedOperatorTokenCases ~ identifierTokenCases;
} }
/// Returns true if the given token type is an identifier or a keyword that
/// would be an identifier if it wasn't reserved.
bool isLikeIdentifier(IdType t)
{
return isKeyword(t) || isBasicType(t) || t == tok!"identifier";
}

View File

@ -0,0 +1,5 @@
int foo() @nogc return
{
}
int foo() @nogc in ()

View File

@ -0,0 +1,5 @@
int foo() @nogc return
{
}
int foo() @nogc in ()

View File

@ -0,0 +1,4 @@
int foo() @nogc return {
}
int foo() @nogc in ()

5
tests/return_attr.d Normal file
View File

@ -0,0 +1,5 @@
int foo() @nogc return
{
}
int foo() @nogc in ()

View File

@ -12,10 +12,11 @@ version (Windows)
else else
enum dfmt = `../bin/dfmt`; enum dfmt = `../bin/dfmt`;
int main() int main(string[] args)
{ {
string pattern = args.length == 2 ? args[1] : "*.d";
foreach (braceStyle; ["allman", "otbs", "knr"]) foreach (braceStyle; ["allman", "otbs", "knr"])
foreach (entry; dirEntries(".", "*.d", SpanMode.shallow).filter!(e => e.baseName(".d") != "test")) foreach (entry; dirEntries(".", pattern, SpanMode.shallow).filter!(e => e.baseName(".d") != "test"))
{ {
const source = entry.baseName; const source = entry.baseName;
const outFileName = buildPath(braceStyle, source ~ ".out"); const outFileName = buildPath(braceStyle, source ~ ".out");