From e69c511e00497ef45ba01f9a04d48134ad41f181 Mon Sep 17 00:00:00 2001 From: Michael Telatynski <7t3chguy@gmail.com> Date: Mon, 17 Apr 2023 12:40:44 +0100 Subject: [PATCH 1/7] RovingTabIndex handle looping around start/end --- src/accessibility/RovingTabIndex.tsx | 15 ++++++++++++--- 1 file changed, 12 insertions(+), 3 deletions(-) diff --git a/src/accessibility/RovingTabIndex.tsx b/src/accessibility/RovingTabIndex.tsx index b449b10710f..e2f1db790b9 100644 --- a/src/accessibility/RovingTabIndex.tsx +++ b/src/accessibility/RovingTabIndex.tsx @@ -156,6 +156,7 @@ export const reducer: Reducer = (state: IState, action: IAction }; interface IProps { + handleLoop?: boolean; handleHomeEnd?: boolean; handleUpDown?: boolean; handleLeftRight?: boolean; @@ -167,6 +168,7 @@ export const findSiblingElement = ( refs: RefObject[], startIndex: number, backwards = false, + loop = false, ): RefObject | undefined => { if (backwards) { for (let i = startIndex; i < refs.length && i >= 0; i--) { @@ -174,12 +176,18 @@ export const findSiblingElement = ( return refs[i]; } } + if (loop) { + return findSiblingElement(refs.slice(startIndex + 1), refs.length - 1, true, false); + } } else { for (let i = startIndex; i < refs.length && i >= 0; i++) { if (refs[i].current?.offsetParent !== null) { return refs[i]; } } + if (loop) { + return findSiblingElement(refs.slice(0, startIndex), 0, false, false); + } } }; @@ -188,6 +196,7 @@ export const RovingTabIndexProvider: React.FC = ({ handleHomeEnd, handleUpDown, handleLeftRight, + handleLoop, onKeyDown, }) => { const [state, dispatch] = useReducer>(reducer, { @@ -252,7 +261,7 @@ export const RovingTabIndexProvider: React.FC = ({ handled = true; if (context.state.refs.length > 0) { const idx = context.state.refs.indexOf(context.state.activeRef!); - focusRef = findSiblingElement(context.state.refs, idx + 1); + focusRef = findSiblingElement(context.state.refs, idx + 1, false, handleLoop); } } break; @@ -266,7 +275,7 @@ export const RovingTabIndexProvider: React.FC = ({ handled = true; if (context.state.refs.length > 0) { const idx = context.state.refs.indexOf(context.state.activeRef!); - focusRef = findSiblingElement(context.state.refs, idx - 1, true); + focusRef = findSiblingElement(context.state.refs, idx - 1, true, handleLoop); } } break; @@ -289,7 +298,7 @@ export const RovingTabIndexProvider: React.FC = ({ }); } }, - [context, onKeyDown, handleHomeEnd, handleUpDown, handleLeftRight], + [context, onKeyDown, handleHomeEnd, handleUpDown, handleLeftRight, handleLoop], ); return ( From 5bcf4d1cce85ff32d980629ccd220785d23b285d Mon Sep 17 00:00:00 2001 From: Michael Telatynski <7t3chguy@gmail.com> Date: Mon, 17 Apr 2023 12:41:08 +0100 Subject: [PATCH 2/7] Make TabbedView expose aria tabpanel/tablist/tab roles --- src/components/structures/TabbedView.tsx | 46 +++++++++++++++++++----- 1 file changed, 37 insertions(+), 9 deletions(-) diff --git a/src/components/structures/TabbedView.tsx b/src/components/structures/TabbedView.tsx index 0d3d01041be..c33a724689c 100644 --- a/src/components/structures/TabbedView.tsx +++ b/src/components/structures/TabbedView.tsx @@ -22,9 +22,9 @@ import { logger } from "matrix-js-sdk/src/logger"; import { _t } from "../../languageHandler"; import AutoHideScrollbar from "./AutoHideScrollbar"; -import AccessibleButton from "../views/elements/AccessibleButton"; import { PosthogScreenTracker, ScreenName } from "../../PosthogTrackers"; import { NonEmptyArray } from "../../@types/common"; +import { RovingAccessibleButton, RovingTabIndexProvider } from "../../accessibility/RovingTabIndex"; /** * Represents a tab for the TabbedView. @@ -98,9 +98,10 @@ export default class TabbedView extends React.Component { } private renderTabLabel(tab: Tab): JSX.Element { - let classes = "mx_TabbedView_tabLabel "; - - if (this.state.activeTabId === tab.id) classes += "mx_TabbedView_tabLabel_active"; + const isActive = this.state.activeTabId === tab.id; + const classes = classNames("mx_TabbedView_tabLabel", { + mx_TabbedView_tabLabel_active: isActive, + }); let tabIcon: JSX.Element | undefined; if (tab.icon) { @@ -108,24 +109,35 @@ export default class TabbedView extends React.Component { } const onClickHandler = (): void => this.setActiveTab(tab); + const id = this.getTabId(tab); const label = _t(tab.label); return ( - {tabIcon} - {label} - + + {label} + + ); } + private getTabId(tab: Tab): string { + return `mx_tabpanel_${tab.label}`; + } + private renderTabPanel(tab: Tab): React.ReactNode { + const id = this.getTabId(tab); return ( -
+
{tab.body}
); @@ -147,7 +159,23 @@ export default class TabbedView extends React.Component { return (
{screenName && } -
{labels}
+ + {({ onKeyDownHandler }) => ( +
+ {labels} +
+ )} +
{panel}
); From 78b6ed9fe52b28d03bc4d8000105ba41d9b57b17 Mon Sep 17 00:00:00 2001 From: Michael Telatynski <7t3chguy@gmail.com> Date: Mon, 17 Apr 2023 12:50:57 +0100 Subject: [PATCH 3/7] Fix right panel being wrongly specified as aria tabs Not all right panels map to the top right header buttons so we cannot describe it as a tabpanel relation --- src/components/views/right_panel/HeaderButton.tsx | 1 - src/components/views/right_panel/HeaderButtons.tsx | 6 +----- 2 files changed, 1 insertion(+), 6 deletions(-) diff --git a/src/components/views/right_panel/HeaderButton.tsx b/src/components/views/right_panel/HeaderButton.tsx index 6d6872bc1e9..3afaed785fb 100644 --- a/src/components/views/right_panel/HeaderButton.tsx +++ b/src/components/views/right_panel/HeaderButton.tsx @@ -55,7 +55,6 @@ export default class HeaderButton extends React.Component { extends React.Component - {this.renderButtons()} -
- ); + return
{this.renderButtons()}
; } } From 93530dfbe8bd3fcf06e73edb393a9dc9251074ee Mon Sep 17 00:00:00 2001 From: Michael Telatynski <7t3chguy@gmail.com> Date: Mon, 17 Apr 2023 13:06:47 +0100 Subject: [PATCH 4/7] tsc strict --- src/components/structures/FilePanel.tsx | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/components/structures/FilePanel.tsx b/src/components/structures/FilePanel.tsx index 9d92892d3bf..78bf9319547 100644 --- a/src/components/structures/FilePanel.tsx +++ b/src/components/structures/FilePanel.tsx @@ -276,7 +276,9 @@ class FilePanel extends React.Component { withoutScrollContainer ref={this.card} > - + {this.card.current && ( + + )} Date: Mon, 17 Apr 2023 13:27:26 +0100 Subject: [PATCH 5/7] Update snapshots --- src/components/structures/TabbedView.tsx | 2 +- .../__snapshots__/TabbedView-test.tsx.snap | 27 +++++-- .../RoomSettingsDialog-test.tsx.snap | 41 ++++++---- .../UserSettingsDialog-test.tsx.snap | 77 +++++++++++++------ 4 files changed, 101 insertions(+), 46 deletions(-) diff --git a/src/components/structures/TabbedView.tsx b/src/components/structures/TabbedView.tsx index c33a724689c..1efada71436 100644 --- a/src/components/structures/TabbedView.tsx +++ b/src/components/structures/TabbedView.tsx @@ -131,7 +131,7 @@ export default class TabbedView extends React.Component { } private getTabId(tab: Tab): string { - return `mx_tabpanel_${tab.label}`; + return `mx_tabpanel_${tab.id}`; } private renderTabPanel(tab: Tab): React.ReactNode { diff --git a/test/components/structures/__snapshots__/TabbedView-test.tsx.snap b/test/components/structures/__snapshots__/TabbedView-test.tsx.snap index 77ead236a31..acd6a4bae64 100644 --- a/test/components/structures/__snapshots__/TabbedView-test.tsx.snap +++ b/test/components/structures/__snapshots__/TabbedView-test.tsx.snap @@ -6,12 +6,16 @@ exports[` renders tabs 1`] = ` class="mx_TabbedView mx_TabbedView_tabsOnLeft" >
renders tabs 1`] = ` /> General
Labs
Security
Settings tabs renders default tabs correctly 1`] = ` NodeList [
General
,
Security & Privacy
,
Roles & Permissions
,
Notifications
,
Poll history diff --git a/test/components/views/dialogs/__snapshots__/UserSettingsDialog-test.tsx.snap b/test/components/views/dialogs/__snapshots__/UserSettingsDialog-test.tsx.snap index fb887a304df..4c1e9b20f94 100644 --- a/test/components/views/dialogs/__snapshots__/UserSettingsDialog-test.tsx.snap +++ b/test/components/views/dialogs/__snapshots__/UserSettingsDialog-test.tsx.snap @@ -3,9 +3,11 @@ exports[` renders tabs correctly 1`] = ` NodeList [
General
,
Appearance
,
Notifications
,
Preferences
,
Keyboard
,
Sidebar
,
Security & Privacy
,
Labs
,
Help & About From ad360cb3b042c9cd9e09064738666efff2b56f2d Mon Sep 17 00:00:00 2001 From: Michael Telatynski <7t3chguy@gmail.com> Date: Mon, 17 Apr 2023 13:48:49 +0100 Subject: [PATCH 6/7] Fix ARIA AXE violation --- src/components/views/right_panel/HeaderButton.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/components/views/right_panel/HeaderButton.tsx b/src/components/views/right_panel/HeaderButton.tsx index 3afaed785fb..03106face21 100644 --- a/src/components/views/right_panel/HeaderButton.tsx +++ b/src/components/views/right_panel/HeaderButton.tsx @@ -54,7 +54,7 @@ export default class HeaderButton extends React.Component { return ( Date: Mon, 17 Apr 2023 14:15:30 +0100 Subject: [PATCH 7/7] Update tests --- cypress/e2e/integration-manager/get-openid-token.spec.ts | 2 +- cypress/e2e/integration-manager/kick.spec.ts | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/cypress/e2e/integration-manager/get-openid-token.spec.ts b/cypress/e2e/integration-manager/get-openid-token.spec.ts index c1026a57876..b2dcb9146ae 100644 --- a/cypress/e2e/integration-manager/get-openid-token.spec.ts +++ b/cypress/e2e/integration-manager/get-openid-token.spec.ts @@ -59,7 +59,7 @@ const INTEGRATION_MANAGER_HTML = ` `; function openIntegrationManager() { - cy.findByRole("tab", { name: "Room info" }).click(); + cy.findByRole("button", { name: "Room info" }).click(); cy.findByRole("button", { name: "Add widgets, bridges & bots" }).click(); } diff --git a/cypress/e2e/integration-manager/kick.spec.ts b/cypress/e2e/integration-manager/kick.spec.ts index 4b29be4b23f..7075c1c199f 100644 --- a/cypress/e2e/integration-manager/kick.spec.ts +++ b/cypress/e2e/integration-manager/kick.spec.ts @@ -62,7 +62,7 @@ const INTEGRATION_MANAGER_HTML = ` `; function openIntegrationManager() { - cy.findByRole("tab", { name: "Room info" }).click(); + cy.findByRole("button", { name: "Room info" }).click(); cy.findByRole("button", { name: "Add widgets, bridges & bots" }).click(); }