Aesthetically a browser's default focus-ring (i.e. the blue outline) isn't always very pretty and many people do not want to see them. BUT, they are very important for keyboard users and play a crucial role in giving these users valuable feedback to show which element currently has focus.
This library aims to give you best of both worlds. It sets a specific CSS class on the webpage's body
element to indicate whether a focus-ring should be shown or not, depending on whether the user interacts with the page via keyboard or a pointing device, like a mouse.
Essentially, you can think of it as, if the user Tabs, the .js-focus-ring
class is set, and when they click anywhere, the .js-no-focus-ring
class is set.
The library will by default set outline: none;
for all elements on the page if the .js-no-focus-ring
class exists. You can very easily override this for specific elements like form <input>
s if you want by setting explicit focus states for these elements.
For instance:
input:focus {
outline: 2px solid blue;
}
Install via NPM or Yarn
npm install keyboard-only-focus-ring
Once you've installed the library, you can use it either as a React.js hook, component or provider. All options take the same optional props.
It is recommended to use this component as close to your top-most React component (like your
App
component) and to use it only once.
import { useKeyboardFocusRing } from 'keyboard-only-focus-ring'
const App = () => {
// Example shows default values for all props (they are all optional)
const enabled = useKeyboardFocusRing({
defaultEnabled: true,
stylesheetId: '__keyboardOnlyFocusRing__',
})
return <div>Focus ring enabled? {enabled ? 'Yes' : 'No'}</div>
}
import { KeyboardFocusRing } from 'keyboard-only-focus-ring'
const App = () => {
return (
<KeyboardFocusRing defaultEnabled={false} stylesheetId="someCustomId">
{({ enabled }) => (
Focus ring enabled? {enabled ? 'Yes' : 'No'}
)}
</KeyboardFocusRing>
);
}
import {
KeyboardFocusRingProvider,
useKeyboardFocusRingEnabled,
} from 'keyboard-only-focus-ring'
const Child = () => {
const enabled = useKeyboardFocusRingEnabled()
return <div>Focus ring enabled? {enabled ? 'Yes' : 'No'}</div>
}
const App = () => {
return (
<KeyboardFocusRingProvider>
<Child />
</KeyboardFocusRingProvider>
)
}
This library uses the CSSStyleSheet[].insertRule
method to inject the snippet of CSS required to disable the default outline. Internet Explorer doesn't support this method, so you can either polyfill it or simply add the following CSS somewhere in your app's global styles:
.js-no-focus-ring * {
outline: none;
}
MIT