Skip to content

Instantly share code, notes, and snippets.

@xizon
Last active May 29, 2023 08:04
Show Gist options
  • Star 1 You must be signed in to star a gist
  • Fork 1 You must be signed in to fork a gist
  • Save xizon/a3be202a1b78c4853b02d07c1c2c2666 to your computer and use it in GitHub Desktop.
Save xizon/a3be202a1b78c4853b02d07c1c2c2666 to your computer and use it in GitHub Desktop.
Vanilla-js Helpers
/**
*************************************
* Core Helpers
*
* @package: https://github.com/xizon/boot-helpers
* @version: 0.1.7
* @last update: May 25, 2023
* @author: UIUX Lab <uiuxlab@gmail.com>
* @license: MIT
*
*************************************
*/
/*
//+++++++++++++++++++++++++++++++++++++++++++
// Page load
//+++++++++++++++++++++++++++++++++++++++++++
__( document ).ready( function() {
//do something
});
__( 'body' ).loader({
imagesSelector: 'body img',
videosSelector: 'body video',
startEvent: function() {
//do something
console.log( '=> loading.' );
},
progressEvent: function(percent) {
//do something
console.log( '=> progress: ' + percent + '%' );
},
endEvent: function() {
console.log( '=> loaded!!!' );
//do something
}
});
//+++++++++++++++++++++++++++++++++++++++++++
// Dom demos
//+++++++++++++++++++++++++++++++++++++++++++
__( '.demo' ).remove();
__( '.demo' ).empty();
__( '.demo' ).show();
__( '.demo' ).hide();
__( '.demo' ).get(-1); //returns all elements
__( '.demo' ).get(0); //returns someone element
__( '.demo' ).len(); //returns length of elements
__( '.demo' ).addClass( 'class-3' );
__( '.demo, .demo2' ).addClass( 'class-4' );
__( '.demo' ).find( 'li' ).addClass( 'class-1 class-2' );
__( '.demo' ).find( 'li' ).removeClass( 'class-2' );
__( '.demo' ).find( '> .demo2' ).addClass( 'class-3' );
__( '.demo' ).find( '> .demo2' ).find( 'li' ).addClass( 'class-3-2' );
__( '.demo' ).closest( '.container' ).addClass( 'class-4' );
__( '.demo' ).css({
'background': '#f00',
'font-size': '18px'
});
__( '#demo' ).css( 'background-color', '#f60' );
__( '.demo' ).data( 'bg', 'red' );
__( '.demo' ).attr( 'disabled', 'disabled' );
__( '.demo' ).width( 300 );
__( '.menu li:first-child' ).width( "50%" );
if( __( '.demo' ).data( 'activated' ) === null ) {
//do something...
}
if( __( '.demo' ).attr( 'data-activated' ) === null ) {
//do something...
}
__( '.menu li' ).eq(1).append( '<span style="color:green">after</span>');
__( '.menu li' ).eq(1).prepend( '<span style="color:red">before</span>');
__( '.menu li' ).first().before( '<li style="color:green">(first)before</li>');
__( '.menu li' ).last().after( '<li style="color:red">(last)after</li>');
__( 'h1' ).wrapInner( '<span class="new-div" />' );
√√
__( 'h1' ).text( 'New H1' );
__( '.demo1' ).prev().addClass( 'prev' );
__( '.demo2' ).next().addClass( 'next' );
__( '.demo3' ).parent().addClass( 'parent' );
__( '.menu' ).parents().addClass( 'all-parents' );
__( '.demo' ).children().addClass( 'children-all' );
__( '.class-1' ).siblings().addClass( 'class-siblings' );
__( '.demo1' ).prev( 'ul' ).addClass( 'prev' );
__( '.demo2' ).next( 'ul' ).addClass( 'next' );
__( '.demo3' ).parent( 'ul' ).addClass( 'parent' );
__( '.menu' ).parents( 'ul' ).addClass( 'all-parents' );
__( '.demo' ).children( '.demo-children2' ).addClass( 'children-single' );
__( '.class-1' ).siblings( 'ul' ).addClass( 'class-siblings' );
__( '.class-1' ).not( '.class-2' ).addClass( 'class-not' );
__( '.class-1' ).filter( '.class-2' ).addClass( 'class-filter' );
__( '.demo' ).trigger( 'click' );
__( '.demo' ).toggleClass( 'class-toggle-1 class-toggle-2' );
//append HTML Element
const htmlObject = document.createElement('div');
htmlObject.innerHTML = '<span style="color:green">after</span>';
__( '.menu li' ).eq(1).append( htmlObject.firstChild );
//clone HTML Element
const cloneHTML = __( '.uix-menu__container' ).clone();
__( cloneHTML ).addClass( 'is-mobile' );
__( 'body' ).prependTo( cloneHTML );
__( 'body' ).appendTo( cloneHTML );
//+++++++++++++++++++++++++++++++++++++++++++
// Click event demos
//+++++++++++++++++++++++++++++++++++++++++++
__( '.menu li a' ).off( 'click' ).one( 'click', function( e ) {
e.preventDefault();
console.log( 'This will be clicked only once' );
});
__( '.menu li' ).off( 'click' ).on( 'click', function( e ) {
console.log('e: ', e);
console.log('this: ', this);
console.log('index(): ', __( this ).index());
console.log('attr(class): ', __( this ).attr( 'class' ));
__( this ).addClass( 'new-class' )
});
__( document ).off( 'click', '.menu li' );
__( document ).on( 'click', '.menu li', function( e ) {
console.log('e: ', e);
console.log('this: ', this);
console.log('index(): ', __( this ).index());
console.log('attr(class): ', __( this ).attr( 'class' ));
__( this ).addClass( 'new-class' )
});
__( '#imghere' ).off( 'click', imgFn);
__( '#imghere' ).on( 'click', imgFn);
function imgFn() {
console.log( 'imgFn' );
}
//+++++++++++++++++++++++++++++++++++++++++++
// Retrieve data demos
//+++++++++++++++++++++++++++++++++++++++++++
console.log( '<h1> content: ' + __( 'h1' ).html());
console.log( '<h1> content: ' + __( 'h1' ).text());
console.log( 'length of .menu li: ' + __( '.menu li' ).len() );
console.log( 'length of #none: ' + __( '#none' ).len() );
console.log( 'width(): ' + __( '.demo' ).width() );
console.log( 'outerWidth(): ' + __( '.demo' ).outerWidth() );
console.log( 'outerWidth( true ): ' + __( '.demo' ).outerWidth(true) );
console.log( 'height(): ' + __( '.demo' ).height() );
console.log( 'outerHeight(): ' + __( '.demo' ).outerHeight() );
console.log( 'outerHeight( true ): ' + __( '.demo' ).outerHeight(true) );
console.log( 'document h: ', __( document ).height() );
console.log( 'document w: ', __( document ).width() );
console.log( 'window h: ', __( window ).height() );
console.log( 'window h: ', __( window ).width() );
console.log( 'window scrollTop: ', __( window ).scrollTop() );
console.log( 'window scrollLeft: ', __( window ).scrollLeft() );
console.log( 'data: [data-bg] value: ' + __( '.demo' ).data( 'bg' ) );
console.log( 'attr: [disabled] value: ' + __( '.demo' ).attr( 'disabled' ) );
console.log( 'allAttrs(): ', __( '.demo' ).allAttrs() );
console.log( __( '.demo' ).hasClass( 'class-1' ) );
console.log( __( 'h1' ).offset() );
console.log( __( 'h1' ).position() );
console.log( __( '.demo' ).maxDimension() );
console.log( __( '.menu li:nth-child(2)' ).index() );
// Traverse all attribute names and values
const allAttrs = __( '#demo' ).allAttrs();
for (let key in allAttrs) {
console.log( key + ' = ' + allAttrs[key] );
}
//+++++++++++++++++++++++++++++++++++++++++++
// Loop demos
//+++++++++++++++++++++++++++++++++++++++++++
__( '.menu li' ).each( function( index, curSelector ) {
console.log( index + ' : ' );
console.log( this );
this.style.background = '#333';
__( this ).css({
'background': '#f00',
'font-size': '18px'
});
//Nested `each() `
//__( curSelector) are generally used for exact each selector
__( curSelector ).find( 'ul > li' ).each( function( index ) {
__( this ).attr( 'id', 'li-id-' + index );
});
//Nested `eq()`
for (let k = 0; k<liNum; k++) {
__( curSelector + ' ul > li' ).eq(k).css({
'font-size' : '18px'
});
__( curSelector + ' ul > li:nth-child('+k+') > a' ).css({
'font-size' : '18px'
});
}
});
//+++++++++++++++++++++++++++++++++++++++++++
// Animation demos
//+++++++++++++++++++++++++++++++++++++++++++
__( '.demo' ).fadeOut(1000, function(){
setTimeout( function() {
__( '.demo' ).fadeIn(3000);
},1000 );
});
__( '.menu' ).animate( 'marginLeft', 0, 100, 'px', 1500, 'ease-out', function(){ console.log(this.className); } );
__( '.menu' ).animate( 'marginTop', 0, 200, 'px', 1500, 'ease-out', function(){ console.log(this.className); } );
//+++++++++++++++++++++++++++++++++++++++++++
// AJAX demos
//+++++++++++++++++++++++++++++++++++++++++++
__.ajax({
url: 'https://restcountries.com/v2/name/Argentina',
method: 'GET',
complete: function( data ) {
console.log( '=> ajax ok!' );
console.log( data );
}
});
//+++++++++++++++++++++++++++++++++++++++++++
// Form demos
//+++++++++++++++++++++++++++++++++++++++++++
__( '#input-name-1' ).val( ); //form control: `<Input />`
__( '#select-name-1' ).val( 'value-3' ); //form control: `<Select />`
__( '#switch-name-1' ).val( true ); //form control: `<Switch />`
__( '#checkbox-name-1' ).val( true ); //form control: `<Checkbox />`
__('input[name="radio-name-1"]').val( 'value-3' ); //form control: `<Radio />`
console.log( '__( val(): ' + __( '#input' ).val());
__( '#checkbox1' ).prop('checked', true);
console.log( '__( prop(): ' + __( '#checkbox2' ).prop( 'checked' ));
__( '#input' ).prop('disabled', true);
// To send data in the application/x-www-form-urlencoded format instead
const formData = new FormData();
const defaultPostData = {
action : 'load_singlepages_ajax_content'
};
for(let k in defaultPostData) {
formData.append(k, defaultPostData[k]);
}
// For multiple form fields data acquisition
const oldFormData = __( '#form' ).serializeArray();
oldFormData.forEach(function(item){
formData.append(item.name, item.value);
});
//+++++++++++++++++++++++++++++++++++++++++++
// Utilities demos
//+++++++++++++++++++++++++++++++++++++++++++
console.log( __.isTouchCapable() ); //true or false
console.log( __.browser.isIE ); //.isMobile, .isAndroid, .isPC, .isSafari, .isIE, .supportsPassive
console.log( __.GUID.create() );
console.log( __.math.evaluate( '100/3' ) );
console.log( __.math.getRandomFloat(1, 10) );
console.log( __.math.getDegree(135) );
console.log( __.math.getRadian(1) );
console.log( __.math.getPolarCoord(10,30,0) );
console.log( __.cssProperty.getAbsoluteCoordinates( __( '.col-12' )[0] ).left );
console.log( __.cssProperty.getTransitionDuration( __( '.col-12' )[0] ) );
console.log( __.styleFormat( 'font-size: 10px;background: #51B801; color:#fff; border-radius: 5px;padding: 2px 3px;display: inline-block;margin-left: 3px;' ) )
console.log( __.trim( 'string string spacing string' ) );
console.log( __.lastUrlParamFormat( 'string-string-spacing_string' ) );
console.log( __.removeFirstLastStr( ',string,string,string,' ) );
console.log( __.toSlug( 'string String2-s' ) );
console.log( __.htmlEncode( '<span style="color:red">text</span>' ) );
console.log( __.htmlDecode( '&lt;span style="color:red"&gt;text&lt;/span&gt;' ) );
console.log( __.validate.isMobile( '13167678787' ) ); //true
console.log( __.validate.isTel( '123-456-7890' ) ); //true
console.log( __.validate.isEmail( 'name@gmail.com' ) ); //true
console.log( __.validate.isNumber( '1421.231' ) ); //true
console.log( __.validate.isInt( '1421.231' ) ); //false
console.log( __.validate.isJSON( '{"a":true}' ) ); //true
// Scroll spy
const myFunc = function(){ console.log('throttle'); }
const throttleFunc = __.throttle(myFunc, 300);
window.removeEventListener('scroll', throttleFunc);
window.removeEventListener('touchmove', throttleFunc);
window.addEventListener('scroll', throttleFunc);
window.addEventListener('touchmove', throttleFunc);
// Click spy
const myFunc = function(){ console.log('debounce'); }
const debounceFunc = __.debounce(myFunc, 300);
function handleClick() {
debounceFunc();
}
// Window spy
const debounceFuncWindow = __.debounce(windowUpdate, 50);
window.removeEventListener('resize', debounceFuncWindow);
window.addEventListener('resize', debounceFuncWindow);
// Deep clone an element
let a = [1,2,3,4], b = __.deepClone(a);
let demo = document.querySelector( '#demo' ), demoCopy = __.deepClone(demo);
// Set a default configuration
function myFun(curElement, config) {
if ( typeof curElement === typeof undefined ) return;
config = __.setDefaultOptions({
"src" : false,
"htmlID" : false,
"fixed" : false,
"ajax" : false
}, config);
console.log( config ); //{src: 'https://google.com', htmlID: true, fixed: false, ajax: false}
}
myFun( __( '.demo-trigger' ), {
src: 'https://google.com',
htmlID: true
});
//+++++++++++++++++++++++++++++++++++++++++++
// Create a new method inherit from the constructor's prototype.
//+++++++++++++++++++++++++++++++++++++++++++
__.fn.myPlugin = function( color ) {
this.each( function(index) {
this.style.background = color;
});
return this;
};
__('body').myPlugin( '#f00' );
*/
const __ = (function () {
'use strict';
/**
* Create the constructor (Wrap the selector)
* @private
*
* @param {String|Element|Array} s - The selector to search for or HTML element to wrap with functionality
* @param {Element} root - OPTIONAL An HTML element to start the element query from
* @return {NodeList} - The collection of elements, wrapped with functionality (see API methods)
*/
const Constructor = function (s, root) {
if ( typeof (s) === 'undefined' ) return;
// Backward compatibility, some methods need to use it
this.storeSelector = this;
//
this.elems = [];
root = root || document;
if ( Array.isArray(s) ) {
//is array
// eg.  [li#demo1, li#demo2, li#demo3]
// [ [li#demo1, li#demo2, li#demo3] ]
//----------
// There may be Nested array, the array needs to be flattened
s = [].concat(...s);
this.elems = s;
} else {
//not array
//----------
if (typeof(s) === 'string') {
//1) string
this.elems = root.querySelectorAll(s);
} else if (s.tagName) {
//2) HTML elements
this.elems = [s];
} else {
//3) window
if ( s === window ) this.elems = [window];
//4) document or other
switch( s.nodeType ) {
case 9:
//if Document
this.elems = [document.body];
break;
default:
}
}
}
};
/**
* Instantiate a new constructor
* @private
*/
const $$ = function (s, root) {
return new Constructor(s, root);
};
/**
* Return a prototype that extending by adding new methods
* @private
*/
$$.fn = Constructor.prototype;
/* ------------- Private Methods -------------- */
/**
* Get real style data
* @private
*/
function getStyle( el, attr ) {
const self = el;
if ( typeof (window) !== 'undefined' ) {
//document or window data
//-----------------
if ( self === document.body ) {
if ( attr == 'height' ) return document.body.clientHeight;
if ( attr == 'width' ) return document.body.clientWidth;
}
if ( self === window ) {
if ( attr == 'height' ) return window.innerHeight;
if ( attr == 'width' ) return window.innerWidth;
}
//element data
//-----------------
let _val = 0;
const computedStyle = window.getComputedStyle
? getComputedStyle(self) // Standards
: self.currentStyle; // Old IE
if (computedStyle) { // This will be true on nearly all browsers
_val = computedStyle[attr]; //return ??px
}
// if getComputedStyle return 'auto'
if ( attr === 'height' && _val === 'auto' ) return actualPropertyValue(self, 'height');
if ( attr === 'width' && _val === 'auto' ) return actualPropertyValue(self, 'width');
//
const newVal = /\d+/.exec( _val ); // Array ["123"]
return parseFloat( newVal );
} else {
return 0;
}
}
/**
* Determine whether it is Child Node
* @private
*/
function isChild( el, p ) {
if (!el || !p || !p.childNodes || !p.childNodes.length) {
return false;
}
return ([].slice.call(p.childNodes).filter(function(n) {
const found = (n === el);
if (!found && n.childNodes && n.childNodes.length) {
return isChild(el, n);
}
return found;
})).length;
}
/**
* Determine whether it is in JSON format
* @private
*/
function isJSON( str ){
if ( typeof(str) === 'string' && str.length > 0 ) {
if ( str.replace( /\"\"/g, '' ).replace( /\,/g, '' ) == '[{}]' ) {
return false;
} else {
if (/^[\],:{}\s]*$/.test( str.replace(/\\["\\\/bfnrtu]/g, '@' ).
replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']').
replace(/(?:^|:|,)(?:\s*\[)+/g, ''))) {
return true;
}else{
return false;
}
}
} else {
if (
typeof(str) === 'object' &&
Object.prototype.toString.call(str) === '[object Object]' &&
! str.length
) {
return true;
} else {
return false;
}
}
}
/**
* Delete elements in the array
* @private
*/
function removeArray(arr) {
let what,
args = arguments,
len = args.length,
index;
while (len > 1 && arr.length) {
what = args[--len];
while ((index = arr.indexOf(what)) !== -1) {
arr.splice(index, 1);
}
}
return arr;
}
/**
* Convert string to camel case
* @private
*/
function toCamelCase(s) {
if (typeof(s) === 'string') {
let firstWord="";
let arr = s.split("-");
for (let i = 1; i < arr.length; i++)
{
firstWord = arr[i].charAt(0).toUpperCase();
arr[i] = firstWord + arr[i].slice(1);
}
arr = arr.join(""); //Convert the arr array to a string
return arr;
} else {
return s;
}
}
/**
* Get CSS path from Dom element
* @private
*/
function elementPath(el) {
if (!(el instanceof Element)) return;
//
const path = [];
let itemIndex = 0;
while (el.nodeType === Node.ELEMENT_NODE) {
const oldSelector = el.nodeName.toLowerCase();
let selector = oldSelector;
if (el.id) {
if ( itemIndex > 0 ) selector += '#' + el.id;
}
if (el.className) {
selector += '.' + el.className.replace(/\s+/g, ".");
}
selector = selector.replace(/\.\./g, ".");
//Add one or more items to the start of an array's result set.
path.unshift(selector);
//
el = el.parentNode;
//
itemIndex++;
}
return path.join(" > ");
}
/**
* Check if a string is a valid number
* @private
*/
function isValidNumeric(str) {
if (typeof str != "string") return false // we only process strings!
if (
!isNaN(str) && // use type coercion to parse the _entirety_ of the string (`parseFloat` alone does not do this)
!isNaN(parseFloat(str)) // ensure strings of whitespace fail
) {
return true;
} else {
return false;
}
}
/**
* Convert string to hump naming
* @private
*/
function stringlineToHump(str) {
if ( typeof str === 'string' && str.length > 0 ) {
const re=/-(\w)/g;
str=str.replace(re,function($0,$1){
return $1.toUpperCase();
});
return str;
} else {
return str;
}
}
/**
* Get the actual value with user specific methed
* it can be 'width', 'height', 'outerWidth', 'outerHeight'
* @private
* @param {Element} el - A DOM node containing one selector to match against.
* @param {String} prop - A string naming the property of style.
* @param {?Json} config - Whether or not margin is included. The key `includeMargin`
takes effect when set to true
* @return {Number} - Returns a pure number.
*/
function actualPropertyValue(el, prop, config) {
const style = window.getComputedStyle ? window.getComputedStyle(el) : el.currentStyle,
display = style.display,
position = style.position,
visibility = style.visibility;
let marginWidth = 0;
let marginHeight = 0;
let maxVal;
let actualVal;
if ( config && config.includeMargin === true ) {
marginWidth = parseFloat(style.marginLeft) + parseFloat(style.marginRight);
marginHeight = parseFloat(style.marginTop) + parseFloat(style.marginBottom);
}
if ( prop === 'width' ) {
maxVal = parseFloat( style.maxWidth );
// if its not hidden we just return normal height
if(display !== 'none' && maxVal !== '0') {
return el.clientWidth;
}
}
if ( prop === 'height' ) {
maxVal = parseFloat( style.maxHeight );
if(display !== 'none' && maxVal !== '0') {
return el.clientHeight;
}
}
if ( prop === 'outerWidth' ) {
maxVal = parseFloat( style.maxWidth );
if(display !== 'none' && maxVal !== '0') {
return el.offsetWidth + marginWidth;
}
}
if ( prop === 'outerHeight' ) {
maxVal = parseFloat( style.maxHeight );
if(display !== 'none' && maxVal !== '0') {
return el.offsetHeight + marginHeight;
}
}
// the element is hidden so:
// making the el block so we can meassure its height but still be hidden
el.style.position = 'absolute';
el.style.visibility = 'hidden';
el.style.display = 'block';
if ( prop === 'width' ) actualVal = el.clientWidth;
if ( prop === 'height' ) actualVal = el.clientHeight;
if ( prop === 'outerWidth' ) actualVal = el.offsetWidth + marginWidth;
if ( prop === 'outerHeight' ) actualVal = el.offsetHeight + marginHeight;
// reverting to the original values
el.style.display = display;
el.style.position = position;
el.style.visibility = visibility;
return actualVal;
}
/**
* Some easing functions
* @private
* @link: https://easings.net
* @param {Number} t - time (Amount of time that has passed since the beginning of the animation. Usually starts at 0 and is slowly increased using a game loop or other update function.)
* @param {Number} b - beginning value (The starting point of the animation. Usually it's a static value, you can start at 0 for example.)
* @param {Number} c - change in value (The amount of change needed to go from starting point to end point. It's also usually a static value.)
* @param {Number} d - duration (Amount of time the animation will take. Usually a static value aswell.)
* @return {Number}
*/
function easeLinear (t, b, c, d) {
return c * t / d + b;
}
function easeInQuart (t, b, c, d) {
return c * (t /= d) * t * t * t + b;
}
function easeOutQuart (t, b, c, d) {
return -c * ((t = t / d - 1) * t * t * t - 1) + b;
}
function easeInOutQuart (t, b, c, d) {
if ((t /= d / 2) < 1) return c / 2 * t * t * t * t + b;
return -c / 2 * ((t -= 2) * t * t * t - 2) + b;
}
function easeInCubic (t, b, c, d) {
return c * (t /= d) * t * t + b;
}
function easeOutCubic (t, b, c, d) {
return c * ((t = t / d - 1) * t * t + 1) + b;
}
function easeInOutCubic (t, b, c, d) {
if ((t /= d / 2) < 1) return c / 2 * t * t * t + b;
return c / 2 * ((t -= 2) * t * t + 2) + b;
}
/* ------------- Independent Methods (public) -------------- */
/**
* Object validation
* @public
*
* @return {Boolean}
*/
$$.validate = $$.validate || ( () => {
function t() { }
return t.version = "0.0.1",
t.isNumber = function(B) {
const A = /^[\d|\.|,]+$/;
return A.test(B);
},
t.isInt = function(B) {
if (B == "") {
return false;
}
const A = /\D+/;
return ! A.test(B);
},
t.isEmail = function(A) {
const B = /^\s*([A-Za-z0-9_-]+(\.\w+)*@(\w+\.)+\w{2,3})\s*$/;
return B.test(A);
},
t.isTel = function(A) {
//const B = /^[\d|\-|\s|\_]+$/;
const B = /^[0-9- ]{7,20}$/;
return B.test(A);
},
t.isMobile = function(A) {
//const B = /^13[0-9]{9}|15[012356789][0-9]{8}|18[0256789][0-9]{8}|147[0-9]{8}$/;
const B = /^1[0-9]{10}$/;
return B.test(A);
},
t.isJSON = function(A) {
return isJSON(A);
},
//
t
})();
/**
* To determine if it is a touch screen.
* @public
*
* @return {Boolean}
*/
$$.isTouchCapable = function() {
return 'ontouchstart' in window ||
window.DocumentTouch && document instanceof window.DocumentTouch ||
window.navigator.maxTouchPoints > 0;
};
/**
* Determine whether it is a special browser
* @public
*
* @return {Json} - Boolean judgment collection of common browsers
*/
$$.browser = $$.browser || ( () => {
let browser = {};
if ( typeof (navigator) !== 'undefined' ) {
// Test via a getter in the options object to see if the passive property is accessed
let supportsPassive = false;
try {
const opts = Object.defineProperty({}, 'passive', {
get: function() {
supportsPassive = true;
}
});
window.addEventListener("testPassive", null, opts);
window.removeEventListener("testPassive", null, opts);
} catch (e) {}
browser = {
isMobile : /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent),
isAndroid : /(android)/i.test(navigator.userAgent),
isPC : !navigator.userAgent.match(/(iPhone|iPod|Android|ios|Mobile)/i),
isSafari : !!navigator.userAgent.match(/Version\/[\d\.]+.*Safari/), /**Test to 9, 10. */
isIE : !!window.ActiveXObject || "ActiveXObject" in window, /**Test to 6 ~ 11 (not edge) */
supportsPassive : supportsPassive
};
}
return browser;
})();
/**
* Create GUID / UUID
* @public
*
* @description This function can be used separately in HTML pages or custom JavaScript.
* @return {String} - The globally-unique identifiers.
*/
$$.GUID = $$.GUID || ( () => {
function t() { }
return t.version = "0.0.1",
/**
* Generate a string of unique characters
*
* @return {String}
*/
t.create = function() {
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
const r = Math.random() * 16 | 0,
v = c == 'x' ? r : (r & 0x3 | 0x8);
return v.toString(16);
});
},
//
t
})();
/**
* Evaluating a string as a mathematical expression in JavaScript
* @public
*
* @description This function can be used separately in HTML pages or custom JavaScript.
* @return {String} - New calculation result.
*/
$$.math = $$.math || ( () => {
function t() { }
return t.version = "0.0.2",
/**
* Expression parsing and evaluation
*
* @param {String} s - A math expression given in string.
* @return {Number} - Returns a pure number calculated.
*/
t.evaluate = function(s) {
const chars = s.replace(/\s/g, '').split("");
let n = [], op = [], index = 0, oplast = true;
n[index] = "";
// Parse the expression
for (let c = 0; c < chars.length; c++) {
if (isNaN(parseInt(chars[c])) && chars[c] !== "." && !oplast) {
op[index] = chars[c];
index++;
n[index] = "";
oplast = true;
} else {
n[index] += chars[c];
oplast = false;
}
}
// Calculate the expression
s = parseFloat(n[0]);
for (let o = 0; o < op.length; o++) {
const num = parseFloat(n[o + 1]);
switch (op[o]) {
case "+":
s = s + num;
break;
case "-":
s = s - num;
break;
case "*":
s = s * num;
break;
case "/":
s = s / num;
break;
}
}
return s;
},
/**
* Generate random number between two numbers
*
* @return {Number} min - Enter a expected minimum.
* @return {Number} max - Enter a expected maximum.
* @return {Number} - A new random number
*/
t.getRandomFloat = function(min, max) {
return Math.random() * (max - min) + min;
},
/**
* Returns the degree from radian.
*
* @return {Number} rad - Value of radian.
* @return {Number}
* @usage:
angle = rad / ( Math.PI / 180 ) = rad * ( 180/Math.PI );
*/
t.getDegree = function(rad) {
return rad / Math.PI * 180;
},
/**
* Returns the radian degree .
*
* @return {Number} deg - Value of degree.
* @return {Number}
* @usage:
rad = Math.PI / 180 * 30 ;
*/
t.getRadian = function(deg) {
return deg * Math.PI / 180;
},
/**
* Convert three.js scene rotation to polar coordinates
*
* @return {Number} x - X coordinate value.
* @return {Number} y - Y coordinate value.
* @return {Number} z - Z coordinate value.
* @return {Json}
* @usage:
x = r * cos(θ)
y = r * sin(θ)
*/
t.getPolarCoord = function(x, y, z) {
const nx = Math.cos(x) * Math.cos(y) * z,
nz = Math.cos(x) * Math.sin(y) * z,
ny = Math.sin(x) * z;
return {
x: nx,
y: ny,
z: nz
};
},
//
t
})();
/**
* Get the CSS property
* @public
*
* @description This function can be used separately in HTML pages or custom JavaScript.
* @param {!Element} el - The Element for which to get the computed style. Using class name or ID to locate.
* @return {String|Object} - The value of property.
*/
$$.cssProperty = $$.cssProperty || ( () => {
function t() { }
return t.version = "0.0.1",
/**
* Get the -webkit-transition-duration property
*
* @param {Element} el - A DOM node containing one selector to match against.
* @return {Number} - Returns a pure number.
*/
t.getTransitionDuration = function( el ) {
if ( typeof el === typeof undefined ) {
return 0;
}
let style = window.getComputedStyle(el),
duration = style.webkitTransitionDuration,
delay = style.webkitTransitionDelay;
if ( typeof duration != typeof undefined ) {
// fix miliseconds vs seconds
duration = (duration.indexOf("ms")>-1) ? parseFloat(duration) : parseFloat(duration)*1000;
delay = (delay.indexOf("ms")>-1) ? parseFloat(delay) : parseFloat(delay)*1000;
return duration;
} else {
return 0;
}
},
/**
* Get an object's absolute position on the page
*
* @param {Element} el - A DOM node containing one selector to match against.
* @return {Json} - An object containing the properties top and left.
*/
t.getAbsoluteCoordinates = function( el ) {
let windowWidth = window.innerWidth,
leftPos = null,
topPos = null;
if ( ! document.getElementsByTagName( 'body' )[0].className.match(/rtl/) ) {
leftPos = ( el.offsetLeft == 0 ) ? el.parentElement.offsetLeft : el.offsetLeft;
topPos = ( el.offsetTop == 0 ) ? el.parentElement.offsetTop : el.offsetTop;
} else {
// width and height in pixels, including padding and border
// Corresponds to outerWidth(), outerHeight()
leftPos = ( el.offsetLeft == 0 ) ? ( windowWidth - ( el.parentElement.offsetLeft + el.parentElement.offsetWidth ) ) : ( windowWidth - ( el.offsetLeft + el.offsetWidth ) );
topPos = ( el.offsetTop == 0 ) ? ( windowWidth - ( el.parentElement.offsetTop + el.parentElement.offsetHeight ) ) : ( windowWidth - ( el.offsetTop + el.offsetHeight ) );
}
return {
'left': leftPos,
'top': topPos
};
},
//
t
})();
/**
* Perform an asynchronous HTTP (Ajax) request.
* @public
*
* @param {Json} props - The attribute and value to be set, the format is JSON
* @return {Void}
*/
$$.ajax = function(props) {
if ( ! isJSON(props) ) return false;
const self = this;
const _url = typeof(props.url) !== 'undefined' ? props.url : '';
const _method = typeof(props.method) !== 'undefined' ? props.method : 'POST';
const _loadedFn = typeof(props.complete) !== 'undefined' ? props.complete : null;
if ( _url != '' ) {
let config = {
mode: 'cors',
method: _method
};
fetch(_url, config )
.then(response => response.json())
.then(result => {
//console.log('Success:', result);
if (_loadedFn && (typeof _loadedFn == 'function')) {
_loadedFn.call(self,result);
}
})
.catch(error => {
console.error('Error:', error);
});
}
};
/**
* Remove all spaces in the string
* @public
*
* @param {String} s - Any string.
* @param {Boolean} isGlobal - If the value is `true`, remove all spaces including the middle
* @return {String} - A new string with no spaces
*/
$$.trim = function(s, isGlobal = false) {
if (typeof(s) === 'string') {
let result;
result = s.replace(/(^\s+)|(\s+$)/g, "");
if ( isGlobal === true ) {
result = result.replace(/\s/g, "");
}
return result;
} else {
return s;
}
};
/**
* Capitalize the first letter of all words in a string
* @public
*
* @param {String} s - Any string.
* @return {String} - A new string.
*/
$$.lastUrlParamFormat = function(s) {
s = s || '';
if ( s.length > 0 ) {
s = s.replace(/\-/g, ' ' ).replace(/\_/g, ' ' );
const pieces = s.split(" ");
for ( let i = 0; i < pieces.length; i++ )
{
const j = pieces[i].charAt(0).toUpperCase();
pieces[i] = j + pieces[i].substr(1);
}
return pieces.join(" ");
} else {
return s;
}
};
/**
* Convert HTML Element's `Style` Attribute to JSON
* @public
*
* @param {String} str - The content of the style attribute in the HTML element, usually a string
* @return {Json} - An HTML element to a JSON object
*/
$$.styleFormat = function(s) {
s = s || '';
if ( s.length > 0 ) {
const styles = s.split(';'),
json = {};
let i = styles.length,
style,
k,
v;
while (i--) {
style = styles[i].split(':');
k = toCamelCase($$.trim(style[0]));
v = $$.trim(style[1]);
if (k.length > 0 && v.length > 0)
{
json[k] = v;
}
}
return json;
} else {
return '';
}
};
/**
* Remove first, last or both symbols
* @public
*
* @param {String} str - Any string.
* @param {String} symbol - The target string to remove.
* @param {Number} type - Type of all or not. if `0`, is all.
* @return {String} - An new string.
*/
$$.removeFirstLastStr = function( str, symbol = ',', type = 0 ) {
if (typeof(str) === 'string') {
if ( type == 0 ) {
const flRegExp = new RegExp( '^\\'+symbol+'|\\'+symbol+'$' , 'g' );
return str.replace( flRegExp, '' );
} else {
const flRegExp = new RegExp( ''+symbol+'\s*$' , 'g' );
return str.replace( flRegExp, '' );
}
} else {
return str;
}
};
/**
* Convert a string to slug.
* @public
*
* @param {String} str - Any string.
* @return {String} - A new string.
*/
$$.toSlug = function( str ) {
if ( typeof( str ) == 'string' && str.length > 0 ) {
return str
.toString()
.replace(/[^\w\s\-!¥【】\u4e00-\u9eff]/gi, '')
.replace(/\s/g, '-')
.replace(/(\-){2,}/g, '-')
.replace(/\-\s*$/, '' )
.toLowerCase();
} else {
return str;
}
};
/**
* Throttle
* @public
*
* @param {Function} fn - A function to be executed within the time limit.
* @param {Number} limit - Waiting time.
* @return {Function} - Returns a new function.
*/
$$.throttle = function( fn, limit = 300 ) {
let waiting = false;
return function () {
if (!waiting) {
fn.apply(this, arguments);
waiting = true;
setTimeout(function () {
waiting = false;
}, limit);
}
}
};
/**
* Debounce
* @public
*
* @param {Function} fn - A function to be executed within the time limit.
* @param {Number} limit - Waiting time.
* @return {Function} - Returns a new function.
*/
$$.debounce = function( fn, limit = 300 ) {
let timer;
return function() {
//Every time this returned function is called, the timer is cleared to ensure that fn is not executed
clearTimeout(timer);
// When the returned function is called for the last time (that is the user stops a continuous operation)
// Execute fn after another delay milliseconds
timer = setTimeout(function() {
fn.apply(this, arguments);
}, limit);
}
};
/**
* Create a deep copy of the set of matched elements.
* @public
*
* @param {Object|Element} obj - The array, JSON or HTML element to be copied.
* @return {Object|Element}
*/
$$.deepClone = function(obj) {
if ( obj.nodeType === 1 ) {
return obj.cloneNode(true);
} else {
let objClone = Array.isArray(obj)?[]:{};
if(obj && typeof obj==="object"){
for(key in obj){
if(obj.hasOwnProperty(key)){
//Determine whether the ojb child element is an object, if it is, copy it recursively
if(obj[key] && typeof obj[key] ==="object"){
objClone[key] = deepClone(obj[key]);
}else{
//If not, simply copy
objClone[key] = obj[key];
}
}
}
}
return objClone;
}
};
/**
* HTML entities encode.
* @public
*
* @param {string} str - Input text.
* @return {string} - Filtered text.
*/
$$.htmlEncode = function(str) {
let res = '';
if (typeof (document) === 'undefined') {
res = str.replace(/[\u00A0-\u9999<>\&]/g, function(i) {
return '&#'+i.charCodeAt(0)+';';
});
} else {
const div = document.createElement('div');
//Creates a new Text node. This method can be used to escape HTML characters.
div.appendChild(document.createTextNode(str));
res = div.innerHTML;
}
//Convert single and double quotes
res = res.replace(/"/g, '&quot;').replace(/'/g, '&#39;');
return res;
};
/**
* HTML entities decode
* @public
*
* @param {string} str - Input text.
* @return {string} - Filtered text.
*/
$$.htmlDecode = function(str) {
let res = '';
if (typeof (document) === 'undefined') {
const entities = [
['amp', '&'],
['apos', '\''],
['#x27', '\''],
['#x2F', '/'],
['#39', '\''],
['#47', '/'],
['lt', '<'],
['gt', '>'],
['nbsp', ' '],
['quot', '"'],
['#60', '<'],
['#62', '>']
];
for (let i = 0, max = entities.length; i < max; i++) {
str = str.replace(new RegExp('&'+entities[i][0]+';', 'g'), entities[i][1]);
}
res = str;
} else {
const txt = document.createElement('textarea');
txt.innerHTML = str;
res = txt.value;
}
return res;
};
/**
* Set a default JSON format configuration
* @public
*
* @param {Json} props - Set some default keys and values.
* @param {Json} options - A JSON variable passed in from outside, including key and value.
* @return {Json} - Merge the new and old values.
*/
$$.setDefaultOptions = function(props, options) {
if ( typeof options === typeof undefined || options === null || options === false ) options = {};
//Set a default configuration
if ( isJSON(props) ) {
const defaultConfigValues = Object.values(props);
Object.keys(props).forEach(function(prop,index) {
// Well-formed string type
Object.keys(options).forEach(function(prop2, index2) {
if ( prop2 === prop ) {
let _v = options[prop2];
if ( _v == 'true' ) _v = true;
if ( _v == 'false' ) _v = false;
if ( isValidNumeric(_v) ) _v = parseFloat(_v);
if ( isJSON(_v) ) _v = Object.prototype.toString.call(_v) === '[object Object]' ? _v : JSON.parse(_v);
options[prop2] = _v;
}
});
//
if ( typeof options[prop] === typeof undefined || options[prop] === null ) options[prop] = defaultConfigValues[index];
});
}
return options;
};
/* ---------------- API methods ----------------- */
/**
* Returns the length of the node
*
* @return {Number}
*/
Constructor.prototype.len = function () {
let { elems } = this;
return elems.length;
};
/**
* Returns a native DOM element
*
* @param {Number} index - A number for index.
* @return {Element}
*/
Constructor.prototype.get = function (index) {
let { elems } = this;
if ( index === -1 ) {
//get all elements
return elems;
} else {
return elems[index];
}
};
/**
* Iterate over an object, executing a function for each matched element.
*
* @param {Function} fn - A function to execute for each matched element.
* @return {NodeList}
*/
Constructor.prototype.each = function (fn) {
if (fn && (typeof fn == 'function')) {
let { elems } = this;
elems = Array.prototype.slice.call(elems);
elems.forEach(function (element, index, array) {
//If the ID does not exist, itemDomsStr cannot be obtained
// Very critical, other possible methods of manipulation dom will use id
if ( (element.id !== undefined && element.id.length === 0) || element.id === undefined ) {
element.id = 'eachitem-' + $$.GUID.create();
}
//!import: The returned HTML element must be current,
//otherwise all HTML elements under document may be queried
const itemDomsStr = '#' + element.id;
fn.call(element, index, itemDomsStr, array);
});
}
};
/**
* Code included inside the code will run once the entire page (all DOM) is ready.
*
* @param {Function} fn - A function to execute after the DOM is ready.
* @return {Void}
*/
Constructor.prototype.ready = function (fn) {
if (document.readyState != 'loading') {
fn();
} else if (document.addEventListener) {
document.addEventListener('DOMContentLoaded', fn);
} else {
document.attachEvent('onreadystatechange', function () {
if (document.readyState != 'loading') fn();
});
}
};
/**
* Detect when images and videos have been loaded.
*
* @param {Json} props - Contains three event functions before loading, loading, and loading completed.
* @return {Void}
*/
Constructor.prototype.loader = function (props) {
this.each(function () {
const self = this;
const sources = [];
let loadingFn = null,
progressFn = null,
loadedFn = null,
imagesSelector = 'body img',
videosSelector = 'body video';
if (isJSON(props)) {
loadingFn = props.startEvent;
progressFn = props.progressEvent;
loadedFn = props.endEvent;
imagesSelector = props.imagesSelector;
videosSelector = props.videosSelector;
}
//count all images on a page
if (typeof (document.images) !== 'undefined' && document.images.length == 0) {
const imgPlaceholder = document.createElement("div");
imgPlaceholder.innerHTML = '<img src="" alt="" style="display:none">';
// insert liLast at the end of <>
document.body.append(imgPlaceholder);
}
//loading
if (loadingFn && (typeof loadingFn == 'function')) {
loadingFn();
}
//Push all images from page
const imgs = document.querySelectorAll(imagesSelector);
for (let i = 0; i < imgs.length; i++) {
sources.push(
{
"url": imgs[i].src,
"type": 'img'
}
);
}
//Push all videos from page
const videos = document.querySelectorAll(videosSelector);
for (let i = 0; i < videos.length; i++) {
const _sources = videos[i].getElementsByTagName('source');
sources.push(
{
"url": _sources.length > 0 ? _sources[0].src : videos[i].src,
"type": 'video'
}
);
}
//Execute after all images and videos have loaded
let per = 0;
let perInit = 1;
if (sources.length == 0) {
per = 100;
}
const loadResources = function () {
let promises = [];
for (let i = 0; i < sources.length; i++) {
if (sources[i].type == 'img') {
///////////
// IMAGE //
///////////
promises.push(
new Promise(function (resolve, reject) {
const img = document.createElement('img');
img.crossOrigin = 'anonymous';
img.src = sources[i].url;
img.onload = function (image) {
//Compatible with safari and firefox
if (typeof image.path === typeof undefined) {
return resolve(image.target.currentSrc);
} else {
return resolve(image.path[0].currentSrc);
}
};
}).then(textureLoaded)
);
} else {
///////////
// VIDEO //
///////////
promises.push(
new Promise(function (resolve, reject) {
const video = document.createElement('video');
video.addEventListener('loadedmetadata', function (video) {
//Compatible with safari and firefox
if (typeof video.path === typeof undefined) {
return resolve(video.target.currentSrc);
} else {
return resolve(video.path[0].currentSrc);
}
}, false);
video.src = sources[i].url;
}).then(textureLoaded)
);
}
}//end for
return Promise.all(promises);
};
const textureLoaded = function (url) {
//progress number
per = parseInt(100 * (perInit / sources.length));
if (isNaN(per)) per = 100;
//Call back progress
/** console.log( 'progress: ' + per + '%' ); */
if (progressFn && (typeof progressFn == 'function')) {
progressFn.call(self, per);
}
perInit++;
return per;
};
//and videos loaded
//Must be placed behind the loadResources()
loadResources().then(function (images) {
if (loadedFn && (typeof loadedFn == 'function')) {
loadedFn();
}
});
});
return this;
};
/**
* Puts data inside an element at the last index (Vanilla JS also has this method)
*
* @param {Element|String} el - An element or string to be parsed as HTML or XML and inserted into the tree.
* @return {Void}
*/
Constructor.prototype.append = function (el) {
this.each(function () {
if (typeof (el) === 'string') {
// Just inside the element, after its last child.
if (document.createElement("div").insertAdjacentHTML) {
this.insertAdjacentHTML("beforeend", el);
}
} else {
const html = (typeof (el) === 'string') ? el : el.outerHTML;
this.innerHTML += html;
}
});
return this;
};
/**
* Puts the prepending element at the first index. (Vanilla JS also has this method)
*
* @param {Element|String} el - An element or string to be parsed as HTML or XML and inserted into the tree.
* @return {Void}
*/
Constructor.prototype.prepend = function (el) {
this.each(function () {
if (typeof (el) === 'string') {
// Just inside the element, before its first child.
if (document.createElement("div").insertAdjacentHTML) {
this.insertAdjacentHTML("afterbegin", el);
}
} else {
const html = (typeof (el) === 'string') ? el : el.outerHTML;
this.innerHTML = html + this.innerHTML;
}
});
return this;
};
/**
* Before the element itself.
*
* @param {String} el - The string to be parsed as HTML or XML and inserted into the tree.
* @return {Void}
*/
Constructor.prototype.before = function (el) {
this.each(function () {
if (typeof (el) === 'string') {
// Before the element itself.
if (document.createElement("div").insertAdjacentHTML) {
this.insertAdjacentHTML("beforebegin", el);
}
} else {
const html = (typeof (el) === 'string') ? el : el.outerHTML;
this.insertAdjacentHTML("beforebegin", html);
}
});
return this;
};
/**
* After the element itself.
*
* @param {String} el - The string to be parsed as HTML or XML and inserted into the tree.
* @return {Void}
*/
Constructor.prototype.after = function (el) {
this.each(function () {
if (typeof (el) === 'string') {
// After the element itself.
if (document.createElement("div").insertAdjacentHTML) {
this.insertAdjacentHTML("afterend", el);
}
} else {
const html = (typeof (el) === 'string') ? el : el.outerHTML;
this.insertAdjacentHTML("afterend", html);
}
});
return this;
};
/**
* Insert an element as the first child node of another
*
* @param {Element} el - A DOM node containing one selector to match against.
*/
Constructor.prototype.prependTo = function (el) {
this.each(function () {
if (this.firstChild) {
this.insertBefore(el, this.firstChild);
}
});
return this;
};
/**
* Insert an element to the end of the target
*
* @param {Element} el - A DOM node containing one selector to match against.
*/
Constructor.prototype.appendTo = function (el) {
this.each(function () {
this.appendChild(el);
});
return this;
};
/**
* Wrap an HTML structure around the content of each element in the set of matched elements.
*
* @param {String} el - An HTML snippet.
* @return {Void}
*/
Constructor.prototype.wrapInner = function (el) {
this.each(function () {
//get old value
const val = this.innerHTML;
//empty default value
this.innerHTML = '';
//The DOMParser() method is awesome, but the parseFromString() method stops at IE10.
const support = (function () {
if (!window.DOMParser) return false;
const parser = new DOMParser();
try {
parser.parseFromString('x', 'text/html');
} catch (err) {
return false;
}
return true;
})();
//Convert a template string into HTML DOM nodes
const stringToHTML = function stringToHTML(str) {
// If DOMParser is supported, use it
if (support) {
const parser = new DOMParser();
const doc = parser.parseFromString(str, 'text/html');
const wrapperEl = doc.body;
return wrapperEl.children[0];
}
// Otherwise, fallback to old-school method
const dom = document.createElement('div');
dom.innerHTML = str;
const wrapperEl = dom;
return wrapperEl.children[0];
};
if (typeof el === 'string') {
const div = this.appendChild(stringToHTML(el));
div.innerHTML = val;
while (this.firstChild !== div) {
div.appendChild(this.firstChild);
}
}
});
return this;
};
/**
* Get the HTML contents of the first element in the set of matched elements or set the HTML contents of every matched element.
*
* @param {?String} el - A string of HTML to set as the content of each matched element.
* @return {Void|String} - The HTML content to set
*/
Constructor.prototype.html = function (el) {
const rootObject = this;
let res = null;
this.each(function () {
if (el === undefined) {
res = this.innerHTML;
} else {
this.innerHTML = el;
res = rootObject;
}
});
return res;
};
/**
* Get the combined text contents of each element in the set of matched elements, including their descendants, or set the text contents of the matched elements.
*
* @param {?String} el - The text to set as the content of each matched element.
* @return {Void|String} - The HTML content to set
*/
Constructor.prototype.text = function (el) {
const rootObject = this;
let res = null;
this.each(function () {
if (el === undefined) {
//Remove HTML Tags
let htmlstr = this.innerHTML;
htmlstr = htmlstr.replace(/(<([^>]+)>)/ig, '');
res = htmlstr;
} else {
this.innerHTML = $$.htmlEncode(el);
res = rootObject;
}
});
return res;
};
/**
* Create a deep copy of the set of matched elements.
*
* @return {Array} - Contains only a collection of HTML elements.
* Returns a duplicate of the node on which this method was called.
*/
Constructor.prototype.clone = function () {
let res = this;
this.each(function () {
res = this.cloneNode(true);
});
return res;
};
/**
* Adds the specified class(es) to each element in the set of matched elements.
*
* @param {String} c - One or more space-separated classes to be added to the class attribute of each matched element.
* @return {Void}
*/
Constructor.prototype.addClass = function (c) {
this.each(function () {
if (! /^\S+$/g.test(c)) {
// It has only whitespace
const classArray = c.split(' ');
let className;
// Loop through the array of classes to add one class at a time
for (let j = 0; j < classArray.length; j++) {
className = classArray[j];
this.classList.add(className);
}
} else {
this.classList.add(c);
}
});
return this;
};
/**
* Remove a single class, multiple classes, or all classes from each element in the set of matched elements.
*
* @param {String} c - One or more space-separated classes to be removed from the class attribute of each matched element.
* @return {Void}
*/
Constructor.prototype.removeClass = function (c) {
this.each(function () {
if (! /^\S+$/g.test(c)) {
// It has only whitespace
const classArray = c.split(' ');
let className;
// Loop through the array of classes to add one class at a time
for (let j = 0; j < classArray.length; j++) {
className = classArray[j];
this.classList.remove(className);
}
} else {
this.classList.remove(c);
}
});
return this;
};
/**
* Add or remove one or more classes from each element in the set of matched elements,
* depending on either the class's presence or the value of the state argument.
*
* @param {String} c - One or more classes (separated by spaces) to be toggled for each element in the matched set.
* @return {Void}
*/
Constructor.prototype.toggleClass = function (c) {
this.each(function () {
if (! /^\S+$/g.test(c)) {
// It has only whitespace
const classArray = c.split(' ');
let className;
// Loop through the array of classes to add one class at a time
for (let j = 0; j < classArray.length; j++) {
className = classArray[j];
if (this.classList.contains(className)) {
this.classList.remove(className);
} else {
this.classList.add(className);
}
}
} else {
if (this.classList.contains(c)) {
this.classList.remove(c);
} else {
this.classList.add(c);
}
}
});
return this;
};
/**
* Set the style properties of elements:
*
* @param {Json} props - The attribute and value to be set, the format is JSON
* @return {Void}
*/
Constructor.prototype.css = function (props, value) {
this.each(function () {
const self = this;
if (isJSON(props)) {
//the json is ok
Object.keys(props).forEach(function (prop) {
self.style[prop] = props[prop];
});
} else {
if (value !== undefined) {
self.style[props] = value;
}
}
});
return this;
};
/**
* Get the descendants of each element in the current set of matched elements
*
* @param {String} s - A string containing a selector expression to match elements against.
* @return {Array} - The collection of elements
*/
Constructor.prototype.find = function (s) {
let res = [];
this.each(function () {
// The symbol ">" is not allowed at the beginning of the find() method.
if (/(^\s*|,\s*)>/.test(s)) {
let removeId;
if (!this.id) {
this.id = 'ID_' + new Date().getTime();
removeId = true;
}
s = s.replace(/(^\s*|,\s*)>/g, '$1#' + this.id + ' >');
[].slice.call(document.querySelectorAll(s)).forEach(function(element){
res.push(element);
});
if (removeId) {
this.id = null;
}
} else {
[].slice.call(this.querySelectorAll(s)).forEach(function(element){
res.push(element);
});
}
});
return $$(res);
};
/**
* Reduce the set of matched elements to the one at the specified index.
*
* @param {Number} index - A number for index.
* @return {Array} - Contains only a collection of HTML elements.
*/
Constructor.prototype.eq = function (index) {
let res = [];
let { elems } = this;
elems = Array.prototype.slice.call(elems);
elems.forEach(function (element, listIndex) {
if ( index === listIndex ) res.push(element);
});
return $$(res);
};
/**
* Returns the Element immediately prior to the specified one in its parent's children list,
* or null if the specified element is the first one in the list.
*
* @param {String} s - A string containing a selector expression to match elements against.
* @return {Array} - Contains only a collection of HTML elements.
*/
Constructor.prototype.prev = function (s) {
let res = [];
this.each(function () {
const el = this.previousElementSibling;
if (s === undefined) {
if (el !== null) res.push(el);
} else {
if (
el !== null &&
//Determine whether the ID, class and HTML nodes match
(el.nodeName.toLowerCase() === s || el.classList.contains(s.replace(/\./g, '')) || '#' + el.id === s)
) {
res.push(el);
}
}
});
return $$(res);
};
/**
* Returns the element immediately following the specified one in its parent's children list,
* or null if the specified element is the last one in the list.
*
* @param {String} s - A string containing a selector expression to match elements against.
* @return {Array} - Contains only a collection of HTML elements.
*/
Constructor.prototype.next = function (s) {
let res = [];
this.each(function () {
const el = this.nextElementSibling;
if (s === undefined) {
if (el !== null) res.push(el);
} else {
if (
el !== null &&
//Determine whether the ID, class and HTML nodes match
(el.nodeName.toLowerCase() === s || el.classList.contains(s.replace(/\./g, '')) || '#' + el.id === s)
) {
res.push(el);
}
}
});
return $$(res);
};
/**
* Returns the DOM node's parent Element, or null if the node either has no parent,
* or its parent isn't a DOM Element.
*
* @param {String} s - A string containing a selector expression to match elements against.
* @return {Array} - Contains only a collection of HTML elements.
*/
Constructor.prototype.parent = function (s) {
let res = [];
this.each(function () {
const el = this.parentElement;
if (s === undefined) {
if (el !== null) res.push(el);
} else {
if (
el !== null &&
//Determine whether the ID, class and HTML nodes match
(el.nodeName.toLowerCase() === s || el.classList.contains(s.replace(/\./g, '')) || '#' + el.id === s)
) {
res.push(el);
}
}
});
return $$(res);
};
/**
* Get the ancestors of each element in the current set of matched elements.
*
* @param {String} s - A string containing a selector expression to match elements against.
* @return {Array} - Contains only a collection of HTML elements.
*/
Constructor.prototype.parents = function (s) {
let res = [];
this.each(function () {
let parentSelector = document.querySelector(s);
// If no parentSelector defined will bubble up all the way to *document*
if (parentSelector === undefined) {
parentSelector = document;
}
let _parent = this.parentNode;
while (_parent !== parentSelector) {
const _currentNode = _parent;
//Determine whether the ID, class and HTML nodes match
if (s !== undefined) {
if (
_currentNode.nodeName.toLowerCase() === s ||
_currentNode.classList.contains(s.replace(/\./g, '')) ||
'#' + _currentNode.id === s
) {
res.push(_currentNode);
}
} else {
res.push(_currentNode);
}
//
_parent = _currentNode.parentNode;
}
// Push that parentSelector you wanted to stop at
if (parentSelector !== null) res.push(parentSelector);
});
return $$(res);
};
/**
* Returns a live HTMLCollection which contains all of the child elements
* of the node upon which it was called.
*
* @param {Element} s - The selector that needs to be filtered. A DOMstring containing
* one selector to match against.
* @return {Array} - The collection of elements
*/
Constructor.prototype.children = function (s) {
let res = [];
this.each(function () {
const self = this;
const childrenList = self.children;
if (childrenList) {
for (let i = 0; i < childrenList.length; i++) {
const _currentNode = childrenList[i];
//Determine whether the ID, class and HTML nodes match
if (s !== undefined) {
if (
_currentNode.nodeName.toLowerCase() === s ||
_currentNode.classList.contains(s.replace(/\./g, '')) ||
'#' + _currentNode.id === s
) {
res.push(_currentNode);
}
} else {
res.push(_currentNode);
}
}
}
});
return $$(res);
};
/**
* Reduce the set of matched elements to those that match the selector or pass the function's test.
*
* @param {String} s - A string containing a selector expression to match elements against.
* @return {Array} - The collection of elements
*/
Constructor.prototype.filter = function (s) {
let res = [];
this.each(function () {
if (s !== undefined) {
if (this.classList.contains(s.replace(/\./g, ''))) {
res.push(this);
}
}
});
return $$(res);
};
/**
* Remove elements from the set of matched elements.
*
* @param {String} s - A string containing a selector expression to match elements against.
* @return {Array} - The collection of elements
*/
Constructor.prototype.not = function (s) {
let res = [];
this.each(function () {
if (s !== undefined) {
if (!this.classList.contains(s.replace(/\./g, ''))) {
res.push(this);
}
}
});
return $$(res);
};
/**
* Get the siblings of each element in the set of matched elements
*
* @param {String} s - A string containing a selector expression to match elements against.
* @return {Array} - The collection of elements
*/
Constructor.prototype.siblings = function (s) {
let res = [];
this.each(function () {
const self = this;
// if no parent, return no sibling
if ( self.parentNode ) {
// first child of the parent node
let sibling = self.parentNode.firstChild;
// collecting siblings
while (sibling) {
if (sibling.nodeType === 1 && sibling !== self) {
//Determine whether the ID, class and HTML nodes match
if (s !== undefined) {
if (
sibling.nodeName.toLowerCase() === s ||
sibling.classList.contains(s.replace(/\./g, '')) ||
'#' + sibling.id === s
) {
res.push(sibling);
}
} else {
res.push(sibling);
}
}
//
sibling = sibling.nextSibling;
}
}
});
return $$(res);
};
/**
* Reduce the set of matched elements to the first in the set.
*
* @return {Array} - Contains only a collection of HTML elements.
*/
Constructor.prototype.first = function () {
let res = [];
let { elems } = this;
elems = Array.prototype.slice.call(elems);
elems.forEach(function (element, listIndex) {
if ( 0 === listIndex ) res.push(element);
});
return $$(res);
};
/**
* Reduce the set of matched elements to the last in the set.
*
* @return {Array} - Contains only a collection of HTML elements.
*/
Constructor.prototype.last = function () {
let res = [];
let { elems, storeSelector } = this;
let max = storeSelector.elems.length;
elems = Array.prototype.slice.call(elems);
elems.forEach(function (element, listIndex) {
if ( max-1 === listIndex ) res.push(element);
});
return $$(res);
};
/**
* Traverses the Element and its parents (heading toward the document root)
* until it finds a node that matches the provided selector string.
*
* @param {String} s - A string containing a selector expression to match elements against.
* @return {Array} - Contains only a collection of HTML elements.
*/
Constructor.prototype.closest = function (s) {
let res = [];
this.each(function () {
//using recursivity
const el = this.closest(s);
//Must judge the result of closest()
if (el !== null) {
res.push(el);
}
});
return $$(res);
};
/**
* Set one or more attributes for the set of matched elements.
*
* @param {String} a - The name of the attribute to set.
* @param {String} v - A value to set for the attribute.
* @return {Void|String} - Get the value of an attribute for the first element in the set of matched elements.
*/
Constructor.prototype.attr = function (a, v) {
const rootObject = this;
let res = null;
this.each(function () {
a = a || '';
if (v === undefined) {
let curVal = this.getAttribute(a);
// Non-existent attributes
res = curVal === null ? null : curVal;
} else {
this.setAttribute(a, v);
res = rootObject;
}
});
return res;
};
/**
* Store arbitrary data associated with the matched elements.
*
* @param {String} a - A string naming the piece of data to set
* @param {String} v - The new data value.
* @return {Void|String} - Return arbitrary data associated with the first element as set by data() or by an HTML5 data-* attribute.
*/
Constructor.prototype.data = function (a, v) {
const rootObject = this;
let res = null;
this.each(function () {
a = a || '';
const _s = stringlineToHump(a);
if (v === undefined) {
let curVal = this.dataset[_s];
if (curVal == 'true') curVal = true;
if (curVal == 'false') curVal = false;
if (isValidNumeric(curVal)) curVal = parseFloat(curVal);
//check if Array or JSON format
if (isJSON(curVal)) {
if (Object.prototype.toString.call(curVal) === '[object Object]') {
curVal = curVal;
} else {
//If the curValult is an array, you need to determine whether it is the expected array
curVal = JSON.parse(curVal);
}
}
// Non-existent attributes
res = curVal === undefined ? null : curVal;
} else {
this.dataset[_s] = v;
res = rootObject;
}
});
return res;
};
/**
* Set one or more attributes for the set of matched Form elements.
*
* @param {String} a - The name of the attribute to set.
* @param {String} v - A value to set for the attribute.
* @return {Void|String} - Get the value of an attribute for the first element in the set of matched elements.
*/
Constructor.prototype.prop = function (a, v) {
const rootObject = this;
let res = null;
this.each(function () {
a = a || '';
if (v === undefined) {
let curVal = this[a];
if (curVal == 'true') curVal = true;
if (curVal == 'false') curVal = false;
if (isValidNumeric(curVal)) curVal = parseFloat(curVal);
//check if Array or JSON format
if (isJSON(curVal)) {
if (Object.prototype.toString.call(curVal) === '[object Object]') {
curVal = curVal;
} else {
//If the curValult is an array, you need to determine whether it is the expected array
curVal = JSON.parse(curVal);
}
}
// Non-existent attributes
res = curVal === undefined ? null : curVal;
} else {
this[a] = v;
res = rootObject;
}
});
return res;
};
/**
* Remove an attribute from each element in the set of matched elements.
*
* @param {String} a - A string naming the piece of data to delete.
* @return {Void}
*/
Constructor.prototype.removeAttr = function (a) {
this.each(function () {
a = a || '';
this.removeAttribute(a);
});
return this;
};
/**
* Remove a previously-stored piece of data.
*
* @param {String} a - A string naming the piece of data to delete.
* @return {Void}
*/
Constructor.prototype.removeData = function (a) {
this.each(function () {
a = a || '';
const _s = stringlineToHump(a);
delete this.dataset[_s];
});
return this;
};
/**
* Attach a handler to an event for the elements. The handler is executed at most once per element per event type.
*
* @param {String} eventType - One event types and optional namespaces, such as "click"
* @param {?String} selector - A selector string to filter the descendants of the selected elements that trigger the event.
* @param {Function} fn - A function to execute when the event is triggered.
* @return {Void}
*/
Constructor.prototype.one = function (eventType, selector, fn) {
this.each(function () {
$$(this).on(eventType, selector, fn, true);
});
return this;
};
/**
* Attach an event handler function for one or more events to the selected elements.
*
* @param {String} eventType - One event types and optional namespaces, such as "click"
* @param {?String} selector - A selector string to filter the descendants of the selected elements
* that trigger the event.
* @param {Function} fn - A function to execute when the event is triggered.
* @param {Boolean} once - A boolean value indicating that the listener should be invoked at most
* once after being added. If true, the listener would be automatically
* removed when invoked.
* @return {Void}
*/
Constructor.prototype.on = function (eventType, selector, fn, once) {
this.each(function () {
if (typeof (once) === 'undefined') once = false;
let _curFun = null;
if (typeof (fn) !== 'function') {
fn = selector;
selector = null;
}
if (!this.myListeners) {
this.myListeners = [];
};
if (selector) {
//if string
_curFun = function (evt) {
[].slice.call(this.querySelectorAll(selector)).forEach(function (el) {
if (el === evt.target) {
fn.call(el, evt);
} else if (isChild(evt.target, el)) {
fn.call(el, evt);
}
});
};
this.myListeners.push({
eType: eventType,
callBack: _curFun,
function: fn,
selector: selector
});
} else {
//if HTML element
_curFun = function (evt) {
fn.call(this, evt);
};
this.myListeners.push({
eType: eventType,
callBack: _curFun,
function: fn,
selector: selector
});
}
if (once) {
this.addEventListener(eventType, _curFun, { once: true });
} else {
this.addEventListener(eventType, _curFun);
}
});
return this;
};
/**
* Remove an event handler.
*
* @param {?String} eventType - One event types and optional namespaces, such as "click"
* @param {?String|Function} curSelector - A selector string or function to filter the descendants of the selected elements that trigger the event.
* @return {Void}
*/
Constructor.prototype.off = function (eventType, curSelector) {
this.each(function () {
if (this.myListeners) {
for (let i = 0; i < this.myListeners.length; i++) {
if (typeof (curSelector) !== 'undefined') {
if (typeof curSelector === 'function') {
//is function
if (curSelector === this.myListeners[i].function) this.removeEventListener(this.myListeners[i].eType, this.myListeners[i].callBack);
} else {
//is string
if (curSelector === this.myListeners[i].selector) this.removeEventListener(this.myListeners[i].eType, this.myListeners[i].callBack);
}
} else {
this.removeEventListener(this.myListeners[i].eType, this.myListeners[i].callBack);
}
};
delete this.myListeners;
}
});
return this;
};
/**
* Get the current coordinates of the first element in the set of matched elements, relative to the document.
*
* @return {Json} - An object containing the properties top and left.
*/
Constructor.prototype.offset = function () {
let res = { top: 0, left: 0 };
this.each(function () {
const box = this.getBoundingClientRect();
let top = 0,
left = 0;
//Include scrollbar and border
top = box.top + window.pageYOffset - document.documentElement.clientTop;
left = box.left + window.pageXOffset - document.documentElement.clientLeft;
res = { top: top, left: left };
});
return res;
};
/**
* Get the current coordinates of the first element in the set of matched elements, relative to the offset parent.
*
* @return {Json} - An object containing the properties top and left.
*/
Constructor.prototype.position = function () {
let res = { top: 0, left: 0 };
this.each(function () {
let top = this.offsetTop ? this.offsetTop : 0,
left = this.offsetLeft ? this.offsetLeft : 0;
res = { top: top, left: left };
});
return res;
};
/**
* Get the number of pixels that an element's content is scrolled vertically.
*
* @return {Number} - Returns a pure number calculated.
*/
Constructor.prototype.scrollTop = function () {
let res = 0;
this.each(function () {
const supportPageOffset = window.pageXOffset !== undefined;
const isCSS1Compat = ((document.compatMode || "") === "CSS1Compat");
const scrollTop = supportPageOffset ? window.pageYOffset : isCSS1Compat ? this.scrollTop : document.body.scrollTop;
res = scrollTop;
});
return res;
};
/**
* Get the number of pixels that an element's content is scrolled from its left edge.
*
@return {Number} - Returns a pure number calculated.
*/
Constructor.prototype.scrollLeft = function (val) {
let res = 0;
this.each(function () {
const supportPageOffset = window.pageXOffset !== undefined;
const isCSS1Compat = ((document.compatMode || "") === "CSS1Compat");
const scrollLeft = supportPageOffset ? window.pageXOffset : isCSS1Compat ? this.scrollLeft : document.body.scrollLeft;
res = scrollLeft;
});
return res;
};
/**
* Get or set the current computed width for elenments
*
* @param {?String|?Number} val - An integer representing the number of pixels, or
* an integer along with an optional unit of measure appended (as a string).
* @return {Void|Number} - Get the current computed width for the first element in the set of matched elements.
*/
Constructor.prototype.width = function (val) {
const rootObject = this;
let res = 0;
this.each(function () {
const self = this;
if (typeof (val) !== 'undefined') {
self.style.width = !isNaN(val) ? val + 'px' : val;
res = rootObject;
} else {
res = getStyle(self, 'width');
}
});
return res;
};
/**
* Get or set the current computed height for elenments
*
* @param {?String|?Number} val - An integer representing the number of pixels, or
* an integer along with an optional unit of measure appended (as a string).
* @return {Void|Number} - Get the current computed height for the first element in the set of matched elements.
*/
Constructor.prototype.height = function (val) {
const rootObject = this;
let res = 0;
this.each(function () {
const self = this;
if (typeof (val) !== 'undefined') {
self.style.height = !isNaN(val) ? val + 'px' : val;
res = rootObject;
} else {
res = getStyle(self, 'height');
}
});
return res;
};
/**
* Get or set the current computed outer width for elenments (including padding, border, and optionally margin)
*
* @param {Boolean} includeMargin - A Boolean indicating whether to include the element's margin in the calculation.
* @return {Number} - Returns the width of the element, including left and right padding, border, and optionally margin, in pixels.
*/
Constructor.prototype.outerWidth = function (includeMargin) {
let res = 0;
this.each(function () {
const self = this;
const width_IncPaddingBorderScrollbar = self.offsetWidth;
const marginLeft = getStyle(self, 'marginLeft');
const marginRight = getStyle(self, 'marginRight');
const borderLeftWidth = getStyle(self, 'borderLeftWidth') || 0;
const borderRightWidth = getStyle(self, 'borderRightWidth') || 0;
let totalWidth = width_IncPaddingBorderScrollbar;
if (typeof (includeMargin) !== 'undefined') {
totalWidth = totalWidth + marginLeft + marginRight;
}
res = totalWidth;
});
return res;
};
/**
* Get or set the current computed outer height for elenments (including padding, border, and optionally margin)
*
* @param {Boolean} includeMargin - A Boolean indicating whether to include the element's margin in the calculation.
* @return {Number} - Returns the height of the element, including left and right padding, border, and optionally margin, in pixels.
*/
Constructor.prototype.outerHeight = function (includeMargin) {
let res = 0;
this.each(function () {
const self = this;
const height_IncPaddingBorderScrollbar = self.offsetHeight;
const marginTop = getStyle(self, 'marginTop');
const marginBottom = getStyle(self, 'marginBottom');
const borderTopWidth = getStyle(self, 'borderTopWidth') || 0;
const borderBottomWidth = getStyle(self, 'borderBottomWidth') || 0;
let totalHeight = height_IncPaddingBorderScrollbar;
if (typeof (includeMargin) !== 'undefined') {
totalHeight = totalHeight + marginTop + marginBottom;
}
res = totalHeight;
});
return res;
};
/**
* Remove the set of matched elements from the DOM.
*
* @return {Void}
*/
Constructor.prototype.remove = function () {
this.each(function () {
this.parentNode.removeChild(this);
});
return this;
};
/**
* Remove all child nodes of the set of matched elements from the DOM.
*
* @return {Void}
*/
Constructor.prototype.empty = function () {
this.each(function () {
while (this.firstChild) {
this.removeChild(this.firstChild);
}
});
return this;
};
/**
* Traverse all the attribute names and values in an HTML element
*
* @return {Array} - A new array containing the properties name and value.
*/
Constructor.prototype.allAttrs = function () {
let res = [];
this.each(function () {
const newArr = [];
Array.from(this.attributes)
.filter(obj => {
return obj.specified;
})
.map(obj => {
newArr[obj.nodeName] = obj.textContent;
});
res = newArr;
});
return res;
};
/**
* Determine whether any of the matched elements are assigned the given class.
*
* @param {String} v - The class name to search for.
* @return {Boolean} - Return true if the class is assigned to an element
*/
Constructor.prototype.hasClass = function (v) {
let res = false;
this.each(function () {
res = this.classList.contains(v) ? true : false;
});
return res;
};
/**
* Get or set the current value of the first element in the set of matched elements.
*
* @param {?String|?Number|?Array} v - Corresponding to the value of each matched element.
* @return {String} - Get the values of form elements.
*/
Constructor.prototype.val = function (v) {
const rootObject = this;
let res = null;
this.each(function () {
const self = this;
let controlType = '';
if (this.tagName == "INPUT" || this.tagName == "TEXTARTA") {
//not `radio`, `checkbox`
if (this.type != 'checkbox' && this.type != 'radio') {
controlType = 'input-textarea';
}
//`checkbox`
if (this.type == 'checkbox') {
controlType = 'checkbox';
}
//`radio`
if (this.type == 'radio') {
controlType = 'radio';
}
}
//`select`
if (this.tagName == "SELECT") {
controlType = 'select';
}
//set value
if (typeof (v) !== 'undefined') {
switch (controlType) {
case "input-textarea":
this.value = v;
res = rootObject;
break;
case "checkbox":
this.checked = v;
res = rootObject;
break;
case "radio":
if (self.value == v.toString()) {
self.checked = true;
}
res = rootObject;
break;
case "select":
this.value = v;
this.dispatchEvent(new Event('change'));
res = rootObject;
break;
default:
this.value = v;
res = rootObject;
}//end switch
} else {
switch (controlType) {
case "input-textarea":
res = this.value;
break;
case "checkbox":
res = this.checked ? 1 : 0;
break;
case "radio":
if (self.checked) {
// do whatever you want with the checked radio
res = self.value;
}
break;
case "select":
res = this.value;
break;
default:
res = this.value;
}//end switch
}
});
return res;
};
/**
* Display the matched elements.
*
* @return {Void}
*/
Constructor.prototype.show = function () {
this.each(function () {
// use inherit so that your CSS controls block/flex/inline-block etc
this.style.display = 'inherit';
});
return this;
};
/**
* Hide the matched elements.
*
* @return {Void}
*/
Constructor.prototype.hide = function () {
this.each(function () {
this.style.display = 'none';
});
return this;
};
/**
* Display the matched elements by fading them to opaque.
*
* @param {Number} speed - A string or number determining how long the animation will run.
* @param {Function} callback - A function to call once the animation is complete, called once per matched element.
* @return {Void}
*/
Constructor.prototype.fadeIn = function (speed, callback) {
this.each(function () {
const elem = this;
let opacity = 0;
if (!elem.style.opacity) {
elem.style.opacity = 0;
}
elem.style.display = "inherit";
const inInterval = setInterval(function () {
opacity += .02;
elem.style.opacity = opacity;
if (opacity >= 1) {
clearInterval(inInterval);
//do something after inInterval()
elem.style.removeProperty("opacity");
if (callback && (typeof callback == 'function')) {
callback();
}
}
}, speed / 50);
});
return this;
};
/**
* Hide the matched elements by fading them to transparent.
*
* @param {Number} speed - A string or number determining how long the animation will run.
* @param {Function} callback - A function to call once the animation is complete, called once per matched element.
* @return {Void}
*/
Constructor.prototype.fadeOut = function (speed, callback) {
this.each(function () {
const elem = this;
let opacity = 1;
if (!elem.style.opacity) {
elem.style.opacity = 1;
}
const outInterval = setInterval(function () {
opacity -= .02;
elem.style.opacity = opacity;
if (opacity <= 0) {
clearInterval(outInterval);
//do something after outInterval()
elem.style.opacity = 0;
elem.style.display = "none"; //adding dispaly property and equall to none
if (callback && (typeof callback == 'function')) {
callback();
}
}
}, speed / 50);
});
return this;
};
/**
* Serialize html form to JSON
*
* @param {Array} types - An array of field strings.
* @return {Array} - A collection of JSON arrays
*/
Constructor.prototype.serializeArray = function (types) {
let res = [];
this.each(function () {
const form = this;
const objects = [];
if (typeof form == 'object' && form.nodeName.toLowerCase() == "form") {
const fieldsTypes = typeof (types) === 'undefined' ? ['input', 'textarea', 'select', 'checkbox', 'progress', 'datalist'] : types;
fieldsTypes.map((item, index) => {
const fields = form.getElementsByTagName(item);
for (let i = 0; i < fields.length; i++) {
const _name = fields[i].getAttribute("name");
let _value = fields[i].value;
// if field is Array
if ( _name !== null && _name.match(/(\[.*?\])/gi) ) {
// foo[], foo[n]
const inputs = form.querySelectorAll("[name='"+_name+"']");
const _arrFieldValue = [];
for (let j = 0; j < inputs.length; j++) {
const _arrField = inputs[j];
_arrFieldValue.push(_arrField.value);
}
_value = _arrFieldValue;
}
//if checkbox or radio
if ( fields[i].type === 'radio' || fields[i].type === 'checkbox' ) {
if ( fields[i].checked === true ) {
objects[objects.length] = {
name: _name,
value: _value
};
}
} else {
objects[objects.length] = {
name: _name,
value: _value
};
}
}
});
}
// remove Duplicate objects from JSON Array
const clean = objects.filter((item, index, self) => index === self.findIndex((t) => (t.name === item.name)));
res = clean;
});
return res;
};
/**
* Search for a given element from among the matched elements.
*
* @return {Number} - The return value is an integer indicating the position of the
* first element within the object relative to its sibling elements.
*/
Constructor.prototype.index = function () {
let res = -1;
this.each(function () {
const self = this;
const children = self.parentNode.childNodes;
let num = 0;
for (let i = 0; i < children.length; i++) {
if (children[i] == self) res = num;
if (children[i].nodeType == 1) num++;
}
});
return res;
};
/**
* Bind an event in the HTML element
*
* @param {String} eventType - One event types and optional namespaces, such as "click"
* @return {Void}
*/
Constructor.prototype.trigger = function (eventType) {
this.each(function () {
const fire = function (elem, type) {
// create and dispatch the event
const event = new CustomEvent(type, {
detail: {
hazcheeseburger: true
}
});
elem.dispatchEvent(event);
};
document.addEventListener("plop", function () { }, false);
fire(this, eventType);
});
return this;
};
/**
* Find the Tallest or widest of all elements
*
* @return {Json} - An object containing the properties width and height.
*/
Constructor.prototype.maxDimension = function () {
const { storeSelector } = this;
let res = {
'height': 0,
'width': 0
};
let traverseIndex = 0;
let max = storeSelector.elems.length;
this.each(function () {
if (traverseIndex === max - 1) {
const elementHeights = Array.prototype.map.call(storeSelector.elems, function (el) {
return el.clientHeight;
});
const elementWidths = Array.prototype.map.call(storeSelector.elems, function (el) {
return el.clientWidth;
});
const maxHeight = Math.max.apply(null, elementHeights);
const maxWidth = Math.max.apply(null, elementWidths);
res = {
'height': maxHeight,
'width': maxWidth
};
}
// Traverse the counter of a selector, reset to 0 when calling `__(selector).XXX()`
traverseIndex++;
});
return res;
};
/**
* Create a new Animation, applies it to the element, then plays the animation
* @param {String} prop - The style property to be set.
* @param {Number} from - The initial offset of the object
* @param {Number} to - The end offset of the object
* @param {String} unit - Unit string, such as `px` and `%`, and so on.
* @param {Number} duration - The number of milliseconds each iteration of the animation takes to complete
* @param {String} easing - The rate of the animation's change over time. Accepts the pre-defined values "linear", "ease-in", "ease-out", and "ease-in-out"
* @param {Function} complete - A function to call once the animation is complete, called once per matched element.
* @return {Void}
*/
Constructor.prototype.animate = function (prop, from, to, unit, duration, easing, complete) {
this.each(function () {
const el = this;
const start = new Date().getTime();
const timer = setInterval(function () {
const time = new Date().getTime() - start;
let val;
switch (easing) {
case "linear":
val = easeLinear(time, from, to - from, duration);
break;
case "ease-in":
val = easeInCubic(time, from, to - from, duration);
break;
case "ease-out":
val = easeOutCubic(time, from, to - from, duration);
break;
case "ease-in-out":
val = easeInOutCubic(time, from, to - from, duration);
break;
default:
val = easeLinear(time, from, to - from, duration);
}
//
const styleValue = val + unit;
el.style[prop] = styleValue;
if (time >= duration) {
clearInterval(timer);
//
if (complete && (typeof complete == 'function')) {
complete.call(el);
}
}
}, 1000 / 60);
});
return this;
};
/* ------------- Global -------------- */
if ( typeof (window) !== 'undefined' ) {
window.__ = $$;
}
//
return $$;
})();
//export default __;
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment