-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathpatterns.js
119 lines (104 loc) · 3.19 KB
/
patterns.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
117
118
119
/**
* This file's primary purpose is to turn a bunch of hard-to-read regular expressions
* in to self-documenting functions with appropriate named capture groups and minimal
* meaning parsing.
*/
/**
* `element(string)` matches the pieces of an initial CSS selector and returns a parsed set of fields.
* e.g., #id, .class or body
*
* It also parses out the piped namespace piece: https://www.w3.org/TR/css3-selectors/#univnmsp
*/
function element(string) {
const CSS_ELEMENT_PATTERN = /^([#.]?)([a-z0-9\\*_-]*)((\|)([a-z0-9\\*_-]*))?/i;
const matches = CSS_ELEMENT_PATTERN.exec(string);
if (!matches) return matches;
const retVal = {
fullGroup: matches[0],
fullNamespaceGroup: matches[3],
namespace: matches[5], // TODO: this is backwards, handle namespace standardization here.
};
// either "#" or "." to indicate id or class selector
if (matches[1] === '#' || matches[1] === '.') {
retVal.specialSelectorType = matches[1];
retVal.specialSelectorValue = matches[2];
} else if (matches[1] === '') {
retVal.elementName = matches[2];
}
return retVal;
}
/**
* `attributePresence(string)` matches the pieces of a CSS selector that represent a attribute presence requirement.
* e.g., [disabled], [x-anything-here]
*/
function attributePresence(string) {
const CSS_ATTRIBUTE_PRESENCE_PATTERN = /^\[([^\]]*)\]/i;
const matches = CSS_ATTRIBUTE_PRESENCE_PATTERN.exec(string);
if (!matches) return matches;
return {
fullGroup: matches[0],
attributeName: matches[1],
};
}
/**
* `attributeValue(string)` matches the pieces of a CSS selector that represent a attribute selector.
* e.g., [disabled='disabled'], [class~='alphaghettis'], [type != 'number']
*
* TODO: this pattern fails on single or unquoted things. Bad!
*/
function attributeValue(string) {
const CSS_ATTRIBUTE_VALUE_PATTERN = /^\[\s*([^~=\s]+)\s*(~?=)\s*"([^"]+)"\s*\]/i;
const matches = CSS_ATTRIBUTE_VALUE_PATTERN.exec(string);
if (!matches) return matches;
return {
fullGroup: matches[0],
field: matches[1],
value: matches[3],
isContains: matches[2] === '~=',
};
}
/**
* `pseudo(string)` matches the pieces of a CSS selector that represent a pseudo selector.
* e.g., :first-child, :visited
*/
// TODO: verify this works with parentheses, e.g., nth-child(2).
function pseudo(string) {
const CSS_PSEUDO_PATTERN = /^:([a-z_-])+/i;
const matches = CSS_PSEUDO_PATTERN.exec(string);
if (!matches) return matches;
return {
fullGroup: matches[0],
selector: matches[1],
};
}
/**
* `combinator(string)` matches the pieces of a CSS selector that represent a combinator.
* e.g., + or >
*/
function combinator(string) {
const CSS_COMBINATOR_PATTERN = /^(\s*[>+\s])?/i;
const matches = CSS_COMBINATOR_PATTERN.exec(string);
if (!matches) return matches;
return {
fullGroup: matches[0],
};
}
/**
* `comma(string)` matches commas in a CSS selector; used for disjunction.
*/
function comma(string) {
const COMMA_PATTERN = /^\s*,/i;
const matches = COMMA_PATTERN.exec(string);
if (!matches) return matches;
return {
fullGroup: matches[0],
};
}
module.exports = {
element,
attributePresence,
attributeValue,
pseudo,
combinator,
comma,
};