415 lines
13 KiB
TypeScript
415 lines
13 KiB
TypeScript
|
/*
|
||
|
Copyright 2024 New Vector Ltd.
|
||
|
Copyright 2020, 2021 The Matrix.org Foundation C.I.C.
|
||
|
|
||
|
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
|
||
|
Please see LICENSE files in the repository root for full details.
|
||
|
*/
|
||
|
|
||
|
import React, { HTMLAttributes } from "react";
|
||
|
import { render } from "jest-matrix-react";
|
||
|
import userEvent from "@testing-library/user-event";
|
||
|
|
||
|
import {
|
||
|
IState,
|
||
|
reducer,
|
||
|
RovingTabIndexProvider,
|
||
|
RovingTabIndexWrapper,
|
||
|
Type,
|
||
|
useRovingTabIndex,
|
||
|
} from "../../../src/accessibility/RovingTabIndex";
|
||
|
|
||
|
const Button = (props: HTMLAttributes<HTMLButtonElement>) => {
|
||
|
const [onFocus, isActive, ref] = useRovingTabIndex<HTMLButtonElement>();
|
||
|
return <button {...props} onFocus={onFocus} tabIndex={isActive ? 0 : -1} ref={ref} />;
|
||
|
};
|
||
|
|
||
|
const checkTabIndexes = (buttons: NodeListOf<HTMLElement>, expectations: number[]) => {
|
||
|
expect([...buttons].map((b) => b.tabIndex)).toStrictEqual(expectations);
|
||
|
};
|
||
|
|
||
|
// give the buttons keys for the fibre reconciler to not treat them all as the same
|
||
|
const button1 = <Button key={1}>a</Button>;
|
||
|
const button2 = <Button key={2}>b</Button>;
|
||
|
const button3 = <Button key={3}>c</Button>;
|
||
|
const button4 = <Button key={4}>d</Button>;
|
||
|
|
||
|
// mock offsetParent
|
||
|
Object.defineProperty(HTMLElement.prototype, "offsetParent", {
|
||
|
get() {
|
||
|
return this.parentNode;
|
||
|
},
|
||
|
});
|
||
|
|
||
|
describe("RovingTabIndex", () => {
|
||
|
it("RovingTabIndexProvider renders children as expected", () => {
|
||
|
const { container } = render(
|
||
|
<RovingTabIndexProvider>
|
||
|
{() => (
|
||
|
<div>
|
||
|
<span>Test</span>
|
||
|
</div>
|
||
|
)}
|
||
|
</RovingTabIndexProvider>,
|
||
|
);
|
||
|
expect(container.textContent).toBe("Test");
|
||
|
expect(container.innerHTML).toBe("<div><span>Test</span></div>");
|
||
|
});
|
||
|
|
||
|
it("RovingTabIndexProvider works as expected with useRovingTabIndex", () => {
|
||
|
const { container, rerender } = render(
|
||
|
<RovingTabIndexProvider>
|
||
|
{() => (
|
||
|
<React.Fragment>
|
||
|
{button1}
|
||
|
{button2}
|
||
|
{button3}
|
||
|
</React.Fragment>
|
||
|
)}
|
||
|
</RovingTabIndexProvider>,
|
||
|
);
|
||
|
|
||
|
// should begin with 0th being active
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [0, -1, -1]);
|
||
|
|
||
|
// focus on 2nd button and test it is the only active one
|
||
|
container.querySelectorAll("button")[2].focus();
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [-1, -1, 0]);
|
||
|
|
||
|
// focus on 1st button and test it is the only active one
|
||
|
container.querySelectorAll("button")[1].focus();
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [-1, 0, -1]);
|
||
|
|
||
|
// check that the active button does not change even on an explicit blur event
|
||
|
container.querySelectorAll("button")[1].blur();
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [-1, 0, -1]);
|
||
|
|
||
|
// update the children, it should remain on the same button
|
||
|
rerender(
|
||
|
<RovingTabIndexProvider>
|
||
|
{() => (
|
||
|
<React.Fragment>
|
||
|
{button1}
|
||
|
{button4}
|
||
|
{button2}
|
||
|
{button3}
|
||
|
</React.Fragment>
|
||
|
)}
|
||
|
</RovingTabIndexProvider>,
|
||
|
);
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [-1, -1, 0, -1]);
|
||
|
|
||
|
// update the children, remove the active button, it should move to the next one
|
||
|
rerender(
|
||
|
<RovingTabIndexProvider>
|
||
|
{() => (
|
||
|
<React.Fragment>
|
||
|
{button1}
|
||
|
{button4}
|
||
|
{button3}
|
||
|
</React.Fragment>
|
||
|
)}
|
||
|
</RovingTabIndexProvider>,
|
||
|
);
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [-1, -1, 0]);
|
||
|
});
|
||
|
|
||
|
it("RovingTabIndexProvider works as expected with RovingTabIndexWrapper", () => {
|
||
|
const { container } = render(
|
||
|
<RovingTabIndexProvider>
|
||
|
{() => (
|
||
|
<React.Fragment>
|
||
|
{button1}
|
||
|
{button2}
|
||
|
<RovingTabIndexWrapper>
|
||
|
{({ onFocus, isActive, ref }) => (
|
||
|
<button
|
||
|
onFocus={onFocus}
|
||
|
tabIndex={isActive ? 0 : -1}
|
||
|
ref={ref as React.RefObject<HTMLButtonElement>}
|
||
|
>
|
||
|
.
|
||
|
</button>
|
||
|
)}
|
||
|
</RovingTabIndexWrapper>
|
||
|
</React.Fragment>
|
||
|
)}
|
||
|
</RovingTabIndexProvider>,
|
||
|
);
|
||
|
|
||
|
// should begin with 0th being active
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [0, -1, -1]);
|
||
|
|
||
|
// focus on 2nd button and test it is the only active one
|
||
|
container.querySelectorAll("button")[2].focus();
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [-1, -1, 0]);
|
||
|
});
|
||
|
|
||
|
describe("reducer functions as expected", () => {
|
||
|
it("SetFocus works as expected", () => {
|
||
|
const ref1 = React.createRef<HTMLElement>();
|
||
|
const ref2 = React.createRef<HTMLElement>();
|
||
|
expect(
|
||
|
reducer(
|
||
|
{
|
||
|
activeRef: ref1,
|
||
|
refs: [ref1, ref2],
|
||
|
},
|
||
|
{
|
||
|
type: Type.SetFocus,
|
||
|
payload: {
|
||
|
ref: ref2,
|
||
|
},
|
||
|
},
|
||
|
),
|
||
|
).toStrictEqual({
|
||
|
activeRef: ref2,
|
||
|
refs: [ref1, ref2],
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it("Unregister works as expected", () => {
|
||
|
const ref1 = React.createRef<HTMLElement>();
|
||
|
const ref2 = React.createRef<HTMLElement>();
|
||
|
const ref3 = React.createRef<HTMLElement>();
|
||
|
const ref4 = React.createRef<HTMLElement>();
|
||
|
|
||
|
let state: IState = {
|
||
|
refs: [ref1, ref2, ref3, ref4],
|
||
|
};
|
||
|
|
||
|
state = reducer(state, {
|
||
|
type: Type.Unregister,
|
||
|
payload: {
|
||
|
ref: ref2,
|
||
|
},
|
||
|
});
|
||
|
expect(state).toStrictEqual({
|
||
|
refs: [ref1, ref3, ref4],
|
||
|
});
|
||
|
|
||
|
state = reducer(state, {
|
||
|
type: Type.Unregister,
|
||
|
payload: {
|
||
|
ref: ref3,
|
||
|
},
|
||
|
});
|
||
|
expect(state).toStrictEqual({
|
||
|
refs: [ref1, ref4],
|
||
|
});
|
||
|
|
||
|
state = reducer(state, {
|
||
|
type: Type.Unregister,
|
||
|
payload: {
|
||
|
ref: ref4,
|
||
|
},
|
||
|
});
|
||
|
expect(state).toStrictEqual({
|
||
|
refs: [ref1],
|
||
|
});
|
||
|
|
||
|
state = reducer(state, {
|
||
|
type: Type.Unregister,
|
||
|
payload: {
|
||
|
ref: ref1,
|
||
|
},
|
||
|
});
|
||
|
expect(state).toStrictEqual({
|
||
|
refs: [],
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it("Register works as expected", () => {
|
||
|
const ref1 = React.createRef<HTMLElement>();
|
||
|
const ref2 = React.createRef<HTMLElement>();
|
||
|
const ref3 = React.createRef<HTMLElement>();
|
||
|
const ref4 = React.createRef<HTMLElement>();
|
||
|
|
||
|
render(
|
||
|
<React.Fragment>
|
||
|
<span ref={ref1} />
|
||
|
<span ref={ref2} />
|
||
|
<span ref={ref3} />
|
||
|
<span ref={ref4} />
|
||
|
</React.Fragment>,
|
||
|
);
|
||
|
|
||
|
let state: IState = {
|
||
|
refs: [],
|
||
|
};
|
||
|
|
||
|
state = reducer(state, {
|
||
|
type: Type.Register,
|
||
|
payload: {
|
||
|
ref: ref1,
|
||
|
},
|
||
|
});
|
||
|
expect(state).toStrictEqual({
|
||
|
activeRef: ref1,
|
||
|
refs: [ref1],
|
||
|
});
|
||
|
|
||
|
state = reducer(state, {
|
||
|
type: Type.Register,
|
||
|
payload: {
|
||
|
ref: ref2,
|
||
|
},
|
||
|
});
|
||
|
expect(state).toStrictEqual({
|
||
|
activeRef: ref1,
|
||
|
refs: [ref1, ref2],
|
||
|
});
|
||
|
|
||
|
state = reducer(state, {
|
||
|
type: Type.Register,
|
||
|
payload: {
|
||
|
ref: ref3,
|
||
|
},
|
||
|
});
|
||
|
expect(state).toStrictEqual({
|
||
|
activeRef: ref1,
|
||
|
refs: [ref1, ref2, ref3],
|
||
|
});
|
||
|
|
||
|
state = reducer(state, {
|
||
|
type: Type.Register,
|
||
|
payload: {
|
||
|
ref: ref4,
|
||
|
},
|
||
|
});
|
||
|
expect(state).toStrictEqual({
|
||
|
activeRef: ref1,
|
||
|
refs: [ref1, ref2, ref3, ref4],
|
||
|
});
|
||
|
|
||
|
// test that the automatic focus switch works for unmounting
|
||
|
state = reducer(state, {
|
||
|
type: Type.SetFocus,
|
||
|
payload: {
|
||
|
ref: ref2,
|
||
|
},
|
||
|
});
|
||
|
expect(state).toStrictEqual({
|
||
|
activeRef: ref2,
|
||
|
refs: [ref1, ref2, ref3, ref4],
|
||
|
});
|
||
|
|
||
|
state = reducer(state, {
|
||
|
type: Type.Unregister,
|
||
|
payload: {
|
||
|
ref: ref2,
|
||
|
},
|
||
|
});
|
||
|
expect(state).toStrictEqual({
|
||
|
activeRef: ref3,
|
||
|
refs: [ref1, ref3, ref4],
|
||
|
});
|
||
|
|
||
|
// test that the insert into the middle works as expected
|
||
|
state = reducer(state, {
|
||
|
type: Type.Register,
|
||
|
payload: {
|
||
|
ref: ref2,
|
||
|
},
|
||
|
});
|
||
|
expect(state).toStrictEqual({
|
||
|
activeRef: ref3,
|
||
|
refs: [ref1, ref2, ref3, ref4],
|
||
|
});
|
||
|
|
||
|
// test that insertion at the edges works
|
||
|
state = reducer(state, {
|
||
|
type: Type.Unregister,
|
||
|
payload: {
|
||
|
ref: ref1,
|
||
|
},
|
||
|
});
|
||
|
state = reducer(state, {
|
||
|
type: Type.Unregister,
|
||
|
payload: {
|
||
|
ref: ref4,
|
||
|
},
|
||
|
});
|
||
|
expect(state).toStrictEqual({
|
||
|
activeRef: ref3,
|
||
|
refs: [ref2, ref3],
|
||
|
});
|
||
|
|
||
|
state = reducer(state, {
|
||
|
type: Type.Register,
|
||
|
payload: {
|
||
|
ref: ref1,
|
||
|
},
|
||
|
});
|
||
|
|
||
|
state = reducer(state, {
|
||
|
type: Type.Register,
|
||
|
payload: {
|
||
|
ref: ref4,
|
||
|
},
|
||
|
});
|
||
|
expect(state).toStrictEqual({
|
||
|
activeRef: ref3,
|
||
|
refs: [ref1, ref2, ref3, ref4],
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe("handles arrow keys", () => {
|
||
|
it("should handle up/down arrow keys work when handleUpDown=true", async () => {
|
||
|
const { container } = render(
|
||
|
<RovingTabIndexProvider handleUpDown>
|
||
|
{({ onKeyDownHandler }) => (
|
||
|
<div onKeyDown={onKeyDownHandler}>
|
||
|
{button1}
|
||
|
{button2}
|
||
|
{button3}
|
||
|
</div>
|
||
|
)}
|
||
|
</RovingTabIndexProvider>,
|
||
|
);
|
||
|
|
||
|
container.querySelectorAll("button")[0].focus();
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [0, -1, -1]);
|
||
|
|
||
|
await userEvent.keyboard("[ArrowDown]");
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [-1, 0, -1]);
|
||
|
|
||
|
await userEvent.keyboard("[ArrowDown]");
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [-1, -1, 0]);
|
||
|
|
||
|
await userEvent.keyboard("[ArrowUp]");
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [-1, 0, -1]);
|
||
|
|
||
|
await userEvent.keyboard("[ArrowUp]");
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [0, -1, -1]);
|
||
|
|
||
|
// Does not loop without
|
||
|
await userEvent.keyboard("[ArrowUp]");
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [0, -1, -1]);
|
||
|
});
|
||
|
|
||
|
it("should call scrollIntoView if specified", async () => {
|
||
|
const { container } = render(
|
||
|
<RovingTabIndexProvider handleUpDown scrollIntoView>
|
||
|
{({ onKeyDownHandler }) => (
|
||
|
<div onKeyDown={onKeyDownHandler}>
|
||
|
{button1}
|
||
|
{button2}
|
||
|
{button3}
|
||
|
</div>
|
||
|
)}
|
||
|
</RovingTabIndexProvider>,
|
||
|
);
|
||
|
|
||
|
container.querySelectorAll("button")[0].focus();
|
||
|
checkTabIndexes(container.querySelectorAll("button"), [0, -1, -1]);
|
||
|
|
||
|
const button = container.querySelectorAll("button")[1];
|
||
|
const mock = jest.spyOn(button, "scrollIntoView");
|
||
|
await userEvent.keyboard("[ArrowDown]");
|
||
|
expect(mock).toHaveBeenCalled();
|
||
|
});
|
||
|
});
|
||
|
});
|