index.js 4.49 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
/**
 * --------------------------------------------------------------------------
 * Bootstrap (v4.3.1): util/index.js
 * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
 * --------------------------------------------------------------------------
 */

const MAX_UID = 1000000
const MILLISECONDS_MULTIPLIER = 1000
const TRANSITION_END = 'transitionend'
XhmikosR's avatar
XhmikosR committed
11
const { jQuery } = window
12
13

// Shoutout AngusCroll (https://goo.gl/pxwQGp)
XhmikosR's avatar
XhmikosR committed
14
const toType = obj => ({}.toString.call(obj).match(/\s([a-z]+)/i)[1].toLowerCase())
15
16
17
18
19
20
21

/**
 * --------------------------------------------------------------------------
 * Public Util Api
 * --------------------------------------------------------------------------
 */

XhmikosR's avatar
XhmikosR committed
22
const getUID = prefix => {
23
24
25
26
  do {
    // eslint-disable-next-line no-bitwise
    prefix += ~~(Math.random() * MAX_UID) // "~~" acts like a faster Math.floor() here
  } while (document.getElementById(prefix))
XhmikosR's avatar
XhmikosR committed
27

28
29
30
  return prefix
}

XhmikosR's avatar
XhmikosR committed
31
const getSelectorFromElement = element => {
32
33
34
35
36
37
38
39
40
41
  let selector = element.getAttribute('data-target')

  if (!selector || selector === '#') {
    const hrefAttr = element.getAttribute('href')

    selector = hrefAttr && hrefAttr !== '#' ? hrefAttr.trim() : ''
  }

  try {
    return document.querySelector(selector) ? selector : null
XhmikosR's avatar
XhmikosR committed
42
  } catch (error) {
43
44
45
46
    return null
  }
}

XhmikosR's avatar
XhmikosR committed
47
const getTransitionDurationFromElement = element => {
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
  if (!element) {
    return 0
  }

  // Get transition-duration of the element
  let {
    transitionDuration,
    transitionDelay
  } = window.getComputedStyle(element)

  const floatTransitionDuration = parseFloat(transitionDuration)
  const floatTransitionDelay = parseFloat(transitionDelay)

  // Return 0 if element or transition duration is not found
  if (!floatTransitionDuration && !floatTransitionDelay) {
    return 0
  }

  // If multiple durations are defined, take the first
  transitionDuration = transitionDuration.split(',')[0]
  transitionDelay = transitionDelay.split(',')[0]

  return (parseFloat(transitionDuration) + parseFloat(transitionDelay)) * MILLISECONDS_MULTIPLIER
}

XhmikosR's avatar
XhmikosR committed
73
const triggerTransitionEnd = element => {
Johann-S's avatar
Johann-S committed
74
75
76
77
  const evt = document.createEvent('HTMLEvents')

  evt.initEvent(TRANSITION_END, true, true)
  element.dispatchEvent(evt)
78
79
}

XhmikosR's avatar
XhmikosR committed
80
const isElement = obj => (obj[0] || obj).nodeType
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100

const emulateTransitionEnd = (element, duration) => {
  let called = false
  const durationPadding = 5
  const emulatedDuration = duration + durationPadding
  function listener() {
    called = true
    element.removeEventListener(TRANSITION_END, listener)
  }

  element.addEventListener(TRANSITION_END, listener)
  setTimeout(() => {
    if (!called) {
      triggerTransitionEnd(element)
    }
  }, emulatedDuration)
}

const typeCheckConfig = (componentName, config, configTypes) => {
  Object.keys(configTypes)
XhmikosR's avatar
XhmikosR committed
101
    .forEach(property => {
102
      const expectedTypes = configTypes[property]
XhmikosR's avatar
XhmikosR committed
103
104
105
106
      const value = config[property]
      const valueType = value && isElement(value) ?
        'element' :
        toType(value)
107
108
109
110
111
112
113
114
115
116

      if (!new RegExp(expectedTypes).test(valueType)) {
        throw new Error(
          `${componentName.toUpperCase()}: ` +
          `Option "${property}" provided type "${valueType}" ` +
          `but expected type "${expectedTypes}".`)
      }
    })
}

XhmikosR's avatar
XhmikosR committed
117
const makeArray = nodeList => {
118
119
120
121
122
123
124
  if (!nodeList) {
    return []
  }

  return [].slice.call(nodeList)
}

XhmikosR's avatar
XhmikosR committed
125
const isVisible = element => {
126
127
128
129
130
131
132
133
134
135
136
137
138
  if (!element) {
    return false
  }

  if (element.style && element.parentNode && element.parentNode.style) {
    return element.style.display !== 'none' &&
      element.parentNode.style.display !== 'none' &&
      element.style.visibility !== 'hidden'
  }

  return false
}

XhmikosR's avatar
XhmikosR committed
139
const findShadowRoot = element => {
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
  if (!document.documentElement.attachShadow) {
    return null
  }

  // Can find the shadow root otherwise it'll return the document
  if (typeof element.getRootNode === 'function') {
    const root = element.getRootNode()
    return root instanceof ShadowRoot ? root : null
  }

  if (element instanceof ShadowRoot) {
    return element
  }

  // when we don't find a shadow root
  if (!element.parentNode) {
    return null
  }

  return findShadowRoot(element.parentNode)
}

// eslint-disable-next-line no-empty-function
const noop = () => function () {}

XhmikosR's avatar
XhmikosR committed
165
const reflow = element => element.offsetHeight
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182

export {
  jQuery,
  TRANSITION_END,
  getUID,
  getSelectorFromElement,
  getTransitionDurationFromElement,
  triggerTransitionEnd,
  isElement,
  emulateTransitionEnd,
  typeCheckConfig,
  makeArray,
  isVisible,
  findShadowRoot,
  noop,
  reflow
}