diff --git a/content/shared/js/skipto.js b/content/shared/js/skipto.js index d5bc6b29bb..d65ab3c766 100644 --- a/content/shared/js/skipto.js +++ b/content/shared/js/skipto.js @@ -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: @@ -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 * ======================================================================== */ @@ -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, @@ -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 { @@ -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; } @@ -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); @@ -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 @@ -1906,6 +1946,8 @@ $skipToId [role="menuitem"]:focus .label { attachNode.insertBefore(this.containerNode, attachNode.firstElementChild); + this.focusMenuitem = null; + return this.containerNode; } @@ -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'); @@ -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 @@ -2172,7 +2214,10 @@ $skipToId [role="menuitem"]:focus .label { */ setFocusToMenuitem(menuitem) { if (menuitem) { + this.removeHoverClass(); + menuitem.classList.add('hover'); menuitem.focus(); + this.focusMenuitem = menuitem; } } @@ -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() { @@ -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(); + } } } @@ -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) { @@ -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)', @@ -2597,7 +2677,8 @@ $skipToId [role="menuitem"]:focus .label { fontFamily: '', fontSize: '', positionLeft: '', - mediaBreakPoint: '', + smallBreakPoint: '', + mediumBreakPoint: '', menuTextColor: '', menuBackgroundColor: '', menuitemFocusTextColor: '', @@ -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', @@ -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%' } },