File size: 2,425 Bytes
8e11898
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
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
/**
 * --------------------------------------------------------------------------
 * Bootstrap (v5.1.3): dom/selector-engine.js
 * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)
 * --------------------------------------------------------------------------
 */

/**
 * ------------------------------------------------------------------------
 * Constants
 * ------------------------------------------------------------------------
 */

import {
    isDisabled,
    isVisible
} from '../util/index'

const NODE_TEXT = 3

const SelectorEngine = {
    find(selector, element = document.documentElement) {
        return [].concat(...Element.prototype.querySelectorAll.call(element, selector))
    },

    findOne(selector, element = document.documentElement) {
        return Element.prototype.querySelector.call(element, selector)
    },

    children(element, selector) {
        return [].concat(...element.children)
            .filter(child => child.matches(selector))
    },

    parents(element, selector) {
        const parents = []

        let ancestor = element.parentNode

        while (ancestor && ancestor.nodeType === Node.ELEMENT_NODE && ancestor.nodeType !== NODE_TEXT) {
            if (ancestor.matches(selector)) {
                parents.push(ancestor)
            }

            ancestor = ancestor.parentNode
        }

        return parents
    },

    prev(element, selector) {
        let previous = element.previousElementSibling

        while (previous) {
            if (previous.matches(selector)) {
                return [previous]
            }

            previous = previous.previousElementSibling
        }

        return []
    },

    next(element, selector) {
        let next = element.nextElementSibling

        while (next) {
            if (next.matches(selector)) {
                return [next]
            }

            next = next.nextElementSibling
        }

        return []
    },

    focusableChildren(element) {
        const focusables = [
            'a',
            'button',
            'input',
            'textarea',
            'select',
            'details',
            '[tabindex]',
            '[contenteditable="true"]'
        ].map(selector => `${selector}:not([tabindex^="-"])`).join(', ')

        return this.find(focusables, element).filter(el => !isDisabled(el) && isVisible(el))
    }
}

export default SelectorEngine