Android develop - Keyboard navigation & Switch Access
Access to interactive elements #
Target: everyone and especially people with motor disabilities who use a keyboard to navigate.
When: during design and development.
Navigation using Switch Access or with keyboard is very useful for people with motor or cognitive difficulties. This navigation makes it possible to move from one interactive element to the next (elements on which an action can be performed).
To manage keyboard (and Switch Access) navigation, there are 2 things to check:
- Allow focus on interactive elements (focus navigation only concerns interactive elements). If, for example, your application has custom views that can be clicked on, you must make sure that these views are focusable by setting the
focusable
attribute totrue
. - Manage the focus display: any interactive element can receive the focus, so the
state_focused
must be defined and allow to easily distinguish which element has the focus.
Note: Android Pie (9, API 28), introduces a specific focus for screen reader to avoid edge effects between the screen reader focus (screenReaderFocusable
) and the keyboard focus (focusable
). Keyboard navigation is not sensitive to the screenReaderFocusable
.
To be verified:
- With the keyboard (and Switch Access) navigation, it is possible to visually determine which element has the focus.
- All the features are accessible with the keyboard (and Switch Access) navigation.
- All interactive elements are highlighted with the keyboard (and Switch Access) navigation.
- Only the interactive elements are highlighted with the keyboard (and Switch Access) navigation.
Examples:
Example of a selector including the state_focused
:
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<!-- Non focused states -->
<item android:state_focused="false" android:state_selected="false" android:state_pressed="false" android:drawable="@drawable/draw_unselected_selector" />
<item android:state_focused="false" android:state_selected="true" android:state_pressed="false" android:drawable="@drawable/draw_selected_selector" />
<!-- Focused states -->
<item android:state_focused="true" android:state_selected="false" android:state_pressed="false" android:drawable="@drawable/draw_unselected_focused_selector" />
<item android:state_focused="true" android:state_selected="true" android:state_pressed="false" android:drawable="@drawable/draw_selected_focused_selector" />
<!-- Pressed -->
<!-- Non focused states -->
<item android:state_focused="false" android:state_selected="false" android:state_pressed="true" android:drawable="@drawable/draw_unselected_pressed_selector" />
<item android:state_focused="false" android:state_selected="true" android:state_pressed="true" android:drawable="@drawable/draw_selected_pressed_selector" />
<!-- Focused states -->
<item android:state_focused="true" android:state_selected="false" android:state_pressed="true" android:drawable="@drawable/draw_unselected_pressed_selector" />
<item android:state_focused="true" android:state_selected="true" android:state_pressed="true" android:drawable="@drawable/draw_selected_pressed_selector" />
</selector>
// Avec Jetpack Compose
@Composable
fun CustomDrawableSelector(
isFocused: Boolean,
isSelected: Boolean,
isPressed: Boolean
) {
val drawableRes = when {
!isFocused && !isSelected && !isPressed -> R.drawable.draw_unselected_selector
!isFocused && isSelected && !isPressed -> R.drawable.draw_selected_selector
isFocused && !isSelected && !isPressed -> R.drawable.draw_unselected_focused_selector
isFocused && isSelected && !isPressed -> R.drawable.draw_selected_focused_selector
!isFocused && !isSelected && isPressed -> R.drawable.draw_unselected_pressed_selector
!isFocused && isSelected && isPressed -> R.drawable.draw_selected_pressed_selector
isFocused && !isSelected && isPressed -> R.drawable.draw_unselected_pressed_selector
isFocused && isSelected && isPressed -> R.drawable.draw_selected_pressed_selector
else -> R.drawable.draw_unselected_selector // Default fallback
}
Image(
painter = painterResource(id = drawableRes),
contentDescription = null,
modifier = Modifier
.size(48.dp) // Adjust size as needed
)
}
Example of a focusable view with keyboard navigation
<View android:focusable="true"
android:layout_width="0px"
android:layout_height="0px" />
Example of a view that is not focusable with keyboard navigation but focusable with touch
<View android:focusableInTouchMode="true"
android:layout_width="0px"
android:layout_height="0px" />
WCAG reference:
Keyboard navigation order #
Target: everyone and especially people with motor disabilities who use a keyboard to navigate.
When: during design and development.
Description:
Keyboard (and Switch Access) navigation, to be understandable and easy to use, must provide a coherent and understandable order of navigation. It is important to manage the focus order : through the options nextFocusDown
, nextFocusUp
, nextFocusRight
and nextFocusLeft
, you can specify which view should take the focus according to the use of the tab, down, up, right and left arrows.
Note: nextFocusDown
, nextFocusUp
, nextFocusRight
, nextFocusLeft
, focusable
and other focus management options are available directly in the XML or in the code through the corresponding methods.
For more information on focus management on Android.
To be verified:
- The keyboard (and Switch Access) navigation order is logical and consistent.
- Items are grouped together when they are linked (example: content within a clickable item in a list).
- It is possible to scroll in a drop-down list to access all the content
- You are never trapped in a part of the screen (what we call a keyboard trap)
Example
<EditText
android:id="@+id/et1"
android:nextFocusDown="@+id/et2"
android:nextFocusUp="@+id/et2"
....../>
<EditText
android:id="@+id/et2"
android:nextFocusDown="@+id/et1"
android:nextFocusUp="@+id/et1"
...../>
WCAG reference: