UNPKG

2.75 kBJavaScriptView Raw
1'use strict';
2
3const atRuleParamIndex = require('../../utils/atRuleParamIndex');
4const declarationValueIndex = require('../../utils/declarationValueIndex');
5const getUnitFromValueNode = require('../../utils/getUnitFromValueNode');
6const report = require('../../utils/report');
7const ruleMessages = require('../../utils/ruleMessages');
8const validateOptions = require('../../utils/validateOptions');
9const valueParser = require('postcss-value-parser');
10
11const ruleName = 'unit-case';
12
13const messages = ruleMessages(ruleName, {
14 expected: (actual, expected) => `Expected "${actual}" to be "${expected}"`,
15});
16
17function rule(expectation, options, context) {
18 return (root, result) => {
19 const validOptions = validateOptions(result, ruleName, {
20 actual: expectation,
21 possible: ['lower', 'upper'],
22 });
23
24 if (!validOptions) {
25 return;
26 }
27
28 function check(node, value, getIndex) {
29 const violations = [];
30
31 function processValue(valueNode) {
32 const unit = getUnitFromValueNode(valueNode);
33
34 if (!unit) {
35 return false;
36 }
37
38 const expectedUnit = expectation === 'lower' ? unit.toLowerCase() : unit.toUpperCase();
39
40 if (unit === expectedUnit) {
41 return false;
42 }
43
44 violations.push({
45 index: getIndex(node) + valueNode.sourceIndex,
46 message: messages.expected(unit, expectedUnit),
47 });
48
49 return true;
50 }
51
52 const parsedValue = valueParser(value).walk((valueNode) => {
53 // Ignore wrong units within `url` function
54 let needFix = false;
55 const value = valueNode.value;
56
57 if (valueNode.type === 'function' && value.toLowerCase() === 'url') {
58 return false;
59 }
60
61 if (value.includes('*')) {
62 value.split('*').some((val) => {
63 return processValue({
64 ...valueNode,
65 sourceIndex: value.indexOf(val) + val.length + 1,
66 value: val,
67 });
68 });
69 }
70
71 needFix = processValue(valueNode);
72
73 if (needFix && context.fix) {
74 valueNode.value = expectation === 'lower' ? value.toLowerCase() : value.toUpperCase();
75 }
76 });
77
78 if (violations.length) {
79 if (context.fix) {
80 if (node.name === 'media') {
81 node.params = parsedValue.toString();
82 } else {
83 node.value = parsedValue.toString();
84 }
85 } else {
86 violations.forEach((err) => {
87 report({
88 index: err.index,
89 message: err.message,
90 node,
91 result,
92 ruleName,
93 });
94 });
95 }
96 }
97 }
98
99 root.walkAtRules((atRule) => {
100 if (!/^media$/i.test(atRule.name) && !atRule.variable) {
101 return;
102 }
103
104 check(atRule, atRule.params, atRuleParamIndex);
105 });
106 root.walkDecls((decl) => check(decl, decl.value, declarationValueIndex));
107 };
108}
109
110rule.ruleName = ruleName;
111rule.messages = messages;
112module.exports = rule;