import { default as sweetAlert } from '../sweetalert2.js'
import { swalClasses, iconTypes } from './classes.js'
import { uniqueArray, error } from './utils.js'
// Remember state in cases where opening and handling a modal will fiddle with it.
export const states = {
previousWindowKeyDown: null,
previousActiveElement: null,
previousBodyPadding: null
}
/*
* Add modal + overlay to DOM
*/
export const init = (params) => {
// Clean up the old modal if it exists
const c = getContainer()
if (c) {
c.parentNode.removeChild(c)
}
if (typeof document === 'undefined') {
error('SweetAlert2 requires document to initialize')
return
}
const container = document.createElement('div')
container.className = swalClasses.container
container.innerHTML = sweetHTML
let targetElement = typeof params.target === 'string' ? document.querySelector(params.target) : params.target
targetElement.appendChild(container)
const modal = getModal()
const input = getChildByClass(modal, swalClasses.input)
const file = getChildByClass(modal, swalClasses.file)
const range = modal.querySelector(`.${swalClasses.range} input`)
const rangeOutput = modal.querySelector(`.${swalClasses.range} output`)
const select = getChildByClass(modal, swalClasses.select)
const checkbox = modal.querySelector(`.${swalClasses.checkbox} input`)
const textarea = getChildByClass(modal, swalClasses.textarea)
input.oninput = () => {
sweetAlert.resetValidationError()
}
file.onchange = () => {
sweetAlert.resetValidationError()
}
range.oninput = () => {
sweetAlert.resetValidationError()
rangeOutput.value = range.value
}
range.onchange = () => {
sweetAlert.resetValidationError()
range.previousSibling.value = range.value
}
select.onchange = () => {
sweetAlert.resetValidationError()
}
checkbox.onchange = () => {
sweetAlert.resetValidationError()
}
textarea.oninput = () => {
sweetAlert.resetValidationError()
}
return modal
}
/*
* Manipulate DOM
*/
const sweetHTML = `
`.replace(/(^|\n)\s*/g, '')
export const getContainer = () => document.body.querySelector('.' + swalClasses.container)
export const getModal = () => getContainer() ? getContainer().querySelector('.' + swalClasses.modal) : null
export const getIcons = () => {
const modal = getModal()
return modal.querySelectorAll('.' + swalClasses.icon)
}
export const elementByClass = (className) => getContainer() ? getContainer().querySelector('.' + className) : null
export const getTitle = () => elementByClass(swalClasses.title)
export const getContent = () => elementByClass(swalClasses.content)
export const getImage = () => elementByClass(swalClasses.image)
export const getProgressSteps = () => elementByClass(swalClasses.progresssteps)
export const getValidationError = () => elementByClass(swalClasses.validationerror)
export const getConfirmButton = () => elementByClass(swalClasses.confirm)
export const getCancelButton = () => elementByClass(swalClasses.cancel)
export const getButtonsWrapper = () => elementByClass(swalClasses.buttonswrapper)
export const getCloseButton = () => elementByClass(swalClasses.close)
export const getFocusableElements = () => {
const focusableElementsWithTabindex = Array.from(
getModal().querySelectorAll('[tabindex]:not([tabindex="-1"]):not([tabindex="0"])')
)
// sort according to tabindex
.sort((a, b) => {
a = parseInt(a.getAttribute('tabindex'))
b = parseInt(b.getAttribute('tabindex'))
if (a > b) {
return 1
} else if (a < b) {
return -1
}
return 0
})
const otherFocusableElements = Array.prototype.slice.call(
getModal().querySelectorAll('button, input:not([type=hidden]), textarea, select, a, [tabindex="0"]')
)
return uniqueArray(focusableElementsWithTabindex.concat(otherFocusableElements))
}
export const hasClass = (elem, className) => {
if (elem.classList) {
return elem.classList.contains(className)
}
return false
}
export const focusInput = (input) => {
input.focus()
// place cursor at end of text in text input
if (input.type !== 'file') {
// http://stackoverflow.com/a/2345915/1331425
const val = input.value
input.value = ''
input.value = val
}
}
export const addClass = (elem, className) => {
if (!elem || !className) {
return
}
const classes = className.split(/\s+/).filter(Boolean)
classes.forEach((className) => {
elem.classList.add(className)
})
}
export const removeClass = (elem, className) => {
if (!elem || !className) {
return
}
const classes = className.split(/\s+/).filter(Boolean)
classes.forEach((className) => {
elem.classList.remove(className)
})
}
export const getChildByClass = (elem, className) => {
for (let i = 0; i < elem.childNodes.length; i++) {
if (hasClass(elem.childNodes[i], className)) {
return elem.childNodes[i]
}
}
}
export const show = (elem, display) => {
if (!display) {
display = 'block'
}
elem.style.opacity = ''
elem.style.display = display
}
export const hide = (elem) => {
elem.style.opacity = ''
elem.style.display = 'none'
}
export const empty = (elem) => {
while (elem.firstChild) {
elem.removeChild(elem.firstChild)
}
}
// borrowed from jquery $(elem).is(':visible') implementation
export const isVisible = (elem) => elem.offsetWidth || elem.offsetHeight || elem.getClientRects().length
export const removeStyleProperty = (elem, property) => {
if (elem.style.removeProperty) {
elem.style.removeProperty(property)
} else {
elem.style.removeAttribute(property)
}
}
export const animationEndEvent = (() => {
const testEl = document.createElement('div')
const transEndEventNames = {
'WebkitAnimation': 'webkitAnimationEnd',
'OAnimation': 'oAnimationEnd oanimationend',
'animation': 'animationend'
}
for (const i in transEndEventNames) {
if (transEndEventNames.hasOwnProperty(i) &&
testEl.style[i] !== undefined) {
return transEndEventNames[i]
}
}
return false
})()
// Reset previous window keydown handler and focued element
export const resetPrevState = () => {
window.onkeydown = states.previousWindowKeyDown
if (states.previousActiveElement && states.previousActiveElement.focus) {
let x = window.scrollX
let y = window.scrollY
states.previousActiveElement.focus()
if (x && y) { // IE has no scrollX/scrollY support
window.scrollTo(x, y)
}
}
}
// Measure width of scrollbar
// https://github.com/twbs/bootstrap/blob/master/js/modal.js#L279-L286
export const measureScrollbar = () => {
const supportsTouch = 'ontouchstart' in window || navigator.msMaxTouchPoints
if (supportsTouch) {
return 0
}
const scrollDiv = document.createElement('div')
scrollDiv.style.width = '50px'
scrollDiv.style.height = '50px'
scrollDiv.style.overflow = 'scroll'
document.body.appendChild(scrollDiv)
const scrollbarWidth = scrollDiv.offsetWidth - scrollDiv.clientWidth
document.body.removeChild(scrollDiv)
return scrollbarWidth
}
/**
* Inject a string of CSS into the page header
*
* @param {String} css
*/
export const injectCSS = (css = '') => {
let head = document.head || document.getElementsByTagName('head')[0]
let style = document.createElement('style')
style.type = 'text/css'
head.appendChild(style)
if (style.styleSheet) {
style.styleSheet.cssText = css
} else {
style.appendChild(document.createTextNode(css))
}
}