Newer
Older
export interface ShortcutLinks {
shortcut_title: string;
shortcut_url: string;
shortcut_is_external: string;
shortcut_aria_label: string;
}
export interface DesktopLinks {
label: string;
shortcutsTitle: string;
closeButtonTitle: string;
import {Component, h, Element, Prop, State, Watch, EventEmitter, Event, Listen} from '@stencil/core';
import ResizeObserver from 'resize-observer-polyfill';
@Component({
tag: 'hy-desktop-menu-links',
styleUrl: 'hy-desktop-menu-links.scss',
shadow: true,
})
export class HyDesktopMenuLinks {
@Element() el: HTMLElement;
/*
First level menu links to be displayed on Desktop screens.
* */
@Prop() dataDesktopLinks: DesktopLinks[] | string;
private _dataDesktopLinks: DesktopLinks[];
@State() firstLevelLinksList: Array<object> = [];
@State() menuLinkItems: Array<object> = [];
@State() hasToolbar: boolean = false;
@State() isDesktopMenuOpen: boolean = false;
@State() currenOpenMenuId: number = 0;
@Event() menuDesktopToggled: EventEmitter;
private _submenuLeftMargin: number = 32;
private _hoverTimer = null;
private ro: ResizeObserver;
@Watch('dataDesktopLinks')
dataDesktopLinksWatcher(data: DesktopLinks[] | string) {
this._dataDesktopLinks = typeof data === 'string' ? JSON.parse(data) : data;
}
componentWillLoad() {
this.dataDesktopLinksWatcher(this.dataDesktopLinks);
}
removeBackdropShadow(size: number) {
// Close backdrop shadow if the screen is < 1200px
if (size < 1200) {
this.showBackdropShadow();
}
}
showBackdropShadow(state = 'close', top = 0) {
let hyHeader = this.el.closest('.hy-site-header') as HTMLElement;
let hyBackdropDiv = hyHeader.children[0] as HTMLElement;
if (hyBackdropDiv) {
if (state === 'open') {
const windowHeight = window.outerHeight;
hyBackdropDiv.style.height = `${windowHeight}px`;
hyBackdropDiv.style.top = `${top}px`;
hyBackdropDiv.style.position = 'absolute';
hyBackdropDiv.classList.add('is-active');
hyBackdropDiv.removeAttribute('style');
hyBackdropDiv.classList.remove('is-active');
showPanel(id) {
this.menuDesktopToggled.emit();
const menuItems = this.el.shadowRoot.querySelectorAll(`.desktop-menu-link`);
const menuPanelItems = this.el.shadowRoot.querySelectorAll('.hy-desktop-menu-panel'); // all panels
const activeMenuItem = this.el.shadowRoot.querySelector(`.desktop-menu-link[link-id="${id}"]`) as HTMLElement;
const activeMenuItemSibling = activeMenuItem.nextElementSibling as HTMLElement; // current panel
// Reset elements by removing the active classes.
menuItems.forEach((item) => {
item.classList.remove('desktop-menu-link--is-active');
item.setAttribute('aria-expanded', 'false');
menuPanelItems.forEach((item) => {
item.classList.remove('hy-desktop-menu-panel--is-active');
item.classList.remove('hy-desktop-menu-panel--overflow');
item.setAttribute('aria-hidden', 'true');
(item as HTMLElement).style.transition = 'none';
(item as HTMLElement).style.opacity = '0';
});
// Add active classes to the currently active item and its sibling element.
activeMenuItem.classList.add('desktop-menu-link--is-active');
activeMenuItem.setAttribute('aria-expanded', 'true');
activeMenuItemSibling.classList.add('hy-desktop-menu-panel--is-active');
(activeMenuItemSibling as HTMLElement).style.opacity = '1';
if (this.hasToolbar) {
activeMenuItemSibling.classList.add('hy-desktop-menu-panel--is-active--has-toolbar');
}
activeMenuItemSibling.setAttribute('aria-hidden', 'false');
//Hide is-active-trail underlining
const activeTrailMenuItem = this.el.shadowRoot.querySelector(
`.desktop-menu-link__label--is-active-trail`
) as HTMLElement;
if (activeTrailMenuItem) {
activeTrailMenuItem.classList.add('desktop-menu-link__label--is-active-trail--disabled');
}
// Add panels top value automatically with the correct header height
let topOffset = window.pageYOffset;
const bodyElementClasses = document.body.classList;
const headerElement = document.querySelector('hy-site-header');
const headerShadowRootElement = headerElement.shadowRoot.querySelector('header');
let headerElementHeight = headerShadowRootElement.offsetHeight + this._headerBorderOffset;
let headerElementOffset = headerShadowRootElement.offsetTop;
let panelElementOffset = headerElementHeight + headerElementOffset;
// Add shadow backdrop
let rect = activeMenuItemSibling.getBoundingClientRect();
let backdropOffset = headerElementHeight + headerElementOffset + rect.height;
let backdropToolbarOffset = backdropOffset - 79;
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
window.addEventListener('scroll', () => {
topOffset = window.pageYOffset;
if (topOffset === 0) {
activeMenuItemSibling.style.top = `${panelElementOffset}px`;
}
if (headerShadowRootElement.classList.contains('hy-site-header--sticky-active') && topOffset > 0) {
if (bodyElementClasses.contains('toolbar-horizontal')) {
activeMenuItemSibling.style.top = `${headerElementHeight}px`;
if (this.isDesktopMenuOpen) {
this.showBackdropShadow('open', backdropToolbarOffset);
}
}
}
});
if (headerShadowRootElement.classList.contains('hy-site-header--sticky-active') && topOffset > 0) {
if (bodyElementClasses.contains('toolbar-horizontal')) {
activeMenuItemSibling.style.top = `${headerElementHeight}px`;
if (this.isDesktopMenuOpen) {
this.showBackdropShadow('open', backdropToolbarOffset);
}
} else {
activeMenuItemSibling.style.top = `${panelElementOffset}px`;
this.showBackdropShadow('open', backdropOffset);
}
} else {
activeMenuItemSibling.style.top = `${panelElementOffset}px`;
this.showBackdropShadow('open', backdropOffset);
}
// Position submenu block under the activated top menu item.
const menuPanelLeftPosition = activeMenuItem.offsetLeft - this._submenuLeftMargin;
activeMenuItemSibling.style.paddingLeft = `${menuPanelLeftPosition}px`;
// Position shortcuts block.
let shortcutsDiv = activeMenuItemSibling.querySelectorAll('ul.shortcuts-panel')[0] as HTMLElement; // shortcuts block
if (shortcutsDiv) {
let subMenuDiv = activeMenuItemSibling.querySelectorAll(
'.hy-desktop-menu-panel__desktop-menu__menu-items'
)[0] as HTMLElement; // 2nd level menu block
let spaceLeftAfterSubmenu = subMenuDiv.getBoundingClientRect().right + shortcutsDiv.offsetWidth;
if (spaceLeftAfterSubmenu >= document.body.scrollWidth) {
// Shortcuts should be placed to the left.
let shortcutsLeftPosition = subMenuDiv.getBoundingClientRect().left - shortcutsDiv.offsetWidth;
shortcutsDiv.style.left = shortcutsLeftPosition.toString().concat('px');
} else {
// Shortcuts should be placed to the right.
let shortcutsLeftPosition = subMenuDiv.getBoundingClientRect().right;
shortcutsDiv.style.left = shortcutsLeftPosition.toString().concat('px');
}
}
}
closePanel(fadeOut = false) {
this.isDesktopMenuOpen = false;
this.currenOpenMenuId = 0;
this.showBackdropShadow();
this.clearPanelItemsStatus(fadeOut);
clearTimeout(this._hoverTimer);
clearPanelItemsStatus(fadeOut = false) {
const menuItems = this.el.shadowRoot.querySelectorAll(`.desktop-menu-link`);
const menuPanelItems = this.el.shadowRoot.querySelectorAll('.hy-desktop-menu-panel');
//Show is-active-trail underlining
const activeTrailMenuItem = this.el.shadowRoot.querySelector(
`.desktop-menu-link__label--is-active-trail`
) as HTMLElement;
if (activeTrailMenuItem) {
activeTrailMenuItem.classList.remove('desktop-menu-link__label--is-active-trail--disabled');
}
// Reset elements by removing the active classes.
menuItems.forEach((item) => {
item.classList.remove('desktop-menu-link--is-active');
item.setAttribute('aria-expanded', 'false');
});
if (fadeOut) {
menuPanelItems.forEach((item) => {
(item as HTMLElement).style.opacity = '0';
(item as HTMLElement).style.transition = 'opacity 1s';
});
this._fadeOutTimer = setTimeout(() => {
menuPanelItems.forEach((item) => {
item.classList.remove('hy-desktop-menu-panel--is-active');
item.setAttribute('aria-hidden', 'true');
});
} else {
menuPanelItems.forEach((item) => {
item.classList.remove('hy-desktop-menu-panel--is-active');
item.setAttribute('aria-hidden', 'true');
(item as HTMLElement).style.opacity = '0';
(item as HTMLElement).style.transition = 'none';
handleDesktopMenuClose(event) {
let fadeOut = true;
this.closePanel(fadeOut);
// CLose the desktop menu panel if user opens the language menu.
@Listen('menuLanguageToggled', {target: 'document'})
menuLanguageToggled() {
let fadeOut = true;
this.closePanel(fadeOut);
}
// Close the desktop menu panel if user opens University main menu
@Listen('universityMainMenuToggled', {target: 'document'})
universityMainMenuPanelToggled() {
let fadeOut = true;
this.closePanel(fadeOut);
}
// Close the desktop menu panel if user opens search panel
@Listen('searchPanelToggled', {target: 'document'})
searchPanelToggled() {
let fadeOut = true;
this.closePanel(fadeOut);
}
// CLose the desktop menu panel if user scrolls Sticky Header till the very top.
@Listen('headerScrollUp', {target: 'document'})
headerScrollUp() {
let fadeOut = false;
this.closePanel(fadeOut);
}
this._hoverTimer = setTimeout(() => {
const activeMenuItem = this.el.shadowRoot.querySelector(`.desktop-menu-link[link-id="${id}"]`) as HTMLElement;
// Set focus to the button.
if (activeMenuItem !== null) activeMenuItem.focus();
this.currenOpenMenuId = id;
this.showPanel(id);
}, 350);
handleDesktopMenuLeave(event) {
let leaveEvent = event as MouseEvent;
let hyHeader = this.el.closest('.hy-site-header') as HTMLElement;
const headerHeight = hyHeader.offsetTop + hyHeader.offsetHeight;
if (leaveEvent.clientY < headerHeight - 4) {
this.closePanel();
}
event.stopPropagation();
}
/*
Close the panel if mouse is moving over the menu label.
* */
if (this.isDesktopMenuOpen) {
let moveEvent = event as MouseEvent;
const activeMenuItem = this.el.shadowRoot.querySelector(
`.desktop-menu-link[link-id="${id}"] .desktop-menu-link__label`
) as HTMLElement;
let topBorder = activeMenuItem.getClientRects()[0].top;
if (this.currenOpenMenuId == id) {
// Mouse moving around the same menu link
if (moveEvent.clientY < topBorder - 4) {
}
handleDesktopMenuFocus(event, id) {
this.currenOpenMenuId = id;
this.showPanel(id);
}
event.stopPropagation();
}
handleDesktopMenuClick(event, id) {
if (!this.isDesktopMenuOpen) {
this.currenOpenMenuId = id;
this.showPanel(id);
} else {
this.handleDesktopMenuClose(event);
}
event.stopPropagation();
}
componentDidLoad() {
// Set the browser resize observer to gather information about browser width.
this.ro = new ResizeObserver((entries) => {
for (const entry of entries) {
this.removeBackdropShadow(entry.contentRect.width);
}
});
this.ro.observe(document.body);
let hyToolbar = document.querySelectorAll('#toolbar-administration')[0];
if (hyToolbar) {
this.hasToolbar = true;
}
const links = this._dataDesktopLinks as Array<DesktopLinks>;
let menuLinkItems = [];
if (links) {
links.map(
({
menuLinkId: id,
shortcuts,
items,
url,
description,
label,
labelExtra,
isActive,
shortcutsTitle,
closeButtonTitle,
}) => {
let classAttributes = [
'desktop-menu-link',
isActive === 'true' ? 'desktop-menu-link--is-active-trail' : '',
].join(' ');
let classAttributesLabel = [
'desktop-menu-link__label',
isActive === 'true' ? 'desktop-menu-link__label--is-active-trail' : '',
].join(' ');
type="button"
class={classAttributes}
link-id={id}
onFocus={(e) => this.handleDesktopMenuFocus(e, id)}
onMouseEnter={(e) => this.handleDesktopMenuEnter(e, id)}
onMouseMove={(e) => this.handleDesktopMenuMove(e, id)}
<span class={classAttributesLabel}>{label}</span>
<div class="hy-desktop-menu-panel" aria-hidden="true">
<div class="hy-desktop-menu-panel__desktop-menu">
<div class="hy-desktop-menu-panel__desktop-menu__menu-items">
<a
aria-current={label}
href={url}
class="hy-desktop-menu-panel__desktop-menu__first-level-menu-item"
menu-link-id={id}
>
<span class="heading-icon">
<hy-icon icon={'hy-icon-arrow-right'} size={40} />
</span>
{labelExtra ? <span class="label">{labelExtra}</span> : <span class="label">{label}</span>}
{description && <span class="description">{description}</span>}
</a>
<ul class={'hy-desktop-menu-panel__desktop-menu__second-level-menu'} menu-link-id={id}>
{items.map(({label, url, isExternal}) => {
let subitemTarget = isExternal ? '_blank' : '_self';
return (
<li>
<a href={url} target={subitemTarget}>
<span class="heading-icon">
<hy-icon icon={'hy-icon-caret-right'} size={12} />
</span>
<span class="label">{label}</span>
{isExternal && (
<span class="external-icon">
<hy-icon icon={'hy-icon-arrow-right'} size={12} />
</span>
)}
</a>
</li>
);
})}
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
</ul>
</div>
{shortcuts.length > 0 && (
<ul class="shortcuts-panel">
<h2 class="shortcuts-panel__title">{shortcutsTitle}</h2>
{shortcuts.map(
({shortcut_title, shortcut_url, shortcut_is_external, shortcut_aria_label}, index) => {
let target = shortcut_is_external ? '_blank' : '_self';
let shortcutClass = [
'shortcuts-panel__shortcut-item',
index == 0 ? 'shortcuts-panel__shortcut-item__first' : '',
].join(' ');
return (
<li class={shortcutClass}>
<a
aria-current={shortcut_aria_label}
href={shortcut_url}
class="shortcut-item__link"
target={target}
aria-label={shortcut_aria_label}
>
<span class="label">{shortcut_title}</span>
<span class="icon">
<hy-icon icon={'hy-icon-arrow-right'} size={24} />
</span>
</a>
</li>
);
}
)}
</ul>
)}
</div>
<button
onClick={(e) => this.handleDesktopMenuClose(e)}
class={{
'hy-desktop-menu-panel__panel-toggle': true,
}}
aria-label="Close menu"
>
<span class="hy-desktop-menu-panel__panel-toggle__label">
<span class="hy-desktop-menu-panel__panel-toggle__label__title">{closeButtonTitle}</span>
<hy-icon icon={'hy-icon-remove'} size={20} fill={ColorVariant.black} />
</span>
</button>
</div>
</li>
);
}
);
}
this.menuLinkItems = menuLinkItems;
<nav
role={'navigation'}
class="hy-site-header__menu-desktop"
onMouseLeave={(e) => this.handleDesktopMenuClose(e)}
>
<ul class="hy-site-header__menu-desktop-container">{this.menuLinkItems}</ul>