Make landmark navigation work with new room list (#30747)

* Make landmark navigation work with new room list

Split out from https://github.com/element-hq/element-web/pull/30640

* Fix landmark selection to work with either room list

* Add test for landmark navigation

* Add test

* Fix test

* Clear mocks between runs
This commit is contained in:
David Baker
2025-09-12 10:24:56 +01:00
committed by GitHub
parent b6710d19c0
commit 34450d513a
3 changed files with 95 additions and 4 deletions

View File

@@ -9,6 +9,7 @@
import { TimelineRenderingType } from "../contexts/RoomContext"; import { TimelineRenderingType } from "../contexts/RoomContext";
import { Action } from "../dispatcher/actions"; import { Action } from "../dispatcher/actions";
import defaultDispatcher from "../dispatcher/dispatcher"; import defaultDispatcher from "../dispatcher/dispatcher";
import SettingsStore from "../settings/SettingsStore";
export const enum Landmark { export const enum Landmark {
// This is the space/home button in the left panel. // This is the space/home button in the left panel.
@@ -72,10 +73,16 @@ export class LandmarkNavigation {
const landmarkToDomElementMap: Record<Landmark, () => HTMLElement | null | undefined> = { const landmarkToDomElementMap: Record<Landmark, () => HTMLElement | null | undefined> = {
[Landmark.ACTIVE_SPACE_BUTTON]: () => document.querySelector<HTMLElement>(".mx_SpaceButton_active"), [Landmark.ACTIVE_SPACE_BUTTON]: () => document.querySelector<HTMLElement>(".mx_SpaceButton_active"),
[Landmark.ROOM_SEARCH]: () => document.querySelector<HTMLElement>(".mx_RoomSearch"), [Landmark.ROOM_SEARCH]: () =>
SettingsStore.getValue("feature_new_room_list")
? document.querySelector<HTMLElement>(".mx_RoomListSearch_search")
: document.querySelector<HTMLElement>(".mx_RoomSearch"),
[Landmark.ROOM_LIST]: () => [Landmark.ROOM_LIST]: () =>
document.querySelector<HTMLElement>(".mx_RoomTile_selected") || SettingsStore.getValue("feature_new_room_list")
document.querySelector<HTMLElement>(".mx_RoomTile"), ? document.querySelector<HTMLElement>(".mx_RoomListItemView_selected") ||
document.querySelector<HTMLElement>(".mx_RoomListItemView")
: document.querySelector<HTMLElement>(".mx_RoomTile_selected") ||
document.querySelector<HTMLElement>(".mx_RoomTile"),
[Landmark.MESSAGE_COMPOSER_OR_HOME]: () => { [Landmark.MESSAGE_COMPOSER_OR_HOME]: () => {
const isComposerOpen = !!document.querySelector(".mx_MessageComposer"); const isComposerOpen = !!document.querySelector(".mx_MessageComposer");

View File

@@ -5,7 +5,7 @@ SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only OR LicenseRef-Element-Com
Please see LICENSE files in the repository root for full details. Please see LICENSE files in the repository root for full details.
*/ */
import React from "react"; import React, { useState, useCallback } from "react";
import { shouldShowComponent } from "../../../../customisations/helpers/UIComponents"; import { shouldShowComponent } from "../../../../customisations/helpers/UIComponents";
import { UIComponent } from "../../../../settings/UIFeature"; import { UIComponent } from "../../../../settings/UIFeature";
@@ -14,6 +14,10 @@ import { RoomListHeaderView } from "./RoomListHeaderView";
import { RoomListView } from "./RoomListView"; import { RoomListView } from "./RoomListView";
import { Flex } from "../../../../shared-components/utils/Flex"; import { Flex } from "../../../../shared-components/utils/Flex";
import { _t } from "../../../../languageHandler"; import { _t } from "../../../../languageHandler";
import { getKeyBindingsManager } from "../../../../KeyBindingsManager";
import { KeyBindingAction } from "../../../../accessibility/KeyboardShortcuts";
import { Landmark, LandmarkNavigation } from "../../../../accessibility/LandmarkNavigation";
import { type IState as IRovingTabIndexState } from "../../../../accessibility/RovingTabIndex";
type RoomListPanelProps = { type RoomListPanelProps = {
/** /**
@@ -28,6 +32,31 @@ type RoomListPanelProps = {
*/ */
export const RoomListPanel: React.FC<RoomListPanelProps> = ({ activeSpace }) => { export const RoomListPanel: React.FC<RoomListPanelProps> = ({ activeSpace }) => {
const displayRoomSearch = shouldShowComponent(UIComponent.FilterContainer); const displayRoomSearch = shouldShowComponent(UIComponent.FilterContainer);
const [focusedElement, setFocusedElement] = useState<Element | null>(null);
const onFocus = useCallback((ev: React.FocusEvent): void => {
setFocusedElement(ev.target as Element);
}, []);
const onBlur = useCallback((): void => {
setFocusedElement(null);
}, []);
const onKeyDown = useCallback(
(ev: React.KeyboardEvent, state?: IRovingTabIndexState): void => {
if (!focusedElement) return;
const navAction = getKeyBindingsManager().getNavigationAction(ev);
if (navAction === KeyBindingAction.PreviousLandmark || navAction === KeyBindingAction.NextLandmark) {
ev.stopPropagation();
ev.preventDefault();
LandmarkNavigation.findAndFocusNextLandmark(
Landmark.ROOM_SEARCH,
navAction === KeyBindingAction.PreviousLandmark,
);
}
},
[focusedElement],
);
return ( return (
<Flex <Flex
@@ -36,6 +65,9 @@ export const RoomListPanel: React.FC<RoomListPanelProps> = ({ activeSpace }) =>
direction="column" direction="column"
align="stretch" align="stretch"
aria-label={_t("room_list|list_title")} aria-label={_t("room_list|list_title")}
onFocus={onFocus}
onBlur={onBlur}
onKeyDown={onKeyDown}
> >
{displayRoomSearch && <RoomListSearch activeSpace={activeSpace} />} {displayRoomSearch && <RoomListSearch activeSpace={activeSpace} />}
<RoomListHeaderView /> <RoomListHeaderView />

View File

@@ -8,21 +8,39 @@
import React from "react"; import React from "react";
import { render, screen } from "jest-matrix-react"; import { render, screen } from "jest-matrix-react";
import { mocked } from "jest-mock"; import { mocked } from "jest-mock";
import userEvent from "@testing-library/user-event";
import { RoomListPanel } from "../../../../../../src/components/views/rooms/RoomListPanel"; import { RoomListPanel } from "../../../../../../src/components/views/rooms/RoomListPanel";
import { shouldShowComponent } from "../../../../../../src/customisations/helpers/UIComponents"; import { shouldShowComponent } from "../../../../../../src/customisations/helpers/UIComponents";
import { MetaSpace } from "../../../../../../src/stores/spaces"; import { MetaSpace } from "../../../../../../src/stores/spaces";
import { LandmarkNavigation } from "../../../../../../src/accessibility/LandmarkNavigation";
import { ReleaseAnnouncementStore } from "../../../../../../src/stores/ReleaseAnnouncementStore";
jest.mock("../../../../../../src/customisations/helpers/UIComponents", () => ({ jest.mock("../../../../../../src/customisations/helpers/UIComponents", () => ({
shouldShowComponent: jest.fn(), shouldShowComponent: jest.fn(),
})); }));
jest.mock("../../../../../../src/accessibility/LandmarkNavigation", () => ({
LandmarkNavigation: {
findAndFocusNextLandmark: jest.fn(),
},
Landmark: {
ROOM_SEARCH: "something",
},
}));
// mock out release announcements as they interfere with what's focused
// (this can be removed once the new room list announcement is gone)
jest.spyOn(ReleaseAnnouncementStore.instance, "getReleaseAnnouncement").mockReturnValue(null);
describe("<RoomListPanel />", () => { describe("<RoomListPanel />", () => {
function renderComponent() { function renderComponent() {
return render(<RoomListPanel activeSpace={MetaSpace.Home} />); return render(<RoomListPanel activeSpace={MetaSpace.Home} />);
} }
beforeEach(() => { beforeEach(() => {
jest.clearAllMocks();
// By default, we consider shouldShowComponent(UIComponent.FilterContainer) should return true // By default, we consider shouldShowComponent(UIComponent.FilterContainer) should return true
mocked(shouldShowComponent).mockReturnValue(true); mocked(shouldShowComponent).mockReturnValue(true);
}); });
@@ -37,4 +55,38 @@ describe("<RoomListPanel />", () => {
renderComponent(); renderComponent();
expect(screen.queryByRole("button", { name: "Search Ctrl K" })).toBeNull(); expect(screen.queryByRole("button", { name: "Search Ctrl K" })).toBeNull();
}); });
it("should move to the next landmark when the shortcut key is pressed", async () => {
renderComponent();
const userEv = userEvent.setup();
// Pick something arbitrary and focusable in the room list component and focus it
const exploreRooms = screen.getByRole("button", { name: "Explore rooms" });
exploreRooms.focus();
expect(exploreRooms).toHaveFocus();
screen.getByRole("navigation", { name: "Room list" }).focus();
await userEv.keyboard("{Control>}{F6}{/Control}");
expect(LandmarkNavigation.findAndFocusNextLandmark).toHaveBeenCalled();
});
it("should not move to the next landmark if room list loses focus", async () => {
renderComponent();
const userEv = userEvent.setup();
// Pick something arbitrary and focusable in the room list component and focus it
const exploreRooms = screen.getByRole("button", { name: "Explore rooms" });
exploreRooms.focus();
expect(exploreRooms).toHaveFocus();
exploreRooms.blur();
expect(exploreRooms).not.toHaveFocus();
await userEv.keyboard("{Control>}{F6}{/Control}");
expect(LandmarkNavigation.findAndFocusNextLandmark).not.toHaveBeenCalled();
});
}); });