Skip to content

Commit

Permalink
Close on enter and space keys
Browse files Browse the repository at this point in the history
  • Loading branch information
tujoworker committed Nov 17, 2024
1 parent d90aa74 commit 24ab19b
Show file tree
Hide file tree
Showing 2 changed files with 71 additions and 34 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -138,13 +138,24 @@ export function HelpButtonInlineContent(
update({ isOpen: false, isUserIntent: false })
}, [update])

const keydownHandler = useCallback(
(event) => {
switch (event.code) {
case 'Escape':
onClose()
buttonRef.current?.focus()
break
const onKeyDown = useCallback(
(event: React.KeyboardEvent<HTMLButtonElement>) => {
if (event.currentTarget === event.target) {
// Firefox returns a whitespace (" ") on event.key when pressing space,
// therefore using .trim() can help normalize this.
// While userEvent.keyboard('{Space}') might return 'Unknown' on event.code,
// making a direct comparison less reliable across all platforms.
switch (event.key.trim() || event.code) {
case 'Enter':
case 'Space':
case 'Escape':
event.preventDefault()
window.requestAnimationFrame(() => {
onClose()
buttonRef.current?.focus()
})
break
}
}
},
[buttonRef, onClose]
Expand Down Expand Up @@ -175,7 +186,7 @@ export function HelpButtonInlineContent(
className="dnb-no-focus"
tabIndex={-1}
innerRef={innerRef}
onKeyDown={keydownHandler}
onKeyDown={onKeyDown}
breakout={breakoutFromLayout}
roundedCorner={!breakoutFromLayout}
innerSpace={
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,13 +28,58 @@ describe('HelpButtonInline', () => {
it('should toggle open state when Space key gets pressed', async () => {
render(<HelpButtonInline help={{ title: 'Help title' }} />)

const button = document.querySelector('button')
expect(document.body).toHaveFocus()

await userEvent.type(button, '{Space}')
expect(button).toHaveClass('dnb-help-button__inline--open')
await userEvent.tab()
expect(document.querySelector('button')).toHaveFocus()

await userEvent.type(button, '{Space}')
expect(button).not.toHaveClass('dnb-help-button__inline--open')
await userEvent.type(document.querySelector('button'), '{Space}')
await waitFor(() => {
expect(document.querySelector('section')).toHaveFocus()
})

await userEvent.keyboard('{Space}')
await waitFor(() => {
expect(document.querySelector('button')).toHaveFocus()
})
})

it('should toggle open state when Enter key gets pressed', async () => {
render(<HelpButtonInline help={{ title: 'Help title' }} />)

expect(document.body).toHaveFocus()

await userEvent.tab()
expect(document.querySelector('button')).toHaveFocus()

await userEvent.keyboard('{Enter}')
await waitFor(() => {
expect(document.querySelector('section')).toHaveFocus()
})

await userEvent.keyboard('{Enter}')
await waitFor(() => {
expect(document.querySelector('button')).toHaveFocus()
})
})

it('should set focus on the button when closing with Escape key', async () => {
render(<HelpButtonInline help={{ title: 'Help title' }} />)

expect(document.body).toHaveFocus()

await userEvent.tab()
expect(document.querySelector('button')).toHaveFocus()

await userEvent.keyboard('{Enter}')
await waitFor(() => {
expect(document.querySelector('section')).toHaveFocus()
})

await userEvent.keyboard('{Escape}')
await waitFor(() => {
expect(document.querySelector('button')).toHaveFocus()
})
})

it('should close when Escape key on the button gets pressed', async () => {
Expand Down Expand Up @@ -161,7 +206,7 @@ describe('HelpButtonInline', () => {
})

it('should set focus on the content when open', async () => {
render(<HelpButtonInline help={{ title: 'Dialog Title' }} />)
render(<HelpButtonInline help={{ title: 'Help title' }} />)

expect(document.body).toHaveFocus()

Expand All @@ -185,27 +230,8 @@ describe('HelpButtonInline', () => {
})
})

it('should set focus on the button when closing with Escape key', async () => {
render(<HelpButtonInline help={{ title: 'Dialog Title' }} />)

expect(document.body).toHaveFocus()

await userEvent.tab()
expect(document.querySelector('button')).toHaveFocus()

await userEvent.keyboard('{Enter}')
await waitFor(() => {
expect(document.querySelector('section')).toHaveFocus()
})

await userEvent.keyboard('{Escape}')
expect(document.querySelector('button')).toHaveFocus()
})

it('should not set focus on the content when open is true', async () => {
render(
<HelpButtonInline help={{ open: true, title: 'Dialog Title' }} />
)
render(<HelpButtonInline help={{ open: true, title: 'Help title' }} />)

expect(document.body).toHaveFocus()

Expand Down

0 comments on commit 24ab19b

Please sign in to comment.