217 lines
5.2 KiB
JavaScript
217 lines
5.2 KiB
JavaScript
|
(function() {
|
|||
|
|
|||
|
/**
|
|||
|
* Debounce
|
|||
|
*
|
|||
|
* @param {Function} func
|
|||
|
* @param {number} wait
|
|||
|
* @param {boolean} immediate
|
|||
|
*/
|
|||
|
function debounce(func, wait, immediate) {
|
|||
|
'use strict';
|
|||
|
|
|||
|
var timeout;
|
|||
|
wait = (typeof wait !== 'undefined') ? wait : 20;
|
|||
|
immediate = (typeof immediate !== 'undefined') ? immediate : true;
|
|||
|
|
|||
|
return function() {
|
|||
|
|
|||
|
var context = this, args = arguments;
|
|||
|
var later = function() {
|
|||
|
timeout = null;
|
|||
|
|
|||
|
if (!immediate) {
|
|||
|
func.apply(context, args);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
var callNow = immediate && !timeout;
|
|||
|
|
|||
|
clearTimeout(timeout);
|
|||
|
timeout = setTimeout(later, wait);
|
|||
|
|
|||
|
if (callNow) {
|
|||
|
func.apply(context, args);
|
|||
|
}
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Prepends an element to a container.
|
|||
|
*
|
|||
|
* @param {Element} container
|
|||
|
* @param {Element} element
|
|||
|
*/
|
|||
|
function prependElement(container, element) {
|
|||
|
if (container.firstChild.nextSibling) {
|
|||
|
return container.insertBefore(element, container.firstChild.nextSibling);
|
|||
|
} else {
|
|||
|
return container.appendChild(element);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Shows an element by adding a hidden className.
|
|||
|
*
|
|||
|
* @param {Element} element
|
|||
|
*/
|
|||
|
function showButton(element) {
|
|||
|
// classList.remove is not supported in IE11
|
|||
|
element.className = element.className.replace('is-empty', '');
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Hides an element by removing the hidden className.
|
|||
|
*
|
|||
|
* @param {Element} element
|
|||
|
*/
|
|||
|
function hideButton(element) {
|
|||
|
// classList.add is not supported in IE11
|
|||
|
if (!element.classList.contains('is-empty')) {
|
|||
|
element.className += ' is-empty';
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Returns the currently available space in the menu container.
|
|||
|
*
|
|||
|
* @returns {number} Available space
|
|||
|
*/
|
|||
|
function getAvailableSpace( button, container ) {
|
|||
|
return container.offsetWidth - button.offsetWidth - 22;
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Returns whether the current menu is overflowing or not.
|
|||
|
*
|
|||
|
* @returns {boolean} Is overflowing
|
|||
|
*/
|
|||
|
function isOverflowingNavivation( list, button, container ) {
|
|||
|
return list.offsetWidth > getAvailableSpace( button, container );
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Set menu container variable
|
|||
|
*/
|
|||
|
var navContainer = document.querySelector('.main-navigation');
|
|||
|
var breaks = [];
|
|||
|
|
|||
|
/**
|
|||
|
* Let’s bail if we our menu doesn't exist
|
|||
|
*/
|
|||
|
if ( ! navContainer ) {
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Refreshes the list item from the menu depending on the menu size
|
|||
|
*/
|
|||
|
function updateNavigationMenu( container ) {
|
|||
|
|
|||
|
/**
|
|||
|
* Let’s bail if our menu is empty
|
|||
|
*/
|
|||
|
if ( ! container.parentNode.querySelector('.main-menu[id]') ) {
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
// Adds the necessary UI to operate the menu.
|
|||
|
var visibleList = container.parentNode.querySelector('.main-menu[id]');
|
|||
|
var hiddenList = visibleList.parentNode.nextElementSibling.querySelector('.hidden-links');
|
|||
|
var toggleButton = visibleList.parentNode.nextElementSibling.querySelector('.main-menu-more-toggle');
|
|||
|
|
|||
|
if ( isOverflowingNavivation( visibleList, toggleButton, container ) ) {
|
|||
|
|
|||
|
// Record the width of the list
|
|||
|
breaks.push( visibleList.offsetWidth );
|
|||
|
// Move last item to the hidden list
|
|||
|
prependElement( hiddenList, ! visibleList.lastChild || null === visibleList.lastChild ? visibleList.previousElementSibling : visibleList.lastChild );
|
|||
|
// Show the toggle button
|
|||
|
showButton( toggleButton );
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
// There is space for another item in the nav
|
|||
|
if ( getAvailableSpace( toggleButton, container ) > breaks[breaks.length - 1] ) {
|
|||
|
// Move the item to the visible list
|
|||
|
visibleList.appendChild( hiddenList.firstChild.nextSibling );
|
|||
|
breaks.pop();
|
|||
|
}
|
|||
|
|
|||
|
// Hide the dropdown btn if hidden list is empty
|
|||
|
if (breaks.length < 2) {
|
|||
|
hideButton( toggleButton );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Recur if the visible list is still overflowing the nav
|
|||
|
if ( isOverflowingNavivation( visibleList, toggleButton, container ) ) {
|
|||
|
updateNavigationMenu( container );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Run our priority+ function as soon as the document is `ready`
|
|||
|
*/
|
|||
|
document.addEventListener( 'DOMContentLoaded', function() {
|
|||
|
|
|||
|
updateNavigationMenu( navContainer );
|
|||
|
|
|||
|
// Also, run our priority+ function on selective refresh in the customizer
|
|||
|
var hasSelectiveRefresh = (
|
|||
|
'undefined' !== typeof wp &&
|
|||
|
wp.customize &&
|
|||
|
wp.customize.selectiveRefresh &&
|
|||
|
wp.customize.navMenusPreview.NavMenuInstancePartial
|
|||
|
);
|
|||
|
|
|||
|
if ( hasSelectiveRefresh ) {
|
|||
|
// Re-run our priority+ function on Nav Menu partial refreshes
|
|||
|
wp.customize.selectiveRefresh.bind( 'partial-content-rendered', function ( placement ) {
|
|||
|
|
|||
|
var isNewNavMenu = (
|
|||
|
placement &&
|
|||
|
placement.partial.id.includes( 'nav_menu_instance' ) &&
|
|||
|
'null' !== placement.container[0].parentNode &&
|
|||
|
placement.container[0].parentNode.classList.contains( 'main-navigation' )
|
|||
|
);
|
|||
|
|
|||
|
if ( isNewNavMenu ) {
|
|||
|
updateNavigationMenu( placement.container[0].parentNode );
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
|
|||
|
/**
|
|||
|
* Run our priority+ function on load
|
|||
|
*/
|
|||
|
window.addEventListener( 'load', function() {
|
|||
|
updateNavigationMenu( navContainer );
|
|||
|
});
|
|||
|
|
|||
|
/**
|
|||
|
* Run our priority+ function every time the window resizes
|
|||
|
*/
|
|||
|
var isResizing = false;
|
|||
|
window.addEventListener( 'resize',
|
|||
|
debounce( function() {
|
|||
|
if ( isResizing ) {
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
isResizing = true;
|
|||
|
setTimeout( function() {
|
|||
|
updateNavigationMenu( navContainer );
|
|||
|
isResizing = false;
|
|||
|
}, 150 );
|
|||
|
} )
|
|||
|
);
|
|||
|
|
|||
|
/**
|
|||
|
* Run our priority+ function
|
|||
|
*/
|
|||
|
updateNavigationMenu( navContainer );
|
|||
|
|
|||
|
})();
|