File Editor
Directories:
.. (Back)
vendors
Files:
sinatra-slider.js
sinatra-wc.js
sinatra.js
skip-link-focus-fix.js
Create New File
Create
Edit File: sinatra.js
//--------------------------------------------------------------------// // Global helper functions //--------------------------------------------------------------------// /** * Matches polyfill. * * @since 1.0.0 */ if (!Element.prototype.matches) { Element.prototype.matches = Element.prototype.msMatchesSelector || Element.prototype.webkitMatchesSelector; } /** * Closest polyfill. * * @since 1.0.0 */ if (!Element.prototype.closest) { Element.prototype.closest = function (s) { var el = this; do { if (el.matches(s)) return el; el = el.parentElement || el.parentNode; } while (el !== null && el.nodeType === 1); return null; }; } /** * Foreach polyfill. * * @since 1.1.0 */ if (window.NodeList && !NodeList.prototype.forEach) { NodeList.prototype.forEach = Array.prototype.forEach; } /** * Element.prototype.classList for IE8/9, Safari. * * @since 1.10 */ (function () { // Helpers. var trim = function (s) { return s.replace(/^\s+|\s+$/g, ''); }, regExp = function (name) { return new RegExp('(^|\\s+)' + name + '(\\s+|$)'); }, forEach = function (list, fn, scope) { for (var i = 0; i < list.length; i++) { fn.call(scope, list[i]); } }; // Class list object with basic methods. function ClassList(element) { this.element = element; } ClassList.prototype = { add: function () { forEach( arguments, function (name) { if (!this.contains(name)) { this.element.className = trim(this.element.className + ' ' + name); } }, this ); }, remove: function () { forEach( arguments, function (name) { this.element.className = trim(this.element.className.replace(regExp(name), ' ')); }, this ); }, toggle: function (name) { return this.contains(name) ? (this.remove(name), false) : (this.add(name), true); }, contains: function (name) { return regExp(name).test(this.element.className); }, item: function (i) { return this.element.className.split(/\s+/)[i] || null; }, // bonus replace: function (oldName, newName) { this.remove(oldName), this.add(newName); }, }; // IE8/9, Safari // Remove this if statements to override native classList. if (!('classList' in Element.prototype)) { // Use this if statement to override native classList that does not have for example replace() method. // See browser compatibility: https://developer.mozilla.org/en-US/docs/Web/API/Element/classList#Browser_compatibility. // if (!('classList' in Element.prototype) || // !('classList' in Element.prototype && Element.prototype.classList.replace)) { Object.defineProperty(Element.prototype, 'classList', { get: function () { return new ClassList(this); }, }); } // For others replace() support. if (window.DOMTokenList && !DOMTokenList.prototype.replace) { DOMTokenList.prototype.replace = ClassList.prototype.replace; } })(); /** * Index polyfill. * * @since 1.0.0 */ var sinatraGetIndex = function (el) { var i = 0; while ((el = el.previousElementSibling)) { i++; } return i; }; /** * Slide Up animation. * * @since 1.0.0 * * @param {[type]} target Element to slide. * @param {Number} duration Animation duration. */ var sinatraSlideUp = (target, duration = 500) => { target.style.transitionProperty = 'height, margin, padding'; target.style.transitionDuration = duration + 'ms'; target.style.boxSizing = 'border-box'; target.style.height = target.offsetHeight + 'px'; target.offsetHeight; target.style.overflow = 'hidden'; target.style.height = 0; target.style.paddingTop = 0; target.style.paddingBottom = 0; target.style.marginTop = 0; target.style.marginBottom = 0; window.setTimeout(() => { target.style.display = null; target.style.removeProperty('height'); target.style.removeProperty('padding-top'); target.style.removeProperty('padding-bottom'); target.style.removeProperty('margin-top'); target.style.removeProperty('margin-bottom'); target.style.removeProperty('overflow'); target.style.removeProperty('transition-duration'); target.style.removeProperty('transition-property'); }, duration); }; /** * Slide Down animation. * * @since 1.0.0 * * @param {[type]} target Element to slide. * @param {Number} duration Animation duration. */ var sinatraSlideDown = (target, duration = 500) => { target.style.removeProperty('display'); let display = window.getComputedStyle(target).display; if (display === 'none') { display = 'block'; } target.style.display = display; let height = target.offsetHeight; target.style.overflow = 'hidden'; target.style.height = 0; target.style.paddingTop = 0; target.style.paddingBottom = 0; target.style.marginTop = 0; target.style.marginBottom = 0; target.offsetHeight; target.style.boxSizing = 'border-box'; target.style.transitionProperty = 'height, margin, padding'; target.style.transitionDuration = duration + 'ms'; target.style.height = height + 'px'; target.style.removeProperty('padding-top'); target.style.removeProperty('padding-bottom'); target.style.removeProperty('margin-top'); target.style.removeProperty('margin-bottom'); window.setTimeout(() => { target.style.removeProperty('height'); target.style.removeProperty('overflow'); target.style.removeProperty('transition-duration'); target.style.removeProperty('transition-property'); }, duration); }; /** * MoveTo - A lightweight scroll animation javascript library without any dependency. * Version 1.8.3 (21-07-2019 00:32) * Licensed under MIT * Copyright 2019 Hasan Aydoğdu <hsnaydd@gmail.com> */ var sinatraScrollTo = (function () { /** * Defaults * @type {object} */ var defaults = { tolerance: 0, duration: 800, easing: 'easeOutQuart', container: window, callback: function callback() {}, }; /** * easeOutQuart Easing Function * @param {number} t - current time * @param {number} b - start value * @param {number} c - change in value * @param {number} d - duration * @return {number} - calculated value */ function easeOutQuart(t, b, c, d) { t /= d; t--; return -c * (t * t * t * t - 1) + b; } /** * Merge two object * * @param {object} obj1 * @param {object} obj2 * @return {object} merged object */ function mergeObject(obj1, obj2) { var obj3 = {}; Object.keys(obj1).forEach(function (propertyName) { obj3[propertyName] = obj1[propertyName]; }); Object.keys(obj2).forEach(function (propertyName) { obj3[propertyName] = obj2[propertyName]; }); return obj3; } /** * Converts camel case to kebab case * @param {string} val the value to be converted * @return {string} the converted value */ function kebabCase(val) { return val.replace(/([A-Z])/g, function ($1) { return '-' + $1.toLowerCase(); }); } /** * Count a number of item scrolled top * @param {Window|HTMLElement} container * @return {number} */ function countScrollTop(container) { if (container instanceof HTMLElement) { return container.scrollTop; } return container.pageYOffset; } /** * sinatraScrollTo Constructor * @param {object} options Options * @param {object} easeFunctions Custom ease functions */ function sinatraScrollTo() { var options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; var easeFunctions = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; this.options = mergeObject(defaults, options); this.easeFunctions = mergeObject( { easeOutQuart: easeOutQuart, }, easeFunctions ); } /** * Register a dom element as trigger * @param {HTMLElement} dom Dom trigger element * @param {function} callback Callback function * @return {function|void} unregister function */ sinatraScrollTo.prototype.registerTrigger = function (dom, callback) { var _this = this; if (!dom) { return; } var href = dom.getAttribute('href') || dom.getAttribute('data-target'); // The element to be scrolled var target = href && href !== '#' ? document.getElementById(href.substring(1)) : document.body; var options = mergeObject(this.options, _getOptionsFromTriggerDom(dom, this.options)); if (typeof callback === 'function') { options.callback = callback; } var listener = function listener(e) { e.preventDefault(); _this.move(target, options); }; dom.addEventListener('click', listener, false); return function () { return dom.removeEventListener('click', listener, false); }; }; /** * Move * Scrolls to given element by using easeOutQuart function * @param {HTMLElement|number} target Target element to be scrolled or target position * @param {object} options Custom options */ sinatraScrollTo.prototype.move = function (target) { var _this2 = this; var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; if (target !== 0 && !target) { return; } options = mergeObject(this.options, options); var distance = typeof target === 'number' ? target : target.getBoundingClientRect().top; var from = countScrollTop(options.container); var startTime = null; var lastYOffset; distance -= options.tolerance; // rAF loop var loop = function loop(currentTime) { var currentYOffset = countScrollTop(_this2.options.container); if (!startTime) { // To starts time from 1, we subtracted 1 from current time // If time starts from 1 The first loop will not do anything, // because easing value will be zero startTime = currentTime - 1; } var timeElapsed = currentTime - startTime; if (lastYOffset) { if ((distance > 0 && lastYOffset > currentYOffset) || (distance < 0 && lastYOffset < currentYOffset)) { return options.callback(target); } } lastYOffset = currentYOffset; var val = _this2.easeFunctions[options.easing](timeElapsed, from, distance, options.duration); options.container.scroll(0, val); if (timeElapsed < options.duration) { window.requestAnimationFrame(loop); } else { options.container.scroll(0, distance + from); options.callback(target); } }; window.requestAnimationFrame(loop); }; /** * Adds custom ease function * @param {string} name Ease function name * @param {function} fn Ease function */ sinatraScrollTo.prototype.addEaseFunction = function (name, fn) { this.easeFunctions[name] = fn; }; /** * Returns options which created from trigger dom element * @param {HTMLElement} dom Trigger dom element * @param {object} options The instance's options * @return {object} The options which created from trigger dom element */ function _getOptionsFromTriggerDom(dom, options) { var domOptions = {}; Object.keys(options).forEach(function (key) { var value = dom.getAttribute('data-mt-'.concat(kebabCase(key))); if (value) { domOptions[key] = isNaN(value) ? value : parseInt(value, 10); } }); return domOptions; } return sinatraScrollTo; })(); /** * Get all of an element's parent elements up the DOM tree * * @since 1.0.0 * * @param {Node} elem The element. * @param {String} selector Selector to match against [optional]. * @return {Array} The parent elements. */ var sinatraGetParents = (elem, selector) => { // Element.matches() polyfill. if (!Element.prototype.matches) { Element.prototype.matches = Element.prototype.matchesSelector || Element.prototype.mozMatchesSelector || Element.prototype.msMatchesSelector || Element.prototype.oMatchesSelector || Element.prototype.webkitMatchesSelector || function (s) { var matches = (this.document || this.ownerDocument).querySelectorAll(s), i = matches.length; while (--i >= 0 && matches.item(i) !== this) {} return i > -1; }; } // Setup parents array. var parents = []; // Get matching parent elements. for (; elem && elem !== document; elem = elem.parentNode) { // Add matching parents to array. if (selector) { if (elem.matches(selector)) { parents.push(elem); } } else { parents.push(elem); } } return parents; }; // CustomEvent() constructor functionality in Internet Explorer 9 and higher. (function () { if (typeof window.CustomEvent === 'function') return false; function CustomEvent(event, params) { params = params || { bubbles: false, cancelable: false, detail: undefined }; var evt = document.createEvent('CustomEvent'); evt.initCustomEvent(event, params.bubbles, params.cancelable, params.detail); return evt; } CustomEvent.prototype = window.Event.prototype; window.CustomEvent = CustomEvent; })(); /** * Trigger custom JS Event. * * @since 1.0.0 * * @link https://developer.mozilla.org/en-US/docs/Web/API/CustomEvent * @param {Node} el Dom Node element on which the event is to be triggered. * @param {Node} typeArg A DOMString representing the name of the event. * @param {String} A CustomEventInit dictionary, having the following fields: * "detail", optional and defaulting to null, of type any, that is an event-dependent value associated with the event. */ var sinatraTriggerEvent = function (el, typeArg) { var customEventInit = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : {}; var event = new CustomEvent(typeArg, customEventInit); el.dispatchEvent(event); }; // Main (function () { //--------------------------------------------------------------------// // Variable caching //--------------------------------------------------------------------// var sinatraScrollButton = document.querySelector('#si-scroll-top'); var pageWrapper = document.getElementById('page'); //--------------------------------------------------------------------// // Local helper functions //--------------------------------------------------------------------// /** * Submenu overflow helper * * @since 1.0.0 */ var sinatraSmartSubmenus = () => { if (document.body.classList.contains('sinatra-is-mobile')) { return; } var el, elPosRight, elPosLeft, winRight; winRight = window.innerWidth; document.querySelectorAll('.sub-menu').forEach((item) => { // Set item to be visible so we can grab offsets item.style.visibility = 'visible'; // Left offset const rect = item.getBoundingClientRect(); elPosLeft = rect.left + window.pageXOffset; // Right offset elPosRight = elPosLeft + rect.width; // Remove styles item.removeAttribute('style'); // Decide where to open if (elPosRight > winRight) { item.closest('li').classList.add('opens-left'); } else if (elPosLeft < 0) { item.closest('li').classList.add('opens-right'); } }); }; /** * Debounce functions for better performance * (c) 2018 Chris Ferdinandi, MIT License, https://gomakethings.com * * @since 1.0.0 * * @param {Function} fn The function to debounce */ var sinatraDebounce = (fn) => { // Setup a timer var timeout; // Return a function to run debounced return function () { // Setup the arguments var context = this; var args = arguments; // If there's a timer, cancel it if (timeout) { window.cancelAnimationFrame(timeout); } // Setup the new requestAnimationFrame() timeout = window.requestAnimationFrame(function () { fn.apply(this, args); }); }; }; /** * Handles Scroll to Top button click * * @since 1.0.0 */ var sinatraScrollTopButton = () => { if (null === sinatraScrollButton) { return; } if (window.pageYOffset > 450 || document.documentElement.scrollTop > 450) { sinatraScrollButton.classList.add('si-visible'); } else { sinatraScrollButton.classList.remove('si-visible'); } }; /** * Handles Sticky Header functionality. * * @since 1.0.0 */ var sinatraStickyHeader = () => { // Check if sticky is enabled. if (!sinatra_vars['sticky-header']['enabled']) { return; } var header = document.getElementById('sinatra-header'); var headerInner = document.getElementById('sinatra-header-inner'); var wpadminbar = document.getElementById('wpadminbar'); // Check for header layout 3. if (document.body.classList.contains('sinatra-header-layout-3')) { header = document.querySelector('#sinatra-header .si-nav-container'); headerInner = document.querySelector('#sinatra-header .si-nav-container .si-container'); } // Mobile nav active. if (window.outerWidth <= sinatra_vars['responsive-breakpoint']) { var header = document.getElementById('sinatra-header'); var headerInner = document.getElementById('sinatra-header-inner'); } // Check if elements exist. if (null === header || null === headerInner) { return; } // Calculate the initial sticky position. var stickyPosition = header.getBoundingClientRect().top; var sticky = stickyPosition - tolerance <= 0; var tolerance; var stickyPlaceholder; // Check if there is a top bar. if (null === wpadminbar) { tolerance = 0; } else if (window.outerWidth <= 600) { tolerance = 0; } else { tolerance = wpadminbar.getBoundingClientRect().height; } var checkPosition = function () { if (null === wpadminbar) { tolerance = 0; } else if (window.outerWidth <= 600) { tolerance = 0; } else { tolerance = wpadminbar.getBoundingClientRect().height; } stickyPosition = header.getBoundingClientRect().top; sticky = stickyPosition - tolerance <= 0; maybeStickHeader(); }; var maybeStickHeader = function () { let hideOn = sinatra_vars['sticky-header']['hide_on']; // Desktop. if (hideOn.includes('desktop') && window.innerWidth >= 992) { sticky = false; } // Tablet. if (hideOn.includes('tablet') && window.innerWidth >= 481 && window.innerWidth < 992) { sticky = false; } // Mobile. if (hideOn.includes('mobile') && window.innerWidth < 481) { sticky = false; } if (sticky) { if (!document.body.classList.contains('si-sticky-header')) { stickyPlaceholder = document.createElement('div'); stickyPlaceholder.setAttribute('id', 'si-sticky-placeholder'); stickyPlaceholder.style.height = headerInner.getBoundingClientRect().height + 'px'; header.appendChild(stickyPlaceholder); document.body.classList.add('si-sticky-header'); // Add sticky header offset variable. document.body.style.setProperty('--si-sticky-h-offset', header.offsetHeight + 20 + 'px'); } } else { if (document.body.classList.contains('si-sticky-header')) { document.body.classList.remove('si-sticky-header'); document.getElementById('si-sticky-placeholder').remove(); } // Remove sticky header offset variable. document.body.style.removeProperty('--si-sticky-h-offset'); } }; // Debounce scroll. if ('true' !== header.getAttribute('data-scroll-listener')) { window.addEventListener('scroll', function () { sinatraDebounce(checkPosition()); }); header.setAttribute('data-scroll-listener', 'true'); } // Debounce resize. if ('true' !== header.getAttribute('data-resize-listener')) { window.addEventListener('resize', function () { sinatraDebounce(checkPosition()); }); header.setAttribute('data-resize-listener', 'true'); } // Trigger scroll. sinatraTriggerEvent(window, 'scroll'); }; /** * Handles smooth scrolling of elements that have 'si-smooth-scroll' class. * * @since 1.0.0 */ var sinatraSmoothScroll = () => { const scrollTo = new sinatraScrollTo({ tolerance: null === document.getElementById('wpadminbar') ? 0 : document.getElementById('wpadminbar').getBoundingClientRect().height, }); const scrollTriggers = document.getElementsByClassName('si-smooth-scroll'); for (var i = 0; i < scrollTriggers.length; i++) { scrollTo.registerTrigger(scrollTriggers[i]); } }; /** * Menu accessibility. * * @since 1.0.0 */ var sinatraMenuAccessibility = () => { if (!document.body.classList.contains('si-menu-accessibility')) { return; } document.querySelectorAll('.sinatra-nav').forEach((menu) => { // aria-haspopup menu.querySelectorAll('ul').forEach((subMenu) => { subMenu.parentNode.setAttribute('aria-haspopup', 'true'); }); // Dropdown visibility on focus menu.querySelectorAll('a').forEach((link) => { link.addEventListener('focus', sinatraMenuFocus, true); link.addEventListener('blur', sinatraMenuFocus, true); }); }); }; /** * Helper function that toggles .hovered on focused/blurred menu items. * * @since 1.0.0 */ function sinatraMenuFocus() { var self = this; // Move up until we find .sinatra-nav while (!self.classList.contains('sinatra-nav')) { if ('li' === self.tagName.toLowerCase()) { if (!self.classList.contains('hovered')) { self.classList.add('hovered'); } else { self.classList.remove('hovered'); } } self = self.parentElement; } } /** * Helps with accessibility for keyboard only users. * * @since 1.0.0 */ var sinatraKeyboardFocus = () => { document.body.addEventListener('keydown', function (e) { document.body.classList.add('using-keyboard'); }); document.body.addEventListener('mousedown', function (e) { document.body.classList.remove('using-keyboard'); }); }; /** * Calculates screen width without scrollbars. * * @since 1.1.4 */ var sinatraCalcScreenWidth = () => { document.body.style.setProperty('--si-screen-width', document.body.clientWidth + 'px'); }; /** * Adds visibility delay on navigation submenus. * * @since 1.0.0 */ var sinatraDropdownDelay = () => { var hoverTimer = null; document.querySelectorAll('.sinatra-nav .menu-item-has-children').forEach((item) => { item.addEventListener('mouseenter', function () { document.querySelectorAll('.menu-item-has-children').forEach((subitem) => { subitem.classList.remove('hovered'); }); }); }); document.querySelectorAll('.sinatra-nav .menu-item-has-children').forEach((item) => { item.addEventListener('mouseleave', function () { item.classList.add('hovered'); if (null !== hoverTimer) { clearTimeout(hoverTimer); hoverTimer = null; } hoverTimer = setTimeout(() => { item.classList.remove('hovered'); item.querySelectorAll('.menu-item-has-children').forEach((childItem) => { childItem.classList.remove('hovered'); }); }, 700); }); }); }; /** * Adds visibility delay for cart widget dropdown. * * @since 1.1.0 */ var sinatraCartDropdownDelay = () => { var hoverTimer = null; document.querySelectorAll('.si-header-widget__cart .si-widget-wrapper').forEach((item) => { item.addEventListener('mouseenter', function () { item.classList.remove('dropdown-visible'); }); }); document.querySelectorAll('.si-header-widget__cart .si-widget-wrapper').forEach((item) => { item.addEventListener('mouseleave', function () { item.classList.add('dropdown-visible'); if (null !== hoverTimer) { clearTimeout(hoverTimer); hoverTimer = null; } hoverTimer = setTimeout(() => { item.classList.remove('dropdown-visible'); }, 700); }); }); }; /** * Handles header search widget functionality. * * @since 1.0.0 */ var sinatraHeaderSearch = () => { var searchButton = document.querySelectorAll('.si-search'); if (0 === searchButton.length) { return; } searchButton.forEach((item) => { item.addEventListener('click', (e) => { e.preventDefault(); if (item.classList.contains('sinatra-active')) { close_search(item); } else { show_search(item); } }); }); // Show search. var show_search = function (item) { // Make search visible document.body.classList.add('si-search-visible'); setTimeout(function () { // Highlight the search icon item.classList.add('sinatra-active'); // Focus the input if (null !== item.nextElementSibling && null !== item.nextElementSibling.querySelector('input')) { item.nextElementSibling.querySelector('input').focus(); item.nextElementSibling.querySelector('input').select(); } }, 100); // Attach the ESC listener document.addEventListener('keydown', esc_close_search); // Attach the outside click listener pageWrapper.addEventListener('click', outside_close_search); }; // Close search var close_search = function (item) { // Animate out document.body.classList.remove('si-search-visible'); // Unhighlight the search icon item.classList.remove('sinatra-active'); // Unhook the ESC listener document.removeEventListener('keydown', esc_close_search); // Unhook the click listener pageWrapper.removeEventListener('click', outside_close_search); }; // Esc support to close search var esc_close_search = function (e) { if (e.keyCode == 27) { document.querySelectorAll('.si-search').forEach((item) => { close_search(item); }); } }; // Close search when clicked anywhere outside the search box var outside_close_search = function (e) { if (null === e.target.closest('.si-search-container') && null === e.target.closest('.si-search')) { document.querySelectorAll('.si-search').forEach((item) => { close_search(item); }); } }; }; /** * Handles mobile menu functionality. * * @since 1.0.0 */ var sinatraMobileMenu = () => { var page = pageWrapper, nav = document.querySelector('#sinatra-header-inner .sinatra-nav'), current; document.querySelectorAll('.si-mobile-nav > button').forEach((item) => { item.addEventListener( 'click', function (e) { e.preventDefault(); if (document.body.parentNode.classList.contains('is-mobile-menu-active')) { close_menu(); } else { show_menu(); } }, false ); }); // Helper functions. var show_menu = function (e) { // Add the active class. document.body.parentNode.classList.add('is-mobile-menu-active'); // Hook the ESC listener document.addEventListener('keyup', esc_close_menu); // Hook the click listener if (null !== page) { page.addEventListener('click', outside_close_menu); } // Hook the click listener for submenu toggle. document.querySelectorAll('#sinatra-header .sinatra-nav').forEach((item) => { item.addEventListener('click', submenu_toggle); }); // Slide down the menu. sinatraSlideDown(nav, 350); }; var close_menu = function (e) { // Remove the active class. document.body.parentNode.classList.remove('is-mobile-menu-active'); // Unhook the ESC listener document.removeEventListener('keyup', esc_close_menu); // Unhook the click listener if (null !== page) { page.removeEventListener('click', outside_close_menu); } // Close submenus document.querySelectorAll('#sinatra-header .sinatra-nav > ul > .si-open').forEach((item) => { submenu_display_toggle(item); }); nav.style.display = null; nav.querySelectorAll('.hovered').forEach((li) => { li.classList.remove('hovered'); }); if (document.body.classList.contains('sinatra-is-mobile')) { // Unhook the click listener for submenu toggle document.querySelectorAll('#sinatra-header .sinatra-nav').forEach((item) => { item.removeEventListener('click', submenu_toggle); }); // Slide up the menu sinatraSlideUp(nav, 250); } }; var outside_close_menu = function (e) { if (null === e.target.closest('.si-hamburger') && null === e.target.closest('.site-navigation')) { close_menu(); } }; var esc_close_menu = function (e) { if (e.keyCode == 27) { close_menu(); } }; var submenu_toggle = function (e) { if (e.target.parentElement.querySelectorAll('.sub-menu').length) { e.preventDefault(); submenu_display_toggle(e.target.parentElement); } }; // Show or hide the sub menu. var submenu_display_toggle = (current) => { if (current.classList.contains('si-open')) { current.classList.remove('si-open'); current.querySelectorAll('.sub-menu').forEach((submenu) => { submenu.style.display = null; }); // Close all submenus automatically. current.querySelectorAll('li').forEach((item) => { item.classList.remove('si-open'); item.querySelectorAll('.sub-menu').forEach((submenu) => { submenu.style.display = null; }); }); } else { current.querySelectorAll('.sub-menu').forEach((submenu) => { // Target first level elements only. if (current === submenu.parentElement) { submenu.style.display = 'block'; } }); current.classList.add('si-open'); } }; // Create custom event for closing mobile menu. document.addEventListener('sinatra-close-mobile-menu', close_menu); }; /** * Sinatra preloader. * * @since 1.0.0 */ var sinatraPreloader = (timeout = 0) => { var preloader = document.getElementById('si-preloader'); if (null === preloader) { return; } var delay = 250; var hide_preloader = () => { if (document.body.classList.contains('si-loaded')) { return; } // Start fade out animation. document.body.classList.add('si-loading'); setTimeout(function () { // Fade out animation completed - set display none document.body.classList.replace('si-loading', 'si-loaded'); // Dispatch event when preloader is done sinatraTriggerEvent(document.body, 'si-preloader-done'); }, delay); }; // Set timeout or hide immediately if (timeout > 0) { setTimeout(function () { hide_preloader(); }, timeout); } else { hide_preloader(); } return false; }; /** * Handles comments toggle functionality. * * @since 1.0.0 */ var sinatraToggleComments = () => { if (!document.body.classList.contains('sinatra-has-comments-toggle')) { return; } if (null == document.getElementById('sinatra-comments-toggle')) { return; } var toggleComments = (e) => { if ('undefined' !== typeof e) { e.preventDefault(); } if (document.body.classList.contains('comments-visible')) { document.body.classList.remove('comments-visible'); document.getElementById('sinatra-comments-toggle').querySelector('span').innerHTML = sinatra_vars.strings.comments_toggle_show; } else { document.body.classList.add('comments-visible'); document.getElementById('sinatra-comments-toggle').querySelector('span').innerHTML = sinatra_vars.strings.comments_toggle_hide; } }; if (null !== document.getElementById('sinatra-comments-toggle') && (-1 !== location.href.indexOf('#comment') || -1 !== location.href.indexOf('respond'))) { toggleComments(); } document.getElementById('sinatra-comments-toggle').addEventListener('click', toggleComments); }; /** * Handles toggling and smooth scrolling when clicked on "Comments" link * * @since 1.0.0 */ var sinatraCommentsClick = () => { var commentsLink = document.querySelector('.single .comments-link'); if (null === commentsLink) { return; } commentsLink.addEventListener('click', function (e) { // Show comments if hidden under a toggle if (document.body.classList.contains('sinatra-has-comments-toggle') && !document.body.classList.contains('comments-visible')) { document.getElementById('sinatra-comments-toggle').click(); } }); }; /** * Removes inline styles on menus on resize. * * @since 1.1.0 */ var sinatraCheckMobileMenu = () => { // Update body class if mobile breakpoint is reached. if (window.innerWidth <= sinatra_vars['responsive-breakpoint']) { document.body.classList.add('sinatra-is-mobile'); } else { if (document.body.classList.contains('sinatra-is-mobile')) { document.body.classList.remove('sinatra-is-mobile'); sinatraTriggerEvent(document, 'sinatra-close-mobile-menu'); } } }; //--------------------------------------------------------------------// // Events //--------------------------------------------------------------------// // DOM ready document.addEventListener('DOMContentLoaded', function () { sinatraPreloader(5000); sinatraMenuAccessibility(); sinatraKeyboardFocus(); sinatraScrollTopButton(); sinatraSmoothScroll(); sinatraDropdownDelay(); sinatraToggleComments(); sinatraHeaderSearch(); sinatraMobileMenu(); sinatraCheckMobileMenu(); sinatraSmartSubmenus(); sinatraCommentsClick(); sinatraCartDropdownDelay(); sinatraStickyHeader(); sinatraCalcScreenWidth(); }); // Window load window.addEventListener('load', function () { sinatraPreloader(); }); // Scroll window.addEventListener('scroll', function () { sinatraDebounce(sinatraScrollTopButton()); }); // Resize window.addEventListener('resize', function () { sinatraDebounce(sinatraSmartSubmenus()); sinatraDebounce(sinatraCheckMobileMenu()); sinatraDebounce(sinatraCalcScreenWidth()); }); // Sinatra ready sinatraTriggerEvent(document.body, 'si-ready'); //--------------------------------------------------------------------// // Global //--------------------------------------------------------------------// window.sinatra = window.sinatra || {}; // Make these function global. window.sinatra.preloader = sinatraPreloader; window.sinatra.stickyHeader = sinatraStickyHeader; })();
Save Changes
Rename File
Rename