fix(a11y): leverage native :focus-visible in CSS
This commit is contained in:
parent
e17d0220c0
commit
0202e8efb0
@ -197,11 +197,10 @@ Here's a list of the properties and what they are used for:
|
|||||||
| ---------------------------------------------- | ------------------------------------------------------------------------------------------------------- | --------------------------------------------------------------------- |
|
| ---------------------------------------------- | ------------------------------------------------------------------------------------------------------- | --------------------------------------------------------------------- |
|
||||||
| `--plyr-color-main` | The primary UI color. |  `#00b3ff` |
|
| `--plyr-color-main` | The primary UI color. |  `#00b3ff` |
|
||||||
| `--plyr-video-background` | The background color of video and poster wrappers for using alpha channel videos and poster images. | `rgba(0, 0, 0, 1)` |
|
| `--plyr-video-background` | The background color of video and poster wrappers for using alpha channel videos and poster images. | `rgba(0, 0, 0, 1)` |
|
||||||
| `--plyr-tab-focus-color` | The color used for the dotted outline when an element is `:focus-visible` (equivalent) keyboard focus. | `--plyr-color-main` |
|
| `--plyr-focus-visible-color` | The color used for the focus styles when an element is `:focus-visible` (keyboard focused). | `--plyr-color-main` |
|
||||||
| `--plyr-badge-background` | The background color for badges in the menu. |  `#4a5464` |
|
| `--plyr-badge-background` | The background color for badges in the menu. |  `#4a5464` |
|
||||||
| `--plyr-badge-text-color` | The text color for badges. |  `#ffffff` |
|
| `--plyr-badge-text-color` | The text color for badges. |  `#ffffff` |
|
||||||
| `--plyr-badge-border-radius` | The border radius used for badges. | `2px` |
|
| `--plyr-badge-border-radius` | The border radius used for badges. | `2px` |
|
||||||
| `--plyr-tab-focus-color` | The color used to highlight tab (keyboard) focus. | `--plyr-color-main` |
|
|
||||||
| `--plyr-captions-background` | The color for the background of captions. | `rgba(0, 0, 0, 0.8)` |
|
| `--plyr-captions-background` | The color for the background of captions. | `rgba(0, 0, 0, 0.8)` |
|
||||||
| `--plyr-captions-text-color` | The color used for the captions text. |  `#ffffff` |
|
| `--plyr-captions-text-color` | The color used for the captions text. |  `#ffffff` |
|
||||||
| `--plyr-control-icon-size` | The size of the icons used in the controls. | `18px` |
|
| `--plyr-control-icon-size` | The size of the icons used in the controls. | `18px` |
|
||||||
|
@ -4,7 +4,6 @@
|
|||||||
// Please see README.md in the root or github.com/sampotts/plyr
|
// Please see README.md in the root or github.com/sampotts/plyr
|
||||||
// ==========================================================================
|
// ==========================================================================
|
||||||
|
|
||||||
import './tab-focus';
|
|
||||||
import 'custom-event-polyfill';
|
import 'custom-event-polyfill';
|
||||||
import 'url-polyfill';
|
import 'url-polyfill';
|
||||||
|
|
||||||
@ -13,7 +12,6 @@ import Shr from 'shr-buttons';
|
|||||||
|
|
||||||
import Plyr from '../../../src/js/plyr';
|
import Plyr from '../../../src/js/plyr';
|
||||||
import sources from './sources';
|
import sources from './sources';
|
||||||
import toggleClass from './toggle-class';
|
|
||||||
|
|
||||||
(() => {
|
(() => {
|
||||||
const production = 'plyr.io';
|
const production = 'plyr.io';
|
||||||
@ -108,10 +106,10 @@ import toggleClass from './toggle-class';
|
|||||||
|
|
||||||
function render(type) {
|
function render(type) {
|
||||||
// Remove active classes
|
// Remove active classes
|
||||||
Array.from(buttons).forEach((button) => toggleClass(button.parentElement, 'active', false));
|
Array.from(buttons).forEach((button) => button.parentElement.classList.toggle('active', false));
|
||||||
|
|
||||||
// Set active on parent
|
// Set active on parent
|
||||||
toggleClass(document.querySelector(`[data-source="${type}"]`), 'active', true);
|
document.querySelector(`[data-source="${type}"]`).classList.toggle('active', true);
|
||||||
|
|
||||||
// Show cite
|
// Show cite
|
||||||
Array.from(document.querySelectorAll('.plyr__cite')).forEach((cite) => {
|
Array.from(document.querySelectorAll('.plyr__cite')).forEach((cite) => {
|
||||||
|
@ -1,31 +0,0 @@
|
|||||||
// Setup tab focus
|
|
||||||
const container = document.getElementById('container');
|
|
||||||
const tabClassName = 'tab-focus';
|
|
||||||
|
|
||||||
// Remove class on blur
|
|
||||||
document.addEventListener('focusout', (event) => {
|
|
||||||
if (!event.target.classList || container.contains(event.target)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
event.target.classList.remove(tabClassName);
|
|
||||||
});
|
|
||||||
|
|
||||||
// Add classname to tabbed elements
|
|
||||||
document.addEventListener('keydown', (event) => {
|
|
||||||
if (event.key !== 'Tab') {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Delay the adding of classname until the focus has changed
|
|
||||||
// This event fires before the focusin event
|
|
||||||
setTimeout(() => {
|
|
||||||
const focused = document.activeElement;
|
|
||||||
|
|
||||||
if (!focused || !focused.classList || container.contains(focused)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
focused.classList.add(tabClassName);
|
|
||||||
}, 10);
|
|
||||||
});
|
|
@ -1,5 +0,0 @@
|
|||||||
// Toggle class on an element
|
|
||||||
const toggleClass = (element, className = '', toggle = false) =>
|
|
||||||
element && element.classList[toggle ? 'add' : 'remove'](className);
|
|
||||||
|
|
||||||
export default toggleClass;
|
|
@ -44,8 +44,8 @@
|
|||||||
outline: 0;
|
outline: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
&.tab-focus {
|
&:focus-visible {
|
||||||
@include tab-focus;
|
@include focus-visible($color-button-background);
|
||||||
}
|
}
|
||||||
|
|
||||||
&:active {
|
&:active {
|
||||||
|
@ -38,8 +38,8 @@ a {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&.tab-focus {
|
&:focus-visible {
|
||||||
@include tab-focus;
|
@include focus-visible($color-link);
|
||||||
}
|
}
|
||||||
|
|
||||||
&.no-border::after {
|
&.no-border::after {
|
||||||
|
@ -58,8 +58,8 @@ aside {
|
|||||||
a {
|
a {
|
||||||
color: $color-twitter;
|
color: $color-twitter;
|
||||||
|
|
||||||
&.tab-focus {
|
&:focus-visible {
|
||||||
@include tab-focus($color-twitter);
|
@include focus-visible($color-twitter);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -25,9 +25,9 @@
|
|||||||
|
|
||||||
// Nicer focus styles
|
// Nicer focus styles
|
||||||
// ---------------------------------------
|
// ---------------------------------------
|
||||||
@mixin tab-focus($color: $tab-focus-default-color) {
|
@mixin focus-visible($color: $focus-default-color) {
|
||||||
box-shadow: 0 0 0 3px rgba($color, 0.35);
|
outline: 2px dashed $color;
|
||||||
outline: 0;
|
outline-offset: 2px;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Use rems for font sizing
|
// Use rems for font sizing
|
||||||
|
@ -39,4 +39,4 @@ $color-button-count-background: #fff;
|
|||||||
$color-button-count-text: $color-gray-600;
|
$color-button-count-text: $color-gray-600;
|
||||||
|
|
||||||
// Focus
|
// Focus
|
||||||
$tab-focus-default-color: #fff;
|
$focus-default-color: $color-brand-primary;
|
||||||
|
4
demo/src/sass/utilities/focus.scss
Normal file
4
demo/src/sass/utilities/focus.scss
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
*:focus-visible {
|
||||||
|
outline: 2px dotted $color-brand-primary;
|
||||||
|
outline-offset: 2px;
|
||||||
|
}
|
@ -379,7 +379,6 @@ const defaults = {
|
|||||||
supported: 'plyr--airplay-supported',
|
supported: 'plyr--airplay-supported',
|
||||||
active: 'plyr--airplay-active',
|
active: 'plyr--airplay-active',
|
||||||
},
|
},
|
||||||
tabFocus: 'plyr__tab-focus',
|
|
||||||
previewThumbnails: {
|
previewThumbnails: {
|
||||||
// Tooltip thumbs
|
// Tooltip thumbs
|
||||||
thumbContainer: 'plyr__preview-thumb',
|
thumbContainer: 'plyr__preview-thumb',
|
||||||
|
8
src/js/controls.js
vendored
8
src/js/controls.js
vendored
@ -1106,7 +1106,7 @@ const controls = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
// Focus the first menu item in a given (or visible) menu
|
// Focus the first menu item in a given (or visible) menu
|
||||||
focusFirstMenuItem(pane, tabFocus = false) {
|
focusFirstMenuItem(pane, focusVisible = false) {
|
||||||
if (this.elements.settings.popup.hidden) {
|
if (this.elements.settings.popup.hidden) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -1119,7 +1119,7 @@ const controls = {
|
|||||||
|
|
||||||
const firstItem = target.querySelector('[role^="menuitem"]');
|
const firstItem = target.querySelector('[role^="menuitem"]');
|
||||||
|
|
||||||
setFocus.call(this, firstItem, tabFocus);
|
setFocus.call(this, firstItem, focusVisible);
|
||||||
},
|
},
|
||||||
|
|
||||||
// Show/hide menu
|
// Show/hide menu
|
||||||
@ -1196,7 +1196,7 @@ const controls = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
// Show a panel in the menu
|
// Show a panel in the menu
|
||||||
showMenuPanel(type = '', tabFocus = false) {
|
showMenuPanel(type = '', focusVisible = false) {
|
||||||
const target = this.elements.container.querySelector(`#plyr-settings-${this.id}-${type}`);
|
const target = this.elements.container.querySelector(`#plyr-settings-${this.id}-${type}`);
|
||||||
|
|
||||||
// Nothing to show, bail
|
// Nothing to show, bail
|
||||||
@ -1247,7 +1247,7 @@ const controls = {
|
|||||||
toggleHidden(target, false);
|
toggleHidden(target, false);
|
||||||
|
|
||||||
// Focus the first item
|
// Focus the first item
|
||||||
controls.focusFirstMenuItem.call(this, target, tabFocus);
|
controls.focusFirstMenuItem.call(this, target, focusVisible);
|
||||||
},
|
},
|
||||||
|
|
||||||
// Set the download URL
|
// Set the download URL
|
||||||
|
@ -21,7 +21,6 @@ class Listeners {
|
|||||||
|
|
||||||
this.handleKey = this.handleKey.bind(this);
|
this.handleKey = this.handleKey.bind(this);
|
||||||
this.toggleMenu = this.toggleMenu.bind(this);
|
this.toggleMenu = this.toggleMenu.bind(this);
|
||||||
this.setTabFocus = this.setTabFocus.bind(this);
|
|
||||||
this.firstTouch = this.firstTouch.bind(this);
|
this.firstTouch = this.firstTouch.bind(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -194,57 +193,6 @@ class Listeners {
|
|||||||
toggleClass(elements.container, player.config.classNames.isTouch, true);
|
toggleClass(elements.container, player.config.classNames.isTouch, true);
|
||||||
};
|
};
|
||||||
|
|
||||||
setTabFocus = (event) => {
|
|
||||||
const { player } = this;
|
|
||||||
const { elements } = player;
|
|
||||||
const { key, type, timeStamp } = event;
|
|
||||||
|
|
||||||
clearTimeout(this.focusTimer);
|
|
||||||
|
|
||||||
// Ignore any key other than tab
|
|
||||||
if (type === 'keydown' && key !== 'Tab') {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Store reference to event timeStamp
|
|
||||||
if (type === 'keydown') {
|
|
||||||
this.lastKeyDown = timeStamp;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove current classes
|
|
||||||
const removeCurrent = () => {
|
|
||||||
const className = player.config.classNames.tabFocus;
|
|
||||||
const current = getElements.call(player, `.${className}`);
|
|
||||||
toggleClass(current, className, false);
|
|
||||||
};
|
|
||||||
|
|
||||||
// Determine if a key was pressed to trigger this event
|
|
||||||
const wasKeyDown = timeStamp - this.lastKeyDown <= 20;
|
|
||||||
|
|
||||||
// Ignore focus events if a key was pressed prior
|
|
||||||
if (type === 'focus' && !wasKeyDown) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove all current
|
|
||||||
removeCurrent();
|
|
||||||
|
|
||||||
// Delay the adding of classname until the focus has changed
|
|
||||||
// This event fires before the focusin event
|
|
||||||
if (type !== 'focusout') {
|
|
||||||
this.focusTimer = setTimeout(() => {
|
|
||||||
const focused = document.activeElement;
|
|
||||||
|
|
||||||
// Ignore if current focus element isn't inside the player
|
|
||||||
if (!elements.container.contains(focused)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
toggleClass(document.activeElement, player.config.classNames.tabFocus, true);
|
|
||||||
}, 10);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Global window & document listeners
|
// Global window & document listeners
|
||||||
global = (toggle = true) => {
|
global = (toggle = true) => {
|
||||||
const { player } = this;
|
const { player } = this;
|
||||||
@ -259,9 +207,6 @@ class Listeners {
|
|||||||
|
|
||||||
// Detect touch by events
|
// Detect touch by events
|
||||||
once.call(player, document.body, 'touchstart', this.firstTouch);
|
once.call(player, document.body, 'touchstart', this.firstTouch);
|
||||||
|
|
||||||
// Tab focus detection
|
|
||||||
toggleListener.call(player, document.body, 'keydown focus blur focusout', this.setTabFocus, toggle, false, true);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// Container listeners
|
// Container listeners
|
||||||
|
@ -268,16 +268,11 @@ export function getElement(selector) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Set focus and tab focus class
|
// Set focus and tab focus class
|
||||||
export function setFocus(element = null, tabFocus = false) {
|
export function setFocus(element = null, focusVisible = false) {
|
||||||
if (!is.element(element)) {
|
if (!is.element(element)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set regular focus
|
// Set regular focus
|
||||||
element.focus({ preventScroll: true });
|
element.focus({ preventScroll: true, focusVisible });
|
||||||
|
|
||||||
// If we want to mimic keyboard focus via tab
|
|
||||||
if (tabFocus) {
|
|
||||||
toggleClass(element, this.config.classNames.tabFocus);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -28,8 +28,8 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Tab focus
|
// Tab focus
|
||||||
&.plyr__tab-focus {
|
&:focus-visible {
|
||||||
@include plyr-tab-focus;
|
@include plyr-focus-visible;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -100,7 +100,7 @@
|
|||||||
right: calc((#{$plyr-control-padding} * 1.5) - #{$plyr-menu-item-arrow-size});
|
right: calc((#{$plyr-control-padding} * 1.5) - #{$plyr-menu-item-arrow-size});
|
||||||
}
|
}
|
||||||
|
|
||||||
&.plyr__tab-focus::after,
|
&:focus-visible::after,
|
||||||
&:hover::after {
|
&:hover::after {
|
||||||
border-left-color: currentColor;
|
border-left-color: currentColor;
|
||||||
}
|
}
|
||||||
@ -132,7 +132,7 @@
|
|||||||
top: 100%;
|
top: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
&.plyr__tab-focus::after,
|
&:focus-visible::after,
|
||||||
&:hover::after {
|
&:hover::after {
|
||||||
border-right-color: currentColor;
|
border-right-color: currentColor;
|
||||||
}
|
}
|
||||||
@ -181,7 +181,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&.plyr__tab-focus::before,
|
&:focus-visible::before,
|
||||||
&:hover::before {
|
&:hover::before {
|
||||||
background: rgba($plyr-color-gray-900, 0.1);
|
background: rgba($plyr-color-gray-900, 0.1);
|
||||||
}
|
}
|
||||||
|
@ -83,17 +83,17 @@
|
|||||||
outline: 0;
|
outline: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
&.plyr__tab-focus {
|
&:focus-visible {
|
||||||
&::-webkit-slider-runnable-track {
|
&::-webkit-slider-runnable-track {
|
||||||
@include plyr-tab-focus;
|
@include plyr-focus-visible;
|
||||||
}
|
}
|
||||||
|
|
||||||
&::-moz-range-track {
|
&::-moz-range-track {
|
||||||
@include plyr-tab-focus;
|
@include plyr-focus-visible;
|
||||||
}
|
}
|
||||||
|
|
||||||
&::-ms-track {
|
&::-ms-track {
|
||||||
@include plyr-tab-focus;
|
@include plyr-focus-visible;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -42,7 +42,7 @@
|
|||||||
|
|
||||||
// Displaying
|
// Displaying
|
||||||
.plyr .plyr__control:hover .plyr__tooltip,
|
.plyr .plyr__control:hover .plyr__tooltip,
|
||||||
.plyr .plyr__control.plyr__tab-focus .plyr__tooltip,
|
.plyr .plyr__control:focus-visible .plyr__tooltip,
|
||||||
.plyr__tooltip--visible {
|
.plyr__tooltip--visible {
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
transform: translate(-50%, 0) scale(1);
|
transform: translate(-50%, 0) scale(1);
|
||||||
@ -82,7 +82,7 @@
|
|||||||
.plyr__controls > .plyr__control:first-child + .plyr__control,
|
.plyr__controls > .plyr__control:first-child + .plyr__control,
|
||||||
.plyr__controls > .plyr__control:last-child {
|
.plyr__controls > .plyr__control:last-child {
|
||||||
&:hover .plyr__tooltip,
|
&:hover .plyr__tooltip,
|
||||||
&.plyr__tab-focus .plyr__tooltip,
|
&:focus-visible .plyr__tooltip,
|
||||||
.plyr__tooltip--visible {
|
.plyr__tooltip--visible {
|
||||||
transform: translate(0, 0) scale(1);
|
transform: translate(0, 0) scale(1);
|
||||||
}
|
}
|
||||||
|
@ -4,8 +4,8 @@
|
|||||||
|
|
||||||
// Nicer focus styles
|
// Nicer focus styles
|
||||||
// ---------------------------------------
|
// ---------------------------------------
|
||||||
@mixin plyr-tab-focus($color: $plyr-tab-focus-color) {
|
@mixin plyr-focus-visible($color: $plyr-focus-visible-color) {
|
||||||
outline: $color dotted 3px;
|
outline: 2px dashed $color;
|
||||||
outline-offset: 2px;
|
outline-offset: 2px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2,4 +2,4 @@
|
|||||||
// Cosmetic
|
// Cosmetic
|
||||||
// ==========================================================================
|
// ==========================================================================
|
||||||
|
|
||||||
$plyr-tab-focus-color: var(--plyr-tab-focus-color, var(--plyr-color-main, $plyr-color-main)) !default;
|
$plyr-focus-visible-color: var(--plyr-focus-visible-color, var(--plyr-color-main, $plyr-color-main)) !default;
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
|
|
||||||
// Control elements
|
// Control elements
|
||||||
.plyr--audio .plyr__control {
|
.plyr--audio .plyr__control {
|
||||||
&.plyr__tab-focus,
|
&:focus-visible,
|
||||||
&:hover,
|
&:hover,
|
||||||
&[aria-expanded='true'] {
|
&[aria-expanded='true'] {
|
||||||
background: $plyr-audio-control-background-hover;
|
background: $plyr-audio-control-background-hover;
|
||||||
|
@ -87,8 +87,7 @@ $embed-padding: (math.div(100, 16) * 9);
|
|||||||
|
|
||||||
// Control elements
|
// Control elements
|
||||||
.plyr--video .plyr__control {
|
.plyr--video .plyr__control {
|
||||||
// Hover and tab focus
|
&:focus-visible,
|
||||||
&.plyr__tab-focus,
|
|
||||||
&:hover,
|
&:hover,
|
||||||
&[aria-expanded='true'] {
|
&[aria-expanded='true'] {
|
||||||
background: $plyr-video-control-background-hover;
|
background: $plyr-video-control-background-hover;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user