A11Y: Switch tabs using the keyboard (#12262)
* Revert "Revert "A11Y: Switch tabs using the keyboard (#12241)" (#12260)"
This reverts commit 4c1e02d412
.
* FIX: Make sure that the "menu-link" is present when a plugin adds a tab.
Other changes:
- We put the notification tab first using JS instead of CSS. It's important because of the tab number data attribute, which the keyboard navigation uses.
- We only set the button id from the attrs object if it's a tab. Otherwise, it conflicts with the topic footer button
This commit is contained in:
parent
1fc67cc26a
commit
5276d432aa
|
@ -6,6 +6,7 @@ import PanEvents, {
|
||||||
import { cancel, later, schedule } from "@ember/runloop";
|
import { cancel, later, schedule } from "@ember/runloop";
|
||||||
import Docking from "discourse/mixins/docking";
|
import Docking from "discourse/mixins/docking";
|
||||||
import MountWidget from "discourse/components/mount-widget";
|
import MountWidget from "discourse/components/mount-widget";
|
||||||
|
import Mousetrap from "mousetrap";
|
||||||
import RerenderOnDoNotDisturbChange from "discourse/mixins/rerender-on-do-not-disturb-change";
|
import RerenderOnDoNotDisturbChange from "discourse/mixins/rerender-on-do-not-disturb-change";
|
||||||
import { observes } from "discourse-common/utils/decorators";
|
import { observes } from "discourse-common/utils/decorators";
|
||||||
import { topicTitleDecorators } from "discourse/components/topic-title";
|
import { topicTitleDecorators } from "discourse/components/topic-title";
|
||||||
|
@ -25,6 +26,7 @@ const SiteHeaderComponent = MountWidget.extend(
|
||||||
_scheduledMovingAnimation: null,
|
_scheduledMovingAnimation: null,
|
||||||
_scheduledRemoveAnimate: null,
|
_scheduledRemoveAnimate: null,
|
||||||
_topic: null,
|
_topic: null,
|
||||||
|
_mousetrap: null,
|
||||||
|
|
||||||
@observes(
|
@observes(
|
||||||
"currentUser.unread_notifications",
|
"currentUser.unread_notifications",
|
||||||
|
@ -209,6 +211,7 @@ const SiteHeaderComponent = MountWidget.extend(
|
||||||
this.dispatch("notifications:changed", "user-notifications");
|
this.dispatch("notifications:changed", "user-notifications");
|
||||||
this.dispatch("header:keyboard-trigger", "header");
|
this.dispatch("header:keyboard-trigger", "header");
|
||||||
this.dispatch("search-autocomplete:after-complete", "search-term");
|
this.dispatch("search-autocomplete:after-complete", "search-term");
|
||||||
|
this.dispatch("user-menu:navigation", "user-menu");
|
||||||
|
|
||||||
this.appEvents.on("dom:clean", this, "_cleanDom");
|
this.appEvents.on("dom:clean", this, "_cleanDom");
|
||||||
|
|
||||||
|
@ -236,6 +239,26 @@ const SiteHeaderComponent = MountWidget.extend(
|
||||||
once: true,
|
once: true,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const header = document.querySelector("header.d-header");
|
||||||
|
const mousetrap = new Mousetrap(header);
|
||||||
|
mousetrap.bind(["right", "left"], (e) => {
|
||||||
|
const activeTab = document.querySelector(".glyphs .menu-link.active");
|
||||||
|
|
||||||
|
if (activeTab) {
|
||||||
|
let focusedTab = document.activeElement;
|
||||||
|
if (!focusedTab.dataset.tabNumber) {
|
||||||
|
focusedTab = activeTab;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.appEvents.trigger("user-menu:navigation", {
|
||||||
|
key: e.key,
|
||||||
|
tabNumber: Number(focusedTab.dataset.tabNumber),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
this.set("_mousetrap", mousetrap);
|
||||||
},
|
},
|
||||||
|
|
||||||
_cleanDom() {
|
_cleanDom() {
|
||||||
|
@ -257,6 +280,8 @@ const SiteHeaderComponent = MountWidget.extend(
|
||||||
cancel(this._scheduledRemoveAnimate);
|
cancel(this._scheduledRemoveAnimate);
|
||||||
window.cancelAnimationFrame(this._scheduledMovingAnimation);
|
window.cancelAnimationFrame(this._scheduledMovingAnimation);
|
||||||
|
|
||||||
|
this._mousetrap.unbind(["right", "left"]);
|
||||||
|
|
||||||
document.removeEventListener("click", this._dismissFirstNotification);
|
document.removeEventListener("click", this._dismissFirstNotification);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -51,6 +51,7 @@ export const ButtonClass = {
|
||||||
attributes["aria-selected"] = tab["aria-selected"];
|
attributes["aria-selected"] = tab["aria-selected"];
|
||||||
attributes["tabindex"] = tab["tabindex"];
|
attributes["tabindex"] = tab["tabindex"];
|
||||||
attributes["aria-controls"] = tab["aria-controls"];
|
attributes["aria-controls"] = tab["aria-controls"];
|
||||||
|
attributes["id"] = attrs.id;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (attrs.disabled) {
|
if (attrs.disabled) {
|
||||||
|
|
|
@ -40,9 +40,13 @@ export default createWidget("quick-access-panel", {
|
||||||
return Promise.resolve([]);
|
return Promise.resolve([]);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
buildId() {
|
||||||
|
return this.key;
|
||||||
|
},
|
||||||
|
|
||||||
buildAttributes() {
|
buildAttributes() {
|
||||||
const attributes = this.attrs;
|
const attributes = this.attrs;
|
||||||
attributes["aria-labelledby"] = this.key;
|
attributes["aria-labelledby"] = attributes.currentQuickAccess;
|
||||||
attributes["tabindex"] = "0";
|
attributes["tabindex"] = "0";
|
||||||
attributes["role"] = "tabpanel";
|
attributes["role"] = "tabpanel";
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
|
import { later } from "@ember/runloop";
|
||||||
import { createWidget } from "discourse/widgets/widget";
|
import { createWidget } from "discourse/widgets/widget";
|
||||||
import { h } from "virtual-dom";
|
import { h } from "virtual-dom";
|
||||||
import { later } from "@ember/runloop";
|
|
||||||
|
|
||||||
const UserMenuAction = {
|
const UserMenuAction = {
|
||||||
QUICK_ACCESS: "quickAccess",
|
QUICK_ACCESS: "quickAccess",
|
||||||
|
@ -50,6 +50,12 @@ createWidget("user-menu-links", {
|
||||||
glyph.href = null;
|
glyph.href = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (glyph.className) {
|
||||||
|
glyph.className += " menu-link";
|
||||||
|
} else {
|
||||||
|
glyph.className = "menu-link";
|
||||||
|
}
|
||||||
|
|
||||||
glyph.role = "tab";
|
glyph.role = "tab";
|
||||||
glyph.tabAttrs = this._tabAttrs(glyph.actionParam);
|
glyph.tabAttrs = this._tabAttrs(glyph.actionParam);
|
||||||
|
|
||||||
|
@ -59,7 +65,8 @@ createWidget("user-menu-links", {
|
||||||
profileGlyph() {
|
profileGlyph() {
|
||||||
return {
|
return {
|
||||||
title: Titles["profile"],
|
title: Titles["profile"],
|
||||||
className: "user-preferences-link",
|
className: "user-preferences-link menu-link",
|
||||||
|
id: QuickAccess.PROFILE,
|
||||||
icon: "user",
|
icon: "user",
|
||||||
action: UserMenuAction.QUICK_ACCESS,
|
action: UserMenuAction.QUICK_ACCESS,
|
||||||
actionParam: QuickAccess.PROFILE,
|
actionParam: QuickAccess.PROFILE,
|
||||||
|
@ -72,7 +79,8 @@ createWidget("user-menu-links", {
|
||||||
notificationsGlyph() {
|
notificationsGlyph() {
|
||||||
return {
|
return {
|
||||||
title: Titles["notifications"],
|
title: Titles["notifications"],
|
||||||
className: "user-notifications-link",
|
className: "user-notifications-link menu-link",
|
||||||
|
id: QuickAccess.NOTIFICATIONS,
|
||||||
icon: "bell",
|
icon: "bell",
|
||||||
action: UserMenuAction.QUICK_ACCESS,
|
action: UserMenuAction.QUICK_ACCESS,
|
||||||
actionParam: QuickAccess.NOTIFICATIONS,
|
actionParam: QuickAccess.NOTIFICATIONS,
|
||||||
|
@ -87,7 +95,8 @@ createWidget("user-menu-links", {
|
||||||
title: Titles["bookmarks"],
|
title: Titles["bookmarks"],
|
||||||
action: UserMenuAction.QUICK_ACCESS,
|
action: UserMenuAction.QUICK_ACCESS,
|
||||||
actionParam: QuickAccess.BOOKMARKS,
|
actionParam: QuickAccess.BOOKMARKS,
|
||||||
className: "user-bookmarks-link",
|
className: "user-bookmarks-link menu-link",
|
||||||
|
id: QuickAccess.BOOKMARKS,
|
||||||
icon: "bookmark",
|
icon: "bookmark",
|
||||||
data: { url: `${this.attrs.path}/activity/bookmarks` },
|
data: { url: `${this.attrs.path}/activity/bookmarks` },
|
||||||
"aria-label": "user.bookmarks",
|
"aria-label": "user.bookmarks",
|
||||||
|
@ -101,7 +110,8 @@ createWidget("user-menu-links", {
|
||||||
title: Titles["messages"],
|
title: Titles["messages"],
|
||||||
action: UserMenuAction.QUICK_ACCESS,
|
action: UserMenuAction.QUICK_ACCESS,
|
||||||
actionParam: QuickAccess.MESSAGES,
|
actionParam: QuickAccess.MESSAGES,
|
||||||
className: "user-pms-link",
|
className: "user-pms-link menu-link",
|
||||||
|
id: QuickAccess.MESSAGES,
|
||||||
icon: "envelope",
|
icon: "envelope",
|
||||||
data: { url: `${this.attrs.path}/messages` },
|
data: { url: `${this.attrs.path}/messages` },
|
||||||
role: "tab",
|
role: "tab",
|
||||||
|
@ -116,15 +126,17 @@ createWidget("user-menu-links", {
|
||||||
return this.attach("link", link);
|
return this.attach("link", link);
|
||||||
},
|
},
|
||||||
|
|
||||||
glyphHtml(glyph) {
|
glyphHtml(glyph, idx) {
|
||||||
if (this.isActive(glyph)) {
|
if (this.isActive(glyph)) {
|
||||||
glyph = this.markAsActive(glyph);
|
glyph = this.markAsActive(glyph);
|
||||||
}
|
}
|
||||||
|
glyph.data["tab-number"] = `${idx}`;
|
||||||
|
|
||||||
return this.attach("flat-button", glyph);
|
return this.attach("flat-button", glyph);
|
||||||
},
|
},
|
||||||
|
|
||||||
html() {
|
html() {
|
||||||
const glyphs = [];
|
const glyphs = [this.notificationsGlyph()];
|
||||||
|
|
||||||
if (extraGlyphs) {
|
if (extraGlyphs) {
|
||||||
extraGlyphs.forEach((g) => {
|
extraGlyphs.forEach((g) => {
|
||||||
|
@ -140,7 +152,6 @@ createWidget("user-menu-links", {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
glyphs.push(this.notificationsGlyph());
|
|
||||||
glyphs.push(this.bookmarksGlyph());
|
glyphs.push(this.bookmarksGlyph());
|
||||||
|
|
||||||
if (this.siteSettings.enable_personal_messages || this.currentUser.staff) {
|
if (this.siteSettings.enable_personal_messages || this.currentUser.staff) {
|
||||||
|
@ -153,7 +164,7 @@ createWidget("user-menu-links", {
|
||||||
h(
|
h(
|
||||||
"div.glyphs",
|
"div.glyphs",
|
||||||
{ attributes: { "aria-label": "Menu links", role: "tablist" } },
|
{ attributes: { "aria-label": "Menu links", role: "tablist" } },
|
||||||
glyphs.map((l) => this.glyphHtml(l))
|
glyphs.map((l, index) => this.glyphHtml(l, index))
|
||||||
),
|
),
|
||||||
]);
|
]);
|
||||||
},
|
},
|
||||||
|
@ -194,6 +205,25 @@ export default createWidget("user-menu", {
|
||||||
showLogoutButton: true,
|
showLogoutButton: true,
|
||||||
},
|
},
|
||||||
|
|
||||||
|
userMenuNavigation(nav) {
|
||||||
|
const maxTabNumber = document.querySelectorAll(".glyphs button").length - 1;
|
||||||
|
const isLeft = nav.key === "ArrowLeft";
|
||||||
|
|
||||||
|
let nextTab = isLeft ? nav.tabNumber - 1 : nav.tabNumber + 1;
|
||||||
|
|
||||||
|
if (isLeft && nextTab < 0) {
|
||||||
|
nextTab = maxTabNumber;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isLeft && nextTab > maxTabNumber) {
|
||||||
|
nextTab = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
document
|
||||||
|
.querySelector(`.menu-link[role='tab'][data-tab-number='${nextTab}']`)
|
||||||
|
.focus();
|
||||||
|
},
|
||||||
|
|
||||||
defaultState() {
|
defaultState() {
|
||||||
return {
|
return {
|
||||||
currentQuickAccess: QuickAccess.NOTIFICATIONS,
|
currentQuickAccess: QuickAccess.NOTIFICATIONS,
|
||||||
|
@ -212,7 +242,7 @@ export default createWidget("user-menu", {
|
||||||
path,
|
path,
|
||||||
currentQuickAccess,
|
currentQuickAccess,
|
||||||
}),
|
}),
|
||||||
this.quickAccessPanel(path, titleKey),
|
this.quickAccessPanel(path, titleKey, currentQuickAccess),
|
||||||
];
|
];
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
|
@ -269,13 +299,14 @@ export default createWidget("user-menu", {
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
quickAccessPanel(path, titleKey) {
|
quickAccessPanel(path, titleKey, currentQuickAccess) {
|
||||||
const { showLogoutButton } = this.settings;
|
const { showLogoutButton } = this.settings;
|
||||||
// This deliberately does NOT fallback to a default quick access panel.
|
// This deliberately does NOT fallback to a default quick access panel.
|
||||||
return this.attach(`quick-access-${this.state.currentQuickAccess}`, {
|
return this.attach(`quick-access-${this.state.currentQuickAccess}`, {
|
||||||
path,
|
path,
|
||||||
showLogoutButton,
|
showLogoutButton,
|
||||||
titleKey,
|
titleKey,
|
||||||
|
currentQuickAccess,
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
|
@ -389,11 +389,6 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.user-notifications-link {
|
|
||||||
// keep this in leftmost position consistently
|
|
||||||
order: -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
div.menu-links-header {
|
div.menu-links-header {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
.menu-links-row {
|
.menu-links-row {
|
||||||
|
@ -416,6 +411,10 @@ div.menu-links-header {
|
||||||
flex: 1 1 auto;
|
flex: 1 1 auto;
|
||||||
padding: 0.65em 0.25em 0.75em;
|
padding: 0.65em 0.25em 0.75em;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
|
|
||||||
|
svg {
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue