Files
2025-06-17 11:53:18 +02:00

1492 lines
58 KiB
JavaScript

/*! skipto - v4.1.7 - 2023-01-13
* https://github.com/paypal/skipto
* Copyright (c) 2023 PayPal Accessibility Team and University of Illinois; Licensed */
/*@cc_on @*/
/*@if (@_jscript_version >= 5.8) @*/
/* ========================================================================
* Copyright (c) <2021> PayPal and University of Illinois
* All rights reserved.
* Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
* Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
* Neither the name of PayPal or any of its subsidiaries or affiliates nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission.
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
* ======================================================================== */
(function() {
'use strict';
var SkipTo = {
skipToId: 'id-skip-to-js-4',
skipToMenuId: 'id-skip-to-menu-4',
domNode: null,
buttonNode: null,
menuNode: null,
tooltipNode: null,
menuitemNodes: [],
firstMenuitem: false,
lastMenuitem: false,
firstChars: [],
headingLevels: [],
skipToIdIndex: 1,
showAllLandmarksSelector: 'main, [role=main], [role=search], nav, [role=navigation], section[aria-label], section[aria-labelledby], section[title], [role=region][aria-label], [role=region][aria-labelledby], [role=region][title], form[aria-label], form[aria-labelledby], aside, [role=complementary], body > header, [role=banner], body > footer, [role=contentinfo]',
showAllHeadingsSelector: 'h1, h2, h3, h4, h5, h6',
showTooltipFocus: false,
showTooltipHover: false,
tooltipTimerDelay: 500, // in milliseconds
// Default configuration values
config: {
// Feature switches
enableActions: false,
enableMofN: true,
enableHeadingLevelShortcuts: true,
enableHelp: true,
enableTooltip: true,
// Customization of button and menu
accesskey: '0', // default is the number zero
attachElement: 'header',
displayOption: 'static', // options: static (default), popup
// container element, use containerClass for custom styling
containerElement: 'div',
containerRole: '',
customClass: '',
// Button labels and messages
buttonTitle: '', // deprecated in favor of buttonTooltip
buttonTitleWithAccesskey: '', // deprecated in favor of buttonTooltipAccesskey
buttonTooltip: '',
buttonTooltipAccesskey: 'Shortcut Key: $key',
buttonLabel: 'Skip To Content',
// Menu labels and messages
menuLabel: 'Landmarks and Headings',
landmarkGroupLabel: 'Landmarks',
headingGroupLabel: 'Headings',
mofnGroupLabel: ' ($m of $n)',
headingLevelLabel: 'Heading level',
mainLabel: 'main',
searchLabel: 'search',
navLabel: 'navigation',
regionLabel: 'region',
asideLabel: 'complementary',
footerLabel: 'contentinfo',
headerLabel: 'banner',
formLabel: 'form',
msgNoLandmarksFound: 'No landmarks found',
msgNoHeadingsFound: 'No headings found',
// Action labels and messages
actionGroupLabel: 'Actions',
actionShowHeadingsHelp: 'Toggles between showing "All" and "Selected" Headings.',
actionShowSelectedHeadingsLabel: 'Show Selected Headings ($num)',
actionShowAllHeadingsLabel: 'Show All Headings ($num)',
actionShowLandmarksHelp: 'Toggles between showing "All" and "Selected" Landmarks.',
actionShowSelectedLandmarksLabel: 'Show Selected Landmarks ($num)',
actionShowAllLandmarksLabel: 'Show All Landmarks ($num)',
actionShowSelectedHeadingsAriaLabel: 'Show $num selected headings',
actionShowAllHeadingsAriaLabel: 'Show all $num headings',
actionShowSelectedLandmarksAriaLabel: 'Show $num selected landmarks',
actionShowAllLandmarksAriaLabel: 'Show all $num landmarks',
// Selectors for landmark and headings sections
landmarks: 'main, [role="main"], [role="search"], nav, [role="navigation"], aside, [role="complementary"]',
headings: 'main h1, [role="main"] h1, main h2, [role="main"] h2',
// Custom CSS position and colors
colorTheme: '',
fontFamily: '',
fontSize: '',
positionLeft: '',
menuTextColor: '',
menuBackgroundColor: '',
menuitemFocusTextColor: '',
menuitemFocusBackgroundColor: '',
focusBorderColor: '',
buttonTextColor: '',
buttonBackgroundColor: '',
},
colorThemes: {
'default': {
fontFamily: 'inherit',
fontSize: 'inherit',
positionLeft: '46%',
menuTextColor: '#1a1a1a',
menuBackgroundColor: '#dcdcdc',
menuitemFocusTextColor: '#eeeeee',
menuitemFocusBackgroundColor: '#1a1a1a',
focusBorderColor: '#1a1a1a',
buttonTextColor: '#1a1a1a',
buttonBackgroundColor: '#eeeeee',
},
'illinois': {
fontFamily: 'inherit',
fontSize: 'inherit',
positionLeft: '46%',
menuTextColor: '#00132c',
menuBackgroundColor: '#cad9ef',
menuitemFocusTextColor: '#eeeeee',
menuitemFocusBackgroundColor: '#00132c',
focusBorderColor: '#ff552e',
buttonTextColor: '#444444',
buttonBackgroundColor: '#dddede',
},
'aria': {
fontFamily: 'sans-serif',
fontSize: '10pt',
positionLeft: '7%',
menuTextColor: '#000',
menuBackgroundColor: '#def',
menuitemFocusTextColor: '#fff',
menuitemFocusBackgroundColor: '#005a9c',
focusBorderColor: '#005a9c',
buttonTextColor: '#005a9c',
buttonBackgroundColor: '#ddd',
}
},
defaultCSS: '.skip-to.popup{position:absolute;top:-30em;left:0}.skip-to,.skip-to.popup.focus{position:absolute;top:0;left:$positionLeft;font-family:$fontFamily;font-size:$fontSize}.skip-to.fixed{position:fixed}.skip-to button{position:relative;margin:0;padding:6px 8px 6px 8px;border-width:0 1px 1px 1px;border-style:solid;border-radius:0 0 6px 6px;border-color:$buttonBackgroundColor;color:$menuTextColor;background-color:$buttonBackgroundColor;z-index:200;font-family:$fontFamily;font-size:$fontSize}.skip-to .skip-to-tooltip{position:absolute;top:2.25em;left:8em;margin:1px;padding:4px;border:1px solid #ccc;box-shadow:2px 3px 5px #ddd;background-color:#eee;color:#000;font-family:Helvetica,Arial,Sans-Serif;font-variant-numeric:slashed-zero;font-size:9pt;width:auto;display:none;white-space:nowrap;z-index:201}.skip-to .skip-to-tooltip.skip-to-show-tooltip{display:block}.skip-to [aria-expanded=true]+.skip-to-tooltip.skip-to-show-tooltip{display:none}.skip-to [role=menu]{position:absolute;min-width:17em;display:none;margin:0;padding:.25rem;background-color:$menuBackgroundColor;border-width:2px;border-style:solid;border-color:$focusBorderColor;border-radius:5px;z-index:1000}.skip-to [role=group]{display:grid;grid-auto-rows:min-content;grid-row-gap:1px}.skip-to [role=separator]:first-child{border-radius:5px 5px 0 0}.skip-to [role=menuitem]{padding:3px;width:auto;border-width:0;border-style:solid;color:$menuTextColor;background-color:$menuBackgroundColor;z-index:1000;display:grid;overflow-y:auto;grid-template-columns:repeat(6,1.2rem) 1fr;grid-column-gap:2px;font-size:1em}.skip-to [role=menuitem] .label,.skip-to [role=menuitem] .level{font-size:100%;font-weight:400;color:$menuTextColor;display:inline-block;background-color:$menuBackgroundColor;line-height:inherit;display:inline-block}.skip-to [role=menuitem] .level{text-align:right;padding-right:4px}.skip-to [role=menuitem] .label{text-align:left;margin:0;padding:0;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}[dir=rtl] .skip-to [role=menuitem] .label{text-align:right}.skip-to [role=menuitem] .label:first-letter,.skip-to [role=menuitem] .level:first-letter{text-decoration:underline;text-transform:uppercase}.skip-to [role=menuitem].skip-to-h1 .level{grid-column:1}.skip-to [role=menuitem].skip-to-h2 .level{grid-column:2}.skip-to [role=menuitem].skip-to-h3 .level{grid-column:3}.skip-to [role=menuitem].skip-to-h4 .level{grid-column:4}.skip-to [role=menuitem].skip-to-h5 .level{grid-column:5}.skip-to [role=menuitem].skip-to-h6 .level{grid-column:8}.skip-to [role=menuitem].skip-to-h1 .label{grid-column:2/8}.skip-to [role=menuitem].skip-to-h2 .label{grid-column:3/8}.skip-to [role=menuitem].skip-to-h3 .label{grid-column:4/8}.skip-to [role=menuitem].skip-to-h4 .label{grid-column:5/8}.skip-to [role=menuitem].skip-to-h5 .label{grid-column:6/8}.skip-to [role=menuitem].skip-to-h6 .label{grid-column:7/8}.skip-to [role=menuitem].skip-to-h1.no-level .label{grid-column:1/8}.skip-to [role=menuitem].skip-to-h2.no-level .label{grid-column:2/8}.skip-to [role=menuitem].skip-to-h3.no-level .label{grid-column:3/8}.skip-to [role=menuitem].skip-to-h4.no-level .label{grid-column:4/8}.skip-to [role=menuitem].skip-to-h5.no-level .label{grid-column:5/8}.skip-to [role=menuitem].skip-to-h6.no-level .label{grid-column:6/8}.skip-to [role=menuitem].skip-to-nesting-level-1 .nesting{grid-column:1}.skip-to [role=menuitem].skip-to-nesting-level-2 .nesting{grid-column:2}.skip-to [role=menuitem].skip-to-nesting-level-3 .nesting{grid-column:3}.skip-to [role=menuitem].skip-to-nesting-level-0 .label{grid-column:1/8}.skip-to [role=menuitem].skip-to-nesting-level-1 .label{grid-column:2/8}.skip-to [role=menuitem].skip-to-nesting-level-2 .label{grid-column:3/8}.skip-to [role=menuitem].skip-to-nesting-level-3 .label{grid-column:4/8}.skip-to [role=menuitem].action .label,.skip-to [role=menuitem].no-items .label{grid-column:1/8}.skip-to [role=separator]{margin:1px 0 1px 0;padding:3px;display:block;width:auto;font-weight:700;border-bottom-width:1px;border-bottom-style:solid;border-bottom-color:$menuTextColor;background-color:$menuBackgroundColor;color:$menuTextColor;z-index:1000}.skip-to [role=separator] .mofn{font-weight:400;font-size:85%}.skip-to [role=separator]:first-child{border-radius:5px 5px 0 0}.skip-to [role=menuitem].last{border-radius:0 0 5px 5px}.skip-to.focus{display:block}.skip-to button:focus,.skip-to button:hover{background-color:$menuBackgroundColor;color:$menuTextColor;outline:0}.skip-to button:focus{padding:6px 7px 5px 7px;border-width:0 2px 2px 2px;border-color:$focusBorderColor}.skip-to [role=menuitem]:focus{padding:1px;border-width:2px;border-style:solid;border-color:$focusBorderColor;background-color:$menuitemFocusBackgroundColor;color:$menuitemFocusTextColor;outline:0}.skip-to [role=menuitem]:focus .label,.skip-to [role=menuitem]:focus .level{background-color:$menuitemFocusBackgroundColor;color:$menuitemFocusTextColor}',
//
// Functions related to configuring the features
// of skipTo
//
isNotEmptyString: function(str) {
return (typeof str === 'string') && str.length && str.trim() && str !== "&nbsp;";
},
isEmptyString: function(str) {
return (typeof str !== 'string') || str.length === 0 && !str.trim();
},
init: function(config) {
var node;
// Check if skipto is already loaded
if (document.querySelector('style#' + this.skipToId)) {
return;
}
var attachElement = document.body;
if (config) {
this.setUpConfig(config);
}
if (typeof this.config.attachElement === 'string') {
node = document.querySelector(this.config.attachElement);
if (node && node.nodeType === Node.ELEMENT_NODE) {
attachElement = node;
}
}
this.addCSSColors();
this.renderStyleElement(this.defaultCSS);
var elem = this.config.containerElement.toLowerCase().trim();
if (!this.isNotEmptyString(elem)) {
elem = 'div';
}
this.domNode = document.createElement(elem);
this.domNode.classList.add('skip-to');
if (this.isNotEmptyString(this.config.customClass)) {
this.domNode.classList.add(this.config.customClass);
}
if (this.isNotEmptyString(this.config.containerRole)) {
this.domNode.setAttribute('role', this.config.containerRole);
}
var displayOption = this.config.displayOption;
if (typeof displayOption === 'string') {
displayOption = displayOption.trim().toLowerCase();
if (displayOption.length) {
switch (this.config.displayOption) {
case 'fixed':
this.domNode.classList.add('fixed');
break;
case 'onfocus': // Legacy option
case 'popup':
this.domNode.classList.add('popup');
break;
default:
break;
}
}
}
// Place skip to at the beginning of the document
if (attachElement.firstElementChild) {
attachElement.insertBefore(this.domNode, attachElement.firstElementChild);
} else {
attachElement.appendChild(this.domNode);
}
this.buttonNode = document.createElement('button');
this.buttonNode.textContent = this.config.buttonLabel;
this.buttonNode.setAttribute('aria-haspopup', 'true');
this.buttonNode.setAttribute('aria-expanded', 'false');
this.buttonNode.setAttribute('aria-controls', this.skipToMenuId);
this.buttonNode.setAttribute('accesskey', this.config.accesskey);
this.domNode.appendChild(this.buttonNode);
this.renderTooltip(this.domNode, this.buttonNode);
this.menuNode = document.createElement('div');
this.menuNode.setAttribute('role', 'menu');
this.menuNode.setAttribute('aria-busy', 'true');
this.menuNode.setAttribute('id', this.skipToMenuId);
this.domNode.appendChild(this.menuNode);
this.buttonNode.addEventListener('keydown', this.handleButtonKeydown.bind(this));
this.buttonNode.addEventListener('click', this.handleButtonClick.bind(this));
this.buttonNode.addEventListener('focus', this.handleButtonFocus.bind(this));
this.buttonNode.addEventListener('blur', this.handleButtonBlur.bind(this));
this.buttonNode.addEventListener('pointerenter', this.handleButtonPointerenter.bind(this));
this.buttonNode.addEventListener('pointerout', this.handleButtonPointerout.bind(this));
this.domNode.addEventListener('focusin', this.handleFocusin.bind(this));
this.domNode.addEventListener('focusout', this.handleFocusout.bind(this));
window.addEventListener('pointerdown', this.handleBackgroundPointerdown.bind(this), true);
},
renderTooltip: function(attachNode, buttonNode) {
var id = 'id-skip-to-tooltip';
var accesskey = this.getBrowserSpecificAccesskey(this.config.accesskey);
var tooltip = this.config.buttonTooltip;
// for backward compatibility, support 'this.config.buttonTitle' if defined
if (this.isNotEmptyString(this.config.buttonTitle)) {
tooltip = this.config.buttonTitle;
}
this.tooltipLeft = buttonNode.getBoundingClientRect().width;
this.tooltipTop = buttonNode.getBoundingClientRect().height;
this.tooltipNode = document.createElement('div');
this.tooltipNode.setAttribute('role', 'tooltip');
this.tooltipNode.id = id;
this.tooltipNode.classList.add('skip-to-tooltip');
if (this.isNotEmptyString(accesskey)) {
tooltip = this.config.buttonTooltipAccesskey.replace('$key', accesskey);
// for backward compatibility support 'buttonTitleWithAccesskey' if defined
if (this.isNotEmptyString(this.config.buttonTitleWithAccesskey)) {
tooltip = this.config.buttonTitleWithAccesskey.replace('$key', accesskey);
}
}
if (this.isEmptyString(tooltip)) {
// if there is no tooltip information
// do not display tooltip
this.config.enableTooltip = false;
} else {
this.tooltipNode.textContent = tooltip;
}
attachNode.appendChild(this.tooltipNode);
this.tooltipNode.style.left = this.tooltipLeft + 'px';
this.tooltipNode.style.top = this.tooltipTop + 'px';
// Temporarily show the tooltip to get rendered height
this.tooltipNode.classList.add('skip-to-show-tooltip');
this.tooltipHeight = this.tooltipNode.getBoundingClientRect().height;
this.tooltipNode.classList.remove('skip-to-show-tooltip');
},
updateStyle: function(stylePlaceholder, value, defaultValue) {
if (typeof value !== 'string' || value.length === 0) {
value = defaultValue;
}
var index1 = this.defaultCSS.indexOf(stylePlaceholder);
var index2 = index1 + stylePlaceholder.length;
while (index1 >= 0 && index2 < this.defaultCSS.length) {
this.defaultCSS = this.defaultCSS.substring(0, index1) + value + this.defaultCSS.substring(index2);
index1 = this.defaultCSS.indexOf(stylePlaceholder, index2);
index2 = index1 + stylePlaceholder.length;
}
},
addCSSColors: function() {
var theme = this.colorThemes['default'];
if (typeof this.colorThemes[this.config.colorTheme] === 'object') {
theme = this.colorThemes[this.config.colorTheme];
}
this.updateStyle('$fontFamily', this.config.fontFamily, theme.fontFamily);
this.updateStyle('$fontSize', this.config.fontSize, theme.fontSize);
this.updateStyle('$positionLeft', this.config.positionLeft, theme.positionLeft);
this.updateStyle('$menuTextColor', this.config.menuTextColor, theme.menuTextColor);
this.updateStyle('$menuBackgroundColor', this.config.menuBackgroundColor, theme.menuBackgroundColor);
this.updateStyle('$menuitemFocusTextColor', this.config.menuitemFocusTextColor, theme.menuitemFocusTextColor);
this.updateStyle('$menuitemFocusBackgroundColor', this.config.menuitemFocusBackgroundColor, theme.menuitemFocusBackgroundColor);
this.updateStyle('$focusBorderColor', this.config.focusBorderColor, theme.focusBorderColor);
this.updateStyle('$buttonTextColor', this.config.buttonTextColor, theme.buttonTextColor);
this.updateStyle('$buttonBackgroundColor', this.config.buttonBackgroundColor, theme.buttonBackgroundColor);
},
getBrowserSpecificAccesskey: function (accesskey) {
var userAgent = navigator.userAgent.toLowerCase();
var platform = navigator.platform.toLowerCase();
var hasWin = platform.indexOf('win') >= 0;
var hasMac = platform.indexOf('mac') >= 0;
var hasLinux = platform.indexOf('linux') >= 0 || platform.indexOf('bsd') >= 0;
var hasAndroid = userAgent.indexOf('android') >= 0;
var hasFirefox = userAgent.indexOf('firefox') >= 0;
var hasChrome = userAgent.indexOf('chrome') >= 0;
var hasOpera = userAgent.indexOf('opr') >= 0;
if (typeof accesskey !== 'string' || accesskey.length === 0) {
return '';
}
if (hasWin || (hasLinux && !hasAndroid)) {
if (hasFirefox) {
return "Shift + Alt + " + accesskey;
} else {
if (hasChrome || hasOpera) {
return "Alt + " + accesskey;
}
}
}
if (hasMac) {
return "Ctrl + Option + " + accesskey;
}
return '';
},
setUpConfig: function(appConfig) {
var localConfig = this.config,
name,
appConfigSettings = typeof appConfig.settings !== 'undefined' ? appConfig.settings.skipTo : {};
for (name in appConfigSettings) {
//overwrite values of our local config, based on the external config
if ((typeof localConfig[name] !== 'undefined') &&
((typeof appConfigSettings[name] === 'string') &&
(appConfigSettings[name].length > 0 ) ||
typeof appConfigSettings[name] === 'boolean')
) {
localConfig[name] = appConfigSettings[name];
} else {
throw new Error('** SkipTo Problem with user configuration option "' + name + '".');
}
}
},
renderStyleElement: function(cssString) {
var styleNode = document.createElement('style');
var headNode = document.getElementsByTagName('head')[0];
var css = document.createTextNode(cssString);
styleNode.setAttribute("type", "text/css");
// ID is used to test whether skipto is already loaded
styleNode.id = this.skipToId;
styleNode.appendChild(css);
headNode.appendChild(styleNode);
},
//
// Functions related to creating and populating the
// the popup menu
//
getFirstChar: function(menuitem) {
var c = '';
var label = menuitem.querySelector('.label');
if (label && this.isNotEmptyString(label.textContent)) {
c = label.textContent.trim()[0].toLowerCase();
}
return c;
},
getHeadingLevelFromAttribute: function(menuitem) {
var level = '';
if (menuitem.hasAttribute('data-level')) {
level = menuitem.getAttribute('data-level');
}
return level;
},
updateKeyboardShortCuts: function () {
var mi;
this.firstChars = [];
this.headingLevels = [];
for(var i = 0; i < this.menuitemNodes.length; i += 1) {
mi = this.menuitemNodes[i];
this.firstChars.push(this.getFirstChar(mi));
this.headingLevels.push(this.getHeadingLevelFromAttribute(mi));
}
},
updateMenuitems: function () {
var menuitemNodes = this.menuNode.querySelectorAll('[role=menuitem');
this.menuitemNodes = [];
for(var i = 0; i < menuitemNodes.length; i += 1) {
this.menuitemNodes.push(menuitemNodes[i]);
}
this.firstMenuitem = this.menuitemNodes[0];
this.lastMenuitem = this.menuitemNodes[this.menuitemNodes.length-1];
this.lastMenuitem.classList.add('last');
this.updateKeyboardShortCuts();
},
renderMenuitemToGroup: function (groupNode, mi) {
var tagNode, tagNodeChild, labelNode, nestingNode;
var menuitemNode = document.createElement('div');
menuitemNode.setAttribute('role', 'menuitem');
menuitemNode.classList.add(mi.class);
if (this.isNotEmptyString(mi.tagName)) {
menuitemNode.classList.add('skip-to-' + mi.tagName.toLowerCase());
}
menuitemNode.setAttribute('data-id', mi.dataId);
menuitemNode.tabIndex = -1;
if (this.isNotEmptyString(mi.ariaLabel)) {
menuitemNode.setAttribute('aria-label', mi.ariaLabel);
}
// add event handlers
menuitemNode.addEventListener('keydown', this.handleMenuitemKeydown.bind(this));
menuitemNode.addEventListener('click', this.handleMenuitemClick.bind(this));
menuitemNode.addEventListener('pointerenter', this.handleMenuitemPointerenter.bind(this));
groupNode.appendChild(menuitemNode);
// add heading level and label
if (mi.class.includes('heading')) {
if (this.config.enableHeadingLevelShortcuts) {
tagNode = document.createElement('span');
tagNodeChild = document.createElement('span');
tagNodeChild.appendChild(document.createTextNode(mi.level));
tagNode.append(tagNodeChild);
tagNode.appendChild(document.createTextNode(')'));
tagNode.classList.add('level');
menuitemNode.append(tagNode);
} else {
menuitemNode.classList.add('no-level');
}
menuitemNode.setAttribute('data-level', mi.level);
if (this.isNotEmptyString(mi.tagName)) {
menuitemNode.classList.add('skip-to-' + mi.tagName);
}
}
// add nesting level for landmarks
if (mi.class.includes('landmark')) {
menuitemNode.setAttribute('data-nesting', mi.nestingLevel);
menuitemNode.classList.add('skip-to-nesting-level-' + mi.nestingLevel);
if (mi.nestingLevel > 0 && (mi.nestingLevel > this.lastNestingLevel)) {
nestingNode = document.createElement('span');
nestingNode.classList.add('nesting');
menuitemNode.append(nestingNode);
}
this.lastNestingLevel = mi.nestingLevel;
}
labelNode = document.createElement('span');
labelNode.appendChild(document.createTextNode(mi.name));
labelNode.classList.add('label');
menuitemNode.append(labelNode);
return menuitemNode;
},
renderGroupLabel: function (groupLabelId, title, m, n) {
var titleNode, mofnNode, s;
var groupLabelNode = document.getElementById(groupLabelId);
titleNode = groupLabelNode.querySelector('.title');
mofnNode = groupLabelNode.querySelector('.mofn');
titleNode.textContent = title;
if (this.config.enableActions && this.config.enableMofN) {
if ((typeof m === 'number') && (typeof n === 'number')) {
s = this.config.mofnGroupLabel;
s = s.replace('$m', m);
s = s.replace('$n', n);
mofnNode.textContent = s;
}
}
},
renderMenuitemGroup: function(groupId, title) {
var labelNode, groupNode, spanNode;
var menuNode = this.menuNode;
if (this.isNotEmptyString(title)) {
labelNode = document.createElement('div');
labelNode.id = groupId + "-label";
labelNode.setAttribute('role', 'separator');
menuNode.appendChild(labelNode);
spanNode = document.createElement('span');
spanNode.classList.add('title');
spanNode.textContent = title;
labelNode.append(spanNode);
spanNode = document.createElement('span');
spanNode.classList.add('mofn');
labelNode.append(spanNode);
groupNode = document.createElement('div');
groupNode.setAttribute('role', 'group');
groupNode.setAttribute('aria-labelledby', labelNode.id);
groupNode.id = groupId;
menuNode.appendChild(groupNode);
menuNode = groupNode;
}
return groupNode;
},
removeMenuitemGroup: function(groupId) {
var node = document.getElementById(groupId);
this.menuNode.removeChild(node);
node = document.getElementById(groupId + "-label");
this.menuNode.removeChild(node);
},
renderMenuitemsToGroup: function(groupNode, menuitems, msgNoItemsFound) {
groupNode.innerHTML = '';
this.lastNestingLevel = 0;
if (menuitems.length === 0) {
var item = {};
item.name = msgNoItemsFound;
item.tagName = '';
item.class = 'no-items';
item.dataId = '';
this.renderMenuitemToGroup(groupNode, item);
}
else {
for (var i = 0; i < menuitems.length; i += 1) {
this.renderMenuitemToGroup(groupNode, menuitems[i]);
}
}
},
getShowMoreHeadingsSelector: function(option) {
if (option === 'all') {
return this.showAllHeadingsSelector;
}
return this.config.headings;
},
getShowMoreHeadingsLabel: function(option, n) {
var label = this.config.actionShowSelectedHeadingsLabel;
if (option === 'all') {
label = this.config.actionShowAllHeadingsLabel;
}
return label.replace('$num', n);
},
getShowMoreHeadingsAriaLabel: function(option, n) {
var label = this.config.actionShowSelectedHeadingsAriaLabel;
if (option === 'all') {
label = this.config.actionShowAllHeadingsAriaLabel;
}
return label.replace('$num', n);
},
renderActionMoreHeadings: function(groupNode) {
var item, menuitemNode;
var option = 'all';
var selectedHeadingsLen = this.getHeadings(this.getShowMoreHeadingsSelector('selected')).length;
var allHeadingsLen = this.getHeadings(this.getShowMoreHeadingsSelector('all')).length;
var noAction = selectedHeadingsLen === allHeadingsLen;
var headingsLen = allHeadingsLen;
if (option !== 'all') {
headingsLen = selectedHeadingsLen;
}
if (!noAction) {
item = {};
item.tagName = '';
item.role = 'menuitem';
item.class = 'action';
item.dataId = 'skip-to-more-headings';
item.name = this.getShowMoreHeadingsLabel(option, headingsLen);
item.ariaLabel = this.getShowMoreHeadingsAriaLabel(option, headingsLen);
menuitemNode = this.renderMenuitemToGroup(groupNode, item);
menuitemNode.setAttribute('data-show-heading-option', option);
menuitemNode.title = this.config.actionShowHeadingsHelp;
}
return noAction;
},
updateHeadingGroupMenuitems: function(option) {
var headings, headingsLen, labelNode, groupNode;
var selectedHeadings = this.getHeadings(this.getShowMoreHeadingsSelector('selected'));
var selectedHeadingsLen = selectedHeadings.length;
var allHeadings = this.getHeadings(this.getShowMoreHeadingsSelector('all'));
var allHeadingsLen = allHeadings.length;
// Update list of headings
if ( option === 'all' ) {
headings = allHeadings;
}
else {
headings = selectedHeadings;
}
this.renderGroupLabel('id-skip-to-group-headings-label', this.config.headingGroupLabel, headings.length, allHeadings.length);
groupNode = document.getElementById('id-skip-to-group-headings');
this.renderMenuitemsToGroup(groupNode, headings, this.config.msgNoHeadingsFound);
this.updateMenuitems();
// Move focus to first heading menuitem
if (groupNode.firstElementChild) {
groupNode.firstElementChild.focus();
}
// Update heading action menuitem
if (option === 'all') {
option = 'selected';
headingsLen = selectedHeadingsLen;
} else {
option = 'all';
headingsLen = allHeadingsLen;
}
var menuitemNode = this.menuNode.querySelector('[data-id=skip-to-more-headings]');
menuitemNode.setAttribute('data-show-heading-option', option);
menuitemNode.setAttribute('aria-label', this.getShowMoreHeadingsAriaLabel(option, headingsLen));
labelNode = menuitemNode.querySelector('span.label');
labelNode.textContent = this.getShowMoreHeadingsLabel(option, headingsLen);
},
getShowMoreLandmarksSelector: function(option) {
if (option === 'all') {
return this.showAllLandmarksSelector;
}
return this.config.landmarks;
},
getShowMoreLandmarksLabel: function(option, n) {
var label = this.config.actionShowSelectedLandmarksLabel;
if (option === 'all') {
label = this.config.actionShowAllLandmarksLabel;
}
return label.replace('$num', n);
},
getShowMoreLandmarksAriaLabel: function(option, n) {
var label = this.config.actionShowSelectedLandmarksAriaLabel;
if (option === 'all') {
label = this.config.actionShowAllLandmarksAriaLabel;
}
return label.replace('$num', n);
},
renderActionMoreLandmarks: function(groupNode) {
var item, menuitemNode;
var option = 'all';
var selectedLandmarksLen = this.getLandmarks(this.getShowMoreLandmarksSelector('selected')).length;
var allLandmarksLen = this.getLandmarks(this.getShowMoreLandmarksSelector('all')).length;
var noAction = selectedLandmarksLen === allLandmarksLen;
var landmarksLen = allLandmarksLen;
if (option !== 'all') {
landmarksLen = selectedLandmarksLen;
}
if (!noAction) {
item = {};
item.tagName = '';
item.role = 'menuitem';
item.class = 'action';
item.dataId = 'skip-to-more-landmarks';
item.name = this.getShowMoreLandmarksLabel(option, landmarksLen);
item.ariaLabel = this.getShowMoreLandmarksAriaLabel(option, landmarksLen);
menuitemNode = this.renderMenuitemToGroup(groupNode, item);
menuitemNode.setAttribute('data-show-landmark-option', option);
menuitemNode.title = this.config.actionShowLandmarksHelp;
}
return noAction;
},
updateLandmarksGroupMenuitems: function(option) {
var landmarks, landmarksLen, labelNode, groupNode;
var selectedLandmarks = this.getLandmarks(this.getShowMoreLandmarksSelector('selected'));
var selectedLandmarksLen = selectedLandmarks.length;
var allLandmarks = this.getLandmarks(this.getShowMoreLandmarksSelector('all'), true);
var allLandmarksLen = allLandmarks.length;
// Update landmark menu items
if ( option === 'all' ) {
landmarks = allLandmarks;
}
else {
landmarks = selectedLandmarks;
}
this.renderGroupLabel('id-skip-to-group-landmarks-label', this.config.landmarkGroupLabel, landmarks.length, allLandmarks.length);
groupNode = document.getElementById('id-skip-to-group-landmarks');
this.renderMenuitemsToGroup(groupNode, landmarks, this.config.msgNoLandmarksFound);
this.updateMenuitems();
// Move focus to first landmark menuitem
if (groupNode.firstElementChild) {
groupNode.firstElementChild.focus();
}
// Update landmark action menuitem
if (option === 'all') {
option = 'selected';
landmarksLen = selectedLandmarksLen;
} else {
option = 'all';
landmarksLen = allLandmarksLen;
}
var menuitemNode = this.menuNode.querySelector('[data-id=skip-to-more-landmarks]');
menuitemNode.setAttribute('data-show-landmark-option', option);
menuitemNode.setAttribute('aria-label', this.getShowMoreLandmarksAriaLabel(option, landmarksLen));
labelNode = menuitemNode.querySelector('span.label');
labelNode.textContent = this.getShowMoreLandmarksLabel(option, landmarksLen);
},
renderMenu: function() {
var groupNode,
selectedLandmarks,
allLandmarks,
landmarkElements,
selectedHeadings,
allHeadings,
headingElements,
selector,
option,
hasNoAction1,
hasNoAction2;
// remove current menu items from menu
while (this.menuNode.lastElementChild) {
this.menuNode.removeChild(this.menuNode.lastElementChild);
}
option = 'selected';
// Create landmarks group
selector = this.getShowMoreLandmarksSelector('all');
allLandmarks = this.getLandmarks(selector, true);
selector = this.getShowMoreLandmarksSelector('selected');
selectedLandmarks = this.getLandmarks(selector);
landmarkElements = selectedLandmarks;
if (option === 'all') {
landmarkElements = allLandmarks;
}
groupNode = this.renderMenuitemGroup('id-skip-to-group-landmarks', this.config.landmarkGroupLabel);
this.renderMenuitemsToGroup(groupNode, landmarkElements, this.config.msgNoLandmarksFound);
this.renderGroupLabel('id-skip-to-group-landmarks-label', this.config.landmarkGroupLabel, landmarkElements.length, allLandmarks.length);
// Create headings group
selector = this.getShowMoreHeadingsSelector('all');
allHeadings = this.getHeadings(selector);
selector = this.getShowMoreHeadingsSelector('selected');
selectedHeadings = this.getHeadings(selector);
headingElements = selectedHeadings;
if (option === 'all') {
headingElements = allHeadings;
}
groupNode = this.renderMenuitemGroup('id-skip-to-group-headings', this.config.headingGroupLabel);
this.renderMenuitemsToGroup(groupNode, headingElements, this.config.msgNoHeadingsFound);
this.renderGroupLabel('id-skip-to-group-headings-label', this.config.headingGroupLabel, headingElements.length, allHeadings.length);
// Create actions, if enabled
if (this.config.enableActions) {
groupNode = this.renderMenuitemGroup('id-skip-to-group-actions', this.config.actionGroupLabel);
hasNoAction1 = this.renderActionMoreLandmarks(groupNode);
hasNoAction2 = this.renderActionMoreHeadings(groupNode);
// Remove action label if no actions are available
if (hasNoAction1 && hasNoAction2) {
this.removeMenuitemGroup('id-skip-to-group-actions');
}
}
// Update list of menuitems
this.updateMenuitems();
},
//
// Menu scripting event functions and utilities
//
setFocusToMenuitem: function(menuitem) {
if (menuitem) {
menuitem.focus();
}
},
setFocusToFirstMenuitem: function() {
this.setFocusToMenuitem(this.firstMenuitem);
},
setFocusToLastMenuitem: function() {
this.setFocusToMenuitem(this.lastMenuitem);
},
setFocusToPreviousMenuitem: function(menuitem) {
var newMenuitem, index;
if (menuitem === this.firstMenuitem) {
newMenuitem = this.lastMenuitem;
} else {
index = this.menuitemNodes.indexOf(menuitem);
newMenuitem = this.menuitemNodes[index - 1];
}
this.setFocusToMenuitem(newMenuitem);
return newMenuitem;
},
setFocusToNextMenuitem: function(menuitem) {
var newMenuitem, index;
if (menuitem === this.lastMenuitem) {
newMenuitem = this.firstMenuitem;
} else {
index = this.menuitemNodes.indexOf(menuitem);
newMenuitem = this.menuitemNodes[index + 1];
}
this.setFocusToMenuitem(newMenuitem);
return newMenuitem;
},
setFocusByFirstCharacter: function(menuitem, char) {
var start, index;
if (char.length > 1) {
return;
}
char = char.toLowerCase();
// Get start index for search based on position of currentItem
start = this.menuitemNodes.indexOf(menuitem) + 1;
if (start >= this.menuitemNodes.length) {
start = 0;
}
// Check remaining items in the menu
index = this.firstChars.indexOf(char, start);
// If not found in remaining items, check headings
if (index === -1) {
index = this.headingLevels.indexOf(char, start);
}
// If not found in remaining items, check from beginning
if (index === -1) {
index = this.firstChars.indexOf(char, 0);
}
// If not found in remaining items, check headings from beginning
if (index === -1) {
index = this.headingLevels.indexOf(char, 0);
}
// If match was found...
if (index > -1) {
this.setFocusToMenuitem(this.menuitemNodes[index]);
}
},
// Utilities
getIndexFirstChars: function(startIndex, char) {
for (var i = startIndex; i < this.firstChars.length; i += 1) {
if (char === this.firstChars[i]) {
return i;
}
}
return -1;
},
// Popup menu methods
openPopup: function() {
this.menuNode.setAttribute('aria-busy', 'true');
this.renderMenu();
this.menuNode.style.display = 'block';
this.menuNode.removeAttribute('aria-busy');
this.buttonNode.setAttribute('aria-expanded', 'true');
},
closePopup: function() {
if (this.isOpen()) {
this.buttonNode.setAttribute('aria-expanded', 'false');
this.menuNode.style.display = 'none';
}
},
isOpen: function() {
return this.buttonNode.getAttribute('aria-expanded') === 'true';
},
// Menu event handlers
handleFocusin: function() {
this.domNode.classList.add('focus');
},
handleFocusout: function() {
this.domNode.classList.remove('focus');
},
handleButtonKeydown: function(event) {
var key = event.key,
flag = false;
switch (key) {
case ' ':
case 'Enter':
case 'ArrowDown':
case 'Down':
this.openPopup();
this.setFocusToFirstMenuitem();
flag = true;
break;
case 'Esc':
case 'Escape':
this.closePopup();
this.buttonNode.focus();
this.hideTooltip();
flag = true;
break;
case 'Up':
case 'ArrowUp':
this.openPopup();
this.setFocusToLastMenuitem();
flag = true;
break;
default:
break;
}
if (flag) {
event.stopPropagation();
event.preventDefault();
}
},
handleButtonClick: function(event) {
if (this.isOpen()) {
this.closePopup();
this.buttonNode.focus();
} else {
this.openPopup();
this.setFocusToFirstMenuitem();
}
event.stopPropagation();
event.preventDefault();
},
isTooltipHidden: function() {
return this.tooltipNode.className.indexOf('skip-to-show-tooltip') < 0;
},
displayTooltip: function() {
if (this.showTooltipFocus || this.showTooltipHover) {
this.tooltipNode.classList.add('skip-to-show-tooltip');
}
},
showTooltip: function() {
this.showTooltipFocus = true;
if (this.config.enableTooltip && this.isTooltipHidden()) {
this.tooltipNode.style.left = this.tooltipLeft + 'px';
this.tooltipNode.style.top = this.tooltipTop + 'px';
setTimeout(this.displayTooltip.bind(this), this.tooltipTimerDelay);
}
},
hideTooltip: function() {
this.showTooltipFocus = false;
if(this.config.enableTooltip) {
this.tooltipNode.classList.remove('skip-to-show-tooltip');
}
},
handleButtonFocus: function() {
this.showTooltip();
},
handleButtonBlur: function() {
this.hideTooltip();
},
handleButtonPointerenter: function(event) {
this.showTooltipHover = true;
if (this.config.enableTooltip && this.isTooltipHidden()) {
var rect = this.buttonNode.getBoundingClientRect();
var left = Math.min(this.tooltipLeft, event.pageX - rect.left + this.tooltipHeight);
this.tooltipNode.style.left = left + 'px';
var top = event.pageY - rect.top;
this.tooltipNode.style.top = top + 'px';
setTimeout(this.showTooltip. bind(this), this.tooltipTimerDelay);
}
},
handleButtonPointerout: function() {
this.showTooltipHover = false;
if(this.config.enableTooltip) {
this.tooltipNode.classList.remove('skip-to-show-tooltip');
}
},
skipToElement: function(menuitem) {
var isVisible = this.isVisible;
var focusNode = false;
var scrollNode = false;
var elem;
function findVisibleElement(e, selectors) {
if (e) {
for (var j = 0; j < selectors.length; j += 1) {
var elems = e.querySelectorAll(selectors[j]);
for(var i = 0; i < elems.length; i +=1) {
if (isVisible(elems[i])) {
return elems[i];
}
}
}
}
return e;
}
var searchSelectors = ['input', 'button', 'input[type=button]', 'input[type=submit]', 'a'];
var navigationSelectors = ['a', 'input', 'button', 'input[type=button]', 'input[type=submit]'];
var landmarkSelectors = ['h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'section', 'article', 'p', 'li', 'a'];
var isLandmark = menuitem.classList.contains('landmark');
var isSearch = menuitem.classList.contains('skip-to-search');
var isNav = menuitem.classList.contains('skip-to-nav');
elem = document.querySelector('[data-skip-to-id="' + menuitem.getAttribute('data-id') + '"]');
if (elem) {
if (isSearch) {
focusNode = findVisibleElement(elem, searchSelectors);
}
if (isNav) {
focusNode = findVisibleElement(elem, navigationSelectors);
}
if (focusNode && this.isVisible(focusNode)) {
focusNode.focus();
focusNode.scrollIntoView({block: 'nearest'});
}
else {
if (isLandmark) {
scrollNode = findVisibleElement(elem, landmarkSelectors);
if (scrollNode) {
elem = scrollNode;
}
}
elem.tabIndex = -1;
elem.focus();
elem.scrollIntoView({block: 'center'});
}
}
},
handleMenuitemAction: function(tgt) {
var option;
switch (tgt.getAttribute('data-id')) {
case '':
// this means there were no headings or landmarks in the list
break;
case 'skip-to-more-headings':
option = tgt.getAttribute('data-show-heading-option');
this.updateHeadingGroupMenuitems(option);
break;
case 'skip-to-more-landmarks':
option = tgt.getAttribute('data-show-landmark-option');
this.updateLandmarksGroupMenuitems(option);
break;
default:
this.closePopup();
this.skipToElement(tgt);
break;
}
},
handleMenuitemKeydown: function(event) {
var tgt = event.currentTarget,
key = event.key,
flag = false;
function isPrintableCharacter(str) {
return str.length === 1 && str.match(/\S/);
}
if (event.ctrlKey || event.altKey || event.metaKey) {
return;
}
if (event.shiftKey) {
if (isPrintableCharacter(key)) {
this.setFocusByFirstCharacter(tgt, key);
flag = true;
}
if (event.key === 'Tab') {
this.buttonNode.focus();
this.closePopup();
flag = true;
}
} else {
switch (key) {
case 'Enter':
case ' ':
this.handleMenuitemAction(tgt);
flag = true;
break;
case 'Esc':
case 'Escape':
this.closePopup();
this.buttonNode.focus();
flag = true;
break;
case 'Up':
case 'ArrowUp':
this.setFocusToPreviousMenuitem(tgt);
flag = true;
break;
case 'ArrowDown':
case 'Down':
this.setFocusToNextMenuitem(tgt);
flag = true;
break;
case 'Home':
case 'PageUp':
this.setFocusToFirstMenuitem();
flag = true;
break;
case 'End':
case 'PageDown':
this.setFocusToLastMenuitem();
flag = true;
break;
case 'Tab':
this.closePopup();
break;
default:
if (isPrintableCharacter(key)) {
this.setFocusByFirstCharacter(tgt, key);
flag = true;
}
break;
}
}
if (flag) {
event.stopPropagation();
event.preventDefault();
}
},
handleMenuitemClick: function(event) {
this.handleMenuitemAction(event.currentTarget);
event.stopPropagation();
event.preventDefault();
},
handleMenuitemPointerenter: function(event) {
var tgt = event.currentTarget;
tgt.focus();
},
handleBackgroundPointerdown: function(event) {
if (!this.domNode.contains(event.target)) {
if (this.isOpen()) {
this.closePopup();
this.buttonNode.focus();
}
}
},
// methods to extract landmarks, headings and ids
normalizeName: function(name) {
if (typeof name === 'string') return name.replace(/\w\S*/g, function(txt) {
return txt.charAt(0).toUpperCase() + txt.substr(1).toLowerCase();
});
return "";
},
getTextContent: function(elem) {
function getText(e, strings) {
// If text node get the text and return
if (e.nodeType === Node.TEXT_NODE) {
strings.push(e.data);
} else {
// if an element for through all the children elements looking for text
if (e.nodeType === Node.ELEMENT_NODE) {
// check to see if IMG or AREA element and to use ALT content if defined
var tagName = e.tagName.toLowerCase();
if ((tagName === 'img') || (tagName === 'area')) {
if (e.alt) {
strings.push(e.alt);
}
} else {
var c = e.firstChild;
while (c) {
getText(c, strings);
c = c.nextSibling;
} // end loop
}
}
}
} // end function getStrings
// Create return object
var str = "Test",
strings = [];
getText(elem, strings);
if (strings.length) str = strings.join(" ");
return str;
},
getAccessibleName: function(elem) {
var labelledbyIds = elem.getAttribute('aria-labelledby'),
label = elem.getAttribute('aria-label'),
title = elem.getAttribute('title'),
name = "";
if (labelledbyIds && labelledbyIds.length) {
var str,
strings = [],
ids = labelledbyIds.split(' ');
if (!ids.length) ids = [labelledbyIds];
for (var i = 0, l = ids.length; i < l; i += 1) {
var e = document.getElementById(ids[i]);
if (e) str = this.getTextContent(e);
if (str && str.length) strings.push(str);
}
name = strings.join(" ");
} else {
if (this.isNotEmptyString(label)) {
name = label;
} else {
if (this.isNotEmptyString(title)) {
name = title;
}
}
}
return name;
},
isVisible: function(element) {
function isVisibleRec(el) {
if (el.nodeType === 9) return true; /*IE8 does not support Node.DOCUMENT_NODE*/
var computedStyle = window.getComputedStyle(el);
var display = computedStyle.getPropertyValue('display');
var visibility = computedStyle.getPropertyValue('visibility');
var hidden = el.getAttribute('hidden');
if ((display === 'none') ||
(visibility === 'hidden') ||
(hidden !== null)) {
return false;
}
return isVisibleRec(el.parentNode);
}
return isVisibleRec(element);
},
getHeadings: function(targets) {
var dataId, level;
if (typeof targets !== 'string') {
targets = this.config.headings;
}
var headingElementsArr = [];
if (typeof targets !== 'string' || targets.length === 0) return;
var headings = document.querySelectorAll(targets);
for (var i = 0, len = headings.length; i < len; i += 1) {
var heading = headings[i];
var role = heading.getAttribute('role');
if ((typeof role === 'string') && (role === 'presentation')) continue;
if (this.isVisible(heading) && this.isNotEmptyString(heading.innerHTML)) {
if (heading.hasAttribute('data-skip-to-id')) {
dataId = heading.getAttribute('data-skip-to-id');
} else {
heading.setAttribute('data-skip-to-id', this.skipToIdIndex);
dataId = this.skipToIdIndex;
}
level = heading.tagName.substring(1);
var headingItem = {};
headingItem.dataId = dataId.toString();
headingItem.class = 'heading';
headingItem.name = this.getTextContent(heading);
headingItem.ariaLabel = headingItem.name + ', ';
headingItem.ariaLabel += this.config.headingLevelLabel + ' ' + level;
headingItem.tagName = heading.tagName.toLowerCase();
headingItem.role = 'heading';
headingItem.level = level;
headingElementsArr.push(headingItem);
this.skipToIdIndex += 1;
}
}
return headingElementsArr;
},
getLocalizedLandmarkName: function(tagName, name) {
var n;
switch (tagName) {
case 'aside':
n = this.config.asideLabel;
break;
case 'footer':
n = this.config.footerLabel;
break;
case 'form':
n = this.config.formLabel;
break;
case 'header':
n = this.config.headerLabel;
break;
case 'main':
n = this.config.mainLabel;
break;
case 'nav':
n = this.config.navLabel;
break;
case 'section':
case 'region':
n = this.config.regionLabel;
break;
case 'search':
n = this.config.searchLabel;
break;
// When an ID is used as a selector, assume for main content
default:
n = tagName;
break;
}
if (this.isNotEmptyString(name)) {
n += ': ' + name;
}
return n;
},
getNestingLevel: function(landmark, landmarks) {
var nestingLevel = 0;
var parentNode = landmark.parentNode;
while (parentNode) {
for (var i = 0; i < landmarks.length; i += 1) {
if (landmarks[i] === parentNode) {
nestingLevel += 1;
// no more than 3 levels of nesting supported
if (nestingLevel === 3) {
return 3;
}
continue;
}
}
parentNode = parentNode.parentNode;
}
return nestingLevel;
},
getLandmarks: function(targets, allFlag) {
if (typeof allFlag !== 'boolean') {
allFlag = false;
}
if (typeof targets !== 'string') {
targets = this.config.landmarks;
}
var landmarks = document.querySelectorAll(targets);
var mainElements = [];
var searchElements = [];
var navElements = [];
var asideElements = [];
var footerElements = [];
var regionElements = [];
var otherElements = [];
var allLandmarks = [];
var dataId = '';
for (var i = 0, len = landmarks.length; i < len; i += 1) {
var landmark = landmarks[i];
// if skipto is a landmark don't include it in the list
if (landmark === this.domNode) {
continue;
}
var role = landmark.getAttribute('role');
var tagName = landmark.tagName.toLowerCase();
if ((typeof role === 'string') && (role === 'presentation')) continue;
if (this.isVisible(landmark)) {
if (!role) role = tagName;
var name = this.getAccessibleName(landmark);
if (typeof name !== 'string') {
name = '';
}
// normalize tagNames
switch (role) {
case 'banner':
tagName = 'header';
break;
case 'complementary':
tagName = 'aside';
break;
case 'contentinfo':
tagName = 'footer';
break;
case 'form':
tagName = 'form';
break;
case 'main':
tagName = 'main';
break;
case 'navigation':
tagName = 'nav';
break;
case 'region':
tagName = 'section';
break;
case 'search':
tagName = 'search';
break;
default:
break;
}
// if using ID for selectQuery give tagName as main
if (['aside', 'footer', 'form', 'header', 'main', 'nav', 'section', 'search'].indexOf(tagName) < 0) {
tagName = 'main';
}
if (landmark.hasAttribute('aria-roledescription')) {
tagName = landmark.getAttribute('aria-roledescription').trim().replace(' ', '-');
}
if (landmark.hasAttribute('data-skip-to-id')) {
dataId = landmark.getAttribute('data-skip-to-id');
} else {
landmark.setAttribute('data-skip-to-id', this.skipToIdIndex);
dataId = this.skipToIdIndex;
}
var landmarkItem = {};
landmarkItem.dataId = dataId.toString();
landmarkItem.class = 'landmark';
landmarkItem.hasName = name.length > 0;
landmarkItem.name = this.getLocalizedLandmarkName(tagName, name);
landmarkItem.tagName = tagName;
landmarkItem.nestingLevel = 0;
if (allFlag) {
landmarkItem.nestingLevel = this.getNestingLevel(landmark, landmarks);
}
this.skipToIdIndex += 1;
allLandmarks.push(landmarkItem);
// For sorting landmarks into groups
switch (tagName) {
case 'main':
mainElements.push(landmarkItem);
break;
case 'search':
searchElements.push(landmarkItem);
break;
case 'nav':
navElements.push(landmarkItem);
break;
case 'aside':
asideElements.push(landmarkItem);
break;
case 'footer':
footerElements.push(landmarkItem);
break;
case 'section':
// Regions must have accessible name to be included
if (landmarkItem.hasName) {
regionElements.push(landmarkItem);
}
break;
default:
otherElements.push(landmarkItem);
break;
}
}
}
if (allFlag) {
return allLandmarks;
}
return [].concat(mainElements, searchElements, navElements, asideElements, regionElements, footerElements, otherElements);
}
};
// Initialize skipto menu button with onload event
window.addEventListener('load', function() {
SkipTo.init(window.SkipToConfig ||
((typeof window.Joomla === 'object' && typeof window.Joomla.getOptions === 'function') ? window.Joomla.getOptions('skipto-settings', {}) : {})
);
});
})();
/*@end @*/