HOME


Mini Shell 1.0
DIR: /home/otwalrll/karismamodernkitchen.com/wp-content/themes/blocksy/static/js/frontend/header/
Upload File :
Current File : /home/otwalrll/karismamodernkitchen.com/wp-content/themes/blocksy/static/js/frontend/header/menu.js
import { isTouchDevice } from '../helpers/is-touch-device'
import { isIosDevice } from '../helpers/is-ios-device'

const isRtl = () => document.querySelector('html').dir === 'rtl'

const isEligibleForSubmenu = (el) =>
	el.className.includes('animated-submenu') &&
	(!el.parentNode.classList.contains('menu') ||
		(el.className.indexOf('ct-mega-menu') === -1 &&
			el.parentNode.classList.contains('menu')))

const getAllParents = (a) => {
	var els = []

	while (a) {
		els.unshift(a)
		a = a.parentNode
	}

	return els
}

function furthest(el, s) {
	var nodes = []

	while (el.parentNode) {
		if (
			el.parentNode &&
			el.parentNode.matches &&
			el.parentNode.matches(s)
		) {
			nodes.push(el.parentNode)
		}

		el = el.parentNode
	}

	return nodes[nodes.length - 1]
}

const getAllParentsUntil = (el, parent) => {
	const parents = []

	while (el.parentNode !== parent) {
		parents.push(el.parentNode)
		el = el.parentNode
	}

	return parents
}

const reversePlacementIfNeeded = (placement) => {
	if (!isRtl()) {
		return placement
	}

	if (placement === 'left') {
		return 'right'
	}

	if (placement === 'right') {
		return 'left'
	}

	return placement
}

const getPreferedPlacementFor = (el) => {
	let farmost = furthest(el, 'li.menu-item')

	if (!farmost) {
		return reversePlacementIfNeeded('right')
	}

	const submenusWithParents = [...farmost.querySelectorAll('.sub-menu')].map(
		(el) => {
			return { el, parents: getAllParentsUntil(el, farmost) }
		}
	)

	if (submenusWithParents.length === 0) {
		return reversePlacementIfNeeded('right')
	}

	const submenusWithParentsSorted = submenusWithParents
		.sort((a, b) => a.parents.length - b.parents.length)
		.reverse()

	const submenuWithMostParents = submenusWithParentsSorted[0]

	const allSubmenus = [
		...submenuWithMostParents.parents.filter((el) =>
			el.matches('.sub-menu')
		),

		...[submenuWithMostParents.el],
	]

	const allSubmenusAlignedWidth = allSubmenus.reduce((acc, el, index) => {
		const style = getComputedStyle(el)

		return (
			acc +
			el.getBoundingClientRect().width +
			(index === 0
				? 0
				: parseFloat(
						style.getPropertyValue(
							'--dropdown-horizontal-offset'
						) || '5px'
				  ))
		)
	}, 0)

	const farmostRect = farmost.getBoundingClientRect()

	let willItFitToTheLeft = allSubmenusAlignedWidth < farmostRect.right

	let willItFitToTheRight =
		innerWidth - farmostRect.left > allSubmenusAlignedWidth

	if (farmost.matches('.animated-submenu-inline')) {
		willItFitToTheRight =
			innerWidth - farmostRect.left - farmostRect.width >
			allSubmenusAlignedWidth
	}

	if (isRtl()) {
		if (!willItFitToTheLeft && !willItFitToTheRight) {
			return 'left'
		}

		return willItFitToTheLeft ? 'left' : 'right'
	}

	if (!willItFitToTheLeft && !willItFitToTheRight) {
		return 'right'
	}

	return willItFitToTheRight ? 'right' : 'left'
}

const computeItemSubmenuFor = (
	reference,
	{
		// left -- 1st level menu items
		// end  -- submenus
		startPosition = 'end',
	}
) => {
	const menu = reference.querySelector('.sub-menu')
	const placement = getPreferedPlacementFor(menu)

	const { left, width, right } = menu.getBoundingClientRect()

	let futurePlacement = placement
	let referenceRect = reference.getBoundingClientRect()

	if (placement === 'left') {
		let referencePoint =
			startPosition === 'end' ? referenceRect.left : referenceRect.right

		if (referencePoint - width < 0) {
			futurePlacement = 'right'
		}
	}

	if (placement === 'right') {
		let referencePoint =
			startPosition === 'end' ? referenceRect.right : referenceRect.left

		if (referencePoint + width > innerWidth) {
			futurePlacement = 'left'
		}
	}

	reference.dataset.submenu = futurePlacement

	reference.addEventListener('click', () => {})
}

const openSubmenu = (e) => {
	const li = e.target.closest('li')

	if (li.__closeSubmenuTimer__) {
		clearTimeout(li.__closeSubmenuTimer__)
		li.__closeSubmenuTimer__ = null
	}

	li.classList.add('ct-active')

	let childIndicator = [...li.children].find((el) =>
		el.matches('.ct-toggle-dropdown-desktop-ghost')
	)

	if (!childIndicator) {
		childIndicator = li.firstElementChild
	}

	if (childIndicator) {
		childIndicator.setAttribute('aria-expanded', 'true')
		if (childIndicator.tagName.toLowerCase() === 'button') {
			childIndicator.setAttribute(
				'aria-label',
				ct_localizations.collapse_submenu
			)
		}
	}

	if (!isEligibleForSubmenu(li)) {
		return
	}

	const menu = li.querySelector('.sub-menu')

	mountMenuLevel(menu)

	if (menu.closest('[data-interaction="hover"]')) {
		menu.parentNode.addEventListener(
			'mouseleave',
			() => {
				;[...menu.children]
					.filter((el) => isEligibleForSubmenu(el))
					.map((el) => el.removeAttribute('data-submenu'))
			},
			{ once: true }
		)
	}
}

const closeSubmenu = (e) => {
	if (!e.target) {
		return
	}

	const li = e.target.closest('li')
	li.classList.remove('ct-active')

	let childIndicator = [...li.children].find((el) =>
		el.matches('.ct-toggle-dropdown-desktop-ghost')
	)

	if (!childIndicator) {
		childIndicator = li.firstElementChild
	}

	if (childIndicator) {
		childIndicator.setAttribute('aria-expanded', 'false')

		if (childIndicator.tagName.toLowerCase() === 'button') {
			childIndicator.setAttribute(
				'aria-label',
				ct_localizations.expand_submenu
			)
		}

		if (e.focusOnIndicator) {
			childIndicator.focus({
				focusVisible: true,
			})
		}
	}

	li.__closeSubmenuTimer__ = setTimeout(() => {
		li.__closeSubmenuTimer__ = null
		;[...li.querySelectorAll('[data-submenu]')].map((el) => {
			el.removeAttribute('data-submenu')
		})
		;[...li.querySelectorAll('.ct-active')].map((el) => {
			el.classList.remove('ct-active')
		})
	}, 30)
}

const mountMenuForElement = (el, args = {}) => {
	if (el.classList.contains('ct-mega-menu-custom-width')) {
		const menu = el.querySelector('.sub-menu')

		const elRect = el.getBoundingClientRect()
		const menuRect = menu.getBoundingClientRect()

		let centerFits =
			elRect.left + elRect.width / 2 > menuRect.width / 2 &&
			innerWidth - (elRect.left + elRect.width / 2) > menuRect.width / 2

		if (!centerFits) {
			const placement = getPreferedPlacementFor(menu)

			let offset = 0

			let edgeOffset = 15

			if (placement === 'right') {
				offset = `${
					Math.round(
						elRect.left - (innerWidth - menuRect.width) + edgeOffset
					) * -1
				}px`

				if (!(elRect.left + elRect.width / 2 > menuRect.width / 2)) {
					offset = `${Math.round(elRect.left - edgeOffset) * -1}px`
				}
			}

			if (placement === 'left') {
				offset = `${
					Math.round(innerWidth - elRect.right - edgeOffset) * -1
				}px`
			}

			el.dataset.submenu = placement

			menu.style.setProperty('--theme-submenu-inline-offset', offset)
		}
	}

	if (isEligibleForSubmenu(el)) {
		computeItemSubmenuFor(el, args)
	}

	if (el.hasSubmenuEventListeners) {
		return
	}

	el.hasSubmenuEventListeners = true

	let hasClickInteraction = el.matches('[data-interaction*="click"] *')

	el.addEventListener('keydown', function (e) {
		if (e.keyCode == 27) {
			closeSubmenu({
				target: el.firstElementChild,
				focusOnIndicator: true,
			})
		}
	})

	el.addEventListener('focusout', (evt) => {
		if (el.contains(evt.relatedTarget)) {
			return
		}

		closeSubmenu({
			target: el.firstElementChild,
		})
	})

	if (!hasClickInteraction) {
		const handleMouseEnter = () => {
			// So that mouseenter event is catched before the open itself
			if (isIosDevice()) {
				openSubmenu({ target: el.firstElementChild })
			} else {
				requestAnimationFrame(() => {
					openSubmenu({ target: el.firstElementChild })
				})
			}

			// If first level
			if (!el.parentNode.classList.contains('.sub-menu')) {
				;[...el.parentNode.children]
					.filter((firstLevelEl) => firstLevelEl !== el)
					.map((firstLevelEl) => {
						closeSubmenu({
							target: firstLevelEl.firstElementChild,
						})
					})
			}

			el.addEventListener(
				'mouseleave',
				() => {
					closeSubmenu({ target: el.firstElementChild })
				},
				{ once: true }
			)
		}

		el.addEventListener('mouseenter', handleMouseEnter)

		if (el.matches(':hover')) {
			handleMouseEnter()
		}

		// On Android devices, allow only 2nd click to open the link.
		// First click will ensure the submenu is opened
		//
		// iOS has this behaviour out of the box.
		//
		// Important: only perform this for touch devices so that keyboard
		// users are not affected.
		if (isTouchDevice()) {
			el.addEventListener('click', (e) => {
				if (!el.classList.contains('ct-active')) {
					e.preventDefault()
				}
			})
		}
	}

	if (hasClickInteraction) {
		let itemTarget = el.matches('[data-interaction*="item"] *')
			? el.firstElementChild
			: el.firstElementChild.querySelector('.ct-toggle-dropdown-desktop')

		itemTarget.addEventListener('click', (e) => {
			e.preventDefault()

			if (e.target.closest('li').classList.contains('ct-active')) {
				closeSubmenu(e)
			} else {
				openSubmenu(e)

				if (isIosDevice()) {
					e.target.closest('li').addEventListener(
						'mouseleave',
						() => {
							closeSubmenu({
								target: el.firstElementChild,
							})
						},
						{ once: true }
					)
				}

				if (!e.target.hasDocumentListener) {
					e.target.hasDocumentListener = true
					// Add the event a bit later
					setTimeout(() => {
						document.addEventListener('click', (evt) => {
							if (!e.target.closest('li').contains(evt.target)) {
								closeSubmenu(e)
							}
						})
					})
				}
			}
		})
	}

	let childIndicator = [...el.children].find((el) =>
		el.matches('.ct-toggle-dropdown-desktop-ghost')
	)

	if (childIndicator) {
		childIndicator.addEventListener('click', (e) => {
			if (e.target.closest('li').classList.contains('ct-active')) {
				closeSubmenu(e)
			} else {
				openSubmenu(e)
			}
		})
	}
}

export const mountMenuLevel = (menuLevel, args = {}) => {
	;[...menuLevel.children]
		.filter((el) =>
			el.matches('.menu-item-has-children, .page_item_has_children')
		)
		.map((el) => {
			mountMenuForElement(el, args)
		})
}