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)) } }