vscode-css-languageservice 5.4.2 → 6.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (84) hide show
  1. package/CHANGELOG.md +5 -1
  2. package/lib/esm/cssLanguageService.d.ts +37 -37
  3. package/lib/esm/cssLanguageService.js +72 -75
  4. package/lib/esm/cssLanguageTypes.d.ts +238 -238
  5. package/lib/esm/cssLanguageTypes.js +42 -42
  6. package/lib/esm/data/webCustomData.js +21965 -21965
  7. package/lib/esm/languageFacts/builtinData.js +142 -142
  8. package/lib/esm/languageFacts/colors.js +469 -472
  9. package/lib/esm/languageFacts/dataManager.js +88 -92
  10. package/lib/esm/languageFacts/dataProvider.js +73 -79
  11. package/lib/esm/languageFacts/entry.js +137 -138
  12. package/lib/esm/languageFacts/facts.js +8 -8
  13. package/lib/esm/parser/cssErrors.js +48 -50
  14. package/lib/esm/parser/cssNodes.js +1502 -2019
  15. package/lib/esm/parser/cssParser.js +1534 -1563
  16. package/lib/esm/parser/cssScanner.js +592 -599
  17. package/lib/esm/parser/cssSymbolScope.js +311 -341
  18. package/lib/esm/parser/lessParser.js +714 -740
  19. package/lib/esm/parser/lessScanner.js +57 -78
  20. package/lib/esm/parser/scssErrors.js +18 -20
  21. package/lib/esm/parser/scssParser.js +796 -818
  22. package/lib/esm/parser/scssScanner.js +95 -116
  23. package/lib/esm/services/cssCodeActions.js +77 -81
  24. package/lib/esm/services/cssCompletion.js +1054 -1149
  25. package/lib/esm/services/cssFolding.js +190 -193
  26. package/lib/esm/services/cssFormatter.js +136 -136
  27. package/lib/esm/services/cssHover.js +148 -151
  28. package/lib/esm/services/cssNavigation.js +378 -470
  29. package/lib/esm/services/cssSelectionRange.js +47 -47
  30. package/lib/esm/services/cssValidation.js +41 -44
  31. package/lib/esm/services/lessCompletion.js +378 -397
  32. package/lib/esm/services/lint.js +518 -532
  33. package/lib/esm/services/lintRules.js +76 -83
  34. package/lib/esm/services/lintUtil.js +196 -205
  35. package/lib/esm/services/pathCompletion.js +157 -231
  36. package/lib/esm/services/scssCompletion.js +354 -378
  37. package/lib/esm/services/scssNavigation.js +82 -154
  38. package/lib/esm/services/selectorPrinting.js +492 -536
  39. package/lib/esm/utils/arrays.js +40 -46
  40. package/lib/esm/utils/objects.js +11 -11
  41. package/lib/esm/utils/resources.js +11 -24
  42. package/lib/esm/utils/strings.js +102 -104
  43. package/lib/umd/cssLanguageService.d.ts +37 -37
  44. package/lib/umd/cssLanguageService.js +99 -102
  45. package/lib/umd/cssLanguageTypes.d.ts +238 -238
  46. package/lib/umd/cssLanguageTypes.js +89 -88
  47. package/lib/umd/data/webCustomData.js +21978 -21978
  48. package/lib/umd/languageFacts/builtinData.js +154 -154
  49. package/lib/umd/languageFacts/colors.js +492 -495
  50. package/lib/umd/languageFacts/dataManager.js +101 -104
  51. package/lib/umd/languageFacts/dataProvider.js +86 -91
  52. package/lib/umd/languageFacts/entry.js +152 -153
  53. package/lib/umd/languageFacts/facts.js +29 -29
  54. package/lib/umd/parser/cssErrors.js +61 -62
  55. package/lib/umd/parser/cssNodes.js +1587 -2034
  56. package/lib/umd/parser/cssParser.js +1547 -1575
  57. package/lib/umd/parser/cssScanner.js +606 -611
  58. package/lib/umd/parser/cssSymbolScope.js +328 -353
  59. package/lib/umd/parser/lessParser.js +727 -752
  60. package/lib/umd/parser/lessScanner.js +70 -90
  61. package/lib/umd/parser/scssErrors.js +31 -32
  62. package/lib/umd/parser/scssParser.js +809 -830
  63. package/lib/umd/parser/scssScanner.js +108 -128
  64. package/lib/umd/services/cssCodeActions.js +90 -93
  65. package/lib/umd/services/cssCompletion.js +1067 -1161
  66. package/lib/umd/services/cssFolding.js +203 -206
  67. package/lib/umd/services/cssFormatter.js +150 -150
  68. package/lib/umd/services/cssHover.js +161 -163
  69. package/lib/umd/services/cssNavigation.js +391 -482
  70. package/lib/umd/services/cssSelectionRange.js +60 -60
  71. package/lib/umd/services/cssValidation.js +54 -56
  72. package/lib/umd/services/lessCompletion.js +391 -409
  73. package/lib/umd/services/lint.js +531 -544
  74. package/lib/umd/services/lintRules.js +91 -95
  75. package/lib/umd/services/lintUtil.js +210 -218
  76. package/lib/umd/services/pathCompletion.js +171 -244
  77. package/lib/umd/services/scssCompletion.js +367 -390
  78. package/lib/umd/services/scssNavigation.js +95 -166
  79. package/lib/umd/services/selectorPrinting.js +510 -550
  80. package/lib/umd/utils/arrays.js +55 -61
  81. package/lib/umd/utils/objects.js +25 -25
  82. package/lib/umd/utils/resources.js +26 -39
  83. package/lib/umd/utils/strings.js +120 -122
  84. package/package.json +10 -10
@@ -1,193 +1,190 @@
1
- /*---------------------------------------------------------------------------------------------
2
- * Copyright (c) Microsoft Corporation. All rights reserved.
3
- * Licensed under the MIT License. See License.txt in the project root for license information.
4
- *--------------------------------------------------------------------------------------------*/
5
- 'use strict';
6
- import { TokenType, Scanner } from '../parser/cssScanner';
7
- import { SCSSScanner, InterpolationFunction } from '../parser/scssScanner';
8
- import { LESSScanner } from '../parser/lessScanner';
9
- export function getFoldingRanges(document, context) {
10
- var ranges = computeFoldingRanges(document);
11
- return limitFoldingRanges(ranges, context);
12
- }
13
- function computeFoldingRanges(document) {
14
- function getStartLine(t) {
15
- return document.positionAt(t.offset).line;
16
- }
17
- function getEndLine(t) {
18
- return document.positionAt(t.offset + t.len).line;
19
- }
20
- function getScanner() {
21
- switch (document.languageId) {
22
- case 'scss':
23
- return new SCSSScanner();
24
- case 'less':
25
- return new LESSScanner();
26
- default:
27
- return new Scanner();
28
- }
29
- }
30
- function tokenToRange(t, kind) {
31
- var startLine = getStartLine(t);
32
- var endLine = getEndLine(t);
33
- if (startLine !== endLine) {
34
- return {
35
- startLine: startLine,
36
- endLine: endLine,
37
- kind: kind
38
- };
39
- }
40
- else {
41
- return null;
42
- }
43
- }
44
- var ranges = [];
45
- var delimiterStack = [];
46
- var scanner = getScanner();
47
- scanner.ignoreComment = false;
48
- scanner.setSource(document.getText());
49
- var token = scanner.scan();
50
- var prevToken = null;
51
- var _loop_1 = function () {
52
- switch (token.type) {
53
- case TokenType.CurlyL:
54
- case InterpolationFunction:
55
- {
56
- delimiterStack.push({ line: getStartLine(token), type: 'brace', isStart: true });
57
- break;
58
- }
59
- case TokenType.CurlyR: {
60
- if (delimiterStack.length !== 0) {
61
- var prevDelimiter = popPrevStartDelimiterOfType(delimiterStack, 'brace');
62
- if (!prevDelimiter) {
63
- break;
64
- }
65
- var endLine = getEndLine(token);
66
- if (prevDelimiter.type === 'brace') {
67
- /**
68
- * Other than the case when curly brace is not on a new line by itself, for example
69
- * .foo {
70
- * color: red; }
71
- * Use endLine minus one to show ending curly brace
72
- */
73
- if (prevToken && getEndLine(prevToken) !== endLine) {
74
- endLine--;
75
- }
76
- if (prevDelimiter.line !== endLine) {
77
- ranges.push({
78
- startLine: prevDelimiter.line,
79
- endLine: endLine,
80
- kind: undefined
81
- });
82
- }
83
- }
84
- }
85
- break;
86
- }
87
- /**
88
- * In CSS, there is no single line comment prefixed with //
89
- * All comments are marked as `Comment`
90
- */
91
- case TokenType.Comment: {
92
- var commentRegionMarkerToDelimiter_1 = function (marker) {
93
- if (marker === '#region') {
94
- return { line: getStartLine(token), type: 'comment', isStart: true };
95
- }
96
- else {
97
- return { line: getEndLine(token), type: 'comment', isStart: false };
98
- }
99
- };
100
- var getCurrDelimiter = function (token) {
101
- var matches = token.text.match(/^\s*\/\*\s*(#region|#endregion)\b\s*(.*?)\s*\*\//);
102
- if (matches) {
103
- return commentRegionMarkerToDelimiter_1(matches[1]);
104
- }
105
- else if (document.languageId === 'scss' || document.languageId === 'less') {
106
- var matches_1 = token.text.match(/^\s*\/\/\s*(#region|#endregion)\b\s*(.*?)\s*/);
107
- if (matches_1) {
108
- return commentRegionMarkerToDelimiter_1(matches_1[1]);
109
- }
110
- }
111
- return null;
112
- };
113
- var currDelimiter = getCurrDelimiter(token);
114
- // /* */ comment region folding
115
- // All #region and #endregion cases
116
- if (currDelimiter) {
117
- if (currDelimiter.isStart) {
118
- delimiterStack.push(currDelimiter);
119
- }
120
- else {
121
- var prevDelimiter = popPrevStartDelimiterOfType(delimiterStack, 'comment');
122
- if (!prevDelimiter) {
123
- break;
124
- }
125
- if (prevDelimiter.type === 'comment') {
126
- if (prevDelimiter.line !== currDelimiter.line) {
127
- ranges.push({
128
- startLine: prevDelimiter.line,
129
- endLine: currDelimiter.line,
130
- kind: 'region'
131
- });
132
- }
133
- }
134
- }
135
- }
136
- // Multiline comment case
137
- else {
138
- var range = tokenToRange(token, 'comment');
139
- if (range) {
140
- ranges.push(range);
141
- }
142
- }
143
- break;
144
- }
145
- }
146
- prevToken = token;
147
- token = scanner.scan();
148
- };
149
- while (token.type !== TokenType.EOF) {
150
- _loop_1();
151
- }
152
- return ranges;
153
- }
154
- function popPrevStartDelimiterOfType(stack, type) {
155
- if (stack.length === 0) {
156
- return null;
157
- }
158
- for (var i = stack.length - 1; i >= 0; i--) {
159
- if (stack[i].type === type && stack[i].isStart) {
160
- return stack.splice(i, 1)[0];
161
- }
162
- }
163
- return null;
164
- }
165
- /**
166
- * - Sort regions
167
- * - Remove invalid regions (intersections)
168
- * - If limit exceeds, only return `rangeLimit` amount of ranges
169
- */
170
- function limitFoldingRanges(ranges, context) {
171
- var maxRanges = context && context.rangeLimit || Number.MAX_VALUE;
172
- var sortedRanges = ranges.sort(function (r1, r2) {
173
- var diff = r1.startLine - r2.startLine;
174
- if (diff === 0) {
175
- diff = r1.endLine - r2.endLine;
176
- }
177
- return diff;
178
- });
179
- var validRanges = [];
180
- var prevEndLine = -1;
181
- sortedRanges.forEach(function (r) {
182
- if (!(r.startLine < prevEndLine && prevEndLine < r.endLine)) {
183
- validRanges.push(r);
184
- prevEndLine = r.endLine;
185
- }
186
- });
187
- if (validRanges.length < maxRanges) {
188
- return validRanges;
189
- }
190
- else {
191
- return validRanges.slice(0, maxRanges);
192
- }
193
- }
1
+ /*---------------------------------------------------------------------------------------------
2
+ * Copyright (c) Microsoft Corporation. All rights reserved.
3
+ * Licensed under the MIT License. See License.txt in the project root for license information.
4
+ *--------------------------------------------------------------------------------------------*/
5
+ 'use strict';
6
+ import { TokenType, Scanner } from '../parser/cssScanner';
7
+ import { SCSSScanner, InterpolationFunction } from '../parser/scssScanner';
8
+ import { LESSScanner } from '../parser/lessScanner';
9
+ export function getFoldingRanges(document, context) {
10
+ const ranges = computeFoldingRanges(document);
11
+ return limitFoldingRanges(ranges, context);
12
+ }
13
+ function computeFoldingRanges(document) {
14
+ function getStartLine(t) {
15
+ return document.positionAt(t.offset).line;
16
+ }
17
+ function getEndLine(t) {
18
+ return document.positionAt(t.offset + t.len).line;
19
+ }
20
+ function getScanner() {
21
+ switch (document.languageId) {
22
+ case 'scss':
23
+ return new SCSSScanner();
24
+ case 'less':
25
+ return new LESSScanner();
26
+ default:
27
+ return new Scanner();
28
+ }
29
+ }
30
+ function tokenToRange(t, kind) {
31
+ const startLine = getStartLine(t);
32
+ const endLine = getEndLine(t);
33
+ if (startLine !== endLine) {
34
+ return {
35
+ startLine,
36
+ endLine,
37
+ kind
38
+ };
39
+ }
40
+ else {
41
+ return null;
42
+ }
43
+ }
44
+ const ranges = [];
45
+ const delimiterStack = [];
46
+ const scanner = getScanner();
47
+ scanner.ignoreComment = false;
48
+ scanner.setSource(document.getText());
49
+ let token = scanner.scan();
50
+ let prevToken = null;
51
+ while (token.type !== TokenType.EOF) {
52
+ switch (token.type) {
53
+ case TokenType.CurlyL:
54
+ case InterpolationFunction:
55
+ {
56
+ delimiterStack.push({ line: getStartLine(token), type: 'brace', isStart: true });
57
+ break;
58
+ }
59
+ case TokenType.CurlyR: {
60
+ if (delimiterStack.length !== 0) {
61
+ const prevDelimiter = popPrevStartDelimiterOfType(delimiterStack, 'brace');
62
+ if (!prevDelimiter) {
63
+ break;
64
+ }
65
+ let endLine = getEndLine(token);
66
+ if (prevDelimiter.type === 'brace') {
67
+ /**
68
+ * Other than the case when curly brace is not on a new line by itself, for example
69
+ * .foo {
70
+ * color: red; }
71
+ * Use endLine minus one to show ending curly brace
72
+ */
73
+ if (prevToken && getEndLine(prevToken) !== endLine) {
74
+ endLine--;
75
+ }
76
+ if (prevDelimiter.line !== endLine) {
77
+ ranges.push({
78
+ startLine: prevDelimiter.line,
79
+ endLine,
80
+ kind: undefined
81
+ });
82
+ }
83
+ }
84
+ }
85
+ break;
86
+ }
87
+ /**
88
+ * In CSS, there is no single line comment prefixed with //
89
+ * All comments are marked as `Comment`
90
+ */
91
+ case TokenType.Comment: {
92
+ const commentRegionMarkerToDelimiter = (marker) => {
93
+ if (marker === '#region') {
94
+ return { line: getStartLine(token), type: 'comment', isStart: true };
95
+ }
96
+ else {
97
+ return { line: getEndLine(token), type: 'comment', isStart: false };
98
+ }
99
+ };
100
+ const getCurrDelimiter = (token) => {
101
+ const matches = token.text.match(/^\s*\/\*\s*(#region|#endregion)\b\s*(.*?)\s*\*\//);
102
+ if (matches) {
103
+ return commentRegionMarkerToDelimiter(matches[1]);
104
+ }
105
+ else if (document.languageId === 'scss' || document.languageId === 'less') {
106
+ const matches = token.text.match(/^\s*\/\/\s*(#region|#endregion)\b\s*(.*?)\s*/);
107
+ if (matches) {
108
+ return commentRegionMarkerToDelimiter(matches[1]);
109
+ }
110
+ }
111
+ return null;
112
+ };
113
+ const currDelimiter = getCurrDelimiter(token);
114
+ // /* */ comment region folding
115
+ // All #region and #endregion cases
116
+ if (currDelimiter) {
117
+ if (currDelimiter.isStart) {
118
+ delimiterStack.push(currDelimiter);
119
+ }
120
+ else {
121
+ const prevDelimiter = popPrevStartDelimiterOfType(delimiterStack, 'comment');
122
+ if (!prevDelimiter) {
123
+ break;
124
+ }
125
+ if (prevDelimiter.type === 'comment') {
126
+ if (prevDelimiter.line !== currDelimiter.line) {
127
+ ranges.push({
128
+ startLine: prevDelimiter.line,
129
+ endLine: currDelimiter.line,
130
+ kind: 'region'
131
+ });
132
+ }
133
+ }
134
+ }
135
+ }
136
+ // Multiline comment case
137
+ else {
138
+ const range = tokenToRange(token, 'comment');
139
+ if (range) {
140
+ ranges.push(range);
141
+ }
142
+ }
143
+ break;
144
+ }
145
+ }
146
+ prevToken = token;
147
+ token = scanner.scan();
148
+ }
149
+ return ranges;
150
+ }
151
+ function popPrevStartDelimiterOfType(stack, type) {
152
+ if (stack.length === 0) {
153
+ return null;
154
+ }
155
+ for (let i = stack.length - 1; i >= 0; i--) {
156
+ if (stack[i].type === type && stack[i].isStart) {
157
+ return stack.splice(i, 1)[0];
158
+ }
159
+ }
160
+ return null;
161
+ }
162
+ /**
163
+ * - Sort regions
164
+ * - Remove invalid regions (intersections)
165
+ * - If limit exceeds, only return `rangeLimit` amount of ranges
166
+ */
167
+ function limitFoldingRanges(ranges, context) {
168
+ const maxRanges = context && context.rangeLimit || Number.MAX_VALUE;
169
+ const sortedRanges = ranges.sort((r1, r2) => {
170
+ let diff = r1.startLine - r2.startLine;
171
+ if (diff === 0) {
172
+ diff = r1.endLine - r2.endLine;
173
+ }
174
+ return diff;
175
+ });
176
+ const validRanges = [];
177
+ let prevEndLine = -1;
178
+ sortedRanges.forEach(r => {
179
+ if (!(r.startLine < prevEndLine && prevEndLine < r.endLine)) {
180
+ validRanges.push(r);
181
+ prevEndLine = r.endLine;
182
+ }
183
+ });
184
+ if (validRanges.length < maxRanges) {
185
+ return validRanges;
186
+ }
187
+ else {
188
+ return validRanges.slice(0, maxRanges);
189
+ }
190
+ }
@@ -1,136 +1,136 @@
1
- /*---------------------------------------------------------------------------------------------
2
- * Copyright (c) Microsoft Corporation. All rights reserved.
3
- * Licensed under the MIT License. See License.txt in the project root for license information.
4
- *--------------------------------------------------------------------------------------------*/
5
- import { Range, Position } from '../cssLanguageTypes';
6
- import { css_beautify } from '../beautify/beautify-css';
7
- import { repeat } from '../utils/strings';
8
- export function format(document, range, options) {
9
- var value = document.getText();
10
- var includesEnd = true;
11
- var initialIndentLevel = 0;
12
- var inRule = false;
13
- var tabSize = options.tabSize || 4;
14
- if (range) {
15
- var startOffset = document.offsetAt(range.start);
16
- // include all leading whitespace iff at the beginning of the line
17
- var extendedStart = startOffset;
18
- while (extendedStart > 0 && isWhitespace(value, extendedStart - 1)) {
19
- extendedStart--;
20
- }
21
- if (extendedStart === 0 || isEOL(value, extendedStart - 1)) {
22
- startOffset = extendedStart;
23
- }
24
- else {
25
- // else keep at least one whitespace
26
- if (extendedStart < startOffset) {
27
- startOffset = extendedStart + 1;
28
- }
29
- }
30
- // include all following whitespace until the end of the line
31
- var endOffset = document.offsetAt(range.end);
32
- var extendedEnd = endOffset;
33
- while (extendedEnd < value.length && isWhitespace(value, extendedEnd)) {
34
- extendedEnd++;
35
- }
36
- if (extendedEnd === value.length || isEOL(value, extendedEnd)) {
37
- endOffset = extendedEnd;
38
- }
39
- range = Range.create(document.positionAt(startOffset), document.positionAt(endOffset));
40
- // Test if inside a rule
41
- inRule = isInRule(value, startOffset);
42
- includesEnd = endOffset === value.length;
43
- value = value.substring(startOffset, endOffset);
44
- if (startOffset !== 0) {
45
- var startOfLineOffset = document.offsetAt(Position.create(range.start.line, 0));
46
- initialIndentLevel = computeIndentLevel(document.getText(), startOfLineOffset, options);
47
- }
48
- if (inRule) {
49
- value = "{\n".concat(trimLeft(value));
50
- }
51
- }
52
- else {
53
- range = Range.create(Position.create(0, 0), document.positionAt(value.length));
54
- }
55
- var cssOptions = {
56
- indent_size: tabSize,
57
- indent_char: options.insertSpaces ? ' ' : '\t',
58
- end_with_newline: includesEnd && getFormatOption(options, 'insertFinalNewline', false),
59
- selector_separator_newline: getFormatOption(options, 'newlineBetweenSelectors', true),
60
- newline_between_rules: getFormatOption(options, 'newlineBetweenRules', true),
61
- space_around_selector_separator: getFormatOption(options, 'spaceAroundSelectorSeparator', false),
62
- brace_style: getFormatOption(options, 'braceStyle', 'collapse'),
63
- indent_empty_lines: getFormatOption(options, 'indentEmptyLines', false),
64
- max_preserve_newlines: getFormatOption(options, 'maxPreserveNewLines', undefined),
65
- preserve_newlines: getFormatOption(options, 'preserveNewLines', true),
66
- wrap_line_length: getFormatOption(options, 'wrapLineLength', undefined),
67
- eol: '\n'
68
- };
69
- var result = css_beautify(value, cssOptions);
70
- if (inRule) {
71
- result = trimLeft(result.substring(2));
72
- }
73
- if (initialIndentLevel > 0) {
74
- var indent = options.insertSpaces ? repeat(' ', tabSize * initialIndentLevel) : repeat('\t', initialIndentLevel);
75
- result = result.split('\n').join('\n' + indent);
76
- if (range.start.character === 0) {
77
- result = indent + result; // keep the indent
78
- }
79
- }
80
- return [{
81
- range: range,
82
- newText: result
83
- }];
84
- }
85
- function trimLeft(str) {
86
- return str.replace(/^\s+/, '');
87
- }
88
- var _CUL = '{'.charCodeAt(0);
89
- var _CUR = '}'.charCodeAt(0);
90
- function isInRule(str, offset) {
91
- while (offset >= 0) {
92
- var ch = str.charCodeAt(offset);
93
- if (ch === _CUL) {
94
- return true;
95
- }
96
- else if (ch === _CUR) {
97
- return false;
98
- }
99
- offset--;
100
- }
101
- return false;
102
- }
103
- function getFormatOption(options, key, dflt) {
104
- if (options && options.hasOwnProperty(key)) {
105
- var value = options[key];
106
- if (value !== null) {
107
- return value;
108
- }
109
- }
110
- return dflt;
111
- }
112
- function computeIndentLevel(content, offset, options) {
113
- var i = offset;
114
- var nChars = 0;
115
- var tabSize = options.tabSize || 4;
116
- while (i < content.length) {
117
- var ch = content.charAt(i);
118
- if (ch === ' ') {
119
- nChars++;
120
- }
121
- else if (ch === '\t') {
122
- nChars += tabSize;
123
- }
124
- else {
125
- break;
126
- }
127
- i++;
128
- }
129
- return Math.floor(nChars / tabSize);
130
- }
131
- function isEOL(text, offset) {
132
- return '\r\n'.indexOf(text.charAt(offset)) !== -1;
133
- }
134
- function isWhitespace(text, offset) {
135
- return ' \t'.indexOf(text.charAt(offset)) !== -1;
136
- }
1
+ /*---------------------------------------------------------------------------------------------
2
+ * Copyright (c) Microsoft Corporation. All rights reserved.
3
+ * Licensed under the MIT License. See License.txt in the project root for license information.
4
+ *--------------------------------------------------------------------------------------------*/
5
+ import { Range, Position } from '../cssLanguageTypes';
6
+ import { css_beautify } from '../beautify/beautify-css';
7
+ import { repeat } from '../utils/strings';
8
+ export function format(document, range, options) {
9
+ let value = document.getText();
10
+ let includesEnd = true;
11
+ let initialIndentLevel = 0;
12
+ let inRule = false;
13
+ const tabSize = options.tabSize || 4;
14
+ if (range) {
15
+ let startOffset = document.offsetAt(range.start);
16
+ // include all leading whitespace iff at the beginning of the line
17
+ let extendedStart = startOffset;
18
+ while (extendedStart > 0 && isWhitespace(value, extendedStart - 1)) {
19
+ extendedStart--;
20
+ }
21
+ if (extendedStart === 0 || isEOL(value, extendedStart - 1)) {
22
+ startOffset = extendedStart;
23
+ }
24
+ else {
25
+ // else keep at least one whitespace
26
+ if (extendedStart < startOffset) {
27
+ startOffset = extendedStart + 1;
28
+ }
29
+ }
30
+ // include all following whitespace until the end of the line
31
+ let endOffset = document.offsetAt(range.end);
32
+ let extendedEnd = endOffset;
33
+ while (extendedEnd < value.length && isWhitespace(value, extendedEnd)) {
34
+ extendedEnd++;
35
+ }
36
+ if (extendedEnd === value.length || isEOL(value, extendedEnd)) {
37
+ endOffset = extendedEnd;
38
+ }
39
+ range = Range.create(document.positionAt(startOffset), document.positionAt(endOffset));
40
+ // Test if inside a rule
41
+ inRule = isInRule(value, startOffset);
42
+ includesEnd = endOffset === value.length;
43
+ value = value.substring(startOffset, endOffset);
44
+ if (startOffset !== 0) {
45
+ const startOfLineOffset = document.offsetAt(Position.create(range.start.line, 0));
46
+ initialIndentLevel = computeIndentLevel(document.getText(), startOfLineOffset, options);
47
+ }
48
+ if (inRule) {
49
+ value = `{\n${trimLeft(value)}`;
50
+ }
51
+ }
52
+ else {
53
+ range = Range.create(Position.create(0, 0), document.positionAt(value.length));
54
+ }
55
+ const cssOptions = {
56
+ indent_size: tabSize,
57
+ indent_char: options.insertSpaces ? ' ' : '\t',
58
+ end_with_newline: includesEnd && getFormatOption(options, 'insertFinalNewline', false),
59
+ selector_separator_newline: getFormatOption(options, 'newlineBetweenSelectors', true),
60
+ newline_between_rules: getFormatOption(options, 'newlineBetweenRules', true),
61
+ space_around_selector_separator: getFormatOption(options, 'spaceAroundSelectorSeparator', false),
62
+ brace_style: getFormatOption(options, 'braceStyle', 'collapse'),
63
+ indent_empty_lines: getFormatOption(options, 'indentEmptyLines', false),
64
+ max_preserve_newlines: getFormatOption(options, 'maxPreserveNewLines', undefined),
65
+ preserve_newlines: getFormatOption(options, 'preserveNewLines', true),
66
+ wrap_line_length: getFormatOption(options, 'wrapLineLength', undefined),
67
+ eol: '\n'
68
+ };
69
+ let result = css_beautify(value, cssOptions);
70
+ if (inRule) {
71
+ result = trimLeft(result.substring(2));
72
+ }
73
+ if (initialIndentLevel > 0) {
74
+ const indent = options.insertSpaces ? repeat(' ', tabSize * initialIndentLevel) : repeat('\t', initialIndentLevel);
75
+ result = result.split('\n').join('\n' + indent);
76
+ if (range.start.character === 0) {
77
+ result = indent + result; // keep the indent
78
+ }
79
+ }
80
+ return [{
81
+ range: range,
82
+ newText: result
83
+ }];
84
+ }
85
+ function trimLeft(str) {
86
+ return str.replace(/^\s+/, '');
87
+ }
88
+ const _CUL = '{'.charCodeAt(0);
89
+ const _CUR = '}'.charCodeAt(0);
90
+ function isInRule(str, offset) {
91
+ while (offset >= 0) {
92
+ const ch = str.charCodeAt(offset);
93
+ if (ch === _CUL) {
94
+ return true;
95
+ }
96
+ else if (ch === _CUR) {
97
+ return false;
98
+ }
99
+ offset--;
100
+ }
101
+ return false;
102
+ }
103
+ function getFormatOption(options, key, dflt) {
104
+ if (options && options.hasOwnProperty(key)) {
105
+ const value = options[key];
106
+ if (value !== null) {
107
+ return value;
108
+ }
109
+ }
110
+ return dflt;
111
+ }
112
+ function computeIndentLevel(content, offset, options) {
113
+ let i = offset;
114
+ let nChars = 0;
115
+ const tabSize = options.tabSize || 4;
116
+ while (i < content.length) {
117
+ const ch = content.charAt(i);
118
+ if (ch === ' ') {
119
+ nChars++;
120
+ }
121
+ else if (ch === '\t') {
122
+ nChars += tabSize;
123
+ }
124
+ else {
125
+ break;
126
+ }
127
+ i++;
128
+ }
129
+ return Math.floor(nChars / tabSize);
130
+ }
131
+ function isEOL(text, offset) {
132
+ return '\r\n'.indexOf(text.charAt(offset)) !== -1;
133
+ }
134
+ function isWhitespace(text, offset) {
135
+ return ' \t'.indexOf(text.charAt(offset)) !== -1;
136
+ }