// RequestAnimationFrame Polyfill (Android 4.3 and below) /*! @author Paul Irish */ /*! @source https://gist.github.com/paulirish/1579671 */ (function() { var rafLastTime = 0; const win: any = window; if (!win.requestAnimationFrame) { win.requestAnimationFrame = function(callback, element) { var currTime = Date.now(); var timeToCall = Math.max(0, 16 - (currTime - rafLastTime)); var id = window.setTimeout(function() { callback(currTime + timeToCall); }, timeToCall); rafLastTime = currTime + timeToCall; return id; }; } if (!win.cancelAnimationFrame) { win.cancelAnimationFrame = function(id) { clearTimeout(id); }; } })(); export const raf = window.requestAnimationFrame.bind(window); export const cancelRaf = window.cancelAnimationFrame.bind(window); export function rafFrames(framesToWait, callback) { framesToWait = Math.ceil(framesToWait); if (framesToWait < 2) { raf(callback); } else { setTimeout(() => { raf(callback); }, (framesToWait - 1) * 17); } } export let CSS: { transform?: string, transition?: string, transitionDuration?: string, transitionDelay?: string, transitionTimingFn?: string, transitionStart?: string, transitionEnd?: string, } = {}; (function() { // transform var i, keys = ['webkitTransform', 'transform', '-webkit-transform', 'webkit-transform', '-moz-transform', 'moz-transform', 'MozTransform', 'mozTransform', 'msTransform']; for (i = 0; i < keys.length; i++) { if (document.documentElement.style[keys[i]] !== undefined) { CSS.transform = keys[i]; break; } } // transition keys = ['webkitTransition', 'mozTransition', 'msTransition', 'transition']; for (i = 0; i < keys.length; i++) { if (document.documentElement.style[keys[i]] !== undefined) { CSS.transition = keys[i]; break; } } // The only prefix we care about is webkit for transitions. var isWebkit = CSS.transition.indexOf('webkit') > -1; // transition duration CSS.transitionDuration = (isWebkit ? '-webkit-' : '') + 'transition-duration'; // transition timing function CSS.transitionTimingFn = (isWebkit ? '-webkit-' : '') + 'transition-timing-function'; // transition delay CSS.transitionDelay = (isWebkit ? '-webkit-' : '') + 'transition-delay'; // To be sure transitionend works everywhere, include *both* the webkit and non-webkit events CSS.transitionEnd = (isWebkit ? 'webkitTransitionEnd ' : '') + 'transitionend'; })(); export function transitionEnd(el: HTMLElement, callback: Function) { if (el) { function unregister() { CSS.transitionEnd.split(' ').forEach(eventName => { el.removeEventListener(eventName, onEvent); }); } function onEvent(ev) { if (el === ev.target) { unregister(); callback(ev); } } CSS.transitionEnd.split(' ').forEach(eventName => { el.addEventListener(eventName, onEvent); }); return unregister; } } export function ready(callback?: Function) { let promise = null; if (!callback) { // a callback wasn't provided, so let's return a promise instead promise = new Promise(resolve => { callback = resolve; }); } if (document.readyState === 'complete' || document.readyState === 'interactive') { callback(); } else { function completed() { document.removeEventListener('DOMContentLoaded', completed, false); window.removeEventListener('load', completed, false); callback(); } document.addEventListener('DOMContentLoaded', completed, false); window.addEventListener('load', completed, false); } return promise; } export function windowLoad(callback?: Function) { let promise = null; if (!callback) { // a callback wasn't provided, so let's return a promise instead promise = new Promise(resolve => { callback = resolve; }); } if (document.readyState === 'complete') { callback(); } else { function completed() { window.removeEventListener('load', completed, false); callback(); } window.addEventListener('load', completed, false); } return promise; } export function pointerCoord(ev: any): {x: number, y: number} { // get coordinates for either a mouse click // or a touch depending on the given event let c = { x: 0, y: 0 }; if (ev) { const touches = ev.touches && ev.touches.length ? ev.touches : [ev]; const e = (ev.changedTouches && ev.changedTouches[0]) || touches[0]; if (e) { c.x = e.clientX || e.pageX || 0; c.y = e.clientY || e.pageY || 0; } } return c; } export function hasPointerMoved(threshold, startCoord, endCoord) { return startCoord && endCoord && (Math.abs(startCoord.x - endCoord.x) > threshold || Math.abs(startCoord.y - endCoord.y) > threshold); } export function isActive(ele) { return !!(ele && (document.activeElement === ele)); } export function hasFocus(ele) { return isActive(ele) && (ele.parentElement.querySelector(':focus') === ele); } export function isTextInput(ele) { return !!ele && (ele.tagName == 'TEXTAREA' || ele.contentEditable === 'true' || (ele.tagName == 'INPUT' && !(/^(radio|checkbox|range|file|submit|reset|color|image|button)$/i).test(ele.type))); } export function hasFocusedTextInput() { let ele = document.activeElement; if (isTextInput(ele)) { return (ele.parentElement.querySelector(':focus') === ele); } return false; } const skipInputAttrsReg = /^(value|checked|disabled|type|class|style|id|autofocus|autocomplete|autocorrect)$/i export function copyInputAttributes(srcElement, destElement) { // copy attributes from one element to another // however, skip over a few of them as they're already // handled in the angular world var attrs = srcElement.attributes; for (var i = 0; i < attrs.length; i++) { var attr = attrs[i]; if (!skipInputAttrsReg.test(attr.name)) { destElement.setAttribute(attr.name, attr.value); } } } let matchesFn: string; let matchesMethods: Array = ['matches','webkitMatchesSelector','mozMatchesSelector','msMatchesSelector']; matchesMethods.some((fn: string) => { if (typeof document.documentElement[fn] === 'function') { matchesFn = fn; return true; } }); export function closest(ele: HTMLElement, selector: string, checkSelf?: boolean) { if (ele && matchesFn) { // traverse parents ele = (checkSelf ? ele : ele.parentElement); while (ele !== null) { if (ele[matchesFn](selector)) { return ele; } ele = ele.parentElement; } } return null; } /** * Get the element offsetWidth and offsetHeight. Values are cached * to reduce DOM reads. Cache is cleared on a window resize. */ export function getDimensions(ele: HTMLElement, id: string): { width: number, height: number, left: number, top: number } { let dimensions = dimensionCache[id]; if (!dimensions) { // make sure we got good values before caching if (ele.offsetWidth && ele.offsetHeight) { dimensions = dimensionCache[id] = { width: ele.offsetWidth, height: ele.offsetHeight, left: ele.offsetLeft, top: ele.offsetTop }; } else { // do not cache bad values return { width: 0, height: 0, left: 0, top: 0 }; } } return dimensions; } export function clearDimensions(id: string) { delete dimensionCache[id]; } export function windowDimensions(): {width: number, height: number} { if (!dimensionCache.win) { // make sure we got good values before caching if (window.innerWidth && window.innerHeight) { dimensionCache.win = { width: window.innerWidth, height: window.innerHeight }; } else { // do not cache bad values return { width: 0, height: 0 }; } } return dimensionCache.win; } export function flushDimensionCache() { dimensionCache = {}; } let dimensionCache:any = {};