selectorEngine.js 1.57 KB
Newer Older
Johann-S's avatar
Johann-S committed
1
2
3
4
5
6
7
/**
 * --------------------------------------------------------------------------
 * Bootstrap (v4.0.0-beta): dom/selectorEngine.js
 * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
 * --------------------------------------------------------------------------
 */

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
// matches polyfill (see: https://mzl.la/2ikXneG)
let fnMatches = null
if (!Element.prototype.matches) {
  fnMatches =
    Element.prototype.msMatchesSelector ||
    Element.prototype.webkitMatchesSelector
} else {
  fnMatches = Element.prototype.matches
}

// closest polyfill (see: https://mzl.la/2vXggaI)
let fnClosest = null
if (!Element.prototype.closest) {
  fnClosest = (element, selector) => {
    let ancestor = element
    if (!document.documentElement.contains(element)) {
      return null
    }

    do {
      if (fnMatches.call(ancestor, selector)) {
        return ancestor
      }

      ancestor = ancestor.parentElement
    } while (ancestor !== null)

    return null
  }
} else {
  fnClosest = (element, selector) => {
    return element.closest(selector)
  }
}

Johann-S's avatar
Johann-S committed
43
const SelectorEngine = {
44
45
46
  matches(element, selector) {
    return fnMatches.call(element, selector)
  },
Johann-S's avatar
Johann-S committed
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61

  find(selector) {
    if (typeof selector !== 'string') {
      return null
    }

    let selectorType = 'querySelectorAll'
    if (selector.indexOf('#') === 0) {
      selectorType = 'getElementById'
      selector = selector.substr(1, selector.length)
    }
    return document[selectorType](selector)
  },

  closest(element, selector) {
62
    return fnClosest(element, selector)
Johann-S's avatar
Johann-S committed
63
64
65
66
  }
}

export default SelectorEngine