From a65700b07ebbf9552794a42b0fc9695dd673064b Mon Sep 17 00:00:00 2001 From: Oliver Foster Date: Tue, 29 Oct 2024 11:59:24 +0000 Subject: [PATCH] Fix: CSS to animate popups, dialog for accessibility (fixes #600) --- js/a11y/popup.js | 13 +++- js/app.js | 1 + js/drawer.js | 1 + js/shadow.js | 37 ++++++++++ js/transitions.js | 8 +++ js/views/ShadowView.js | 55 +++++++++++++++ js/views/drawerView.js | 135 ++++++++++++++---------------------- js/views/notifyPopupView.js | 100 ++++++++------------------ js/views/notifyPushView.js | 5 +- less/_defaults/base.less | 15 ++++ less/core/drawer.less | 1 + less/core/notify.less | 9 ++- templates/notifyPopup.hbs | 4 +- templates/shadow.hbs | 3 +- 14 files changed, 224 insertions(+), 163 deletions(-) create mode 100644 js/shadow.js create mode 100644 js/views/ShadowView.js diff --git a/js/a11y/popup.js b/js/a11y/popup.js index f6a14997..07d1ee9c 100644 --- a/js/a11y/popup.js +++ b/js/a11y/popup.js @@ -1,4 +1,5 @@ import Adapt from 'core/js/adapt'; +import logging from '../logging'; /** * Tabindex and aria-hidden manager for popups. @@ -94,6 +95,12 @@ export default class Popup extends Backbone.Controller { } this._floorStack.push($popupElement); this._focusStack.push($(document.activeElement)); + if ($popupElement.is('dialog')) { + $popupElement[0].addEventListener('cancel', event => event.preventDefault()); + $popupElement[0].showModal(); + return; + } + logging.deprecated('a11y/popup opened: Use native dialog tag for', $popupElement); let $elements = $(config._options._tabbableElements).filter(config._options._tabbableElementsExcludes); const $branch = $popupElement.add($popupElement.parents()); const $siblings = $branch.siblings().filter(config._options._tabbableElementsExcludes); @@ -167,7 +174,11 @@ export default class Popup extends Backbone.Controller { if (this._floorStack.length <= 1) { return; } - this._floorStack.pop(); + const $popupElement = this._floorStack.pop(); + if ($popupElement.is('dialog')) { + $popupElement[0].close(); + return this._focusStack.pop(); + } $(config._options._tabbableElements).filter(config._options._tabbableElementsExcludes).each((index, item) => { const $item = $(item); let previousTabIndex = ''; diff --git a/js/app.js b/js/app.js index acf8e0d5..0b349809 100644 --- a/js/app.js +++ b/js/app.js @@ -22,6 +22,7 @@ import 'core/js/navigation'; import 'core/js/startController'; import 'core/js/DOMElementModifications'; import 'core/js/tooltips'; +import 'core/js/shadow'; import 'plugins'; $('body').append(Handlebars.templates.loading()); diff --git a/js/drawer.js b/js/drawer.js index 79603dc3..1f13179b 100644 --- a/js/drawer.js +++ b/js/drawer.js @@ -18,6 +18,7 @@ class Drawer extends Backbone.Controller { onAdaptStart() { this._drawerView = new DrawerView({ collection: DrawerCollection }); + this._drawerView.$el.insertAfter('#shadow'); } onLanguageChanged() { diff --git a/js/shadow.js b/js/shadow.js new file mode 100644 index 00000000..d57ad8d9 --- /dev/null +++ b/js/shadow.js @@ -0,0 +1,37 @@ +import Backbone from 'backbone'; +import Adapt from 'core/js/adapt'; +import ShadowView from './views/ShadowView'; + +class Shadow extends Backbone.Controller { + + initialize() { + this.listenTo(Adapt, { + 'adapt:start': this.onAdaptStart + }); + } + + onAdaptStart() { + this._shadowView = new ShadowView(); + this._shadowView.$el.prependTo('body'); + } + + get isOpen() { + return this._shadowView?.isOpen ?? false; + } + + async show() { + return this._shadowView?.showShadow(); + } + + async hide() { + return this._shadowView?.hideShadow(); + } + + remove() { + this._shadowView?.remove(); + this._shadowView = null; + } + +} + +export default new Shadow(); diff --git a/js/transitions.js b/js/transitions.js index 1b9b6518..be0265d2 100644 --- a/js/transitions.js +++ b/js/transitions.js @@ -1,5 +1,13 @@ import logging from 'core/js/logging'; +/** + * Wait for one requestAnimationFrame to allow classes to settle + * @returns {Promise} + */ +export async function transitionNextFrame() { + return new Promise(resolve => requestAnimationFrame(resolve)); +} + /** * Handler to await completion of active `CSSTransitions`. * An optional `transition-property` to await can be specified, else all properties will be evaluated. diff --git a/js/views/ShadowView.js b/js/views/ShadowView.js new file mode 100644 index 00000000..5f32058b --- /dev/null +++ b/js/views/ShadowView.js @@ -0,0 +1,55 @@ +import Adapt from 'core/js/adapt'; +import Backbone from 'backbone'; +import { transitionNextFrame, transitionsEnded } from '../transitions'; + +class ShadowView extends Backbone.View { + + className() { + return 'shadow js-shadow u-display-none'; + } + + attributes() { + return { + id: 'shadow' + }; + } + + initialize() { + this._isOpen = false; + this.render(); + } + + render() { + const template = Handlebars.templates.shadow; + this.$el.html(template({ _globals: Adapt.course.get('_globals') })); + return this; + } + + get isOpen() { + return this._isOpen; + } + + async showShadow() { + this._isOpen = true; + this.$el.addClass('anim-open-before'); + await transitionNextFrame(); + this.$el.removeClass('u-display-none'); + await transitionNextFrame(); + this.$el.addClass('anim-open-after'); + await transitionsEnded(this.$el); + } + + async hideShadow() { + this._isOpen = false; + this.$el.addClass('anim-close-before'); + await transitionNextFrame(); + this.$el.addClass('anim-close-after'); + await transitionsEnded(this.$el); + this.$el.addClass('u-display-none'); + await transitionNextFrame(); + this.$el.removeClass('anim-open-before anim-open-after anim-close-before anim-close-after'); + } + +} + +export default ShadowView; diff --git a/js/views/drawerView.js b/js/views/drawerView.js index a845d5c9..ff44acaa 100644 --- a/js/views/drawerView.js +++ b/js/views/drawerView.js @@ -1,10 +1,19 @@ import Adapt from 'core/js/adapt'; +import shadow from '../shadow'; import a11y from 'core/js/a11y'; import DrawerItemView from 'core/js/views/drawerItemView'; import Backbone from 'backbone'; +import { + transitionNextFrame, + transitionsEnded +} from '../transitions'; class DrawerView extends Backbone.View { + tagName() { + return 'dialog'; + } + className() { return [ 'drawer', @@ -14,7 +23,6 @@ class DrawerView extends Backbone.View { attributes() { return { - role: 'dialog', 'aria-modal': 'true', 'aria-labelledby': 'drawer-heading', 'aria-hidden': 'true', @@ -32,6 +40,7 @@ class DrawerView extends Backbone.View { initialize() { this._isVisible = false; this.disableAnimation = Adapt.config.has('_disableAnimation') ? Adapt.config.get('_disableAnimation') : false; + this.$el.toggleClass('disable-animation', Boolean(this.disableAnimation)); this._globalDrawerPosition = Adapt.config.get('_drawer')?._position ?? 'auto'; this.drawerDuration = Adapt.config.get('_drawer')?._duration ?? 400; this.setupEventListeners(); @@ -41,6 +50,7 @@ class DrawerView extends Backbone.View { setupEventListeners() { this.onKeyUp = this.onKeyUp.bind(this); $(window).on('keyup', this.onKeyUp); + this.el.addEventListener('click', this.onShadowClicked.bind(this), { capture: true }); } onKeyUp(event) { @@ -49,18 +59,24 @@ class DrawerView extends Backbone.View { this.hideDrawer(); } + onShadowClicked(event) { + const dialog = this.el; + const rect = dialog.getBoundingClientRect(); + const isInDialog = (rect.top <= event.clientY && event.clientY <= rect.top + rect.height && + rect.left <= event.clientX && event.clientX <= rect.left + rect.width); + if (isInDialog) return; + event.preventDefault(); + this.hideDrawer(); + } + render() { const template = Handlebars.templates.drawer; - $(this.el).html(template({ _globals: Adapt.course.get('_globals') })).prependTo('body'); - const shadowTemplate = Handlebars.templates.shadow; - $(shadowTemplate()).prependTo('body'); + this.$el.html(template({ _globals: Adapt.course.get('_globals') })); _.defer(this.postRender.bind(this)); return this; } postRender() { - this.$('a, button, input, select, textarea').attr('tabindex', -1); - this.checkIfDrawerIsAvailable(); } @@ -74,7 +90,6 @@ class DrawerView extends Backbone.View { .removeClass(`is-position-${this.drawerPosition}`) .addClass(`is-position-${position}`); this.drawerPosition = position; - this.drawerAnimationDir = (position === 'auto') ? (isRTL ? 'left' : 'right') : position; } openCustomView(view, hasBackButton = true, position) { @@ -109,7 +124,8 @@ class DrawerView extends Backbone.View { return (this._isVisible && this._isCustomViewVisible === false); } - showDrawer(emptyDrawer, position = null) { + async showDrawer(emptyDrawer, position = null) { + shadow.show(); this.setDrawerPosition(position); this.$el .removeClass('u-display-none') @@ -122,9 +138,6 @@ class DrawerView extends Backbone.View { this._isVisible = true; } - // Sets tab index to 0 for all tabbable elements in Drawer - this.$('a, button, input, select, textarea').attr('tabindex', 0); - if (emptyDrawer) { this.$('.drawer__back').addClass('u-display-none'); this._isCustomViewVisible = false; @@ -148,34 +161,15 @@ class DrawerView extends Backbone.View { Adapt.trigger('drawer:openedCustomView'); } - $('.js-shadow').removeClass('u-display-none'); $('.js-drawer-holder').scrollTop(0); - const direction = {}; - direction[this.drawerAnimationDir] = 0; - - const complete = () => { - this.addShadowEvent(); - $('.js-nav-drawer-btn').attr('aria-expanded', true); - Adapt.trigger('drawer:opened'); - // focus on first tabbable element in drawer - a11y.focusFirst(this.$el, { defer: true }); - }; + $('.js-nav-drawer-btn').attr('aria-expanded', true); + Adapt.trigger('drawer:opened'); + + this.$el.addClass('anim-open-before'); + await transitionNextFrame(); + this.$el.addClass('anim-open-after'); + await transitionsEnded(this.$el); - // delay drawer animation until after background fadeout animation is complete - if (this.disableAnimation) { - this.$el.css(direction); - complete(); - } else { - const easing = Adapt.config.get('_drawer')?._showEasing || 'easeOutQuart'; - this.$el.velocity(direction, this.drawerDuration, easing); - - $('.js-shadow').velocity({ opacity: 1 }, { - duration: this.drawerDuration, - begin: () => { - complete(); - } - }); - } } emptyDrawer() { @@ -191,56 +185,34 @@ class DrawerView extends Backbone.View { this.collection.forEach(model => new DrawerItemView({ model })); } - hideDrawer($toElement) { + async hideDrawer($toElement) { if (!this._isVisible) return; this._useMenuPosition = false; - const direction = {}; - a11y.popupClosed($toElement); - this._isVisible = false; - a11y.scrollEnable('body'); - direction[this.drawerAnimationDir] = -this.$el.width(); - - const complete = () => { - this.$el - .removeAttr('style') - .addClass('u-display-none') - .attr('aria-hidden', 'true') - .attr('aria-expanded', 'false'); - this.$('.js-drawer-holder').removeAttr('role'); - this._customView = null; - $('.js-nav-drawer-btn').attr('aria-expanded', false); - Adapt.trigger('drawer:closed'); - this.setDrawerPosition(this._globalDrawerPosition); - }; - - if (this.disableAnimation) { - this.$el.css(direction); - $('.js-shadow').addClass('u-display-none'); - complete(); - } else { - const easing = Adapt.config.get('_drawer')?._hideEasing || 'easeInQuart'; - this.$el.velocity(direction, this.drawerDuration, easing, () => { - complete(); - }); - - $('.js-shadow').velocity({ opacity: 0 }, { - duration: this.drawerDuration, - complete() { - $('.js-shadow').addClass('u-display-none'); - } - }); - } this._isCustomViewVisible = false; - this.removeShadowEvent(); - } + shadow.hide(); - addShadowEvent() { - $('.js-shadow').one('click touchstart', () => this.hideDrawer()); - } + this.$el.addClass('anim-close-before'); + await transitionNextFrame(); + this.$el.addClass('anim-close-after'); + await transitionsEnded(this.$el); - removeShadowEvent() { - $('.js-shadow').off('click touchstart'); + this.$el.removeClass('anim-open-before anim-open-after anim-close-before anim-close-after'); + + a11y.popupClosed($toElement); + this._isVisible = false; + a11y.scrollEnable('body'); + + this.$el + .removeAttr('style') + .addClass('u-display-none') + .attr('aria-hidden', 'true') + .attr('aria-expanded', 'false'); + this.$('.js-drawer-holder').removeAttr('role'); + this._customView = null; + $('.js-nav-drawer-btn').attr('aria-expanded', false); + Adapt.trigger('drawer:closed'); + this.setDrawerPosition(this._globalDrawerPosition); } remove() { @@ -249,7 +221,6 @@ class DrawerView extends Backbone.View { $(window).off('keyup', this.onKeyUp); Adapt.trigger('drawer:empty'); this.collection.reset(); - $('.js-shadow').remove(); } } diff --git a/js/views/notifyPopupView.js b/js/views/notifyPopupView.js index ca0530d5..2c4a6b06 100644 --- a/js/views/notifyPopupView.js +++ b/js/views/notifyPopupView.js @@ -4,6 +4,7 @@ import data from 'core/js/data'; import a11y from 'core/js/a11y'; import AdaptView from 'core/js/views/adaptView'; import Backbone from 'backbone'; +import { transitionNextFrame, transitionsEnded } from '../transitions'; export default class NotifyPopupView extends Backbone.View { @@ -12,19 +13,14 @@ export default class NotifyPopupView extends Backbone.View { } attributes() { - return Object.assign({ - role: 'dialog', - 'aria-labelledby': 'notify-heading', - 'aria-modal': 'true' - }, this.model.get('_attributes')); + return this.model.get('_attributes'); } events() { return { 'click .js-notify-btn-alert': 'onAlertButtonClicked', 'click .js-notify-btn-prompt': 'onPromptButtonClicked', - 'click .js-notify-close-btn': 'onCloseButtonClicked', - 'click .js-notify-shadow-click': 'onShadowClicked' + 'click .js-notify-close-btn': 'onCloseButtonClicked' }; } @@ -32,19 +28,20 @@ export default class NotifyPopupView extends Backbone.View { this.notify = notify; _.bindAll(this, 'resetNotifySize', 'onKeyUp'); this.disableAnimation = Adapt.config.get('_disableAnimation') || false; + this.$el.toggleClass('disable-animation', Boolean(this.disableAnimation)); this.isOpen = false; this.hasOpened = false; this.setupEventListeners(); this.render(); + this.$('.notify__popup')[0].addEventListener('click', this.onShadowClicked.bind(this), { capture: true }); } setupEventListeners() { this.listenTo(Adapt, { remove: this.closeNotify, - 'notify:resize': this.resetNotifySize, + 'notify:resize device:resize': this.resetNotifySize, 'notify:cancel': this.cancelNotify, - 'notify:close': this.closeNotify, - 'device:resize': this.resetNotifySize + 'notify:close': this.closeNotify }); this.setupEscapeKey(); } @@ -62,14 +59,7 @@ export default class NotifyPopupView extends Backbone.View { render() { const data = this.model.toJSON(); const template = Handlebars.templates.notifyPopup; - // hide notify container - this.$el.css('visibility', 'hidden'); - // attach popup + shadow this.$el.html(template(data)).appendTo('.notify__popup-container'); - // hide popup - this.$('.notify__popup').css('visibility', 'hidden'); - // show notify container - this.$el.css('visibility', 'visible'); this.showNotify(); return this; } @@ -95,6 +85,11 @@ export default class NotifyPopupView extends Backbone.View { } onShadowClicked(event) { + const dialog = this.$('.notify__popup')[0]; + const rect = dialog.getBoundingClientRect(); + const isInDialog = (rect.top <= event.clientY && event.clientY <= rect.top + rect.height && + rect.left <= event.clientX && event.clientX <= rect.left + rect.width); + if (isInDialog) return; event.preventDefault(); if (this.model.get('_closeOnShadowClick') === false) return; this.cancelNotify(); @@ -117,9 +112,7 @@ export default class NotifyPopupView extends Backbone.View { const notifyHeight = this.$('.notify__popup-inner').outerHeight(); const isFullWindow = (notifyHeight >= windowHeight); this.$('.notify__popup').css({ - height: isFullWindow ? '100%' : 'auto', - top: isFullWindow ? 0 : '', - 'margin-top': isFullWindow ? '' : -(notifyHeight / 2), + height: isFullWindow ? '100%' : notifyHeight, 'overflow-y': isFullWindow ? 'scroll' : '', '-webkit-overflow-scrolling': isFullWindow ? 'touch' : '' }); @@ -136,42 +129,17 @@ export default class NotifyPopupView extends Backbone.View { this.$el.imageready(this.onLoaded.bind(this)); } - onLoaded() { - if (this.disableAnimation) { - this.$('.notify__shadow').css('display', 'block'); - } else { - this.$('.notify__shadow').velocity({ opacity: 0 }, { duration: 0 }).velocity({ opacity: 1 }, { - duration: 400, - begin: () => { - this.$('.notify__shadow').css('display', 'block'); - } - }); - } - this.resizeNotify(); - if (this.disableAnimation) { - this.$('.notify__popup').css('visibility', 'visible'); - this.onOpened(); - } else { - this.$('.notify__popup').velocity({ opacity: 0 }, { duration: 0 }).velocity({ opacity: 1 }, { - duration: 400, - begin: () => { - // Make sure to make the notify visible and then set - // focus, disabled scroll and manage tabs - this.$('.notify__popup').css('visibility', 'visible'); - this.onOpened(); - } - }); - } - } - - onOpened() { + async onLoaded() { this.hasOpened = true; // Allows popup manager to control focus - a11y.popupOpened(this.$el); + a11y.popupOpened(this.$('.notify__popup')); a11y.scrollDisable('body'); $('html').addClass('notify'); - // Set focus to first accessible element - a11y.focusFirst(this.$('.notify__popup'), { defer: false }); + + this.$el.addClass('anim-open-before'); + await transitionNextFrame(); + this.$el.addClass('anim-open-after'); + await transitionsEnded(this.$('.notify__popup, .notify__shadow')); } async addSubView() { @@ -217,26 +185,14 @@ export default class NotifyPopupView extends Backbone.View { }); } - onCloseReady() { - if (this.disableAnimation) { - this.$('.notify__popup').css('visibility', 'hidden'); - this.$el.css('visibility', 'hidden'); - this.remove(); - } else { - this.$('.notify__popup').velocity({ opacity: 0 }, { - duration: 400, - complete: () => { - this.$('.notify__popup').css('visibility', 'hidden'); - } - }); - this.$('.notify__shadow').velocity({ opacity: 0 }, { - duration: 400, - complete: () => { - this.$el.css('visibility', 'hidden'); - this.remove(); - } - }); - } + async onCloseReady() { + this.$el.addClass('anim-close-before'); + await transitionNextFrame(); + this.$el.addClass('anim-close-after'); + await transitionsEnded(this.$('.notify__popup, .notify__shadow')); + + this.remove(); + a11y.scrollEnable('body'); $('html').removeClass('notify'); // Return focus to previous active element diff --git a/js/views/notifyPushView.js b/js/views/notifyPushView.js index 47c1c7fd..67acdea0 100644 --- a/js/views/notifyPushView.js +++ b/js/views/notifyPushView.js @@ -2,6 +2,10 @@ import Adapt from 'core/js/adapt'; export default class NotifyPushView extends Backbone.View { + tagName() { + return 'dialog'; + } + className() { const classes = [ 'notify-push', @@ -13,7 +17,6 @@ export default class NotifyPushView extends Backbone.View { attributes() { return { - role: 'dialog', 'aria-labelledby': 'notify-push-heading', 'aria-modal': 'false' }; diff --git a/less/_defaults/base.less b/less/_defaults/base.less index 9d330118..dbeb14bf 100644 --- a/less/_defaults/base.less +++ b/less/_defaults/base.less @@ -74,3 +74,18 @@ zw { nb { white-space: nowrap; } + +dialog { + background: transparent; + padding: 0; + margin: 0; + max-width: inherit; + max-height: inherit; + width: 100%; + height: 100%; + border: 0; + + &::backdrop { + background: transparent; + } +} diff --git a/less/core/drawer.less b/less/core/drawer.less index f857bee2..6ba44c18 100644 --- a/less/core/drawer.less +++ b/less/core/drawer.less @@ -1,6 +1,7 @@ .drawer { position: fixed; top: 0; + opacity: 0; // Update JS to set start point // Update animation to be css rather than js height: 100%; diff --git a/less/core/notify.less b/less/core/notify.less index c55beac4..3e9c43d4 100644 --- a/less/core/notify.less +++ b/less/core/notify.less @@ -1,12 +1,15 @@ .notify { - position: relative; z-index: 100; &__popup { position: fixed; top: 50%; + transform: translateY(-50%); + opacity: 0; + height: fit-content; width: 100%; - visibility: hidden; + overflow-y: hidden; + max-height: 100vh; z-index: 100; background-color: @background-inverted; } @@ -45,7 +48,7 @@ .notify__text { width: 60%; } - + .notify__image-container { width: 40%; } diff --git a/templates/notifyPopup.hbs b/templates/notifyPopup.hbs index e0042f40..5bfdaf4c 100644 --- a/templates/notifyPopup.hbs +++ b/templates/notifyPopup.hbs @@ -1,7 +1,7 @@ {{! make the _globals object in course.json available to this template}} {{import_globals}} -
+
@@ -101,6 +101,6 @@ {{{a11y_wrap_focus}}} -
+
diff --git a/templates/shadow.hbs b/templates/shadow.hbs index a5e368e5..8b137891 100644 --- a/templates/shadow.hbs +++ b/templates/shadow.hbs @@ -1,2 +1 @@ -{{! drawerView.js will appends this on document's body}} -
+