-
Notifications
You must be signed in to change notification settings - Fork 5
/
plugin.html-validate.bem.js
114 lines (103 loc) · 3.53 KB
/
plugin.html-validate.bem.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
const { Rule } = require('html-validate');
const { nodeIgnore } = require('./plugin.html-validate.utils');
/**
* @typedef { import('html-validate').DOMReadyEvent } DOMReadyEvent
*/
const MODIFIER_PATTERN = /(--([a-zA-Z0-9]+(?:-[a-zA-Z0-9]+)*))$/;
const ELEMENT_PATTERN = /(__([a-zA-Z0-9]+(?:-[a-zA-Z0-9]+)*))$/;
/**
* Lint missing BEM-element.
*/
class NoMissingElement extends Rule {
/**
* @param {object} options - plugin options
*/
constructor(options) {
super({ ignore: '', ...options });
this.domReady = this.domReady.bind(this);
}
/**
* Setup plugin events.
*/
setup() {
this.on('dom:ready', this.domReady);
}
/**
* Lint html document.
* @param {DOMReadyEvent.document} document - document object
*/
domReady({ document }) {
const nodes = document.querySelectorAll('body [class]');
const ignores = this.options.ignore ? document.querySelectorAll(this.options.ignore) : [];
nodes.forEach((node) => {
if (nodeIgnore(node, ignores)) {
return;
}
const classList = [...node.classList];
classList.forEach((className) => {
if (!ELEMENT_PATTERN.test(className)) return;
const [, elementMatch] = className.match(ELEMENT_PATTERN);
const blockName = className.substring(0, className.length - elementMatch.length);
const closest = node.closest(`.${blockName}`);
if (!(closest && closest !== node)) {
this.report(
node,
`Class-element references missing block ${JSON.stringify(
blockName
)} (element is ${JSON.stringify(className)}).`
);
}
});
});
}
}
/**
*Lint missing BEM-modifier.
*/
class NoMissingModifier extends Rule {
/**
* @param {object} options - plugin options
*/
constructor(options) {
super({ ignore: '', ...options });
this.domReady = this.domReady.bind(this);
}
/**
* Setup plugin events.
*/
setup() {
this.on('dom:ready', this.domReady);
}
/**
* Lint html document.
* @param {DOMReadyEvent.document} document - document object
*/
domReady({ document }) {
const nodes = document.querySelectorAll('body [class]');
const ignores = this.options.ignore ? document.querySelectorAll(this.options.ignore) : [];
nodes.forEach((node) => {
if (nodeIgnore(node, ignores)) {
return;
}
const classList = [...node.classList];
classList.forEach((className) => {
if (!MODIFIER_PATTERN.test(className)) return;
const [, modifierMatch] = className.match(MODIFIER_PATTERN);
const blockName = className.substring(0, className.length - modifierMatch.length);
if (!classList.includes(blockName)) {
this.report(
node,
`Class-modifier references missing block ${JSON.stringify(
blockName
)} (modifier is ${JSON.stringify(className)}).`
);
}
});
});
}
}
module.exports = { NoMissingElement, NoMissingModifier };
module.exports.rules = {
'pitcher/bem-no-missing-element': NoMissingElement,
'pitcher/bem-no-missing-modifier': NoMissingModifier,
};