Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

WB-1780: Refactor combobox to use Announcer [WIP] #2390

Draft
wants to merge 60 commits into
base: announcer-pt1
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
60 commits
Select commit Hold shift + click to select a range
42a2d7b
Initial commit of Announcer
marcysutton Nov 13, 2024
5c67d98
WIP: append messages
marcysutton Nov 15, 2024
3581692
Leverage React for tests
marcysutton Nov 15, 2024
fc86083
Refactor to use dictionary
marcysutton Nov 15, 2024
23f00ef
Cleanup, move types, add comments
marcysutton Nov 15, 2024
859f5e3
Fix outdated param in story
marcysutton Nov 15, 2024
2f3e6b4
Put testing styles in Storybook preview.css
marcysutton Nov 19, 2024
53c9518
Remove console.log
marcysutton Nov 19, 2024
c119abe
Add working test for auto-removal of messages
marcysutton Nov 19, 2024
2428b60
Added changeset
marcysutton Nov 20, 2024
2557a7e
Remove manually created changelog file
marcysutton Nov 22, 2024
3fbf47e
Restructure files based on PR feedback
marcysutton Nov 22, 2024
d6248c4
Rename sendMessage to announceMessage
marcysutton Nov 22, 2024
a01d18d
Rename clear-messages file to match function
marcysutton Nov 22, 2024
1f42473
Move utility functions into separate files
marcysutton Nov 22, 2024
6e1752d
Append regions to end of document.body
marcysutton Nov 22, 2024
8775ba2
Renaming timeouts, adding comments for clarity
marcysutton Nov 22, 2024
5d6e5ca
Reformat files for linter
marcysutton Nov 22, 2024
92beca9
Try kicking the linter one more time
marcysutton Nov 22, 2024
f06ef66
Expand tests
marcysutton Nov 23, 2024
2611248
Make document check more consistent
marcysutton Nov 23, 2024
30edef8
Add comments, types, and a few more tests
marcysutton Nov 23, 2024
48bcf35
Implement debounce / async logic
marcysutton Nov 26, 2024
c36d79a
Implement debounce / async logic
marcysutton Nov 26, 2024
e978123
Get async tests working
marcysutton Nov 26, 2024
147a364
Add missing test utility file
marcysutton Nov 26, 2024
32774c3
Update docs in Storybook for latest API changes
marcysutton Nov 26, 2024
e526e3b
Firm up debounce logic
marcysutton Nov 27, 2024
260b6b1
Clean up stray log and setTimeout testing approach
marcysutton Nov 27, 2024
c95f6f3
Add test file I somehow missed
marcysutton Nov 27, 2024
1ca62d7
Suppress story artifacts from announcements
marcysutton Dec 3, 2024
1dd4106
Add initial timeout back to help Safari/VO
marcysutton Dec 3, 2024
a334d5c
Fix typo in reattachment selector
marcysutton Dec 10, 2024
c7b2162
Rename Announcer filenames to lowercase
marcysutton Dec 10, 2024
698665e
Remove console.log
marcysutton Dec 10, 2024
3bc42da
Remove commented-out test code
marcysutton Dec 10, 2024
36e614f
Update tests from review feedback
marcysutton Dec 10, 2024
06ccdcb
Refactor combobox to use Announcer
jandrade Dec 11, 2024
0cb27cf
Clean up WIP wonder-blocks-style code
marcysutton Dec 12, 2024
cbb4c2d
Refactor debounce logic and tests
marcysutton Dec 12, 2024
7584543
Clean up storybook styling with custom body class
marcysutton Dec 12, 2024
ebe284c
Update jsdoc comments for debounce utility
marcysutton Dec 12, 2024
4957581
Fix incorrect object in debounce test
marcysutton Dec 12, 2024
7492c69
Merge conflicts
jandrade Dec 12, 2024
b5fb657
Merge announcer-pt1 into announcer-combobox
jandrade Dec 12, 2024
b6009b7
[wb1812.1.deprecate] Mark ID stuff as deprecated (#2388)
somewhatabstract Dec 16, 2024
897686b
[wb1812.2.idcomponent] Add the Id component (#2389)
somewhatabstract Dec 16, 2024
56d961f
[wb1812.3.migratewb] Migrate Wonder Blocks off old id providers (#2391)
somewhatabstract Dec 16, 2024
8237972
Version Packages (#2396)
khan-actions-bot Dec 16, 2024
3c1682f
QoL: Turn on automatic commit of changesets (#2394)
jeremywiebe Dec 16, 2024
d23c9c5
[wb1812.4.delete] Delete the custom identifier generation API (#2398)
somewhatabstract Dec 17, 2024
193d8a4
RELEASING: Releasing 27 package(s) (#2402)
khan-actions-bot Dec 17, 2024
e095558
Empty commit to trigger release action
somewhatabstract Dec 17, 2024
361cb52
RadioGroup: Add flexible width to legend element to fill available sp…
jandrade Dec 19, 2024
2cfb36f
Version Packages (#2406)
khan-actions-bot Dec 19, 2024
b2f03d9
Tooling: Move Storybook reusable components to __docs__ to optimize T…
jandrade Dec 20, 2024
1aa4fef
Update changesets' auto-commit feature to not suppress CI on commits …
jeremywiebe Dec 20, 2024
53b4197
Changesets: Bump form package to test release process (#2410)
jandrade Dec 20, 2024
c58e3fa
Version Packages (#2411)
khan-actions-bot Dec 20, 2024
7fc7a6e
Merge announcer-pt1 into announcer-combobox
jandrade Dec 20, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Implement debounce / async logic
1. Keeps announce from being called too frequently. We can play with the timeout duration.
2. Makes the returned IDREF more reliable in a browser.
  • Loading branch information
marcysutton committed Nov 26, 2024
commit 48bcf351169bd1aeba98728e4392a92e653eb7a5
11 changes: 8 additions & 3 deletions __docs__/wonder-blocks-announcer/announcer.stories.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -19,9 +19,14 @@ const AnnouncerExample = ({
}: AnnounceMessageProps) => {
return (
<Button
onClick={() => {
// TODO: explore making method async for consistent return string
announceMessage({message, level, removalDelay});
onClick={async () => {
const idRef = await announceMessage({
message,
level,
removalDelay,
});
/* eslint-disable-next-line */
console.log(idRef);
}}
>
Save
Expand Down
78 changes: 56 additions & 22 deletions packages/wonder-blocks-announcer/src/Announcer.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ import {
import {alternateIndex} from "./util/util";

const REMOVAL_TIMEOUT_DELAY = 5000;
const WAIT_THRESHOLD = 250;

/**
* Internal class to manage screen reader announcements.
Expand Down Expand Up @@ -124,35 +125,44 @@ class Announcer {
* @param {number} removalDelay How long to wait before removing message
* @returns {string} IDREF for targeted element or empty string if it failed
*/
announce(
async announce(
message: string,
level: PolitenessLevel,
removalDelay?: number,
): string {
if (!this.node) {
this.reattachNodes();
}
): Promise<string> {
const announceCB = (
message: string,
level: PolitenessLevel,
removalDelay?: number,
) => {
if (!this.node) {
this.reattachNodes();
}
// Filter region elements to the selected level
const regions: RegionDef[] = [...this.dictionary.values()].filter(
(entry: RegionDef) => entry.level === level,
);

// Filter region elements to the selected level
const regions: RegionDef[] = [...this.dictionary.values()].filter(
(entry: RegionDef) => entry.level === level,
);
const newIndex = this.appendMessage(
message,
level,
regions,
removalDelay,
);

const newIndex = this.appendMessage(
message,
level,
regions,
removalDelay,
);
// overwrite central index for the given level
if (level === "assertive") {
this.regionFactory.aIndex = newIndex;
} else {
this.regionFactory.pIndex = newIndex;
}

// overwrite central index for the given level
if (level === "assertive") {
this.regionFactory.aIndex = newIndex;
} else {
this.regionFactory.pIndex = newIndex;
}
return regions[newIndex].id || "";
};

return regions[newIndex].id || "";
const safeAnnounce = this.debounce(announceCB, WAIT_THRESHOLD);
const result = await safeAnnounce({message, level, removalDelay});
return result;
}

/**
Expand Down Expand Up @@ -212,6 +222,30 @@ class Announcer {
return index;
}

/**
* Keep announcements from happening too often.
* Anytime the announcer is called repeatedly, this will slow down the results.
* @param {Function} callback Announce method to call with argments
* @param {number} wait Length of time to wait before calling callback again
* @returns {string} idRef of targeted live region element
*/
debounce(callback: (...args: any[]) => string, wait: number) {
let timeoutId: ReturnType<typeof setTimeout> | null = null;

return (...args: any[]): Promise<string> => {
return new Promise((resolve) => {
if (timeoutId !== null) {
clearTimeout(timeoutId);
}

timeoutId = setTimeout(() => {
const result = callback(...args);
resolve(result);
}, wait);
});
};
}

/**
* Reset state to defaults.
* Useful for testing.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -86,13 +86,13 @@ describe("Announcer class", () => {
});

describe("clearing messages", () => {
test("clearing by IDREF", () => {
test("clearing by IDREF", async () => {
// Arrange
const announcer = Announcer.getInstance();
expect(announcer.regionFactory.pIndex).toBe(0);

// Act
const idRef = announcer.announce("something", "polite");
const idRef = await announcer.announce("something", "polite");
const firstRegion = announcer.dictionary.get(idRef)?.element;
expect(firstRegion?.textContent).toBe("something");

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -75,11 +75,9 @@ describe("Announcer.announceMessage", () => {
// ACT
const announcement1Id = announceMessage({
message: message1,
timeoutDelay: 0,
});
const announcement2Id = announceMessage({
message: message2,
timeoutDelay: 0,
});

// ASSERT
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,21 +3,20 @@ import {announceMessage} from "../announce-message";
import {clearMessages} from "../clear-messages";

describe("Announcer.clearMessages", () => {
test("empties a targeted live region element by IDREF", () => {
test("empties a targeted live region element by IDREF", async () => {
// ARRANGE
const message1 = "Shine a million stars";
const message2 = "Dull no stars";

// ACT
const announcement1Id = announceMessage({
const announcement1Id = await announceMessage({
message: message1,
timeoutDelay: 0,
});

const region1 = screen.getByTestId("wbARegion-polite1");
expect(region1).toHaveTextContent(message1);

announceMessage({message: message2, timeoutDelay: 0});
announceMessage({message: message2});
const region2 = screen.getByTestId("wbARegion-polite0");

clearMessages(announcement1Id);
Expand All @@ -33,19 +32,18 @@ describe("Announcer.clearMessages", () => {
const message2 = "Red fish blue fish";

// ACT
announceMessage({message: message1, timeoutDelay: 0});
announceMessage({message: message1});

const region1 = screen.getByTestId("wbARegion-polite1");
expect(region1).toHaveTextContent(message1);

announceMessage({message: message2, timeoutDelay: 0});
announceMessage({message: message2});
const region2 = screen.getByTestId("wbARegion-polite0");
expect(region2).toHaveTextContent(message2);

announceMessage({
message: message1,
level: "assertive",
timeoutDelay: 0,
});
const region3 = screen.getByTestId("wbARegion-assertive1");
expect(region3).toHaveTextContent(message1);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,10 +10,5 @@ type AnnounceMessageButtonProps = {

export const AnnounceMessageButton = (props: AnnounceMessageButtonProps) => {
const {buttonText = "Click"} = props;
// add timeoutDelay: 0 to skip browser setTimeout in Jest tests
return (
<button onClick={() => announceMessage({timeoutDelay: 0, ...props})}>
{buttonText}
</button>
);
return <button onClick={() => announceMessage(props)}>{buttonText}</button>;
};
19 changes: 3 additions & 16 deletions packages/wonder-blocks-announcer/src/announce-message.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,6 @@ export type AnnounceMessageProps = {
message: string;
level?: PolitenessLevel;
removalDelay?: number;
timeoutDelay?: number;
};

export const defaultTimeout = 100;
Expand All @@ -15,25 +14,13 @@ export const defaultTimeout = 100;
* @param {string} message The message to announce.
* @param {PolitenessLevel} level Polite or assertive announcements
* @param {number} removalDelay Optional duration to remove a message after sending. Defaults to 5000ms.
* @param {number} timeoutDelay Optional duration to alter an announcement timeout. Useful in tests for rendering immediately.
* @returns {string} IDREF for targeted live region element
*/
export function announceMessage({
export async function announceMessage({
message,
level = "polite", // TODO: decide whether to allow other roles, i.e. role=`timer`
removalDelay,
timeoutDelay = defaultTimeout,
}: AnnounceMessageProps): string {
}: AnnounceMessageProps): Promise<string> {
const announcer = Announcer.getInstance();

// This timeout is for Safari/Voiceover to improve reliability of the first appended message.
if (timeoutDelay > 0) {
setTimeout(() => {
return announcer.announce(message, level, removalDelay);
}, timeoutDelay);
} else {
return announcer.announce(message, level, removalDelay);
}

return "";
return announcer.announce(message, level, removalDelay);
}