fix: use temporary buffer for conditional newline

Signed-off-by: Prajwal S N <prajwalnadig21@gmail.com>
This commit is contained in:
Prajwal S N 2024-01-02 14:47:20 +05:30
parent 3eb09287b2
commit 90bb91d2c6
No known key found for this signature in database
GPG Key ID: 60701A603988FAC2
1 changed files with 48 additions and 14 deletions

View File

@ -16,13 +16,15 @@ import dfmt.config;
import dfmt.editorconfig; import dfmt.editorconfig;
import std.range; import std.range;
import std.format : format; import std.format : format;
import std.stdio : writeln, File; import std.stdio : File;
extern (C++) class FormatVisitor : SemanticTimeTransitiveVisitor extern (C++) class FormatVisitor : SemanticTimeTransitiveVisitor
{ {
File.LockingTextWriter buf; File.LockingTextWriter buf;
const Config* config; const Config* config;
string eol; string eol;
string tempBuf; // a string buffer to temporarily store data for line splitting
bool useTempBuf; // toggles the buffer being written to
uint depth; // the current indentation level uint depth; // the current indentation level
uint length; // the length of the current line of code uint length; // the length of the current line of code
bool declString; // set while declaring alias for string,wstring or dstring bool declString; // set while declaring alias for string,wstring or dstring
@ -81,14 +83,24 @@ extern (C++) class FormatVisitor : SemanticTimeTransitiveVisitor
isNewline = true; isNewline = true;
} }
void conditionalNewline(T)(T data) // Writes a newline only if the data will exceed the maximum allowed line length
bool conditionalNewline()
{ {
// If the current length is crosses the soft limit OR // If the current length crosses the soft limit OR
// if the current length + data length crosses the hard limit, // if the current length + data length crosses the hard limit,
// insert a newline. // insert a newline.
if (length > config.dfmt_soft_max_line_length if (length > config.dfmt_soft_max_line_length
|| (length + data.length) > config.max_line_length) || (length + tempBuf.length) > config.max_line_length) {
newline(); newline();
return true;
}
return false;
}
void writeTempBuf() {
useTempBuf = false;
write(tempBuf);
tempBuf = "";
} }
void write(T)(T data) if (is(T : char) || is(T : dchar)) void write(T)(T data) if (is(T : char) || is(T : dchar))
@ -98,7 +110,10 @@ extern (C++) class FormatVisitor : SemanticTimeTransitiveVisitor
indent(); indent();
isNewline = false; isNewline = false;
} }
buf.put(data); if (useTempBuf)
tempBuf ~= data;
else
buf.put(data);
length += 1; length += 1;
} }
@ -109,7 +124,10 @@ extern (C++) class FormatVisitor : SemanticTimeTransitiveVisitor
indent(); indent();
isNewline = false; isNewline = false;
} }
buf.put(data); if (useTempBuf)
tempBuf ~= data;
else
buf.put(data);
length += data.length; length += data.length;
} }
@ -232,8 +250,6 @@ extern (C++) class FormatVisitor : SemanticTimeTransitiveVisitor
void visitInteger(ASTCodegen.IntegerExp e) void visitInteger(ASTCodegen.IntegerExp e)
{ {
import core.stdc.stdio : sprintf;
auto v = e.toInteger(); auto v = e.toInteger();
if (e.type) if (e.type)
{ {
@ -3241,6 +3257,7 @@ extern (C++) class FormatVisitor : SemanticTimeTransitiveVisitor
void visitTemplateConstraint(ASTCodegen.Expression constraint) void visitTemplateConstraint(ASTCodegen.Expression constraint)
{ {
if (!constraint) if (!constraint)
return; return;
@ -3249,7 +3266,7 @@ extern (C++) class FormatVisitor : SemanticTimeTransitiveVisitor
case TemplateConstraintStyle._unspecified: case TemplateConstraintStyle._unspecified:
// Fallthrough to the default case // Fallthrough to the default case
case TemplateConstraintStyle.conditional_newline_indent: case TemplateConstraintStyle.conditional_newline_indent:
conditionalNewline(); useTempBuf = true;
depth++; depth++;
break; break;
case TemplateConstraintStyle.always_newline_indent: case TemplateConstraintStyle.always_newline_indent:
@ -3257,24 +3274,41 @@ extern (C++) class FormatVisitor : SemanticTimeTransitiveVisitor
depth++; depth++;
break; break;
case TemplateConstraintStyle.conditional_newline: case TemplateConstraintStyle.conditional_newline:
conditionalNewline(); useTempBuf = true;
break; break;
case TemplateConstraintStyle.always_newline: case TemplateConstraintStyle.always_newline:
newline(); newline();
break; break;
} }
write(" if"); write("if");
if (config.dfmt_space_after_keywords) if (config.dfmt_space_after_keywords)
write(' '); write(' ');
write('('); write('(');
writeExpr(constraint); writeExpr(constraint);
write(')'); write(')');
if (config.dfmt_template_constraint_style == TemplateConstraintStyle.always_newline_indent final switch (config.dfmt_template_constraint_style)
|| config.dfmt_template_constraint_style {
== TemplateConstraintStyle.conditional_newline_indent) case TemplateConstraintStyle._unspecified:
// Fallthrough to the default case
case TemplateConstraintStyle.conditional_newline_indent:
if (!conditionalNewline())
buf.put(' ');
writeTempBuf();
depth--; depth--;
break;
case TemplateConstraintStyle.always_newline_indent:
depth--;
break;
case TemplateConstraintStyle.conditional_newline:
if (!conditionalNewline())
buf.put(' ');
writeTempBuf();
break;
case TemplateConstraintStyle.always_newline:
break;
}
} }
override void visitBaseClasses(ASTCodegen.ClassDeclaration d) override void visitBaseClasses(ASTCodegen.ClassDeclaration d)