//--------------------------------------------------------------------// // 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 (null !== el && 1 === el.nodeType); return null; }; } /** * Foreach polyfill. * * @since 1.0.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 bloghashGetIndex = 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 bloghashSlideUp = (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 bloghashSlideDown = (target, duration = 500) => { target.style.removeProperty('display'); let display = window.getComputedStyle(target).display; if ('none' === display) { 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 */ var bloghashScrollTo = (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; } /** * bloghashScrollTo Constructor * @param {object} options Options * @param {object} easeFunctions Custom ease functions */ function bloghashScrollTo() { var options = 0 < arguments.length && arguments[0] !== undefined ? arguments[0] : {}; var easeFunctions = 1 < arguments.length && 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 */ bloghashScrollTo.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 ('function' === typeof callback) { 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 */ bloghashScrollTo.prototype.move = function (target) { var _this2 = this; var options = 1 < arguments.length && arguments[1] !== undefined ? arguments[1] : {}; if (0 !== target && !target) { return; } options = mergeObject(this.options, options); var distance = 'number' === typeof target ? 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 ((0 < distance && lastYOffset > currentYOffset) || (0 > distance && 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 */ bloghashScrollTo.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 bloghashScrollTo; }()); /** * 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 bloghashGetParents = (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 (0 <= --i && matches.item(i) !== this) { } return -1 < i; }; } // 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 ('function' === typeof window.CustomEvent) { 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 bloghashTriggerEvent = function (el, typeArg) { var customEventInit = 2 < arguments.length && arguments[2] !== undefined ? arguments[2] : {}; var event = new CustomEvent(typeArg, customEventInit); el.dispatchEvent(event); }; // Main (function () { //--------------------------------------------------------------------// // Variable caching //--------------------------------------------------------------------// var bloghashScrollButton = document.querySelector('#bloghash-scroll-top'); var pageWrapper = document.getElementById('page'); //--------------------------------------------------------------------// // Local helper functions //--------------------------------------------------------------------// /** * * @param {*} button */ function initLoadingTargetButton(button) { let entry = document.createElement('span'); entry.classList.add('wait'); let wrapButtonText = document.createElement('span'); wrapButtonText.innerHTML = button.innerHTML; wrapButtonText.style.display = "none"; button.innerHTML = ''; button.append(entry); button.append(wrapButtonText); button.disabled = true; } /** * * @param {*} button */ function endLoadingTargetButton(button) { button.children[0].remove(); const text = button.children[0].innerHTML; button.children[0].remove(); button.innerHTML = text; button.disabled = false; } /** * Submenu overflow helper * * @since 1.0.0 */ var bloghashSmartSubmenus = () => { if (document.body.classList.contains('bloghash-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 (0 > elPosLeft) { 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 bloghashDebounce = (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 bloghashScrollTopButton = () => { if (null === bloghashScrollButton) { return; } if (450 < window.pageYOffset || 450 < document.documentElement.scrollTop) { bloghashScrollButton.classList.add('bloghash-visible'); } else { bloghashScrollButton.classList.remove('bloghash-visible'); } }; /** * Handles Sticky Header functionality. * * @since 1.0.0 */ var bloghashStickyHeader = () => { // Check if sticky is enabled. if (!bloghash_vars['sticky-header'].enabled) { return; } var header = document.getElementById('bloghash-header'); var headerInner = document.getElementById('bloghash-header-inner'); var wpadminbar = document.getElementById('wpadminbar'); // Check for header layout 3 & 4. if (document.body.classList.contains('bloghash-header-layout-3') || document.body.classList.contains('bloghash-header-layout-4') || document.body.classList.contains('bloghash-header-layout-6')) { header = document.querySelector('#bloghash-header .bloghash-nav-container'); headerInner = document.querySelector('#bloghash-header .bloghash-nav-container .bloghash-container'); } // Mobile nav active. if (window.outerWidth <= bloghash_vars['responsive-breakpoint']) { var header = document.getElementById('bloghash-header'); var headerInner = document.getElementById('bloghash-header-inner'); } // Check if elements exist. if (null === header || null === headerInner) { return; } // Calculate the initial sticky position. var stickyPosition = header.getBoundingClientRect().bottom; var sticky = 0 >= stickyPosition - tolerance; var tolerance; var stickyPlaceholder; // Check if there is a top bar. if (null === wpadminbar) { tolerance = 0; } else if (600 >= window.outerWidth) { tolerance = 0; } else { tolerance = wpadminbar.getBoundingClientRect().height; } var checkPosition = function () { if (null === wpadminbar) { tolerance = 0; } else if (600 >= window.outerWidth) { tolerance = 0; } else { tolerance = wpadminbar.getBoundingClientRect().height; } stickyPosition = header.getBoundingClientRect().bottom; sticky = 0 >= stickyPosition - tolerance; maybeStickHeader(); }; var maybeStickHeader = function () { let hideOn = bloghash_vars['sticky-header']['hide_on']; // Desktop. if (hideOn.includes('desktop') && 992 <= window.innerWidth) { sticky = false; } // Tablet. if (hideOn.includes('tablet') && 481 <= window.innerWidth && 992 > window.innerWidth) { sticky = false; } // Mobile. if (hideOn.includes('mobile') && 481 > window.innerWidth) { sticky = false; } if (sticky) { if (!document.body.classList.contains('bloghash-sticky-header')) { stickyPlaceholder = document.createElement('div'); stickyPlaceholder.setAttribute('id', 'bloghash-sticky-placeholder'); stickyPlaceholder.style.height = headerInner.getBoundingClientRect().height + 'px'; header.appendChild(stickyPlaceholder); document.body.classList.add('bloghash-sticky-header'); // Add sticky header offset variable. document.body.style.setProperty('--bloghash-sticky-h-offset', header.offsetHeight + 20 + 'px'); } } else { if (document.body.classList.contains('bloghash-sticky-header')) { document.body.classList.remove('bloghash-sticky-header'); document.getElementById('bloghash-sticky-placeholder').remove(); } // Remove sticky header offset variable. document.body.style.removeProperty('--bloghash-sticky-h-offset'); } }; // Debounce scroll. if ('true' !== header.getAttribute('data-scroll-listener')) { window.addEventListener('scroll', function () { bloghashDebounce(checkPosition()); }); header.setAttribute('data-scroll-listener', 'true'); } // Debounce resize. if ('true' !== header.getAttribute('data-resize-listener')) { window.addEventListener('resize', function () { bloghashDebounce(checkPosition()); }); header.setAttribute('data-resize-listener', 'true'); } // Trigger scroll. bloghashTriggerEvent(window, 'scroll'); }; /** * Handles smooth scrolling of elements that have 'bloghash-smooth-scroll' class. * * @since 1.0.0 */ var bloghashSmoothScroll = () => { const scrollTo = new bloghashScrollTo({ tolerance: null === document.getElementById('wpadminbar') ? 0 : document.getElementById('wpadminbar').getBoundingClientRect().height }); const scrollTriggers = document.getElementsByClassName('bloghash-smooth-scroll'); for (var i = 0; i < scrollTriggers.length; i++) { scrollTo.registerTrigger(scrollTriggers[i]); } }; /** * Menu accessibility. * * @since 1.0.0 */ var bloghashMenuAccessibility = () => { if (!document.body.classList.contains('bloghash-menu-accessibility')) { return; } document.querySelectorAll('.bloghash-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', bloghashMenuFocus, true); link.addEventListener('blur', bloghashMenuFocus, true); }); }); }; /** * Helper function that toggles .hovered on focused/blurred menu items. * * @since 1.0.0 */ function bloghashMenuFocus() { var self = this; // Move up until we find .bloghash-nav while (!self.classList.contains('bloghash-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 bloghashKeyboardFocus = () => { 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.0.0 */ var bloghashCalcScreenWidth = () => { document.body.style.setProperty('--bloghash-screen-width', document.body.clientWidth + 'px'); }; /** * Adds visibility delay on navigation submenus. * * @since 1.0.0 */ var bloghashDropdownDelay = () => { var hoverTimer = null; document.querySelectorAll('.bloghash-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('.bloghash-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.0.0 */ var bloghashCartDropdownDelay = () => { var hoverTimer = null; document.querySelectorAll('.bloghash-header-widget__cart .bloghash-widget-wrapper').forEach((item) => { item.addEventListener('mouseenter', function () { item.classList.remove('dropdown-visible'); }); }); document.querySelectorAll('.bloghash-header-widget__cart .bloghash-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 bloghashHeaderSearch = () => { var searchButton = document.querySelectorAll('.bloghash-search'); if (0 === searchButton.length) { return; } searchButton.forEach((item) => { item.addEventListener('click', (e) => { e.preventDefault(); if (item.classList.contains('bloghash-active')) { close_search(item); } else { show_search(item); } }); }); document.querySelectorAll('.bloghash-search-close').forEach(item => item.addEventListener('click', function (e) { e.preventDefault(); if (!item.classList.contains('bloghash-active')) { close_search(document.querySelector('.bloghash-search')); document.querySelector('.bloghash-search').focus(); } })); // Show search. var show_search = function (item) { var bloghashsearch = document.querySelectorAll('.bloghash-search-form'); bloghashsearch.forEach(function (bloghashsearch) { if (!bloghashsearch) { return false; } document.addEventListener('keydown', function (e) { var firstEl, selectors, elements, lastEl, tabKey, shiftKey; selectors = 'input, a, button'; elements = bloghashsearch.querySelectorAll(selectors); lastEl = elements[elements.length - 1]; firstEl = elements[0]; tabKey = 9 === e.keyCode; shiftKey = e.shiftKey; if (!shiftKey && tabKey && lastEl === document.activeElement) { e.preventDefault(); firstEl.focus(); } if (shiftKey && tabKey && firstEl === document.activeElement) { e.preventDefault(); lastEl.focus(); } }); }); // Make search visible document.body.classList.add('bloghash-search-visible'); setTimeout(function () { // Highlight the search icon item.classList.add('bloghash-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('bloghash-search-visible'); // Unhighlight the search icon item.classList.remove('bloghash-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 (27 == e.keyCode) { document.querySelectorAll('.bloghash-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('.bloghash-search-container') && null === e.target.closest('.bloghash-search')) { document.querySelectorAll('.bloghash-search').forEach((item) => { close_search(item); }); } }; }; /** * Handles mobile menu functionality. * * @since 1.0.0 */ var bloghashMobileMenu = () => { var page = pageWrapper, nav = document.querySelector('#bloghash-header-inner .bloghash-nav'), current; document.querySelectorAll('.bloghash-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) { var bloghashnav = document.querySelectorAll('.site-header'); if (document.body.classList.contains('bloghash-is-mobile')) { bloghashnav.forEach(function (bloghashnav) { if (!bloghashnav) { return false; } document.addEventListener('keydown', function (e) { var firstEl, selectors, elements, lastEl, tabKey, shiftKey, closeIcon; selectors = '.bloghash-nav a, .bloghash-nav button'; elements = bloghashnav.querySelectorAll(selectors); lastEl = elements[elements.length - 1]; firstEl = elements[0]; tabKey = 9 === e.keyCode; shiftKey = e.shiftKey; closeIcon = document.querySelector('.bloghash-hamburger-bloghash-primary-nav'); if (!shiftKey && tabKey && lastEl === document.activeElement) { e.preventDefault(); closeIcon.focus(); } if (shiftKey && tabKey && firstEl === document.activeElement) { e.preventDefault(); closeIcon.focus(); } // If there are no elements in the menu, don't move the focus if (tabKey && firstEl === lastEl) { e.preventDefault(); } }); }); } // 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('#bloghash-header .bloghash-nav').forEach((item) => { item.addEventListener('click', submenu_toggle); }); // Slide down the menu. bloghashSlideDown(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('#bloghash-header .bloghash-nav > ul > .bloghash-open').forEach((item) => { submenu_display_toggle(item); }); nav.querySelectorAll('.hovered').forEach((li) => { li.classList.remove('hovered'); }); if (document.body.classList.contains('bloghash-is-mobile')) { // Unhook the click listener for submenu toggle document.querySelectorAll('#bloghash-header .bloghash-nav').forEach((item) => { item.removeEventListener('click', submenu_toggle); }); // Slide up the menu bloghashSlideUp(nav, 250); } else { nav.style.display = null; } }; var outside_close_menu = function (e) { if (null === e.target.closest('.bloghash-hamburger') && null === e.target.closest('.site-navigation')) { close_menu(); } }; var esc_close_menu = function (e) { if (27 == e.keyCode) { close_menu(); } }; var submenu_toggle = function (e) { if (e.target.parentElement.querySelectorAll('.sub-menu').length) { e.preventDefault(); if ('button' === e.target.type) { submenu_display_toggle(e.target.parentElement); } } }; // Show or hide the sub menu. var submenu_display_toggle = (current) => { if (current.classList.contains('bloghash-open')) { current.classList.remove('bloghash-open'); current.querySelectorAll('.sub-menu').forEach((submenu) => { //submenu.style.display = null; bloghashSlideUp(submenu, 350); }); // Close all submenus automatically. current.querySelectorAll('li').forEach((item) => { item.classList.remove('bloghash-open'); item.querySelectorAll('.sub-menu').forEach((submenu) => { //submenu.style.display = null; bloghashSlideUp(submenu, 350); }); }); } else { current.querySelectorAll('.sub-menu').forEach((submenu) => { // Target first level elements only. if (current === submenu.parentElement) { //submenu.style.display = 'block'; bloghashSlideDown(submenu, 350); } }); current.classList.add('bloghash-open'); } }; // Create custom event for closing mobile menu. document.addEventListener('bloghash-close-mobile-menu', close_menu); }; /** * Bloghash preloader. * * @since 1.0.0 */ var bloghashPreloader = (timeout = 0) => { var preloader = document.getElementById('bloghash-preloader'); if (null === preloader) { return; } var delay = 250; var hide_preloader = () => { if (document.body.classList.contains('bloghash-loaded')) { return; } // Start fade out animation. document.body.classList.add('bloghash-loading'); setTimeout(function () { // Fade out animation completed - set display none document.body.classList.replace('bloghash-loading', 'bloghash-loaded'); // Dispatch event when preloader is done bloghashTriggerEvent(document.body, 'bloghash-preloader-done'); }, delay); }; // Set timeout or hide immediately if (0 < timeout) { setTimeout(function () { hide_preloader(); }, timeout); } else { hide_preloader(); } return false; }; /** * Handles comments toggle functionality. * * @since 1.0.0 */ var bloghashToggleComments = () => { if (!document.body.classList.contains('bloghash-has-comments-toggle')) { return; } if (null == document.getElementById('bloghash-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('bloghash-comments-toggle').querySelector('span').innerHTML = bloghash_vars.strings.comments_toggle_show; } else { document.body.classList.add('comments-visible'); document.getElementById('bloghash-comments-toggle').querySelector('span').innerHTML = bloghash_vars.strings.comments_toggle_hide; } }; if (null !== document.getElementById('bloghash-comments-toggle') && (-1 !== location.href.indexOf('#comment') || -1 !== location.href.indexOf('respond'))) { toggleComments(); } document.getElementById('bloghash-comments-toggle').addEventListener('click', toggleComments); }; /** * Handles toggling and smooth scrolling when clicked on "Comments" link * * @since 1.0.0 */ var bloghashCommentsClick = () => { 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('bloghash-has-comments-toggle') && !document.body.classList.contains('comments-visible')) { document.getElementById('bloghash-comments-toggle').click(); } }); }; /** * Removes inline styles on menus on resize. * * @since 1.0.0 */ var bloghashCheckMobileMenu = () => { // Update body class if mobile breakpoint is reached. if (window.innerWidth <= bloghash_vars['responsive-breakpoint']) { document.body.classList.add('bloghash-is-mobile'); } else { if (document.body.classList.contains('bloghash-is-mobile')) { document.body.classList.remove('bloghash-is-mobile'); bloghashTriggerEvent(document, 'bloghash-close-mobile-menu'); } } }; /** * Set Slider Init function * * @since 1.0.0 */ var bloghashSliderInit = (sliderElement) => { if (!sliderElement) return; // Check if the slider element exists let swiperOptions = JSON.parse(sliderElement.dataset.swiperOptions); if (sliderElement.classList.contains('swiper-top')) { const bottomSwiperElement = document.querySelector('.swiper-bottom'); if (bottomSwiperElement) { // Ensure the bottom swiper element exists const bottomSwiperOptions = JSON.parse(bottomSwiperElement.dataset.swiperOptions); const bottomSwiper = new Swiper(bottomSwiperElement, bottomSwiperOptions); swiperOptions.thumbs = { swiper: bottomSwiper }; } } const mySwiper = new Swiper(sliderElement, swiperOptions); } /** * Time and Date Update * * @since 1.0.0 */ var bloghashUpdateTimeAndDate = () => { if (document.getElementById("bloghash-time")) { document.getElementById("bloghash-time").textContent = new Date().toLocaleTimeString(); } if (document.getElementById("bloghash-date")) { const options = { weekday: 'short', month: 'short', day: 'numeric', year: 'numeric' }; document.getElementById("bloghash-date").textContent = new Date().toLocaleString(document.documentElement.lang, options); } } /** * Handle Quick Like * * @since 1.0.0 */ var bloghashHandleQuickLike = (element) => { if ((element.classList.contains('liked'))) { element.classList.remove('liked', 'heartBeat'); element.lastElementChild.textContent = parseInt(element.lastElementChild.textContent) - 1; } else { element.classList.add('liked', 'heartBeat'); element.lastElementChild.textContent = parseInt(element.lastElementChild.textContent) + 1; } } /** * Ticker News with vanilla.marquee * * @since 1.0.0 */ var bloghashTicker = () => { const hdir = document.body.classList.contains("rtl") ? "right" : "left"; const htc = document.querySelector(".bloghash-ticker.one-ticker"); if (htc) { const htcM = new Marquee(htc.querySelector(".ticker-slider-wrap"), { speed: 50, duration: 14000, gap: 0, delayBeforeStart: 0, direction: hdir, duplicated: true, startVisible: true, pauseOnHover: true, }); htc.addEventListener("click", (e) => { const pausePlayIcon = e.target.closest(".ticker-slider-pause")?.querySelector("i"); if (pausePlayIcon) { pausePlayIcon.classList.toggle("fa-pause"); pausePlayIcon.classList.toggle("fa-play"); htcM.toggle(); } }); } } //--------------------------------------------------------------------// // Events //--------------------------------------------------------------------// // DOM ready document.addEventListener('DOMContentLoaded', function () { bloghashPreloader(5000); bloghashMenuAccessibility(); bloghashKeyboardFocus(); bloghashScrollTopButton(); bloghashSmoothScroll(); bloghashDropdownDelay(); bloghashToggleComments(); bloghashHeaderSearch(); bloghashMobileMenu(); bloghashCheckMobileMenu(); bloghashSmartSubmenus(); bloghashCommentsClick(); bloghashCartDropdownDelay(); bloghashStickyHeader(); bloghashCalcScreenWidth(); // Initialize all sliders, including those that are not thumbs document.querySelectorAll('.bloghash-swiper:not(.swiper-bottom)').forEach(bloghashSliderInit); // Then, initialize thumb (bottom) sliders to ensure they can be linked document.querySelectorAll('.bloghash-swiper.swiper-bottom').forEach(bloghashSliderInit); setInterval(bloghashUpdateTimeAndDate, 1000); bloghashUpdateTimeAndDate(); // Initial call to display immediately bloghashTicker(); }); // Window load window.addEventListener('load', function () { bloghashPreloader(); }); // Scroll window.addEventListener('scroll', function () { bloghashDebounce(bloghashScrollTopButton()); }); // Resize window.addEventListener('resize', function () { bloghashDebounce(bloghashSmartSubmenus()); bloghashDebounce(bloghashCheckMobileMenu()); bloghashDebounce(bloghashCalcScreenWidth()); }); // Bloghash ready bloghashTriggerEvent(document.body, 'bloghash-ready'); //--------------------------------------------------------------------// // Global //--------------------------------------------------------------------// window.bloghash = window.bloghash || {}; // Make these function global. window.bloghash.preloader = bloghashPreloader; window.bloghash.stickyHeader = bloghashStickyHeader; window.bloghashSliderInit = bloghashSliderInit; window.App = {}; App.config = { headroom: { enabled: true, options: { classes: { initial: 'headroom', pinned: 'is-pinned', unpinned: 'is-unpinned', top: 'is-top', notTop: 'is-not-top', bottom: 'is-bottom', notBottom: 'is-not-bottom', frozen: 'is-frozen' } } }, ajax: { enabled: true }, cursorFollower: { enabled: true, disableBreakpoint: '992' } }; App.html = document.querySelector('html'); App.body = document.querySelector('body'); /****** // Cursor // ******/ window.onload = () => { if (App.config.cursorFollower?.enabled) Cursor.init(); }; const Cursor = (() => { const cursor = document.querySelector('.bloghash-js-cursor'); let follower, label, icon, clientX = -100, clientY = -100, cursorWidth, cursorHeight, cursorTriggers, state = false; const setVariables = () => { if (!cursor) return; [follower, label, icon] = ['.bloghash-js-follower', '.bloghash-js-label', '.bloghash-js-icon'].map(sel => cursor.querySelector(sel)); [cursorWidth, cursorHeight] = [cursor.offsetWidth / 2, cursor.offsetHeight / 2]; }; const addEventListeners = () => { document.addEventListener('mousedown', () => cursor.classList.add('is-mouse-down')); document.addEventListener('mouseup', () => cursor.classList.remove('is-mouse-down')); document.addEventListener('mousemove', ({ clientX: x, clientY: y }) => [clientX, clientY] = [x, y]); }; const render = () => { cursor.style.transform = `translate(${clientX - cursorWidth}px, ${clientY - cursorHeight}px)`; requestAnimationFrame(render); }; const enterHandler = ({ target }) => { const updateCursor = (cls, sel, attr) => { if (target.getAttribute(attr)) { App.body.classList.add(cls); cursor.classList.add(sel); return target.getAttribute(attr); } }; cursor.classList.add('is-active'); label.innerHTML = updateCursor('is-cursor-active', 'has-label', 'data-cursor-label') || ''; icon.innerHTML = updateCursor('is-cursor-active', 'has-icon', 'data-cursor-icon') || ''; }; const leaveHandler = () => { ['is-cursor-active', 'is-active', 'has-label', 'has-icon'].forEach(cls => cursor.classList.remove(cls)); [label.innerHTML, icon.innerHTML] = ['', '']; }; const updateCursorTriggers = () => { cursorTriggers?.forEach(el => el.removeEventListener('mouseenter', enterHandler)); cursorTriggers = document.querySelectorAll('button, a, input, [data-cursor], [data-cursor-label], [data-cursor-icon], textarea'); cursorTriggers.forEach(el => { el.addEventListener('mouseenter', enterHandler); el.addEventListener('mouseleave', leaveHandler); }); }; const breakpointCheck = () => { const updateState = () => { let width = window.innerWidth || screen.width; state = width >= App.config.cursorFollower?.disableBreakpoint; cursor.classList.toggle('is-enabled', state); state ? updateCursorTriggers() : cursorTriggers.forEach(el => el.removeEventListener('mouseenter', enterHandler)); }; updateState(); window.addEventListener('resize', updateState); }; return { init: () => { if (!cursor) return; setVariables(); state = true; cursor.classList.add('is-enabled'); addEventListeners(); requestAnimationFrame(render); updateCursorTriggers(); breakpointCheck(); }, update: updateCursorTriggers, clear: () => cursorTriggers?.forEach(el => el.removeEventListener('mouseenter', enterHandler)), hide: () => cursor?.classList.add('is-hidden'), show: () => cursor?.classList.remove('is-hidden') }; })(); /****** // Start --> Dark / Light Mode Setup // ******/ document.addEventListener('DOMContentLoaded', function () { const toggleButton = document.querySelector('.bloghash-darkmode'); if (toggleButton) { // Initialize dark mode based on customizer setting or local storage let darkModeEnabled = bloghash_vars.dark_mode || localStorage.getItem('darkmode') === 'dark'; updateDarkMode(darkModeEnabled); // Toggle button click handler toggleButton.addEventListener('click', function () { darkModeEnabled = !darkModeEnabled; // Toggle dark mode state updateDarkMode(darkModeEnabled); // Update UI and store preference }); function updateDarkMode(enabled) { if (enabled) { document.documentElement.setAttribute('data-darkmode', 'dark'); localStorage.setItem('darkmode', 'dark'); toggleButton.classList.add('active'); } else { document.documentElement.setAttribute('data-darkmode', 'light'); localStorage.setItem('darkmode', 'light'); toggleButton.classList.remove('active'); } } } }); }());