Better enum formatting and whitespace cleanup
This commit is contained in:
parent
b82ef4ad60
commit
9c8abe55fa
154
src/dfmt.d
154
src/dfmt.d
|
@ -1,28 +1,28 @@
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Boost Software License - Version 1.0 - August 17th, 2003
|
* Boost Software License - Version 1.0 - August 17th, 2003
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person or organization
|
* Permission is hereby granted, free of charge, to any person or organization
|
||||||
* obtaining a copy of the software and accompanying documentation covered by
|
* obtaining a copy of the software and accompanying documentation covered by
|
||||||
* this license (the "Software") to use, reproduce, display, distribute,
|
* this license (the "Software") to use, reproduce, display, distribute,
|
||||||
* execute, and transmit the Software, and to prepare derivative works of the
|
* execute, and transmit the Software, and to prepare derivative works of the
|
||||||
* Software, and to permit third-parties to whom the Software is furnished to
|
* Software, and to permit third-parties to whom the Software is furnished to
|
||||||
* do so, all subject to the following:
|
* do so, all subject to the following:
|
||||||
*
|
*
|
||||||
* The copyright notices in the Software and this entire statement, including
|
* The copyright notices in the Software and this entire statement, including
|
||||||
* the above license grant, this restriction and the following disclaimer,
|
* the above license grant, this restriction and the following disclaimer,
|
||||||
* must be included in all copies of the Software, in whole or in part, and
|
* must be included in all copies of the Software, in whole or in part, and
|
||||||
* all derivative works of the Software, unless such copies or derivative
|
* all derivative works of the Software, unless such copies or derivative
|
||||||
* works are solely in the form of machine-executable object code generated by
|
* works are solely in the form of machine-executable object code generated by
|
||||||
* a source language processor.
|
* a source language processor.
|
||||||
*
|
*
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
|
* FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
|
||||||
* SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
|
* SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
|
||||||
* FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
|
* FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
|
||||||
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||||
* DEALINGS IN THE SOFTWARE.
|
* DEALINGS IN THE SOFTWARE.
|
||||||
******************************************************************************/
|
******************************************************************************/
|
||||||
|
|
||||||
module dfmt;
|
module dfmt;
|
||||||
|
|
||||||
|
@ -36,14 +36,26 @@ import std.array;
|
||||||
|
|
||||||
int main(string[] args)
|
int main(string[] args)
|
||||||
{
|
{
|
||||||
if (args.length < 2)
|
ubyte[] buffer;
|
||||||
|
if (args.length == 1)
|
||||||
{
|
{
|
||||||
writeln("File name is a required argument");
|
ubyte[4096] inputBuffer;
|
||||||
return 1;
|
ubyte[] b;
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
b = stdin.rawRead(inputBuffer);
|
||||||
|
if (b.length)
|
||||||
|
buffer ~= b;
|
||||||
|
else
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
File f = File(args[1]);
|
||||||
|
buffer = new ubyte[](f.size);
|
||||||
|
f.rawRead(buffer);
|
||||||
}
|
}
|
||||||
File f = File(args[1]);
|
|
||||||
ubyte[] buffer = new ubyte[](f.size);
|
|
||||||
f.rawRead(buffer);
|
|
||||||
LexerConfig config;
|
LexerConfig config;
|
||||||
config.stringBehavior = StringBehavior.source;
|
config.stringBehavior = StringBehavior.source;
|
||||||
config.whitespaceBehavior = WhitespaceBehavior.skip;
|
config.whitespaceBehavior = WhitespaceBehavior.skip;
|
||||||
|
@ -52,15 +64,15 @@ int main(string[] args)
|
||||||
parseConfig.whitespaceBehavior = WhitespaceBehavior.skip;
|
parseConfig.whitespaceBehavior = WhitespaceBehavior.skip;
|
||||||
StringCache cache = StringCache(StringCache.defaultBucketCount);
|
StringCache cache = StringCache(StringCache.defaultBucketCount);
|
||||||
ASTInformation astInformation;
|
ASTInformation astInformation;
|
||||||
FormatterConfig formatterConfig;
|
FormatterConfig formatterConfig;
|
||||||
auto parseTokens = getTokensForParser(buffer, parseConfig, &cache);
|
auto parseTokens = getTokensForParser(buffer, parseConfig, &cache);
|
||||||
auto mod = parseModule(parseTokens, args[1]);
|
auto mod = parseModule(parseTokens, args.length > 1 ? args[1] : "stdin");
|
||||||
auto visitor = new FormatVisitor(&astInformation);
|
auto visitor = new FormatVisitor(&astInformation);
|
||||||
visitor.visit(mod);
|
visitor.visit(mod);
|
||||||
astInformation.cleanup();
|
astInformation.cleanup();
|
||||||
auto tokens = byToken(buffer, config, &cache).array();
|
auto tokens = byToken(buffer, config, &cache).array();
|
||||||
auto tokenFormatter = TokenFormatter(tokens, stdout, &astInformation,
|
auto tokenFormatter = TokenFormatter(tokens, stdout, &astInformation,
|
||||||
&formatterConfig);
|
&formatterConfig);
|
||||||
tokenFormatter.format();
|
tokenFormatter.format();
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -68,12 +80,12 @@ int main(string[] args)
|
||||||
struct TokenFormatter
|
struct TokenFormatter
|
||||||
{
|
{
|
||||||
this(const(Token)[] tokens, File output, ASTInformation* astInformation,
|
this(const(Token)[] tokens, File output, ASTInformation* astInformation,
|
||||||
FormatterConfig* config)
|
FormatterConfig* config)
|
||||||
{
|
{
|
||||||
this.tokens = tokens;
|
this.tokens = tokens;
|
||||||
this.output = output;
|
this.output = output;
|
||||||
this.astInformation = astInformation;
|
this.astInformation = astInformation;
|
||||||
this.config = config;
|
this.config = config;
|
||||||
}
|
}
|
||||||
|
|
||||||
void format()
|
void format()
|
||||||
|
@ -87,6 +99,8 @@ struct TokenFormatter
|
||||||
assert (indentLevel >= 0);
|
assert (indentLevel >= 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
void formatStep()
|
void formatStep()
|
||||||
{
|
{
|
||||||
import std.range:assumeSorted;
|
import std.range:assumeSorted;
|
||||||
|
@ -144,20 +158,20 @@ struct TokenFormatter
|
||||||
case tok!"cast":
|
case tok!"cast":
|
||||||
writeToken();
|
writeToken();
|
||||||
break;
|
break;
|
||||||
case tok!"mixin":
|
case tok!"mixin":
|
||||||
writeToken();
|
writeToken();
|
||||||
write(" ");
|
write(" ");
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
if (index + 1 < tokens.length)
|
if (index + 1 < tokens.length)
|
||||||
{
|
{
|
||||||
auto next = tokens[index + 1];
|
auto next = tokens[index + 1];
|
||||||
if (next.type == tok!";" || next.type == tok!"("
|
if (next.type == tok!";" || next.type == tok!"("
|
||||||
|| next.type == tok!")" || next.type == tok!","
|
|| next.type == tok!")" || next.type == tok!","
|
||||||
|| next.type == tok!"{")
|
|| next.type == tok!"{")
|
||||||
{
|
{
|
||||||
writeToken();
|
writeToken();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
writeToken();
|
writeToken();
|
||||||
|
@ -312,18 +326,21 @@ struct TokenFormatter
|
||||||
assert (false, str(current.type));
|
assert (false, str(current.type));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Pushes a temporary indent level
|
||||||
void pushIndent()
|
void pushIndent()
|
||||||
{
|
{
|
||||||
if (tempIndent == 0)
|
if (tempIndent == 0)
|
||||||
tempIndent++;
|
tempIndent++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Pops a temporary indent level
|
||||||
void popIndent()
|
void popIndent()
|
||||||
{
|
{
|
||||||
if (tempIndent > 0)
|
if (tempIndent > 0)
|
||||||
tempIndent--;
|
tempIndent--;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Writes balanced braces
|
||||||
void writeBraces()
|
void writeBraces()
|
||||||
{
|
{
|
||||||
import std.range : assumeSorted;
|
import std.range : assumeSorted;
|
||||||
|
@ -349,29 +366,32 @@ struct TokenFormatter
|
||||||
}
|
}
|
||||||
else if (current.type == tok!"}")
|
else if (current.type == tok!"}")
|
||||||
{
|
{
|
||||||
|
// Silly hack to format enums better.
|
||||||
|
if (peekBackIs(tok!"identifier"))
|
||||||
|
newline();
|
||||||
write("}");
|
write("}");
|
||||||
depth--;
|
depth--;
|
||||||
if (index < tokens.length &&
|
if (index < tokens.length &&
|
||||||
assumeSorted(astInformation.doubleNewlineLocations)
|
assumeSorted(astInformation.doubleNewlineLocations)
|
||||||
.equalRange(tokens[index].index).length)
|
.equalRange(tokens[index].index).length)
|
||||||
{
|
{
|
||||||
output.write("\n");
|
output.write("\n");
|
||||||
}
|
}
|
||||||
if (config.braceStyle == BraceStyle.otbs)
|
if (config.braceStyle == BraceStyle.otbs)
|
||||||
{
|
{
|
||||||
index++;
|
index++;
|
||||||
if (index < tokens.length && current.type == tok!"else")
|
if (index < tokens.length && current.type == tok!"else")
|
||||||
write(" ");
|
write(" ");
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (peekIs(tok!"case") || peekIs(tok!"default"))
|
if (peekIs(tok!"case") || peekIs(tok!"default"))
|
||||||
indentLevel--;
|
indentLevel--;
|
||||||
newline();
|
newline();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
index++;
|
index++;
|
||||||
if (peekIs(tok!"case") || peekIs(tok!"default"))
|
if (peekIs(tok!"case") || peekIs(tok!"default"))
|
||||||
indentLevel--;
|
indentLevel--;
|
||||||
newline();
|
newline();
|
||||||
|
@ -411,7 +431,7 @@ struct TokenFormatter
|
||||||
else if (current.type == tok!")")
|
else if (current.type == tok!")")
|
||||||
{
|
{
|
||||||
if (peekIs(tok!"identifier") || (index + 1 < tokens.length
|
if (peekIs(tok!"identifier") || (index + 1 < tokens.length
|
||||||
&& isKeyword(tokens[index + 1].type)))
|
&& isKeyword(tokens[index + 1].type)))
|
||||||
{
|
{
|
||||||
writeToken();
|
writeToken();
|
||||||
write(" ");
|
write(" ");
|
||||||
|
@ -623,21 +643,21 @@ struct TokenFormatter
|
||||||
/// Information about the AST
|
/// Information about the AST
|
||||||
ASTInformation* astInformation;
|
ASTInformation* astInformation;
|
||||||
|
|
||||||
/// Configuration
|
/// Configuration
|
||||||
FormatterConfig* config;
|
FormatterConfig* config;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The only good brace styles
|
/// The only good brace styles
|
||||||
enum BraceStyle
|
enum BraceStyle
|
||||||
{
|
{
|
||||||
allman,
|
allman,
|
||||||
otbs
|
otbs
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Configuration options for formatting
|
/// Configuration options for formatting
|
||||||
struct FormatterConfig
|
struct FormatterConfig
|
||||||
{
|
{
|
||||||
/// Number of spaces used for indentation
|
/// Number of spaces used for indentation
|
||||||
uint indentSize = 4;
|
uint indentSize = 4;
|
||||||
|
|
||||||
/// Use tabs or spaces
|
/// Use tabs or spaces
|
||||||
|
@ -652,8 +672,8 @@ struct FormatterConfig
|
||||||
/// Hard line wrap limit
|
/// Hard line wrap limit
|
||||||
uint columnHardLimit = 120;
|
uint columnHardLimit = 120;
|
||||||
|
|
||||||
/// Use the One True Brace Style
|
/// Use the One True Brace Style
|
||||||
BraceStyle braceStyle = BraceStyle.otbs;
|
BraceStyle braceStyle = BraceStyle.allman;
|
||||||
}
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
|
@ -700,6 +720,12 @@ final class FormatVisitor : ASTVisitor
|
||||||
functionBody.accept(this);
|
functionBody.accept(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override void visit(const EnumBody enumBody)
|
||||||
|
{
|
||||||
|
astInformation.doubleNewlineLocations ~= enumBody.endLocation;
|
||||||
|
enumBody.accept(this);
|
||||||
|
}
|
||||||
|
|
||||||
override void visit(const Unittest unittest_)
|
override void visit(const Unittest unittest_)
|
||||||
{
|
{
|
||||||
astInformation.doubleNewlineLocations ~= unittest_.blockStatement.endLocation;
|
astInformation.doubleNewlineLocations ~= unittest_.blockStatement.endLocation;
|
||||||
|
@ -718,7 +744,7 @@ final class FormatVisitor : ASTVisitor
|
||||||
structBody.accept(this);
|
structBody.accept(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
override void visit(const TemplateDeclaration templateDeclaration)
|
override void visit(const TemplateDeclaration templateDeclaration)
|
||||||
{
|
{
|
||||||
astInformation.doubleNewlineLocations ~= templateDeclaration.endLocation;
|
astInformation.doubleNewlineLocations ~= templateDeclaration.endLocation;
|
||||||
templateDeclaration.accept(this);
|
templateDeclaration.accept(this);
|
||||||
|
|
Loading…
Reference in New Issue