A no-fuss accessible keyboard navigation handling for a flexible grid component. This is not a spreadsheet-like grid or table, but provides keyboard navigation appropriate for one. If you have a grid (as in a CSS grid) that the user will want to navigate between items, this provides a simple way to provide keyboard navigation following W3C recommendations.
- Provides "standard" W3C keyboard navigation
- Simple integration with any App, including React; it requires a single delegation of keydown events.
- Responsive. Handles row size changes automatically
- Dynamic. Handles dynamic lists of elements efficiently
- Rigorously tested
- Customizable
For our users' sake, it's convenient and recommended for data grids to have keyboard navigation. The
W3 recommends certain behaviors, but
these aren't built in to HTML-- nor are they easily obtainable. You have to write Javascript and switch
statements.
Once you start writing this, you realize the code gets complicated-- or at least tedious. With any sort of responsive
layout, moving the the "next row" is a dynamic calculation, not just an addition, as is finding the first or last
element of the row.
This package handles all that with a simple TypeScript package.
A proper demo will come, but in the meantime, see https://amp-what.com
To add keyboard navigation to a grid of elements, images or tiles, here are the steps:
npm add --save grid-navigator
(Yarn and Typescript also work.)
Your code must provide a function to tell the Grid Navigator which elements the user is navigating. This is specified as a function that returns a NodeList of elements. (There's no performance penalty if this list of elements never changes.) Here is a simple example function:
const elementsProvider = () => document.getElementsByTagName('li')
This will be called lazily when needed.
You are also responsible for reflecting the focus, or "select" status, to the user.
This is done by implementing a callback function called selectCallback
. The W3C
recommends using tabIndex
values of '-1' and '0' to keep track of this, but
it's up to you. This callback function will receive two arguments: the element,
and a boolean about whether the element is being selected or not. As the user
moves from one element to the next, this will function will be called
twice-- once for leaving the old element, and once for entering the new one:
const selectCallback = (e, selectNow) => {
if (selectNow) // entering/focusing
e.tabindex = 0
else // leaving/blur
e.tabindex = -1
}
Typically you will use some accompanying CSS, such as:
[role="grid"] [tabindex="0"]:focus {
outline: #005a9c;
outline-style: dotted;
outline-width: 3px;
}
To stitch it all together, you need a GridNavigator object, fully configured.
import { GridNavigator } from 'grid-navigator'
const myNav = new GridNavigator({ elementProvider, selectCallback })
The final required step is to delegate keyboard events to the GridNavigator. From the keydown handler of your grid container, you need to add a keydown handler:
const myNav = new GridNavigator({ elementProvider, selectCallback })
const grid = document.getElementById('grid')
grid.addEventHandler('keydown', (e) => {
if (myNav.onKeyDown(e)) return // `onKeyDown` returns TRUE if it handles the event
// do you normal stuff
...
})
This keydown handler will look for applicable keys, and if present, call the selectCallback
appropriately. It returns true
if it handles the keyboard event, and false
otherwise.
The code uses the elementProvider
to determine the elements being navigated. Once they are
calculated, it assumes they are stable unless you tell it otherwise. To indicate that elements have changed,
call myNav.markStale()
, and GridNavigator will call elementProvider
(along with other appropriate callbacks).
Note, this calculation is done lazily, only during navigation, which means there is minimal overhead
for a heavily dynamic grid.
By default, rows widths are determine by looking at the position of elements on the page,
and uses a simple heuristic to see where columns line up. This works well
for responsive layouts, where the row width is determined within the CSS, and not readily available
to the Javascript. If you have this information available in JS, or just want to calculate it yourself,
provide a function columnCountCalculator: (elems: NodeListOf<E>) => number
. It will be
called whenever a layout change has been detected.
Here's a sketch of a hook:
const useGridNavigator = (gridRef, selectCallback) => {
const [gridNav, setGridNav] = useState()
useEffect(() => {
if (gridRef.current) {
const elementProvider = () => gridRef.current.getElementsByTagName('li')
setGridNav(new GridNavigator({ elementProvider, selectCallback }))
gridRef.current.addEventHandler('keydown', (e) => {
if (gridNav.onKeyDown(e)) e.stopImmediatePropagation()
})
}
}, [gridRef])
useEffect(() => gridNav?.markStale())
}
...
const selectCallback = ...
useGridNavigator(myGridRef, selectCallback)
Out of the box, the default key bindings are those suggested by the w3 documentation. These are specified within grid-navigator using a Javascript object, interpreted as a map:
export const DEFAULT_STANDARD: KeyToMoveOpMap = {
'ArrowLeft': 'prev',
'ArrowRight': 'next',
'ArrowUp': 'up',
'ArrowDown': 'down',
'PageDown': 'pageDown',
'PageUp': 'pageUp',
'Home': 'startOfLine',
'End': 'endOfLine',
'Ctrl + Home': 'first',
'Ctrl + End': 'last',
}
You can amend this map with any of your own.
For convenience, a few supplementary key bindings are provided:
VI
-ishEMACS
-ishNUMPAD
, for navigation using a numeric keypad. This is outside the standard recommendation, but convenient for some users.
To use any of these, import them and pass them in to the GridNavigator
constructor:
import { KeyMaps } from 'grid-navigator'
myKeyMaps = [...KeyMaps.DEFAULT_STANDARD, ...KeyMaps.VI, ...KeyMaps.EMACS]
const myNav = new GridNavigator({
keyMap: myKeyMaps,
elementProvider,
selectCallback
})
The grid navigator accepts a keyMap
property to so that you can control in fine detail how keystrokes are handled. As
described above, this is a Javascript object used as a map from a key value to a "moveOp".
- The keyboard value is provided by keyboard-event-to-string.
- The "move" is encapsulated by the
MoveOp
type.
Contact me for licensing or consulting.
See package.json
.
To release, yarn publish
Copyright (c) 2021–2023 Andrew J. Peterson, NDP Software. All Rights Reserved.