Skip to content

Commit

Permalink
Infrastructure: Update skipto.js to fix shortcut key bug for non-engl…
Browse files Browse the repository at this point in the history
…ish keyboard layouts (pull #2975)

* Disables shortcut key for skipto when focus is on a text input.
* When menu opens, prevents hover from triggering focus change.
* Remove hover styling when pointer leaves menu.
  • Loading branch information
jongund authored May 22, 2024
1 parent e21ef7f commit 504391a
Showing 1 changed file with 163 additions and 70 deletions.
233 changes: 163 additions & 70 deletions content/shared/js/skipto.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
/* ========================================================================
* Version: 5.2.1
* Copyright (c) 2022, 2023 Jon Gunderson; Licensed BSD
* Version: 5.3.2
* Copyright (c) 2022, 2023, 2024 Jon Gunderson; Licensed BSD
* Copyright (c) 2021 PayPal Accessibility Team and University of Illinois; Licensed BSD
* All rights reserved.
* Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
Expand All @@ -9,7 +9,8 @@
* Neither the name of PayPal or any of its subsidiaries or affiliates, nor the name of the University of Illinois, nor the names of any other contributors 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.
*
* Documenation: https://skipto-landmarks-headings.github.io/page-script-5
* CDN: https://skipto-landmarks-headings.github.io/page-script-5/dist/skipto.min.js
* Documentation: https://skipto-landmarks-headings.github.io/page-script-5
* Code: https://github.com/skipto-landmarks-headings/page-script-5
* Report Issues: https://github.com/skipto-landmarks-headings/page-script-5/issues
* ======================================================================== */
Expand Down Expand Up @@ -122,16 +123,19 @@ $skipToId.popup {
transition: top 0.35s ease;
}
$skipToId button .text {
$skipToId button .skipto-text {
padding: 6px 8px 6px 8px;
display: inline-block;
}
$skipToId button img {
height: 24px;
padding: 2px 4px 2px 4px;
$skipToId button .skipto-small {
padding: 6px 8px 6px 8px;
display: none;
}
$skipToId button .skipto-medium {
padding: 6px 8px 6px 8px;
display: none;
background-color: #e8e9ea;
}
$skipToId,
Expand Down Expand Up @@ -165,18 +169,54 @@ $skipToId button {
z-index: $zIndex !important;
}
@media screen and (max-width: $mediaBreakPointpx) {
$skipToId button img {
display: block;
@media screen and (max-width: $smallBreakPointpx) {
$skipToId:not(.popup) button .skipto-small {
transition: top 0.35s ease;
display: inline-block;
}
$skipToId:not(.popup) button .skipto-text,
$skipToId:not(.popup) button .skipto-medium {
transition: top 0.35s ease;
display: none;
}
$skipToId:not(.popup).focus button .skipto-text {
transition: top 0.35s ease;
display: inline-block;
}
$skipToId:not(.popup).focus button .skipto-small,
$skipToId:not(.popup).focus button .skipto-medium {
transition: top 0.35s ease;
display: none;
}
}
@media screen and (min-width: $smallBreakPointpx) and (max-width: $mediumBreakPointpx) {
$skipToId:not(.popup) button .skipto-medium {
transition: top 0.35s ease;
display: inline-block;
}
$skipToId button {
border-color: #e8e9ea;
$skipToId:not(.popup) button .skipto-text,
$skipToId:not(.popup) button .skipto-small {
transition: top 0.35s ease;
display: none;
}
$skipToId button .text {
$skipToId:not(.popup).focus button .skipto-text {
transition: top 0.35s ease;
display: inline-block;
}
$skipToId:not(.popup).focus button .skipto-small,
$skipToId:not(.popup).focus button .skipto-medium {
transition: top 0.35s ease;
display: none;
}
}
$skipToId.fixed {
Expand Down Expand Up @@ -329,37 +369,31 @@ $skipToId button:hover {
background-color: $menuBackgroundColor;
color: $menuTextColor;
outline: none;
}
$skipToId button:focus,
$skipToId button:hover {
border-width: 0px 2px 2px 2px;
border-color: $focusBorderColor;
}
$skipToId button:focus .text,
$skipToId button:hover .text {
padding: 6px 7px 5px 7px;
}
$skipToId button:focus img,
$skipToId button:hover img {
padding: 2px 3px 4px 3px;
$skipToId button:focus .skipto-text,
$skipToId button:hover .skipto-text,
$skipToId button:focus .skipto-small,
$skipToId button:hover .skipto-small,
$skipToId button:focus .skipto-medium,
$skipToId button:hover .skipto-medium {
padding: 6px 7px 5px 7px;
}
$skipToId [role="menuitem"]:focus {
padding: 1px;
border-width: 2px;
border-style: solid;
border-color: $focusBorderColor;
background-color: $menuitemFocusBackgroundColor;
color: $menuitemFocusTextColor;
outline: none;
}
$skipToId [role="menuitem"]:focus .level,
$skipToId [role="menuitem"]:focus .label {
$skipToId [role="menuitem"].hover,
$skipToId [role="menuitem"].hover .level,
$skipToId [role="menuitem"].hover .label {
background-color: $menuitemFocusBackgroundColor;
color: $menuitemFocusTextColor;
}
Expand Down Expand Up @@ -499,7 +533,8 @@ $skipToId [role="menuitem"]:focus .label {
updateStyle('$fontSize', config.fontSize, theme.fontSize, defaultTheme.fontSize);

updateStyle('$positionLeft', config.positionLeft, theme.positionLeft, defaultTheme.positionLeft);
updateStyle('$mediaBreakPoint', config.mediaBreakPoint, theme.mediaBreakPoint, defaultTheme.mediaBreakPoint);
updateStyle('$smallBreakPoint', config.smallBreakPoint, theme.smallBreakPoint, defaultTheme.smallBreakPoint);
updateStyle('$mediumBreakPoint', config.mediumBreakPoint, theme.mediumBreakPoint, defaultTheme.mediumBreakPoint);

updateStyle('$menuTextColor', config.menuTextColor, theme.menuTextColor, defaultTheme.menuTextColor);
updateStyle('$menuBackgroundColor', config.menuBackgroundColor, theme.menuBackgroundColor, defaultTheme.menuBackgroundColor);
Expand Down Expand Up @@ -1851,14 +1886,19 @@ $skipToId [role="menuitem"]:focus .label {
this.containerNode.appendChild(this.buttonNode);

this.buttonTextNode = document.createElement('span');
this.buttonTextNode.classList.add('text');
this.buttonTextNode.classList.add('skipto-text');
this.buttonTextNode.textContent = buttonVisibleLabel;
this.buttonNode.appendChild(this.buttonTextNode);

const imageNode = document.createElement('img');
imageNode.src = "";
imageNode.setAttribute('alt', '');
this.buttonNode.appendChild(imageNode);
const smallButtonNode = document.createElement('span');
smallButtonNode.classList.add('skipto-small');
smallButtonNode.textContent = config.smallButtonLabel;
this.buttonNode.appendChild(smallButtonNode);

const mediumButtonNode = document.createElement('span');
mediumButtonNode.classList.add('skipto-medium');
mediumButtonNode.textContent = config.buttonLabel;
this.buttonNode.appendChild(mediumButtonNode);

// Create menu container

Expand Down Expand Up @@ -1906,6 +1946,8 @@ $skipToId [role="menuitem"]:focus .label {

attachNode.insertBefore(this.containerNode, attachNode.firstElementChild);

this.focusMenuitem = null;

return this.containerNode;

}
Expand Down Expand Up @@ -2019,8 +2061,8 @@ $skipToId [role="menuitem"]:focus .label {
/*
* @method updateMenuitems
*
* @desc Updates the menu information with the current manu items
* used for menu navgation commands
* @desc Updates the menu information with the current menu items
* used for menu navigation commands
*/
updateMenuitems () {
let menuitemNodes = this.menuNode.querySelectorAll('[role=menuitem');
Expand Down Expand Up @@ -2063,7 +2105,7 @@ $skipToId [role="menuitem"]:focus .label {
menuitemNode.addEventListener('keydown', this.handleMenuitemKeydown.bind(this));
menuitemNode.addEventListener('click', this.handleMenuitemClick.bind(this));
menuitemNode.addEventListener('pointerenter', this.handleMenuitemPointerenter.bind(this));

menuitemNode.addEventListener('pointerleave', this.handleMenuitemPointerleave.bind(this));
groupNode.appendChild(menuitemNode);

// add heading level and label
Expand Down Expand Up @@ -2172,7 +2214,10 @@ $skipToId [role="menuitem"]:focus .label {
*/
setFocusToMenuitem(menuitem) {
if (menuitem) {
this.removeHoverClass();
menuitem.classList.add('hover');
menuitem.focus();
this.focusMenuitem = menuitem;
}
}

Expand Down Expand Up @@ -2341,7 +2386,18 @@ $skipToId [role="menuitem"]:focus .label {
isOpen() {
return this.buttonNode.getAttribute('aria-expanded') === 'true';
}


/*
* @method removeHoverClass
*
* @desc Removes hover class for menuitems
*/
removeHoverClass() {
this.menuitemNodes.forEach( node => {
node.classList.remove('hover');
});
}

// Menu event handlers

handleFocusin() {
Expand Down Expand Up @@ -2396,34 +2452,51 @@ $skipToId [role="menuitem"]:focus .label {
}

handleDocumentKeydown (event) {
let key = event.key,
flag = false;

let altPressed =
this.usesAltKey &&
event.altKey &&
!event.ctrlKey &&
!event.shiftKey &&
!event.metaKey;

let optionPressed =
this.usesOptionKey &&
event.altKey &&
!event.ctrlKey &&
!event.shiftKey &&
!event.metaKey;

if (
(optionPressed && this.config.optionShortcut === key) ||
(altPressed && this.config.altShortcut === key)
) {
this.openPopup();
this.setFocusToFirstMenuitem();
flag = true;
}
if (flag) {
event.stopPropagation();
event.preventDefault();
const enabledInputTypes = [
'button',
'checkbox',
'color',
'file',
'image',
'radio',
'range',
'reset',
'submit'
];

const target = event.target;
const tagName = target.tagName ? target.tagName.toLowerCase() : '';
const type = tagName === 'input' ? target.type.toLowerCase() : '';

if ((tagName !== 'textarea') &&
((tagName !== 'input') ||
((tagName === 'input') && enabledInputTypes.includes(type))
)) {

const altPressed =
this.usesAltKey &&
event.altKey &&
!event.ctrlKey &&
!event.shiftKey &&
!event.metaKey;

const optionPressed =
this.usesOptionKey &&
event.altKey &&
!event.ctrlKey &&
!event.shiftKey &&
!event.metaKey;

if ((optionPressed && this.config.optionShortcut === event.key) ||
(altPressed && this.config.altShortcut === event.key) ||
((optionPressed || altPressed) && (48 === event.keyCode))
) {
this.openPopup();
this.setFocusToFirstMenuitem();
event.stopPropagation();
event.preventDefault();
}
}
}

Expand Down Expand Up @@ -2519,7 +2592,13 @@ $skipToId [role="menuitem"]:focus .label {

handleMenuitemPointerenter(event) {
let tgt = event.currentTarget;
tgt.focus();
this.removeHoverClass();
tgt.classList.add('hover');
}

handleMenuitemPointerleave(event) {
let tgt = event.currentTarget;
tgt.classList.remove('hover');
}

handleBackgroundPointerdown(event) {
Expand Down Expand Up @@ -2566,6 +2645,7 @@ $skipToId [role="menuitem"]:focus .label {

// Button labels and messages
buttonLabel: 'Skip To Content',
smallButtonLabel: 'SkipTo',
altLabel: 'Alt',
optionLabel: 'Option',
buttonShortcut: ' ($modifier+$key)',
Expand Down Expand Up @@ -2597,7 +2677,8 @@ $skipToId [role="menuitem"]:focus .label {
fontFamily: '',
fontSize: '',
positionLeft: '',
mediaBreakPoint: '',
smallBreakPoint: '',
mediumBreakPoint: '',
menuTextColor: '',
menuBackgroundColor: '',
menuitemFocusTextColor: '',
Expand All @@ -2612,7 +2693,8 @@ $skipToId [role="menuitem"]:focus .label {
fontFamily: 'inherit',
fontSize: 'inherit',
positionLeft: '46%',
mediaBreakPoint: '540',
smallBreakPoint: '576',
mediumBreakPoint: '992',
menuTextColor: '#1a1a1a',
menuBackgroundColor: '#dcdcdc',
menuitemFocusTextColor: '#eeeeee',
Expand Down Expand Up @@ -2686,6 +2768,17 @@ $skipToId [role="menuitem"]:focus .label {
focusBorderColor: '#dd3444',
buttonTextColor: '#fff',
buttonBackgroundColor: '#036',
},
'openweba11y': {
hostnameSelector: 'openweba11y.com',
buttonTextColor: '#13294B',
buttonBackgroundColor: '#dddddd',
focusBorderColor: '#C5050C',
menuTextColor: '#13294B',
menuBackgroundColor: '#dddddd',
menuitemFocusTextColor: '#dddddd',
menuitemFocusBackgroundColor: '#13294B',
fontSize: '90%'
}
},

Expand Down

0 comments on commit 504391a

Please sign in to comment.