1
- import type { TSESTree } from '@typescript-eslint/types'
1
+ import type { TSESLint } from '@typescript-eslint/utils'
2
+
3
+ import { TSESTree } from '@typescript-eslint/types'
2
4
3
5
import type { SortingNode } from '../types/sorting-node'
6
+ import type { Options } from './sort-maps/types'
4
7
5
8
import {
9
+ buildUseConfigurationIfJsonSchema ,
10
+ buildCustomGroupsArrayJsonSchema ,
6
11
partitionByCommentJsonSchema ,
7
12
partitionByNewLineJsonSchema ,
8
13
specialCharactersJsonSchema ,
14
+ newlinesBetweenJsonSchema ,
9
15
ignoreCaseJsonSchema ,
10
16
buildTypeJsonSchema ,
11
17
alphabetJsonSchema ,
12
18
localesJsonSchema ,
19
+ groupsJsonSchema ,
13
20
orderJsonSchema ,
14
21
} from '../utils/common-json-schemas'
22
+ import { validateGeneratedGroupsConfiguration } from '../utils/validate-generated-groups-configuration'
15
23
import { validateCustomSortConfiguration } from '../utils/validate-custom-sort-configuration'
24
+ import { getCustomGroupsCompareOptions } from '../utils/get-custom-groups-compare-options'
25
+ import { getMatchingContextOptions } from '../utils/get-matching-context-options'
16
26
import { getEslintDisabledLines } from '../utils/get-eslint-disabled-lines'
27
+ import { doesCustomGroupMatch } from './sort-maps/does-custom-group-match'
17
28
import { isNodeEslintDisabled } from '../utils/is-node-eslint-disabled'
18
29
import { hasPartitionComment } from '../utils/has-partition-comment'
19
30
import { createNodeIndexMap } from '../utils/create-node-index-map'
31
+ import { sortNodesByGroups } from '../utils/sort-nodes-by-groups'
20
32
import { getCommentsBefore } from '../utils/get-comments-before'
33
+ import { getNewlinesErrors } from '../utils/get-newlines-errors'
34
+ import { singleCustomGroupJsonSchema } from './sort-maps/types'
21
35
import { createEslintRule } from '../utils/create-eslint-rule'
22
36
import { getLinesBetween } from '../utils/get-lines-between'
37
+ import { getGroupNumber } from '../utils/get-group-number'
23
38
import { getSourceCode } from '../utils/get-source-code'
24
39
import { toSingleLine } from '../utils/to-single-line'
25
40
import { rangeToDiff } from '../utils/range-to-diff'
26
41
import { getSettings } from '../utils/get-settings'
27
42
import { isSortable } from '../utils/is-sortable'
28
43
import { makeFixes } from '../utils/make-fixes'
29
- import { sortNodes } from '../utils/sort-nodes '
44
+ import { useGroups } from '../utils/use-groups '
30
45
import { complete } from '../utils/complete'
31
46
import { pairwise } from '../utils/pairwise'
32
47
33
- type Options = [
34
- Partial < {
35
- partitionByComment :
36
- | {
37
- block ?: string [ ] | boolean | string
38
- line ?: string [ ] | boolean | string
39
- }
40
- | string [ ]
41
- | boolean
42
- | string
43
- type : 'alphabetical' | 'line-length' | 'natural' | 'custom'
44
- specialCharacters : 'remove' | 'trim' | 'keep'
45
- locales : NonNullable < Intl . LocalesArgument >
46
- partitionByNewLine : boolean
47
- order : 'desc' | 'asc'
48
- ignoreCase : boolean
49
- alphabet : string
50
- } > ,
51
- ]
52
-
53
- type MESSAGE_ID = 'unexpectedMapElementsOrder'
48
+ type MESSAGE_ID =
49
+ | 'missedSpacingBetweenMapElementsMembers'
50
+ | 'extraSpacingBetweenMapElementsMembers'
51
+ | 'unexpectedMapElementsGroupOrder'
52
+ | 'unexpectedMapElementsOrder'
54
53
55
54
let defaultOptions : Required < Options [ 0 ] > = {
56
55
specialCharacters : 'keep' ,
57
56
partitionByComment : false ,
58
57
partitionByNewLine : false ,
58
+ newlinesBetween : 'ignore' ,
59
+ useConfigurationIf : { } ,
59
60
type : 'alphabetical' ,
60
61
ignoreCase : true ,
62
+ customGroups : [ ] ,
61
63
locales : 'en-US' ,
62
64
alphabet : '' ,
63
65
order : 'asc' ,
66
+ groups : [ ] ,
64
67
}
65
68
66
69
export default createEslintRule < Options , MESSAGE_ID > ( {
@@ -79,11 +82,29 @@ export default createEslintRule<Options, MESSAGE_ID>({
79
82
return
80
83
}
81
84
85
+ let sourceCode = getSourceCode ( context )
82
86
let settings = getSettings ( context . settings )
83
- let options = complete ( context . options . at ( 0 ) , settings , defaultOptions )
87
+
88
+ let matchedContextOptions = getMatchingContextOptions ( {
89
+ nodeNames : elements
90
+ . filter (
91
+ element =>
92
+ element !== null &&
93
+ element . type !== TSESTree . AST_NODE_TYPES . SpreadElement ,
94
+ )
95
+ . map ( element => getNodeName ( { sourceCode, element } ) ) ,
96
+ contextOptions : context . options ,
97
+ } )
98
+
99
+ let options = complete ( matchedContextOptions [ 0 ] , settings , defaultOptions )
84
100
validateCustomSortConfiguration ( options )
101
+ validateGeneratedGroupsConfiguration ( {
102
+ customGroups : options . customGroups ,
103
+ groups : options . groups ,
104
+ selectors : [ ] ,
105
+ modifiers : [ ] ,
106
+ } )
85
107
86
- let sourceCode = getSourceCode ( context )
87
108
let eslintDisabledLines = getEslintDisabledLines ( {
88
109
ruleName : context . id ,
89
110
sourceCode,
@@ -106,29 +127,39 @@ export default createEslintRule<Options, MESSAGE_ID>({
106
127
for ( let part of parts ) {
107
128
let formattedMembers : SortingNode [ ] [ ] = [ [ ] ]
108
129
for ( let element of part ) {
109
- let name : string
130
+ let name : string = getNodeName ( {
131
+ sourceCode,
132
+ element,
133
+ } )
110
134
111
- if ( element . type === 'ArrayExpression' ) {
112
- let [ left ] = element . elements
135
+ let lastSortingNode = formattedMembers . at ( - 1 ) ?. at ( - 1 )
113
136
114
- if ( ! left ) {
115
- name = `${ left } `
116
- } else if ( left . type === 'Literal' ) {
117
- name = left . raw
118
- } else {
119
- name = sourceCode . getText ( left )
137
+ let { defineGroup, getGroup } = useGroups ( options )
138
+ for ( let customGroup of options . customGroups ) {
139
+ if (
140
+ doesCustomGroupMatch ( {
141
+ elementName : name ,
142
+ customGroup,
143
+ } )
144
+ ) {
145
+ defineGroup ( customGroup . groupName , true )
146
+ /**
147
+ * If the custom group is not referenced in the `groups` option, it
148
+ * will be ignored
149
+ */
150
+ if ( getGroup ( ) === customGroup . groupName ) {
151
+ break
152
+ }
120
153
}
121
- } else {
122
- name = sourceCode . getText ( element )
123
154
}
124
155
125
- let lastSortingNode = formattedMembers . at ( - 1 ) ?. at ( - 1 )
126
156
let sortingNode : SortingNode = {
127
157
isEslintDisabled : isNodeEslintDisabled (
128
158
element ,
129
159
eslintDisabledLines ,
130
160
) ,
131
161
size : rangeToDiff ( element , sourceCode ) ,
162
+ group : getGroup ( ) ,
132
163
node : element ,
133
164
name,
134
165
}
@@ -155,7 +186,11 @@ export default createEslintRule<Options, MESSAGE_ID>({
155
186
let sortNodesExcludingEslintDisabled = (
156
187
ignoreEslintDisabledNodes : boolean ,
157
188
) : SortingNode [ ] =>
158
- sortNodes ( nodes , options , { ignoreEslintDisabledNodes } )
189
+ sortNodesByGroups ( nodes , options , {
190
+ getGroupCompareOptions : groupNumber =>
191
+ getCustomGroupsCompareOptions ( options , groupNumber ) ,
192
+ ignoreEslintDisabledNodes,
193
+ } )
159
194
let sortedNodes = sortNodesExcludingEslintDisabled ( false )
160
195
let sortedNodesExcludingEslintDisabled =
161
196
sortNodesExcludingEslintDisabled ( true )
@@ -166,69 +201,131 @@ export default createEslintRule<Options, MESSAGE_ID>({
166
201
let leftIndex = nodeIndexMap . get ( left ) !
167
202
let rightIndex = nodeIndexMap . get ( right ) !
168
203
204
+ let leftNumber = getGroupNumber ( options . groups , left )
205
+ let rightNumber = getGroupNumber ( options . groups , right )
206
+
169
207
let indexOfRightExcludingEslintDisabled =
170
208
sortedNodesExcludingEslintDisabled . indexOf ( right )
209
+
210
+ let messageIds : MESSAGE_ID [ ] = [ ]
211
+
171
212
if (
172
- leftIndex < rightIndex &&
173
- leftIndex < indexOfRightExcludingEslintDisabled
213
+ leftIndex > rightIndex ||
214
+ leftIndex >= indexOfRightExcludingEslintDisabled
174
215
) {
175
- return
216
+ messageIds . push (
217
+ leftNumber === rightNumber
218
+ ? 'unexpectedMapElementsOrder'
219
+ : 'unexpectedMapElementsGroupOrder' ,
220
+ )
176
221
}
177
222
178
- context . report ( {
179
- fix : fixer =>
180
- makeFixes ( {
181
- sortedNodes : sortedNodesExcludingEslintDisabled ,
182
- sourceCode,
183
- options,
184
- fixer,
185
- nodes,
186
- } ) ,
187
- data : {
188
- right : toSingleLine ( right . name ) ,
189
- left : toSingleLine ( left . name ) ,
190
- } ,
191
- messageId : 'unexpectedMapElementsOrder' ,
192
- node : right . node ,
193
- } )
223
+ messageIds = [
224
+ ...messageIds ,
225
+ ...getNewlinesErrors ( {
226
+ missedSpacingError : 'missedSpacingBetweenMapElementsMembers' ,
227
+ extraSpacingError : 'extraSpacingBetweenMapElementsMembers' ,
228
+ rightNum : rightNumber ,
229
+ leftNum : leftNumber ,
230
+ sourceCode,
231
+ options,
232
+ right,
233
+ left,
234
+ } ) ,
235
+ ]
236
+
237
+ for ( let messageId of messageIds ) {
238
+ context . report ( {
239
+ fix : fixer =>
240
+ makeFixes ( {
241
+ sortedNodes : sortedNodesExcludingEslintDisabled ,
242
+ sourceCode,
243
+ options,
244
+ fixer,
245
+ nodes,
246
+ } ) ,
247
+ data : {
248
+ right : toSingleLine ( right . name ) ,
249
+ left : toSingleLine ( left . name ) ,
250
+ rightGroup : right . group ,
251
+ leftGroup : left . group ,
252
+ } ,
253
+ node : right . node ,
254
+ messageId,
255
+ } )
256
+ }
194
257
} )
195
258
}
196
259
}
197
260
} ,
198
261
} ) ,
199
262
meta : {
200
- schema : [
201
- {
263
+ schema : {
264
+ items : {
202
265
properties : {
203
266
partitionByComment : {
204
267
...partitionByCommentJsonSchema ,
205
268
description :
206
269
'Allows you to use comments to separate the maps members into logical groups.' ,
207
270
} ,
271
+ customGroups : buildCustomGroupsArrayJsonSchema ( {
272
+ singleCustomGroupJsonSchema,
273
+ } ) ,
274
+ useConfigurationIf : buildUseConfigurationIfJsonSchema ( ) ,
208
275
partitionByNewLine : partitionByNewLineJsonSchema ,
209
276
specialCharacters : specialCharactersJsonSchema ,
277
+ newlinesBetween : newlinesBetweenJsonSchema ,
210
278
ignoreCase : ignoreCaseJsonSchema ,
211
279
alphabet : alphabetJsonSchema ,
212
280
type : buildTypeJsonSchema ( ) ,
213
281
locales : localesJsonSchema ,
282
+ groups : groupsJsonSchema ,
214
283
order : orderJsonSchema ,
215
284
} ,
216
285
additionalProperties : false ,
217
286
type : 'object' ,
218
287
} ,
219
- ] ,
220
- docs : {
221
- url : 'https://perfectionist.dev/rules/sort-maps' ,
222
- description : 'Enforce sorted Map elements.' ,
223
- recommended : true ,
288
+ uniqueItems : true ,
289
+ type : 'array' ,
224
290
} ,
225
291
messages : {
292
+ unexpectedMapElementsGroupOrder :
293
+ 'Expected "{{right}}" ({{rightGroup}}) to come before "{{left}}" ({{leftGroup}}).' ,
294
+ missedSpacingBetweenMapElementsMembers :
295
+ 'Missed spacing between "{{left}}" and "{{right}}" members.' ,
296
+ extraSpacingBetweenMapElementsMembers :
297
+ 'Extra spacing between "{{left}}" and "{{right}}" members.' ,
226
298
unexpectedMapElementsOrder :
227
299
'Expected "{{right}}" to come before "{{left}}".' ,
228
300
} ,
301
+ docs : {
302
+ url : 'https://perfectionist.dev/rules/sort-maps' ,
303
+ description : 'Enforce sorted Map elements.' ,
304
+ recommended : true ,
305
+ } ,
229
306
type : 'suggestion' ,
230
307
fixable : 'code' ,
231
308
} ,
232
309
defaultOptions : [ defaultOptions ] ,
233
310
name : 'sort-maps' ,
234
311
} )
312
+
313
+ let getNodeName = ( {
314
+ sourceCode,
315
+ element,
316
+ } : {
317
+ sourceCode : TSESLint . SourceCode
318
+ element : TSESTree . Expression
319
+ } ) : string => {
320
+ if ( element . type === 'ArrayExpression' ) {
321
+ let [ left ] = element . elements
322
+
323
+ if ( ! left ) {
324
+ return `${ left } `
325
+ } else if ( left . type === 'Literal' ) {
326
+ return left . raw
327
+ }
328
+ return sourceCode . getText ( left )
329
+ }
330
+ return sourceCode . getText ( element )
331
+ }
0 commit comments