Skip to content

Commit

Permalink
fix: arrow-key navigation behavior in search
Browse files Browse the repository at this point in the history
The "contacts" section items were not included
in the roving tabindex winget: they all had individual tab stops,
which is very confusing.
  • Loading branch information
WofWca committed Nov 27, 2024
1 parent 9895870 commit e1e6e08
Show file tree
Hide file tree
Showing 3 changed files with 37 additions and 5 deletions.
2 changes: 1 addition & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
## [Unreleased][unreleased]

## Added
- accessibility: arrow-key navigation for the list of chats, list of accounts #4224, #4291
- accessibility: arrow-key navigation for the list of chats, list of accounts #4224, #4291, #4361
- Add "Learn More" button to "Disappearing Messages" dialog #4330
- new icon for Mac users
- smooth-scroll to newly arriving messages instead of jumping instantly #4125
Expand Down
37 changes: 33 additions & 4 deletions packages/frontend/src/components/contact/ContactListItem.tsx
Original file line number Diff line number Diff line change
@@ -1,16 +1,18 @@
import React, { MouseEventHandler } from 'react'
import React, { MouseEventHandler, useRef } from 'react'
import Contact from './Contact'
import classNames from 'classnames'
import { Type } from '../../backend-com'
import Icon from '../Icon'
import useTranslationFunction from '../../hooks/useTranslationFunction'
import { useRovingTabindex } from '../../contexts/RovingTabindex'

export const DeltaCheckbox = (props: {
checked: boolean
disabled?: boolean
onClick?: (event: React.SyntheticEvent) => void
tabIndex?: 0 | -1
}) => {
const { checked, disabled } = props
const { checked, disabled, tabIndex } = props
const _onClick = props.onClick
const onClick = (
event: React.ChangeEvent<any> | React.MouseEvent<any, MouseEvent>
Expand All @@ -24,6 +26,7 @@ export const DeltaCheckbox = (props: {
disabled={disabled === true}
onChange={onClick}
checked={checked}
tabIndex={tabIndex}
/>
<div
className='checkmark'
Expand Down Expand Up @@ -58,6 +61,11 @@ export function ContactListItem(props: {
disabled,
onContextMenu,
} = props

const checkboxDisabled = disabled || contact.id === 1

const refMain = useRef<HTMLButtonElement>(null)

const onCheckboxClick = () => {
if (disabled) return
if (!showCheckbox) return
Expand All @@ -69,18 +77,37 @@ export function ContactListItem(props: {
if (!showRemove) return
typeof props.onRemoveClick === 'function' && props.onRemoveClick(contact)
}

// Keep in mind that this component is not always placed inside of
// `RovingTabindexContext`. It's fine, because `useRovingTabindex`
// will simply always return `tabIndex === 0` in this case.
const rovingTabindex = useRovingTabindex(refMain)

return (
<div
className={classNames('contact-list-item', { disabled })}
key={contact.id}
// Apply these to the wrapper element,
// because there may be several interactive elements in this component.
onKeyDown={rovingTabindex.onKeydown}
onFocus={rovingTabindex.setAsActiveElement}
>
<button
className={classNames('contact-list-item-button', { disabled })}
ref={refMain}
className={classNames(
'contact-list-item-button',
rovingTabindex.className,
{ disabled }
)}
// `aria-disabled` instead of just `disabled` because we probably
// still want to keep it focusable so that the context menu can be
// activated, and for screen-readers.
aria-disabled={disabled}
disabled={disabled && !onContextMenu}
// FYI this makes this element keyboard-navigarble
// regardless of whether it is disabled.
// This is probably fine.
tabIndex={rovingTabindex.tabIndex}
onClick={() => {
if (disabled) return
onClick && onClick(contact)
Expand All @@ -100,7 +127,8 @@ export function ContactListItem(props: {
{showCheckbox && (
<DeltaCheckbox
checked={checked}
disabled={disabled || contact.id === 1}
disabled={checkboxDisabled}
tabIndex={checkboxDisabled ? undefined : rovingTabindex.tabIndex}
onClick={onCheckboxClick}
/>
)}
Expand All @@ -109,6 +137,7 @@ export function ContactListItem(props: {
className='btn-remove'
onClick={onRemoveClick}
disabled={disabled}
tabIndex={disabled ? undefined : rovingTabindex.tabIndex}
aria-label={tx('remove_desktop')}
>
<Icon icon='cross' coloring='remove' />
Expand Down
3 changes: 3 additions & 0 deletions packages/frontend/src/contexts/RovingTabindex.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,9 @@ import React, {
* https://developer.mozilla.org/en-US/docs/Web/Accessibility/Keyboard-navigable_JavaScript_widgets#technique_1_roving_tabindex
* ).
*
* It is OK to use this outside of `RovingTabindexContext`.
* In this case it will simply always return `tabIndex === 0`.
*
* This is similar to https://www.npmjs.com/package/react-roving-tabindex, but
* - it handles elements changing their order in DOM:
* 'ArrowDown' will always focus the element that is next in DOM,
Expand Down

0 comments on commit e1e6e08

Please sign in to comment.