diff --git a/src/dfmt/indentation.d b/src/dfmt/indentation.d index cdd78e4..50d7ab8 100644 --- a/src/dfmt/indentation.d +++ b/src/dfmt/indentation.d @@ -7,25 +7,82 @@ module dfmt.indentation; import dfmt.config; import dfmt.editorconfig; -import dparse.lexer; +import dmd.tokens; import std.bitmanip : bitfields; /** * Returns: true if the given token type is a wrap indent type */ -bool isWrapIndent(IdType type) pure nothrow @nogc @safe +bool isWrapIndent(TOK type) pure nothrow @nogc @safe { - return type != tok!"{" && type != tok!"case" && type != tok!"@" - && type != tok!"]" && type != tok!"(" && type != tok!")" && isOperator(type); + switch (type) + { + case TOK.leftCurly: + case TOK.case_: + case TOK.at: + case TOK.rightBracket: + case TOK.leftParenthesis: + case TOK.rightParenthesis: + return false; + + // Operators + case TOK.lessThan: + case TOK.greaterThan: + case TOK.lessOrEqual: + case TOK.greaterOrEqual: + case TOK.equal: + case TOK.notEqual: + case TOK.identity: + case TOK.notIdentity: + case TOK.is_: + + case TOK.leftShift: + case TOK.rightShift: + case TOK.leftShiftAssign: + case TOK.rightShiftAssign: + case TOK.unsignedRightShift: + case TOK.unsignedRightShiftAssign: + case TOK.concatenateAssign: + case TOK.add: + case TOK.min: + case TOK.addAssign: + case TOK.minAssign: + case TOK.mul: + case TOK.div: + case TOK.mod: + case TOK.mulAssign: + case TOK.divAssign: + case TOK.modAssign: + case TOK.and: + case TOK.or: + case TOK.xor: + case TOK.andAssign: + case TOK.orAssign: + case TOK.xorAssign: + case TOK.assign: + case TOK.not: + case TOK.tilde: + case TOK.plusPlus: + case TOK.minusMinus: + case TOK.dot: + case TOK.comma: + case TOK.question: + case TOK.andAnd: + case TOK.orOr: + return true; + default: + return false; + } } /** * Returns: true if the given token type is a temporary indent type */ -bool isTempIndent(IdType type) pure nothrow @nogc @safe +bool isTempIndent(TOK type) pure nothrow @nogc @safe { - return type != tok!")" && type != tok!"{" && type != tok!"case" && type != tok!"@"; + return type != TOK.rightParenthesis && type != TOK.leftCurly && type != TOK.case_ && type != TOK + .at; } /** @@ -44,24 +101,20 @@ struct IndentStack static struct Details { mixin(bitfields!( - // generally true for all operators except {, case, @, ], (, ) - bool, "wrap", 1, - // temporary indentation which get's reverted when a block starts - // generally true for all tokens except ), {, case, @ - bool, "temp", 1, - // emit minimal newlines - bool, "mini", 1, - // for associative arrays or arrays containing them, break after every item - bool, "breakEveryItem", 1, - // when an item inside an array would break mid-item, definitely break at the comma first - bool, "preferLongBreaking", 1, - uint, "", 27)); + // generally true for all operators except {, case, @, ], (, ) + bool, "wrap", 1, // temporary indentation which get's reverted when a block starts + // generally true for all tokens except ), {, case, @ + bool, "temp", 1, // emit minimal newlines + bool, "mini", 1, // for associative arrays or arrays containing them, break after every item + bool, "breakEveryItem", 1, // when an item inside an array would break mid-item, definitely break at the comma first + bool, "preferLongBreaking", 1, + uint, "", 27)); } /** * Get the indent size at the most recent occurrence of the given indent type */ - int indentToMostRecent(IdType item) const + int indentToMostRecent(TOK item) const { if (index == 0) return -1; @@ -84,7 +137,7 @@ struct IndentStack int tempIndentCount = 0; for (size_t i = index; i > 0; i--) { - if (!details[i - 1].wrap && arr[i - 1] != tok!"]") + if (!details[i - 1].wrap && arr[i - 1] != TOK.rightBracket) break; tempIndentCount++; } @@ -94,7 +147,7 @@ struct IndentStack /** * Pushes the given indent type on to the stack. */ - void push(IdType item) pure nothrow + void push(TOK item) pure nothrow { Details detail; detail.wrap = isWrapIndent(item); @@ -105,7 +158,7 @@ struct IndentStack /** * Pushes the given indent type on to the stack. */ - void push(IdType item, Details detail) pure nothrow + void push(TOK item, Details detail) pure nothrow { arr[index] = item; details[index] = detail; @@ -145,7 +198,7 @@ struct IndentStack index--; } - bool topAre(IdType[] types...) + bool topAre(TOK[] types...) { if (types.length > index) return false; @@ -156,7 +209,7 @@ struct IndentStack /** * Returns: `true` if the top of the indent stack is the given indent type. */ - bool topIs(IdType type) const pure nothrow @safe @nogc + bool topIs(TOK type) const pure nothrow @safe @nogc { return index > 0 && index <= arr.length && arr[index - 1] == type; } @@ -172,9 +225,10 @@ struct IndentStack /** * Returns: `true` if the top of the indent stack is a temporary indent with the specified token */ - bool topIsTemp(IdType item) + bool topIsTemp(TOK item) { - return index > 0 && index <= arr.length && arr[index - 1] == item && details[index - 1].temp; + return index > 0 && index <= arr.length && arr[index - 1] == item && details[index - 1] + .temp; } /** @@ -188,16 +242,17 @@ struct IndentStack /** * Returns: `true` if the top of the indent stack is a temporary indent with the specified token */ - bool topIsWrap(IdType item) + bool topIsWrap(TOK item) { - return index > 0 && index <= arr.length && arr[index - 1] == item && details[index - 1].wrap; + return index > 0 && index <= arr.length && arr[index - 1] == item && details[index - 1] + .wrap; } /** * Returns: `true` if the top of the indent stack is one of the given token * types. */ - bool topIsOneOf(IdType[] types...) const pure nothrow @safe @nogc + bool topIsOneOf(TOK[] types...) const pure nothrow @safe @nogc { if (index == 0) return false; @@ -208,7 +263,7 @@ struct IndentStack return false; } - IdType top() const pure nothrow @property @safe @nogc + TOK top() const pure nothrow @property @safe @nogc { return arr[index - 1]; } @@ -233,26 +288,28 @@ struct IndentStack */ void dump(size_t pos = size_t.max, string file = __FILE__, uint line = __LINE__) const { - import dparse.lexer : str; import std.algorithm.iteration : map; import std.stdio : stderr; if (pos == size_t.max) - stderr.writefln("\033[31m%s:%d %(%s %)\033[0m", file, line, arr[0 .. index].map!(a => str(a))); + stderr.writefln("\033[31m%s:%d %(%s %)\033[0m", file, line, arr[0 .. index].map!( + a => Token.toString(a))); else - stderr.writefln("\033[31m%s:%d at %d %(%s %)\033[0m", file, line, pos, arr[0 .. index].map!(a => str(a))); + stderr.writefln("\033[31m%s:%d at %d %(%s %)\033[0m", file, line, pos, arr[0 .. index].map!( + a => Token.toString(a))); } private: size_t index; - IdType[256] arr; + TOK[256] arr; Details[arr.length] details; int indentSize(const size_t k = size_t.max) const pure nothrow @safe @nogc { import std.algorithm : among; + if (index == 0 || k == 0) return 0; immutable size_t j = k == size_t.max ? index : k; @@ -260,9 +317,9 @@ private: int parenCount; foreach (i; 0 .. j) { - immutable int pc = (arr[i] == tok!"!" || arr[i] == tok!"(" || arr[i] == tok!")") ? parenCount + 1 - : parenCount; - if ((details[i].wrap || arr[i] == tok!"(") && parenCount > 1) + immutable int pc = (arr[i] == TOK.not || arr[i] == TOK.leftParenthesis || arr[i] == TOK + .rightParenthesis) ? parenCount + 1 : parenCount; + if ((details[i].wrap || arr[i] == TOK.leftParenthesis) && parenCount > 1) { parenCount = pc; continue; @@ -277,31 +334,33 @@ private: } immutable currentIsNonWrapTemp = !details[i].wrap - && details[i].temp && arr[i] != tok!")" && arr[i] != tok!"!"; + && details[i].temp && arr[i] != TOK.rightParenthesis && arr[i] != TOK.not; - if (currentIsNonWrapTemp && arr[i + 1] == tok!"]") + if (currentIsNonWrapTemp && arr[i + 1] == TOK.rightBracket) { parenCount = pc; continue; } - if (arr[i] == tok!"static" - && arr[i + 1].among!(tok!"if", tok!"else", tok!"foreach", tok!"foreach_reverse") - && (i + 2 >= index || arr[i + 2] != tok!"{")) + if (arr[i] == TOK.static_ + && arr[i + 1].among!(TOK.if_, TOK.else_, TOK.foreach_, TOK.foreach_reverse_) + && (i + 2 >= index || arr[i + 2] != TOK.leftCurly)) { parenCount = pc; continue; } - if (currentIsNonWrapTemp && (arr[i + 1] == tok!"switch" - || arr[i + 1] == tok!"{" || arr[i + 1] == tok!")")) + + if (currentIsNonWrapTemp && (arr[i + 1] == TOK.switch_ + || arr[i + 1] == TOK.leftCurly || arr[i + 1] == TOK.rightParenthesis)) { parenCount = pc; continue; } } - else if (parenCount == 0 && arr[i] == tok!"(" && config.dfmt_single_indent == OptionalBoolean.f) + else if (parenCount == 0 && arr[i] == TOK.leftParenthesis && config.dfmt_single_indent == OptionalBoolean + .f) size++; - if (arr[i] == tok!"!") + if (arr[i] == TOK.not) size++; parenCount = pc; @@ -312,21 +371,21 @@ private: 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!"("); + return (details[i + 1].wrap && arr[i] == TOK.rightParenthesis) + || (parenCount == 0 && arr[i + 1] == TOK.comma && arr[i] == TOK.leftParenthesis); } } unittest { IndentStack stack; - stack.push(tok!"{"); + stack.push(TOK.leftCurly); assert(stack.length == 1); assert(stack.indentLevel == 1); stack.pop(); assert(stack.length == 0); assert(stack.indentLevel == 0); - stack.push(tok!"if"); + stack.push(TOK.if_); assert(stack.topIsTemp()); stack.popTempIndents(); assert(stack.length == 0); diff --git a/src/dfmt/wrapping.d b/src/dfmt/wrapping.d index 79fb85f..1d3f436 100644 --- a/src/dfmt/wrapping.d +++ b/src/dfmt/wrapping.d @@ -5,14 +5,14 @@ module dfmt.wrapping; -import dparse.lexer; +import dmd.tokens; import dfmt.tokens; import dfmt.config; struct State { this(uint breaks, const Token[] tokens, immutable short[] depths, - const Config* config, int currentLineLength, int indentLevel) pure @safe + const Config* config, int currentLineLength, int indentLevel) @safe { import std.math : abs; import core.bitop : popcnt, bsf; @@ -42,8 +42,8 @@ struct State { if (((1 << i) & breaks) == 0) continue; - immutable prevType = i > 0 ? tokens[i - 1].type : tok!""; - immutable currentType = tokens[i].type; + immutable prevType = i > 0 ? tokens[i - 1].value : TOK.error; + immutable currentType = tokens[i].value; immutable p = abs(depths[i]); immutable bc = breakCost(prevType, currentType) * (p == 0 ? 1 : p * 2); this._cost += bc + newlinePenalty; @@ -126,7 +126,7 @@ private enum ALGORITHMIC_COMPLEXITY_SUCKS = uint.sizeof * 8; * continuation indents are reduced. This is used for array literals. */ size_t[] chooseLineBreakTokens(size_t index, const Token[] tokens, - immutable short[] depths, const Config* config, int currentLineLength, int indentLevel) + immutable short[] depths, const Config* config, int currentLineLength, int indentLevel) { import std.container.rbtree : RedBlackTree; import std.algorithm : filter, min; @@ -159,7 +159,7 @@ size_t[] chooseLineBreakTokens(size_t index, const Token[] tokens, if (current < lowest) lowest = current; validMoves!(typeof(open))(open, tokens[0 .. tokensEnd], depths[0 .. tokensEnd], - current.breaks, config, currentLineLength, indentLevel); + current.breaks, config, currentLineLength, indentLevel); } foreach (r; open[].filter!(a => a.solved)) return genRetVal(r.breaks, index); @@ -170,11 +170,11 @@ size_t[] chooseLineBreakTokens(size_t index, const Token[] tokens, } void validMoves(OR)(auto ref OR output, const Token[] tokens, immutable short[] depths, - uint current, const Config* config, int currentLineLength, int indentLevel) + uint current, const Config* config, int currentLineLength, int indentLevel) { foreach (i, token; tokens) { - if (!isBreakToken(token.type) || (((1 << i) & current) != 0)) + if (!isBreakToken(token.value) || (((1 << i) & current) != 0)) continue; immutable uint breaks = current | (1 << i); output.insert(State(breaks, tokens, depths, config, currentLineLength, indentLevel));