Skip to content

Commit 909964b

Browse files
committed
feat: add force focus utility and integrate it with delete all conversations modal for improved accessibility
1 parent 8a7bcaa commit 909964b

File tree

3 files changed

+32
-0
lines changed

3 files changed

+32
-0
lines changed

src/webchat-ui/components/WebchatUI.tsx

+7
Original file line numberDiff line numberDiff line change
@@ -69,6 +69,7 @@ import speechOutput from "./plugins/speech-output";
6969
import getMessagesListWithoutControlCommands from "../utils/filter-out-control-commands";
7070
import { isValidMarkdown, removeMarkdownChars } from "../../webchat/helper/handleMarkdown";
7171
import DeleteAllConversationsModal from "./presentational/previous-conversations/DeleteAllConversations";
72+
import { forceFocus } from "../utils/force-focus";
7273

7374
export interface WebchatUIProps {
7475
currentSession: string;
@@ -259,6 +260,7 @@ export class WebchatUI extends React.PureComponent<
259260
chatToggleButtonRef: React.RefObject<HTMLButtonElement>;
260261
closeButtonInHeaderRef: React.RefObject<HTMLButtonElement>;
261262
menuButtonInHeaderRef: React.RefObject<HTMLButtonElement>;
263+
deleteButtonInHeaderRef: React.RefObject<HTMLButtonElement>;
262264
ratingButtonInHeaderRef: React.RefObject<HTMLButtonElement>;
263265
webchatWindowRef: React.RefObject<HTMLDivElement>;
264266
homeScreenCloseButtonRef: React.RefObject<HTMLButtonElement>;
@@ -277,6 +279,7 @@ export class WebchatUI extends React.PureComponent<
277279
this.chatToggleButtonRef = React.createRef();
278280
this.closeButtonInHeaderRef = React.createRef();
279281
this.menuButtonInHeaderRef = React.createRef();
282+
this.deleteButtonInHeaderRef = React.createRef();
280283
this.ratingButtonInHeaderRef = React.createRef();
281284
this.webchatWindowRef = React.createRef();
282285
this.homeScreenCloseButtonRef = React.createRef();
@@ -1248,6 +1251,9 @@ export class WebchatUI extends React.PureComponent<
12481251
isOpen
12491252
onOpenChange={open => {
12501253
this.setState({ showDeleteAllConversationsModal: open });
1254+
if (!open && this.deleteButtonInHeaderRef.current) {
1255+
forceFocus(this.deleteButtonInHeaderRef.current);
1256+
}
12511257
}}
12521258
/>
12531259
);
@@ -1415,6 +1421,7 @@ export class WebchatUI extends React.PureComponent<
14151421
title={getTitles()}
14161422
closeButtonRef={this.closeButtonInHeaderRef}
14171423
menuButtonRef={this.menuButtonInHeaderRef}
1424+
deleteButtonRef={this.deleteButtonInHeaderRef}
14181425
chatToggleButtonRef={this.chatToggleButtonRef}
14191426
hideBackButton={hideBackButton}
14201427
showChatScreen={showChatScreen}

src/webchat-ui/components/presentational/Header.tsx

+3
Original file line numberDiff line numberDiff line change
@@ -109,6 +109,7 @@ interface HeaderProps {
109109
onSetShowChatOptionsScreen?: (show: boolean) => void;
110110
closeButtonRef?: React.RefObject<HTMLButtonElement>;
111111
menuButtonRef?: React.RefObject<HTMLButtonElement>;
112+
deleteButtonRef?: React.RefObject<HTMLButtonElement>;
112113
chatToggleButtonRef?: React.RefObject<HTMLButtonElement>;
113114
hideBackButton?: boolean;
114115
deleteIconColor?: string;
@@ -189,6 +190,8 @@ const Header: FC<HeaderProps> = props => {
189190
aria-label="Delete All Conversations"
190191
className="webchat-header-delete-all-conversations-button"
191192
iconColor={rest.deleteIconColor}
193+
tabIndex={0}
194+
ref={rest.deleteButtonRef}
192195
>
193196
<DeleteIcon></DeleteIcon>
194197
</HeaderIconButton>

src/webchat-ui/utils/force-focus.ts

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
/**
2+
* Forces focus on a specified element and prevents focus from being applied to any other element.
3+
* This function applies focus to the target element and sets up event listeners to maintain that focus.
4+
* @param targetElement - The HTML element that should maintain focus
5+
*/
6+
export function forceFocus(targetElement: HTMLElement) {
7+
// Initially focus the element
8+
9+
// Function to handle any focus event in the document
10+
setTimeout(() => {
11+
targetElement.focus();
12+
13+
const preventFocusChange = e => {
14+
e.preventDefault();
15+
targetElement.focus();
16+
document.removeEventListener("focusin", preventFocusChange);
17+
};
18+
19+
// Listen for any focus changes immediately after
20+
document.addEventListener("focusin", preventFocusChange);
21+
}, 10);
22+
}

0 commit comments

Comments
 (0)