| OLD | NEW |
| (Empty) |
| 1 // Copyright (c) 2014, the Dart project authors. Please see the AUTHORS file | |
| 2 // for details. All rights reserved. Use of this source code is governed by a | |
| 3 // BSD-style license that can be found in the LICENSE file. | |
| 4 | |
| 5 library dart_style.src.line_prefix; | |
| 6 | |
| 7 import 'chunk.dart'; | |
| 8 import 'nesting.dart'; | |
| 9 import 'rule/rule.dart'; | |
| 10 | |
| 11 /// A prefix of a series of chunks and the context needed to uniquely describe | |
| 12 /// any shared state between the preceding and following chunks. | |
| 13 /// | |
| 14 /// This is used by [LineSplitter] to memoize suffixes whose best splits have | |
| 15 /// previously been calculated. For each unique [LinePrefix], there is a single | |
| 16 /// set of best splits for the remainder of the line following it. | |
| 17 /// | |
| 18 /// [LinePrefix] is a value type. It overloads [hashCode] and [==] and it's | |
| 19 /// critical that those be correct and efficient. These objects are used as | |
| 20 /// keys in the [LineSplitter]'s memoization table. | |
| 21 class LinePrefix { | |
| 22 /// The number of chunks in the prefix. | |
| 23 /// | |
| 24 /// The suffix is the remaining chunks starting at index [length]. | |
| 25 final int length; | |
| 26 | |
| 27 /// The [Rule]s that apply to chunks in the prefix and have thus already had | |
| 28 /// their values selected. | |
| 29 /// | |
| 30 /// Does not include rules that do not also appear in the suffix since they | |
| 31 /// don't affect the suffix. | |
| 32 final Map<Rule, int> ruleValues; | |
| 33 | |
| 34 /// The number of characters of "statement-based" indentation of the line | |
| 35 /// after the prefix. | |
| 36 /// | |
| 37 /// This handles things like control flow, switch cases, and constructor | |
| 38 /// initialization lists that tweak the per-line indentation. | |
| 39 /// | |
| 40 /// For nested blocks, this also includes the indentation to push the entire | |
| 41 /// block over. | |
| 42 final int _indent; | |
| 43 | |
| 44 final NestingSplitter _nesting; | |
| 45 | |
| 46 /// The absolute starting column of the line after this chunk. | |
| 47 /// | |
| 48 /// This takes into account whether the line should be flush left or not. | |
| 49 int get column => _flushLeft ? 0 : _indent + _nesting.indent; | |
| 50 final bool _flushLeft; | |
| 51 | |
| 52 /// Creates a new zero-length prefix with initial [indent] whose suffix is | |
| 53 /// the entire line. | |
| 54 LinePrefix(int indent, {bool flushLeft}) | |
| 55 : this._(0, {}, indent, new NestingSplitter(), flushLeft: flushLeft); | |
| 56 | |
| 57 LinePrefix._(this.length, this.ruleValues, this._indent, this._nesting, | |
| 58 {bool flushLeft: false}) | |
| 59 : _flushLeft = flushLeft; | |
| 60 | |
| 61 bool operator ==(other) { | |
| 62 if (other is! LinePrefix) return false; | |
| 63 | |
| 64 if (length != other.length) return false; | |
| 65 if (_indent != other._indent) return false; | |
| 66 if (_flushLeft != other._flushLeft) return false; | |
| 67 if (_nesting != other._nesting) return false; | |
| 68 | |
| 69 // Compare rule values. | |
| 70 if (ruleValues.length != other.ruleValues.length) return false; | |
| 71 | |
| 72 for (var key in ruleValues.keys) { | |
| 73 if (other.ruleValues[key] != ruleValues[key]) return false; | |
| 74 } | |
| 75 | |
| 76 return true; | |
| 77 } | |
| 78 | |
| 79 int get hashCode => length.hashCode ^ _indent ^ _nesting.hashCode; | |
| 80 | |
| 81 /// Create a new LinePrefix one chunk longer than this one using [ruleValues], | |
| 82 /// and assuming that we do not split before that chunk. | |
| 83 LinePrefix extend(Map<Rule, int> ruleValues) => | |
| 84 new LinePrefix._(length + 1, ruleValues, _indent, _nesting, | |
| 85 flushLeft: _flushLeft); | |
| 86 | |
| 87 /// Create a series of new LinePrefixes one chunk longer than this one using | |
| 88 /// [ruleValues], and assuming that the new [chunk] splits at an expression | |
| 89 /// boundary so there may be multiple possible different nesting stacks. | |
| 90 /// | |
| 91 /// If this prefix is for a nested block, [blockIndentation] may be nonzero | |
| 92 /// to push the output to the right. | |
| 93 Iterable<LinePrefix> split( | |
| 94 Chunk chunk, int blockIndentation, Map<Rule, int> ruleValues) { | |
| 95 var indent = chunk.indent + blockIndentation; | |
| 96 var flushLeft = chunk.flushLeft; | |
| 97 | |
| 98 // If the chunk has a block, then its flushLeft property is for the first | |
| 99 // line of the block, not the line after the block. The line after the block | |
| 100 // is never flush left since it will always be for a `}` or `]`. | |
| 101 if (chunk.blockChunks.isNotEmpty) flushLeft = false; | |
| 102 | |
| 103 return _nesting.update(chunk.nesting).map((nesting) => new LinePrefix._( | |
| 104 length + 1, ruleValues, indent, nesting, | |
| 105 flushLeft: flushLeft)); | |
| 106 } | |
| 107 | |
| 108 String toString() { | |
| 109 var result = "prefix $length"; | |
| 110 if (_indent != 0) result += " indent ${_indent}"; | |
| 111 if (_nesting.indent != 0) result += " nesting ${_nesting.indent}"; | |
| 112 if (ruleValues.isNotEmpty) { | |
| 113 var rules = | |
| 114 ruleValues.keys.map((key) => "$key:${ruleValues[key]}").join(" "); | |
| 115 | |
| 116 result += " rules $rules"; | |
| 117 } | |
| 118 return result; | |
| 119 } | |
| 120 } | |
| OLD | NEW |