* Use firstVisible category for roving tab index * Adding category keyboard navigation tests * Reduce repetition in categories definition and add tests * Remove ternary operators * Simplify
495 lines
19 KiB
TypeScript
495 lines
19 KiB
TypeScript
/*
|
|
Copyright 2024 New Vector Ltd.
|
|
Copyright 2023 The Matrix.org Foundation C.I.C.
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only OR LicenseRef-Element-Commercial
|
|
Please see LICENSE files in the repository root for full details.
|
|
*/
|
|
|
|
import React, { createRef } from "react";
|
|
import { render, waitFor, act } from "jest-matrix-react";
|
|
import userEvent from "@testing-library/user-event";
|
|
|
|
import EmojiPicker from "../../../../../src/components/views/emojipicker/EmojiPicker";
|
|
import { stubClient } from "../../../../test-utils";
|
|
import SettingsStore from "../../../../../src/settings/SettingsStore";
|
|
|
|
describe("EmojiPicker", function () {
|
|
stubClient();
|
|
|
|
// Helper to get the currently active emoji's text content from the grid
|
|
const getActiveEmojiText = (container: HTMLElement): string =>
|
|
container.querySelector('.mx_EmojiPicker_body .mx_EmojiPicker_item_wrapper [tabindex="0"]')?.textContent || "";
|
|
|
|
beforeEach(() => {
|
|
// Clear recent emojis to prevent test pollution
|
|
jest.spyOn(SettingsStore, "getValue").mockImplementation((settingName) => {
|
|
if (settingName === "recent_emoji") return [] as any;
|
|
return jest.requireActual("../../../../../src/settings/SettingsStore").default.getValue(settingName);
|
|
});
|
|
});
|
|
|
|
afterEach(() => {
|
|
jest.restoreAllMocks();
|
|
});
|
|
|
|
it("should initialize categories with correct state when no recent emojis", () => {
|
|
const ref = createRef<EmojiPicker>();
|
|
render(<EmojiPicker ref={ref} onChoose={(str: string) => false} onFinished={jest.fn()} />);
|
|
|
|
//@ts-ignore private access
|
|
const categories = ref.current!.categories;
|
|
|
|
// Verify we have all expected categories
|
|
expect(categories).toHaveLength(9);
|
|
expect(categories.map((c) => c.id)).toEqual([
|
|
"recent",
|
|
"people",
|
|
"nature",
|
|
"foods",
|
|
"activity",
|
|
"places",
|
|
"objects",
|
|
"symbols",
|
|
"flags",
|
|
]);
|
|
|
|
// Recent category should be disabled when empty
|
|
const recentCategory = categories.find((c) => c.id === "recent");
|
|
expect(recentCategory).toMatchObject({
|
|
id: "recent",
|
|
enabled: false,
|
|
visible: false,
|
|
firstVisible: false,
|
|
});
|
|
|
|
// People category should be the first visible when no recent emojis
|
|
const peopleCategory = categories.find((c) => c.id === "people");
|
|
expect(peopleCategory).toMatchObject({
|
|
id: "people",
|
|
enabled: true,
|
|
visible: true,
|
|
firstVisible: true,
|
|
});
|
|
|
|
// Other categories should start as not visible and not firstVisible
|
|
const natureCategory = categories.find((c) => c.id === "nature");
|
|
expect(natureCategory).toMatchObject({
|
|
id: "nature",
|
|
enabled: true,
|
|
visible: false,
|
|
firstVisible: false,
|
|
});
|
|
|
|
const flagsCategory = categories.find((c) => c.id === "flags");
|
|
expect(flagsCategory).toMatchObject({
|
|
id: "flags",
|
|
enabled: true,
|
|
visible: false,
|
|
firstVisible: false,
|
|
});
|
|
|
|
// All categories should have refs and names
|
|
categories.forEach((cat) => {
|
|
expect(cat.ref).toBeTruthy();
|
|
expect(cat.name).toBeTruthy();
|
|
});
|
|
});
|
|
|
|
it("should initialize categories with recent as firstVisible when recent emojis exist", () => {
|
|
// Mock recent emojis
|
|
jest.spyOn(SettingsStore, "getValue").mockImplementation((settingName) => {
|
|
if (settingName === "recent_emoji") return ["😀", "🎉", "❤️"] as any;
|
|
return jest.requireActual("../../../../../src/settings/SettingsStore").default.getValue(settingName);
|
|
});
|
|
|
|
const ref = createRef<EmojiPicker>();
|
|
render(<EmojiPicker ref={ref} onChoose={(str: string) => false} onFinished={jest.fn()} />);
|
|
|
|
//@ts-ignore private access
|
|
const categories = ref.current!.categories;
|
|
|
|
// Recent category should be enabled and firstVisible
|
|
const recentCategory = categories.find((c) => c.id === "recent");
|
|
expect(recentCategory).toMatchObject({
|
|
id: "recent",
|
|
enabled: true,
|
|
visible: true,
|
|
firstVisible: true,
|
|
});
|
|
|
|
// People category should be visible but NOT firstVisible when recent exists
|
|
const peopleCategory = categories.find((c) => c.id === "people");
|
|
expect(peopleCategory).toMatchObject({
|
|
id: "people",
|
|
enabled: true,
|
|
visible: true,
|
|
firstVisible: false,
|
|
});
|
|
});
|
|
|
|
it("should not mangle default order after filtering", async () => {
|
|
const ref = createRef<EmojiPicker>();
|
|
const { container } = render(
|
|
<EmojiPicker ref={ref} onChoose={(str: string) => false} onFinished={jest.fn()} />,
|
|
);
|
|
|
|
// Record the HTML before filtering
|
|
const beforeHtml = container.innerHTML;
|
|
|
|
// Apply a filter and assert that the HTML has changed
|
|
//@ts-ignore private access
|
|
act(() => ref.current!.onChangeFilter("test"));
|
|
expect(beforeHtml).not.toEqual(container.innerHTML);
|
|
|
|
// Clear the filter and assert that the HTML matches what it was before filtering
|
|
//@ts-ignore private access
|
|
act(() => ref.current!.onChangeFilter(""));
|
|
await waitFor(() => expect(beforeHtml).toEqual(container.innerHTML));
|
|
});
|
|
|
|
it("sort emojis by shortcode and size", function () {
|
|
const ep = new EmojiPicker({ onChoose: (str: string) => false, onFinished: jest.fn() });
|
|
|
|
//@ts-ignore private access
|
|
act(() => ep.onChangeFilter("heart"));
|
|
|
|
//@ts-ignore private access
|
|
expect(ep.memoizedDataByCategory["people"][0].shortcodes[0]).toEqual("heart");
|
|
//@ts-ignore private access
|
|
expect(ep.memoizedDataByCategory["people"][1].shortcodes[0]).toEqual("heartbeat");
|
|
});
|
|
|
|
it("should allow keyboard navigation using arrow keys", async () => {
|
|
// mock offsetParent
|
|
Object.defineProperty(HTMLElement.prototype, "offsetParent", {
|
|
get() {
|
|
return this.parentNode;
|
|
},
|
|
});
|
|
|
|
const onChoose = jest.fn();
|
|
const onFinished = jest.fn();
|
|
const { container } = render(<EmojiPicker onChoose={onChoose} onFinished={onFinished} />);
|
|
|
|
const input = container.querySelector("input")!;
|
|
expect(input).toHaveFocus();
|
|
|
|
function getEmoji(): string {
|
|
return getActiveEmojiText(container);
|
|
}
|
|
|
|
expect(getEmoji()).toEqual("😀");
|
|
// First arrow key press shows highlight without navigating
|
|
await userEvent.keyboard("[ArrowDown]");
|
|
expect(getEmoji()).toEqual("😀");
|
|
// Subsequent arrow keys navigate
|
|
await userEvent.keyboard("[ArrowDown]");
|
|
expect(getEmoji()).toEqual("🙂");
|
|
await userEvent.keyboard("[ArrowUp]");
|
|
expect(getEmoji()).toEqual("😀");
|
|
await userEvent.keyboard("Flag");
|
|
await userEvent.keyboard("[ArrowRight]");
|
|
await userEvent.keyboard("[ArrowRight]");
|
|
expect(getEmoji()).toEqual("📫️");
|
|
await userEvent.keyboard("[ArrowDown]");
|
|
expect(getEmoji()).toEqual("🇦🇨");
|
|
await userEvent.keyboard("[ArrowLeft]");
|
|
expect(getEmoji()).toEqual("📭️");
|
|
await userEvent.keyboard("[ArrowUp]");
|
|
expect(getEmoji()).toEqual("⛳️");
|
|
await userEvent.keyboard("[ArrowRight]");
|
|
expect(getEmoji()).toEqual("📫️");
|
|
await userEvent.keyboard("[Enter]");
|
|
|
|
expect(onChoose).toHaveBeenCalledWith("📫️");
|
|
expect(onFinished).toHaveBeenCalled();
|
|
});
|
|
|
|
it("should move actual focus when navigating between emojis after Tab", async () => {
|
|
// mock offsetParent
|
|
Object.defineProperty(HTMLElement.prototype, "offsetParent", {
|
|
get() {
|
|
return this.parentNode;
|
|
},
|
|
});
|
|
|
|
const onChoose = jest.fn();
|
|
const onFinished = jest.fn();
|
|
const { container } = render(<EmojiPicker onChoose={onChoose} onFinished={onFinished} />);
|
|
|
|
const input = container.querySelector("input")!;
|
|
expect(input).toHaveFocus();
|
|
|
|
// Wait for emojis to render
|
|
await waitFor(() => {
|
|
expect(container.querySelector('[role="gridcell"]')).toBeInTheDocument();
|
|
});
|
|
|
|
function getEmoji(): string {
|
|
return document.activeElement?.textContent || "";
|
|
}
|
|
|
|
function getVirtuallyFocusedEmoji(): string {
|
|
return getActiveEmojiText(container);
|
|
}
|
|
|
|
// Initially, arrow keys use virtual focus (aria-activedescendant)
|
|
// The first emoji is virtually focused by default
|
|
expect(input).toHaveFocus();
|
|
expect(getVirtuallyFocusedEmoji()).toEqual("😀");
|
|
expect(getEmoji()).toEqual(""); // No actual emoji has focus
|
|
|
|
// First arrow key press shows highlight without navigating
|
|
await userEvent.keyboard("[ArrowDown]");
|
|
expect(input).toHaveFocus(); // Input still has focus
|
|
expect(getVirtuallyFocusedEmoji()).toEqual("😀"); // Virtual focus stayed on first emoji
|
|
expect(getEmoji()).toEqual(""); // No actual emoji has focus
|
|
|
|
// Second arrow key press navigates
|
|
await userEvent.keyboard("[ArrowDown]");
|
|
expect(input).toHaveFocus(); // Input still has focus
|
|
expect(getVirtuallyFocusedEmoji()).toEqual("🙂"); // Virtual focus moved
|
|
expect(getEmoji()).toEqual(""); // No actual emoji has focus
|
|
|
|
// Tab to move actual focus to the emoji
|
|
await userEvent.keyboard("[Tab]");
|
|
expect(input).not.toHaveFocus();
|
|
expect(getEmoji()).toEqual("🙂"); // Now emoji has actual focus
|
|
|
|
// Arrow keys now move actual DOM focus between emojis
|
|
await userEvent.keyboard("[ArrowDown]");
|
|
expect(getEmoji()).toEqual("🤩"); // Actual focus moved down one row
|
|
expect(input).not.toHaveFocus();
|
|
|
|
await userEvent.keyboard("[ArrowUp]");
|
|
expect(getEmoji()).toEqual("🙂"); // Actual focus moved back up
|
|
expect(input).not.toHaveFocus();
|
|
|
|
await userEvent.keyboard("[ArrowRight]");
|
|
expect(getEmoji()).toEqual("🙃"); // Actual focus moved right
|
|
expect(input).not.toHaveFocus();
|
|
});
|
|
|
|
it("should not select emoji on Enter press before highlight is shown", async () => {
|
|
// mock offsetParent
|
|
Object.defineProperty(HTMLElement.prototype, "offsetParent", {
|
|
get() {
|
|
return this.parentNode;
|
|
},
|
|
});
|
|
|
|
const onChoose = jest.fn();
|
|
const onFinished = jest.fn();
|
|
const { container } = render(<EmojiPicker onChoose={onChoose} onFinished={onFinished} />);
|
|
|
|
const input = container.querySelector("input")!;
|
|
expect(input).toHaveFocus();
|
|
|
|
// Wait for emojis to render
|
|
await waitFor(() => {
|
|
expect(container.querySelector('[role="gridcell"]')).toBeInTheDocument();
|
|
});
|
|
|
|
// Press Enter immediately without interacting with arrow keys or search
|
|
await userEvent.keyboard("[Enter]");
|
|
|
|
// onChoose and onFinished should not be called
|
|
expect(onChoose).not.toHaveBeenCalled();
|
|
expect(onFinished).not.toHaveBeenCalled();
|
|
|
|
// Now press arrow key to show highlight
|
|
await userEvent.keyboard("[ArrowDown]");
|
|
|
|
// Press Enter again - now it should work
|
|
await userEvent.keyboard("[Enter]");
|
|
|
|
// onChoose and onFinished should be called
|
|
expect(onChoose).toHaveBeenCalledWith("😀");
|
|
expect(onFinished).toHaveBeenCalled();
|
|
});
|
|
|
|
it("should reset to first emoji when filter is cleared after navigation", async () => {
|
|
// mock offsetParent
|
|
Object.defineProperty(HTMLElement.prototype, "offsetParent", {
|
|
get() {
|
|
return this.parentNode;
|
|
},
|
|
});
|
|
|
|
const onChoose = jest.fn();
|
|
const onFinished = jest.fn();
|
|
const { container } = render(<EmojiPicker onChoose={onChoose} onFinished={onFinished} />);
|
|
|
|
const input = container.querySelector("input")!;
|
|
expect(input).toHaveFocus();
|
|
|
|
function getEmoji(): string {
|
|
return getActiveEmojiText(container);
|
|
}
|
|
|
|
// Initially on first emoji
|
|
expect(getEmoji()).toEqual("😀");
|
|
|
|
// Show highlight with first arrow press
|
|
await userEvent.keyboard("[ArrowDown]");
|
|
expect(getEmoji()).toEqual("😀");
|
|
|
|
// Navigate to a different emoji
|
|
await userEvent.keyboard("[ArrowDown]");
|
|
expect(getEmoji()).toEqual("🙂");
|
|
await userEvent.keyboard("[ArrowDown]");
|
|
expect(getEmoji()).toEqual("🤩");
|
|
|
|
// Type a search query to filter emojis (this sets showHighlight=true)
|
|
await userEvent.type(input, "think");
|
|
await waitFor(() => {
|
|
// After filtering, we should be on the "thinking" emoji
|
|
expect(getEmoji()).toEqual("🤔");
|
|
});
|
|
|
|
// Clear the search filter
|
|
await userEvent.clear(input);
|
|
|
|
// After clearing, showHighlight is false, so the highlight is hidden
|
|
// The activeNode might still be on 🤔, but we can't see it
|
|
|
|
// Press arrow key - this should reset to first emoji AND show highlight
|
|
await userEvent.keyboard("[ArrowDown]");
|
|
await waitFor(() => {
|
|
expect(getEmoji()).toEqual("😀"); // Should now be on first emoji with highlight shown
|
|
});
|
|
|
|
// Next arrow key should navigate from first emoji
|
|
await userEvent.keyboard("[ArrowDown]");
|
|
expect(getEmoji()).toEqual("🙂");
|
|
});
|
|
|
|
describe("Category keyboard selection", () => {
|
|
beforeEach(() => {
|
|
// mock offsetParent
|
|
Object.defineProperty(HTMLElement.prototype, "offsetParent", {
|
|
get() {
|
|
return this.parentNode;
|
|
},
|
|
});
|
|
});
|
|
|
|
it("check tabindex for the first category when no recent emojis", async () => {
|
|
const { container } = render(<EmojiPicker onChoose={jest.fn()} onFinished={jest.fn()} />);
|
|
|
|
await waitFor(() => {
|
|
expect(container.querySelector('[data-category-id="people"]')).toBeInTheDocument();
|
|
});
|
|
|
|
// People category should have tabindex="0"
|
|
const peopleTab = container.querySelector('[title*="Smileys"]');
|
|
expect(peopleTab).toHaveAttribute("tabindex", "0");
|
|
expect(peopleTab).toHaveAttribute("aria-selected", "true");
|
|
|
|
// Other categories should have tabindex="-1"
|
|
const natureTab = container.querySelector('[title*="Animals"]');
|
|
expect(natureTab).toHaveAttribute("tabindex", "-1");
|
|
});
|
|
|
|
it("check tabindex for recent category when recent emojis exist", async () => {
|
|
// Mock recent emojis
|
|
jest.spyOn(SettingsStore, "getValue").mockImplementation((settingName) => {
|
|
if (settingName === "recent_emoji") return ["😀", "🎉"] as any;
|
|
return jest.requireActual("../../../../../src/settings/SettingsStore").default.getValue(settingName);
|
|
});
|
|
|
|
const { container } = render(<EmojiPicker onChoose={jest.fn()} onFinished={jest.fn()} />);
|
|
|
|
await waitFor(() => {
|
|
expect(container.querySelector('[data-category-id="recent"]')).toBeInTheDocument();
|
|
});
|
|
|
|
// Recent category should have tabindex="0"
|
|
const recentTab = container.querySelector('[title*="Frequently"]');
|
|
expect(recentTab).toHaveAttribute("tabindex", "0");
|
|
expect(recentTab).toHaveAttribute("aria-selected", "true");
|
|
|
|
// People category should have tabindex="-1"
|
|
const peopleTab = container.querySelector('[title*="Smileys"]');
|
|
expect(peopleTab).toHaveAttribute("tabindex", "-1");
|
|
});
|
|
|
|
it("should update table position when clicking on a different category tab", async () => {
|
|
const { container } = render(<EmojiPicker onChoose={jest.fn()} onFinished={jest.fn()} />);
|
|
|
|
await waitFor(() => {
|
|
expect(container.querySelector('[data-category-id="people"]')).toBeInTheDocument();
|
|
});
|
|
|
|
// Initially, people category should be visible
|
|
const peopleTab = container.querySelector('[title*="Smileys"]') as HTMLButtonElement;
|
|
expect(peopleTab).toHaveAttribute("tabindex", "0");
|
|
|
|
// Click on nature category tab
|
|
const natureTab = container.querySelector('[title*="Animals"]') as HTMLButtonElement;
|
|
await userEvent.click(natureTab);
|
|
|
|
// Wait for scroll and visibility update
|
|
await waitFor(() => {
|
|
const natureCategory = container.querySelector('[data-category-id="nature"]');
|
|
expect(natureCategory).toBeInTheDocument();
|
|
});
|
|
});
|
|
|
|
it("should navigate between category tabs using arrow keys", async () => {
|
|
const { container } = render(<EmojiPicker onChoose={jest.fn()} onFinished={jest.fn()} />);
|
|
|
|
await waitFor(() => {
|
|
expect(container.querySelector('[data-category-id="people"]')).toBeInTheDocument();
|
|
});
|
|
|
|
// Focus on the category header
|
|
const peopleTab = container.querySelector('[title*="Smileys"]') as HTMLButtonElement;
|
|
peopleTab.focus();
|
|
expect(peopleTab).toHaveFocus();
|
|
|
|
// Press ArrowRight to move to next category
|
|
await userEvent.keyboard("[ArrowRight]");
|
|
|
|
// Should focus on next enabled category and trigger scroll
|
|
await waitFor(() => {
|
|
// Verify focus moved away from people tab
|
|
expect(peopleTab).not.toHaveFocus();
|
|
|
|
// Verify some other category tab now has focus
|
|
const focusedTab = document.activeElement;
|
|
expect(focusedTab?.getAttribute("role")).toBe("tab");
|
|
expect(focusedTab).not.toBe(peopleTab);
|
|
});
|
|
});
|
|
|
|
it("should navigate to first/last category using Home/End keys", async () => {
|
|
const { container } = render(<EmojiPicker onChoose={jest.fn()} onFinished={jest.fn()} />);
|
|
|
|
await waitFor(() => {
|
|
expect(container.querySelector('[data-category-id="people"]')).toBeInTheDocument();
|
|
});
|
|
|
|
// Focus on the category header
|
|
const peopleTab = container.querySelector('[title*="Smileys"]') as HTMLButtonElement;
|
|
peopleTab.focus();
|
|
|
|
// Press End to jump to last category
|
|
await userEvent.keyboard("[End]");
|
|
|
|
await waitFor(() => {
|
|
const flagsTab = container.querySelector('[title*="Flags"]') as HTMLButtonElement;
|
|
expect(flagsTab).toHaveFocus();
|
|
});
|
|
|
|
// Press Home to jump to first category
|
|
await userEvent.keyboard("[Home]");
|
|
|
|
await waitFor(() => {
|
|
expect(peopleTab).toHaveFocus();
|
|
});
|
|
});
|
|
});
|
|
});
|