-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathspacing.js
116 lines (96 loc) · 3.55 KB
/
spacing.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
import stylelint from 'stylelint'
import {declarationValueIndex} from 'stylelint/lib/utils/nodeFieldIndices.cjs'
import valueParser from 'postcss-value-parser'
import {primitivesVariables, walkGroups} from './lib/utils.js'
const {
createPlugin,
utils: {report, ruleMessages, validateOptions},
} = stylelint
export const ruleName = 'primer/spacing'
export const messages = ruleMessages(ruleName, {
rejected: (value, replacement) => {
if (!replacement) {
return `Please use a primer size variable instead of '${value}'. Consult the primer docs for a suitable replacement. https://primer.style/foundations/primitives/size`
}
return `Please replace '${value}' with size variable '${replacement['name']}'. https://primer.style/foundations/primitives/size`
},
})
// Props that we want to check
const propList = ['padding', 'margin', 'top', 'right', 'bottom', 'left']
// Values that we want to ignore
const valueList = ['${']
const sizes = primitivesVariables('spacing')
// Add +-1px to each value
for (const size of sizes) {
const values = size['values']
const px = parseInt(values.find(value => value.includes('px')))
if (![2, 6].includes(px)) {
values.push(`${px + 1}px`)
values.push(`${px - 1}px`)
}
}
/** @type {import('stylelint').Rule} */
const ruleFunction = primary => {
return (root, result) => {
const validOptions = validateOptions(result, ruleName, {
actual: primary,
possible: [true],
})
if (!validOptions) return
root.walkDecls(declNode => {
const {prop, value} = declNode
if (!propList.some(spacingProp => prop.startsWith(spacingProp))) return
if (valueList.some(valueToIgnore => value.includes(valueToIgnore))) return
const parsedValue = walkGroups(valueParser(value), node => {
// Only check word types. https://github.com/TrySound/postcss-value-parser#word
if (node.type !== 'word') {
return
}
// Exact values to ignore.
if (['*', '+', '-', '/', '0', 'auto', 'inherit', 'initial'].includes(node.value)) {
return
}
const valueUnit = valueParser.unit(node.value)
if (valueUnit && (valueUnit.unit === '' || !/^-?[0-9.]+$/.test(valueUnit.number))) {
return
}
// Skip if the value unit isn't a supported unit.
if (valueUnit && !['px', 'rem', 'em'].includes(valueUnit.unit)) {
return
}
// If the variable is found in the value, skip it.
if (
sizes.some(variable =>
new RegExp(`${variable['name'].replace(/[.*+?^${}()|[\]\\]/g, '\\$&')}\\b`).test(node.value),
)
) {
return
}
const replacement = sizes.find(variable => variable.values.includes(node.value.replace('-', '')))
const fixable = replacement && valueUnit && !valueUnit.number.includes('-')
let fix = undefined
if (fixable) {
fix = () => {
node.value = node.value.replace(node.value, `var(${replacement['name']})`)
}
}
report({
index: declarationValueIndex(declNode) + node.sourceIndex,
endIndex: declarationValueIndex(declNode) + node.sourceIndex + node.value.length,
message: messages.rejected(node.value, replacement),
node: declNode,
result,
ruleName,
fix,
})
})
declNode.value = parsedValue.toString()
})
}
}
ruleFunction.ruleName = ruleName
ruleFunction.messages = messages
ruleFunction.meta = {
fixable: true,
}
export default createPlugin(ruleName, ruleFunction)