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
|
/**
* --------------------------------------------------------------------------
* Bootstrap (v4.3.1): dom/selectorEngine.js
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
* --------------------------------------------------------------------------
*/
import Polyfill from './polyfill'
import {
makeArray
} from '../util/index'
/**
* ------------------------------------------------------------------------
* Constants
* ------------------------------------------------------------------------
*/
const findFn = Polyfill.find
const findOne = Polyfill.findOne
const NODE_TEXT = 3
const SelectorEngine = {
matches(element, selector) {
return element.matches(selector)
},
find(selector, element = document.documentElement) {
if (typeof selector !== 'string') {
return null
}
return findFn.call(element, selector)
},
findOne(selector, element = document.documentElement) {
if (typeof selector !== 'string') {
return null
}
return findOne.call(element, selector)
},
children(element, selector) {
if (typeof selector !== 'string') {
return null
}
const children = makeArray(element.children)
return children.filter((child) => this.matches(child, selector))
},
parents(element, selector) {
if (typeof selector !== 'string') {
return null
}
const parents = []
let ancestor = element.parentNode
while (ancestor && ancestor.nodeType === Node.ELEMENT_NODE && ancestor.nodeType !== NODE_TEXT) {
if (this.matches(ancestor, selector)) {
parents.push(ancestor)
}
ancestor = ancestor.parentNode
}
return parents
},
closest(element, selector) {
if (typeof selector !== 'string') {
return null
}
return element.closest(selector)
},
prev(element, selector) {
if (typeof selector !== 'string') {
return null
}
const siblings = []
let previous = element.previousSibling
while (previous && previous.nodeType === Node.ELEMENT_NODE && previous.nodeType !== NODE_TEXT) {
if (this.matches(previous, selector)) {
siblings.push(previous)
}
previous = previous.previousSibling
}
return siblings
}
}
export default SelectorEngine
|