2020-03-07 10:50:58 +00:00
|
|
|
/**
|
|
|
|
* Extend object a with the properties of object b.
|
|
|
|
* If there's a conflict, object b takes precedence.
|
|
|
|
*
|
|
|
|
* @param {object} a
|
|
|
|
* @param {object} b
|
|
|
|
*/
|
|
|
|
export const extend = ( a, b ) => {
|
|
|
|
|
2020-03-07 10:58:18 +00:00
|
|
|
for( let i in b ) {
|
2020-03-07 10:50:58 +00:00
|
|
|
a[ i ] = b[ i ];
|
|
|
|
}
|
|
|
|
|
|
|
|
return a;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-03-16 14:18:47 +00:00
|
|
|
* querySelectorAll but returns an Array.
|
2020-03-07 10:50:58 +00:00
|
|
|
*/
|
2020-03-16 14:18:47 +00:00
|
|
|
export const queryAll = ( el, selector ) => {
|
2020-03-07 10:50:58 +00:00
|
|
|
|
2020-03-16 14:18:47 +00:00
|
|
|
return Array.from( el.querySelectorAll( selector ) );
|
2020-03-07 10:50:58 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-05-08 18:12:52 +00:00
|
|
|
/**
|
|
|
|
* classList.toggle() with cross browser support
|
|
|
|
*/
|
|
|
|
export const toggleClass = ( el, className, value ) => {
|
|
|
|
if( value ) {
|
|
|
|
el.classList.add( className );
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
el.classList.remove( className );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-07 10:50:58 +00:00
|
|
|
/**
|
|
|
|
* Utility for deserializing a value.
|
|
|
|
*
|
|
|
|
* @param {*} value
|
|
|
|
* @return {*}
|
|
|
|
*/
|
|
|
|
export const deserialize = ( value ) => {
|
|
|
|
|
|
|
|
if( typeof value === 'string' ) {
|
|
|
|
if( value === 'null' ) return null;
|
|
|
|
else if( value === 'true' ) return true;
|
|
|
|
else if( value === 'false' ) return false;
|
|
|
|
else if( value.match( /^-?[\d\.]+$/ ) ) return parseFloat( value );
|
|
|
|
}
|
|
|
|
|
|
|
|
return value;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Measures the distance in pixels between point a
|
|
|
|
* and point b.
|
|
|
|
*
|
|
|
|
* @param {object} a point with x/y properties
|
|
|
|
* @param {object} b point with x/y properties
|
|
|
|
*
|
|
|
|
* @return {number}
|
|
|
|
*/
|
|
|
|
export const distanceBetween = ( a, b ) => {
|
|
|
|
|
2020-03-07 10:58:18 +00:00
|
|
|
let dx = a.x - b.x,
|
2020-03-07 10:50:58 +00:00
|
|
|
dy = a.y - b.y;
|
|
|
|
|
|
|
|
return Math.sqrt( dx*dx + dy*dy );
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Applies a CSS transform to the target element.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} element
|
|
|
|
* @param {string} transform
|
|
|
|
*/
|
|
|
|
export const transformElement = ( element, transform ) => {
|
|
|
|
|
|
|
|
element.style.transform = transform;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-05-26 07:46:50 +00:00
|
|
|
/**
|
|
|
|
* Element.matches with IE support.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} target The element to match
|
|
|
|
* @param {String} selector The CSS selector to match
|
|
|
|
* the element against
|
|
|
|
*
|
|
|
|
* @return {Boolean}
|
|
|
|
*/
|
2020-05-26 08:45:05 +00:00
|
|
|
export const matches = ( target, selector ) => {
|
2020-05-26 07:46:50 +00:00
|
|
|
|
2020-05-26 08:45:05 +00:00
|
|
|
let matchesMethod = target.matches || target.matchesSelector || target.msMatchesSelector;
|
2020-05-26 07:46:50 +00:00
|
|
|
|
|
|
|
return !!( matchesMethod && matchesMethod.call( target, selector ) );
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-03-07 10:50:58 +00:00
|
|
|
/**
|
|
|
|
* Find the closest parent that matches the given
|
|
|
|
* selector.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} target The child element
|
|
|
|
* @param {String} selector The CSS selector to match
|
|
|
|
* the parents against
|
|
|
|
*
|
|
|
|
* @return {HTMLElement} The matched parent or null
|
|
|
|
* if no matching parent was found
|
|
|
|
*/
|
2020-05-26 08:45:05 +00:00
|
|
|
export const closest = ( target, selector ) => {
|
2020-03-07 10:50:58 +00:00
|
|
|
|
2020-05-26 08:45:05 +00:00
|
|
|
// Native Element.closest
|
|
|
|
if( typeof target.closest === 'function' ) {
|
|
|
|
return target.closest( selector );
|
|
|
|
}
|
2020-03-07 10:50:58 +00:00
|
|
|
|
2020-05-26 08:45:05 +00:00
|
|
|
// Polyfill
|
|
|
|
while( target ) {
|
|
|
|
if( matches( target, selector ) ) {
|
|
|
|
return target;
|
2020-03-07 10:50:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Keep searching
|
2020-05-26 08:45:05 +00:00
|
|
|
target = target.parentNode;
|
2020-03-07 10:50:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handling the fullscreen functionality via the fullscreen API
|
|
|
|
*
|
|
|
|
* @see http://fullscreen.spec.whatwg.org/
|
|
|
|
* @see https://developer.mozilla.org/en-US/docs/DOM/Using_fullscreen_mode
|
|
|
|
*/
|
2020-05-05 18:54:40 +00:00
|
|
|
export const enterFullscreen = element => {
|
2020-03-07 10:50:58 +00:00
|
|
|
|
2020-05-05 18:54:40 +00:00
|
|
|
element = element || document.documentElement;
|
2020-03-07 10:50:58 +00:00
|
|
|
|
|
|
|
// Check which implementation is available
|
|
|
|
let requestMethod = element.requestFullscreen ||
|
|
|
|
element.webkitRequestFullscreen ||
|
|
|
|
element.webkitRequestFullScreen ||
|
|
|
|
element.mozRequestFullScreen ||
|
|
|
|
element.msRequestFullscreen;
|
|
|
|
|
|
|
|
if( requestMethod ) {
|
|
|
|
requestMethod.apply( element );
|
|
|
|
}
|
|
|
|
|
2020-03-07 13:18:03 +00:00
|
|
|
}
|
|
|
|
|
2020-03-10 20:08:11 +00:00
|
|
|
/**
|
|
|
|
* Creates an HTML element and returns a reference to it.
|
|
|
|
* If the element already exists the existing instance will
|
|
|
|
* be returned.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} container
|
|
|
|
* @param {string} tagname
|
|
|
|
* @param {string} classname
|
|
|
|
* @param {string} innerHTML
|
|
|
|
*
|
|
|
|
* @return {HTMLElement}
|
|
|
|
*/
|
|
|
|
export const createSingletonNode = ( container, tagname, classname, innerHTML='' ) => {
|
|
|
|
|
|
|
|
// Find all nodes matching the description
|
|
|
|
let nodes = container.querySelectorAll( '.' + classname );
|
|
|
|
|
|
|
|
// Check all matches to find one which is a direct child of
|
|
|
|
// the specified container
|
|
|
|
for( let i = 0; i < nodes.length; i++ ) {
|
|
|
|
let testNode = nodes[i];
|
|
|
|
if( testNode.parentNode === container ) {
|
|
|
|
return testNode;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If no node was found, create it now
|
|
|
|
let node = document.createElement( tagname );
|
|
|
|
node.className = classname;
|
|
|
|
node.innerHTML = innerHTML;
|
|
|
|
container.appendChild( node );
|
|
|
|
|
|
|
|
return node;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-03-07 13:18:03 +00:00
|
|
|
/**
|
|
|
|
* Injects the given CSS styles into the DOM.
|
|
|
|
*
|
|
|
|
* @param {string} value
|
|
|
|
*/
|
2020-03-09 13:44:57 +00:00
|
|
|
export const createStyleSheet = ( value ) => {
|
2020-03-07 13:18:03 +00:00
|
|
|
|
|
|
|
let tag = document.createElement( 'style' );
|
|
|
|
tag.type = 'text/css';
|
2020-03-09 13:44:57 +00:00
|
|
|
|
|
|
|
if( value && value.length > 0 ) {
|
|
|
|
if( tag.styleSheet ) {
|
|
|
|
tag.styleSheet.cssText = value;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
tag.appendChild( document.createTextNode( value ) );
|
|
|
|
}
|
2020-03-07 13:18:03 +00:00
|
|
|
}
|
2020-03-09 13:44:57 +00:00
|
|
|
|
|
|
|
document.head.appendChild( tag );
|
|
|
|
|
|
|
|
return tag;
|
2020-03-07 13:18:03 +00:00
|
|
|
|
2020-03-10 19:40:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a key:value hash of all query params.
|
|
|
|
*/
|
|
|
|
export const getQueryHash = () => {
|
|
|
|
|
|
|
|
let query = {};
|
|
|
|
|
|
|
|
location.search.replace( /[A-Z0-9]+?=([\w\.%-]*)/gi, a => {
|
|
|
|
query[ a.split( '=' ).shift() ] = a.split( '=' ).pop();
|
|
|
|
} );
|
|
|
|
|
|
|
|
// Basic deserialization
|
|
|
|
for( let i in query ) {
|
|
|
|
let value = query[ i ];
|
|
|
|
|
|
|
|
query[ i ] = deserialize( unescape( value ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do not accept new dependencies via query config to avoid
|
|
|
|
// the potential of malicious script injection
|
|
|
|
if( typeof query['dependencies'] !== 'undefined' ) delete query['dependencies'];
|
|
|
|
|
|
|
|
return query;
|
|
|
|
|
2020-03-16 14:18:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the remaining height within the parent of the
|
|
|
|
* target element.
|
|
|
|
*
|
|
|
|
* remaining height = [ configured parent height ] - [ current parent height ]
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} element
|
|
|
|
* @param {number} [height]
|
|
|
|
*/
|
|
|
|
export const getRemainingHeight = ( element, height = 0 ) => {
|
|
|
|
|
|
|
|
if( element ) {
|
|
|
|
let newHeight, oldHeight = element.style.height;
|
|
|
|
|
|
|
|
// Change the .stretch element height to 0 in order find the height of all
|
|
|
|
// the other elements
|
|
|
|
element.style.height = '0px';
|
|
|
|
|
|
|
|
// In Overview mode, the parent (.slide) height is set of 700px.
|
|
|
|
// Restore it temporarily to its natural height.
|
|
|
|
element.parentNode.style.height = 'auto';
|
|
|
|
|
|
|
|
newHeight = height - element.parentNode.offsetHeight;
|
|
|
|
|
|
|
|
// Restore the old height, just in case
|
|
|
|
element.style.height = oldHeight + 'px';
|
|
|
|
|
|
|
|
// Clear the parent (.slide) height. .removeProperty works in IE9+
|
|
|
|
element.parentNode.style.removeProperty('height');
|
|
|
|
|
|
|
|
return newHeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
return height;
|
|
|
|
|
2021-11-24 10:07:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const fileExtensionToMimeMap = {
|
|
|
|
'mp4': 'video/mp4',
|
|
|
|
'm4a': 'video/mp4',
|
|
|
|
'ogv': 'video/ogg',
|
|
|
|
'mpeg': 'video/mpeg',
|
|
|
|
'webm': 'video/webm'
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Guess the MIME type for common file formats.
|
|
|
|
*/
|
|
|
|
export const getMimeTypeFromFile = ( filename='' ) => {
|
|
|
|
return fileExtensionToMimeMap[filename.split('.').pop()]
|
2023-02-13 09:02:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Encodes a string for RFC3986-compliant URL format.
|
|
|
|
* https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/encodeURI#encoding_for_rfc3986
|
|
|
|
*
|
|
|
|
* @param {string} url
|
|
|
|
*/
|
|
|
|
export const encodeRFC3986URI = ( url='' ) => {
|
|
|
|
return encodeURI(url)
|
|
|
|
.replace(/%5B/g, "[")
|
|
|
|
.replace(/%5D/g, "]")
|
|
|
|
.replace(
|
|
|
|
/[!'()*]/g,
|
|
|
|
(c) => `%${c.charCodeAt(0).toString(16).toUpperCase()}`
|
|
|
|
);
|
2020-03-07 10:50:58 +00:00
|
|
|
}
|