Merge pull request #535 from dlang-community/single-indent

Add single_indent option
This commit is contained in:
Razvan Nitu 2021-10-22 16:22:40 +03:00 committed by GitHub
commit 2beb819851
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
13 changed files with 296 additions and 3 deletions

View File

@ -116,6 +116,7 @@ dfmt_template_constraint_style | **`conditional_newline_indent`** `conditional_n
dfmt_single_template_constraint_indent | `true`, **`false`** | Set if the constraints are indented by a single tab instead of two. Has only an effect if the style set to `always_newline_indent` or `conditional_newline_indent`. dfmt_single_template_constraint_indent | `true`, **`false`** | Set if the constraints are indented by a single tab instead of two. Has only an effect if the style set to `always_newline_indent` or `conditional_newline_indent`.
dfmt_space_before_aa_colon | `true`, **`false`** | Adds a space after an associative array key before the `:` like in older dfmt versions. dfmt_space_before_aa_colon | `true`, **`false`** | Adds a space after an associative array key before the `:` like in older dfmt versions.
dfmt_keep_line_breaks | `true`, **`false`** | Keep existing line breaks if these don't violate other formatting rules. dfmt_keep_line_breaks | `true`, **`false`** | Keep existing line breaks if these don't violate other formatting rules.
dfmt_single_indent | `true`, **`false`** | Set if the code in parens is indented by a single tab instead of two.
## Terminology ## Terminology
* Braces - `{` and `}` * Braces - `{` and `}`

View File

@ -61,6 +61,8 @@ struct Config
OptionalBoolean dfmt_space_before_aa_colon; OptionalBoolean dfmt_space_before_aa_colon;
/// ///
OptionalBoolean dfmt_keep_line_breaks; OptionalBoolean dfmt_keep_line_breaks;
///
OptionalBoolean dfmt_single_indent;
mixin StandardEditorConfigFields; mixin StandardEditorConfigFields;
@ -90,6 +92,7 @@ struct Config
dfmt_single_template_constraint_indent = OptionalBoolean.f; dfmt_single_template_constraint_indent = OptionalBoolean.f;
dfmt_space_before_aa_colon = OptionalBoolean.f; dfmt_space_before_aa_colon = OptionalBoolean.f;
dfmt_keep_line_breaks = OptionalBoolean.f; dfmt_keep_line_breaks = OptionalBoolean.f;
dfmt_single_indent = OptionalBoolean.f;
} }
/** /**

View File

@ -111,6 +111,7 @@ struct TokenFormatter(OutputRange)
this.output = output; this.output = output;
this.astInformation = astInformation; this.astInformation = astInformation;
this.config = config; this.config = config;
this.indents = IndentStack(config);
{ {
auto eol = config.end_of_line; auto eol = config.end_of_line;

View File

@ -5,6 +5,8 @@
module dfmt.indentation; module dfmt.indentation;
import dfmt.config;
import dfmt.editorconfig;
import dparse.lexer; import dparse.lexer;
import std.bitmanip : bitfields; import std.bitmanip : bitfields;
@ -31,6 +33,14 @@ bool isTempIndent(IdType type) pure nothrow @nogc @safe
*/ */
struct IndentStack struct IndentStack
{ {
/// Configuration
private const Config* config;
this(const Config* config)
{
this.config = config;
}
static struct Details static struct Details
{ {
mixin(bitfields!( mixin(bitfields!(
@ -221,7 +231,7 @@ struct IndentStack
/** /**
* Dumps the current state of the indentation stack to `stderr`. Used for debugging. * Dumps the current state of the indentation stack to `stderr`. Used for debugging.
*/ */
void dump(size_t pos = size_t.max, string file = __FILE__, uint line = __LINE__) void dump(size_t pos = size_t.max, string file = __FILE__, uint line = __LINE__) const
{ {
import dparse.lexer : str; import dparse.lexer : str;
import std.algorithm.iteration : map; import std.algorithm.iteration : map;
@ -260,6 +270,12 @@ private:
if (i + 1 < index) if (i + 1 < index)
{ {
if (config.dfmt_single_indent == OptionalBoolean.t && skipDoubleIndent(i, parenCount))
{
parenCount = pc;
continue;
}
immutable currentIsNonWrapTemp = !details[i].wrap immutable currentIsNonWrapTemp = !details[i].wrap
&& details[i].temp && arr[i] != tok!")" && arr[i] != tok!"!"; && details[i].temp && arr[i] != tok!")" && arr[i] != tok!"!";
if (arr[i] == tok!"static" if (arr[i] == tok!"static"
@ -276,7 +292,7 @@ private:
continue; continue;
} }
} }
else if (parenCount == 0 && arr[i] == tok!"(") else if (parenCount == 0 && arr[i] == tok!"(" && config.dfmt_single_indent == OptionalBoolean.f)
size++; size++;
if (arr[i] == tok!"!") if (arr[i] == tok!"!")
@ -287,6 +303,12 @@ private:
} }
return size; return size;
} }
bool skipDoubleIndent(size_t i, int parenCount) const pure nothrow @safe @nogc
{
return (details[i + 1].wrap && arr[i] == tok!")")
|| (parenCount == 0 && arr[i + 1] == tok!"," && arr[i] == tok!"(");
}
} }
unittest unittest

View File

@ -95,6 +95,9 @@ else
case "keep_line_breaks": case "keep_line_breaks":
optConfig.dfmt_keep_line_breaks = optVal; optConfig.dfmt_keep_line_breaks = optVal;
break; break;
case "single_indent":
optConfig.dfmt_single_indent = optVal;
break;
default: default:
assert(false, "Invalid command-line switch"); assert(false, "Invalid command-line switch");
} }
@ -125,7 +128,8 @@ else
"space_before_aa_colon", &handleBooleans, "space_before_aa_colon", &handleBooleans,
"tab_width", &optConfig.tab_width, "tab_width", &optConfig.tab_width,
"template_constraint_style", &optConfig.dfmt_template_constraint_style, "template_constraint_style", &optConfig.dfmt_template_constraint_style,
"keep_line_breaks", &handleBooleans); "keep_line_breaks", &handleBooleans,
"single_indent", &handleBooleans);
// dfmt on // dfmt on
} }
catch (GetOptException e) catch (GetOptException e)
@ -329,6 +333,7 @@ Formatting Options:
--compact_labeled_statements --compact_labeled_statements
--template_constraint_style --template_constraint_style
--space_before_aa_colon --space_before_aa_colon
--single_indent
`, `,
optionsToString!(typeof(Config.dfmt_template_constraint_style))); optionsToString!(typeof(Config.dfmt_template_constraint_style)));
} }

View File

@ -0,0 +1,49 @@
unittest
{
{
bool anotherTemplatedFunction(One, Two, Three)(One alpha, Two bravo,
Three charlie, double delta)
{
if (isNumeric!One && isNumeric!Two && isNumeric!Three && echo
&& foxtrot && golf && hotel && india && juliet)
{
}
}
}
}
void f()
{
string a = "foo"
~ "bar" /* bar */
~ "baz";
}
unittest
{
if (a)
{
while (sBraceDepth == 0 && indents.topIsTemp()
&& ((indents.top != tok!"if" && indents.top != tok!"version")
|| !peekIs(tok!"else")))
a();
}
}
unittest
{
callFunc({ int i = 10; return i; });
callFunc({
int i = 10;
foo(alpha, bravo, charlie, delta, echo, foxtrot, golf, echo);
doStuff(withThings, andOtherStuff);
return i;
});
callFunc({
int i = 10;
foo(alpha_longVarName, bravo_longVarName, charlie_longVarName, delta_longVarName,
echo_longVarName, foxtrot_longVarName, golf_longVarName, echo_longVarName);
doStuff(withThings, andOtherStuff);
return i;
}, more_stuff);
}

View File

@ -0,0 +1,41 @@
unittest
{
{
bool anotherTemplatedFunction(One, Two, Three)(One alpha, Two bravo,
Three charlie, double delta)
{
if (isNumeric!One && isNumeric!Two && isNumeric!Three && echo
&& foxtrot && golf && hotel && india && juliet)
{
}
}
}
}
unittest
{
if (a)
{
while (sBraceDepth == 0 && indents.topIsTemp()
&& ((indents.top != tok!"if" && indents.top != tok!"version") || !peekIs(tok!"else")))
a();
}
}
unittest
{
callFunc({ int i = 10; return i; });
callFunc({
int i = 10;
foo(alpha, bravo, charlie, delta, echo, foxtrot, golf, echo);
doStuff(withThings, andOtherStuff);
return i;
});
callFunc({
int i = 10;
foo(alpha_longVarName, bravo_longVarName, charlie_longVarName, delta_longVarName,
echo_longVarName, foxtrot_longVarName, golf_longVarName, echo_longVarName);
doStuff(withThings, andOtherStuff);
return i;
}, more_stuff);
}

View File

@ -0,0 +1,2 @@
--single_indent true
--keep_line_breaks true

View File

@ -0,0 +1,49 @@
unittest
{
{
bool anotherTemplatedFunction(One, Two, Three)(One alpha, Two bravo,
Three charlie, double delta)
{
if (isNumeric!One && isNumeric!Two && isNumeric!Three && echo
&& foxtrot && golf && hotel && india && juliet)
{
}
}
}
}
void f()
{
string a = "foo"
~ "bar" /* bar */
~ "baz";
}
unittest
{
if (a)
{
while (sBraceDepth == 0 && indents.topIsTemp()
&& ((indents.top != tok!"if" && indents.top != tok!"version")
|| !peekIs(tok!"else")))
a();
}
}
unittest
{
callFunc({ int i = 10; return i; });
callFunc({
int i = 10;
foo(alpha, bravo, charlie, delta, echo, foxtrot, golf, echo);
doStuff(withThings, andOtherStuff);
return i;
});
callFunc({
int i = 10;
foo(alpha_longVarName, bravo_longVarName, charlie_longVarName, delta_longVarName,
echo_longVarName, foxtrot_longVarName, golf_longVarName, echo_longVarName);
doStuff(withThings, andOtherStuff);
return i;
}, more_stuff);
}

View File

@ -0,0 +1,42 @@
unittest {
{
bool anotherTemplatedFunction(One, Two, Three)(One alpha, Two bravo,
Three charlie, double delta) {
if (isNumeric!One && isNumeric!Two && isNumeric!Three && echo
&& foxtrot && golf && hotel && india && juliet) {
}
}
}
}
void f() {
string a = "foo"
~ "bar" /* bar */
~ "baz";
}
unittest {
if (a) {
while (sBraceDepth == 0 && indents.topIsTemp()
&& ((indents.top != tok!"if" && indents.top != tok!"version")
|| !peekIs(tok!"else")))
a();
}
}
unittest {
callFunc({ int i = 10; return i; });
callFunc({
int i = 10;
foo(alpha, bravo, charlie, delta, echo, foxtrot, golf, echo);
doStuff(withThings, andOtherStuff);
return i;
});
callFunc({
int i = 10;
foo(alpha_longVarName, bravo_longVarName, charlie_longVarName, delta_longVarName,
echo_longVarName, foxtrot_longVarName, golf_longVarName, echo_longVarName);
doStuff(withThings, andOtherStuff);
return i;
}, more_stuff);
}

View File

@ -0,0 +1,35 @@
unittest {
{
bool anotherTemplatedFunction(One, Two, Three)(One alpha, Two bravo,
Three charlie, double delta) {
if (isNumeric!One && isNumeric!Two && isNumeric!Three && echo
&& foxtrot && golf && hotel && india && juliet) {
}
}
}
}
unittest {
if (a) {
while (sBraceDepth == 0 && indents.topIsTemp()
&& ((indents.top != tok!"if" && indents.top != tok!"version") || !peekIs(tok!"else")))
a();
}
}
unittest {
callFunc({ int i = 10; return i; });
callFunc({
int i = 10;
foo(alpha, bravo, charlie, delta, echo, foxtrot, golf, echo);
doStuff(withThings, andOtherStuff);
return i;
});
callFunc({
int i = 10;
foo(alpha_longVarName, bravo_longVarName, charlie_longVarName, delta_longVarName,
echo_longVarName, foxtrot_longVarName, golf_longVarName, echo_longVarName);
doStuff(withThings, andOtherStuff);
return i;
}, more_stuff);
}

1
tests/single_indent.args Normal file
View File

@ -0,0 +1 @@
--single_indent true

42
tests/single_indent.d Normal file
View File

@ -0,0 +1,42 @@
unittest
{
{
bool anotherTemplatedFunction(One, Two, Three)(One alpha, Two bravo,
Three charlie, double delta)
{
if (isNumeric!One && isNumeric!Two && isNumeric!Three && echo
&& foxtrot && golf && hotel && india && juliet)
{
}
}
}
}
unittest
{
if (a)
{
while (sBraceDepth == 0 && indents.topIsTemp()
&& ((indents.top != tok!"if" && indents.top != tok!"version")
|| !peekIs(tok!"else")))
a();
}
}
unittest
{
callFunc({ int i = 10; return i; });
callFunc({
int i = 10;
foo(alpha, bravo, charlie, delta, echo, foxtrot, golf, echo);
doStuff(withThings, andOtherStuff);
return i;
});
callFunc({
int i = 10;
foo(alpha_longVarName, bravo_longVarName, charlie_longVarName, delta_longVarName,
echo_longVarName, foxtrot_longVarName, golf_longVarName, echo_longVarName);
doStuff(withThings, andOtherStuff);
return i;
}, more_stuff);
}