iOS
Developer notes
- A sheet helps people perform a distinct task that’s related to the parent view without taking them away from their current context
- Use native elements when at all possible vs a custom element, as it will handle expected behavior without additional development effort
- Most sheets appear as a modal that partially covers the underlying content
- The screen reader is usually confined in the sheet/drawer if it covers underlining content. If a sheet does not cover other content, the screen reader can move out of it
- Ensure there is a way to collapse or close the sheet for the screen reader if covering other content. Initial focus is to an invisible dismiss button in later versions
- A grabber/card controller is used for sheets that expand/collapse. When implemented, the initial focus can be on the grabber, then the next swipe focus is moved to the invisible dismiss button or close X.
- A card controller button expands/collapses the sheet to cover half the screen or the whole screen. After expanding or collapsing, only the name and state gets announced, not the full announcement
- Keyboard users would not be able to expand the sheet without the grabber
Name
- Programmatic name describes the purpose of the control.
-
Ensure that the button that activates the sheet has a programmatic name. The sheet does not have its own programmatic name as it is implied by the button’s programmatic name.
- UIKit
- Set the programmatic name of the button that activates the sheet.
- You can programmatically set the visible label with
setTitle()
.- The button’s title will overwrite the button’s
accessibilityLabel
.
- The button’s title will overwrite the button’s
- If a visible label is not applicable in this case, set the triggering button’s
accessibilityLabel
to the label of your choice.- To do this in Interface Builder, set the label using the Identity Inspector
- To hide labels from VoiceOver programmatically, set the label’s
isAccessibilityElement
property tofalse
- To hide labels from VoiceOver using Interface Builder, uncheck
Accessibility Enabled
in the Identity Inspector.
- SwiftUI
- Set the programmatic name of the button that activates the sheet.
- If there is no visible label, use view modifier
accessibilityLabel(_:)
. - If button has icon(s), hide the icon(s) from VoiceOver by using view modifier
accessibilityHidden(true)
.
Role
- Assign a role to the button that activates the sheet
-
The role of the sheet is implied when the sheet opens, and the initial focus lands on the invisible dismiss button (or sheet grabber, if it exists).
- UIKit
- Use
UIButton
for the button that activates the sheet. - Use
UISheetPresentationController
to implement the sheet. - If necessary, set
accessibilityTraits
to.button
for the button that activates the sheet.
- Use
- SwiftUI
- Use native
Button
view for the button that activates the sheet - Use native
sheet
view modifier for implementation of the sheet - If necessary, use view modifier
accessibilityAddTraits(.isButton)
to assign the role for the button that activates the sheet. - If applicable, use view modifier
accessibilityRemoveTraits(:)
to remove unwanted traits.
- Use native
Groupings
-
Group elements together logically to better understand the context of the screen
- UIKit
- Ensure that the child elements of the overarching view you want to group in has their
isAccessibilityElement
properties set to false. - Set
isAccessibilityElement
totrue
for the parent view. Then, adjustaccessibilityLabel
andaccessibilityTraits
accordingly.- If frame does not exist due to custom button, use
accessibilityFrameInContainer
to set the custom control’s frame to the parent view’s container or view of your choice.- You can also unionize two frames with
frame.union
(i.e.titleLabel.frame.union(subtitleLabel.frame)
).
- You can also unionize two frames with
- Use
shouldGroupAccessibilityElement
for a precise order if the native order should be disrupted. - Use
shouldGroupAccessibilityChildren
to indicate whether VoiceOver must group its children views. This allows making unique vocalizations or define a particular reading order for a part of the page.
- If frame does not exist due to custom button, use
- Ensure that the child elements of the overarching view you want to group in has their
- SwiftUI
- Use view modifier
accessibilityElement(children: .combine)
to merge the child accessibility element’s properties into the new accessibilityElement.
- Use view modifier
State
- The state of the sheet is implied when the sheet opens, and the initial focus lands on the invisible dismiss button.
- For a sheet with adjustable height, the action to increase sheet height or decrease sheet height must be announced when the user is interacting with the sheet grabber/card controller.
-
The accessibility label or value must be updated with each interaction with the sheet grabber/card controller to adjust the sheet height.
- UIKit
- For enabled state of the button that activates the sheet: Set
isEnabled
totrue
. - For disabled state of the button that activates the sheet: Set
isEnabled
tofalse
. Announcement for disabled is “Dimmed”.- If necessary, you may change the accessibility trait of the button to
notEnabled
, but this may overwrite the current accessibility role of the button.
- If necessary, you may change the accessibility trait of the button to
- For enabled state of the button that activates the sheet: Set
- SwiftUI
- The open and closed state of the sheet is implied when the user’s initial focus is on the invisible dismiss button (or sheet grabber, if it exists), and when the user closes the sheet with the invisible dismiss button or developer-implemented close button, if it exists.
- For disabled state of the button that activates the sheet, use view modifier
disabled()
.
Focus
- Use the device’s default focus functionality.
- When the sheet is closed, the focus should return to the triggering element.
- External keyboard tab order often follows the screen reader focus, but sometimes this functionality requires additional development to manage focus.
- Initial focus should land on the sheet grabber, if it exists. Then, the invisible dismiss button. Then, the developer-implemented close button, if it exists.
-
If the sheet grabber does not exist, the initial focus should land on the invisible dismiss button, and then the implemented close button.
- UIKit
- If VoiceOver is not reaching a particular element, set the element’s
isAccessibilityElement
totrue
- Note: You may need to adjust the programmatic name, role, state, and/or value after doing this, as this action may overwrite previously configured accessibility.
- To move screen reader focus to newly revealed content, use
UIAccessibility.post(notification:argument:)
that takes in.screenChanged
and the newly revealed content as the parameter arguments. - To NOT move focus, but dynamically announce new content: use
UIAccessibility.post(notification:argument:)
that takes in.announcement
and the announcement text as the parameter arguments. UIAccessibilityContainer
protocol: Have a table of elements that defines the reading order of the elements.
- If VoiceOver is not reaching a particular element, set the element’s
- SwiftUI
- For general focus management that impacts both screen readers and non-screen readers, use the property wrapper
@FocusState
to assign an identity of a focus state.- Use the property wrapper
@FocusState
in conjunction with the view modifierfocused(_:)
to assign focus on a view with@FocusState
as the source of truth. - Use the property wrapper
@FocusState
in conjunction with the view modifierfocused(_:equals:)
to assign focus on a view, when the view is equal to a specific value.
- Use the property wrapper
- If necessary, use property wrapper
@AccessibilityFocusState
to assign identifiers to specific views to manually shift focus from one view to another as the user interacts with the screen with VoiceOver on.
- For general focus management that impacts both screen readers and non-screen readers, use the property wrapper
Announcement examples
- Announcement order can vary.
- “Double tap to dismiss pop-up window, button” (Invisible dismiss button- later versions)
- “Close, button” (If Close X is available)
- “Card controller, minimized, button, adjust the size of the card overlaying screen” (When focus is on the card controller in minimized to go to half screen)
- “Card controller, half screen, button, adjust the size of the card overlaying screen” (When focus is on the card controller in half screen to go to full screen)
- “Card controller, full screen, button, adjust the size of the card overlaying screen” (When focus is on the card controller in full screen to minimize)
Android
Developer notes
- A sheet helps people perform a distinct task that’s related to the parent view without taking them away from their current context
- Use native elements when at all possible vs a custom element, as it will handle expected behavior without additional development effort
- Most sheets appear as a modal that partially covers the underlying content.
- The screen reader is usually confined in the sheet/drawer if it covers underlining content. If a sheet does not cover other content, the screen reader can move out of it
- Swipe anywhere on the screen dismisses the sheet
- A grabber at the top of the sheet is used to expand/collapse if needed. When implemented, the initial focus is on the grabber
- A grabber expands/collapses the sheet to cover half the screen or the whole screen. Even though two finger swipe will expand the sheet, the user has to start the swipe with fingers on the drawer, which may be a challenge to those who cannot see the screen. Keyboard users would not be able to expand the sheet without the grabber
Name
- Name describes the purpose of the control
-
Ensure that the button that activates the sheet has a programmatic name. The sheet does not have one as it is implied by the button’s programmatic name.
- Android Views
android:text
XML attribute- Optional: use
contentDescription
for a more descriptive name, depending on type of view and for elements (icons) without a visible label contentDescription
overridesandroid:text
- Use
labelFor
attribute to associate the visible label with the control
- Jetpack Compose
- Compose uses semantics properties to pass information to accessibility services.
- The built-in Button composable will fill the semantics properties with information inferred from the composable by default.
- Optional: use
contentDescription
for a more descriptive name to override the default visible label of the button text. - Example specification of contentDescription in compose:
modifier = Modifier.semantics { contentDescription = "" }
Role
- When not using native controls (custom controls), roles will need to be manually coded.
- Android Views
ModalBottomSheet
- Grabber announces as “double tap to activate”
- Jetpack Compose
ModalBottomSheet
BottomSheetScaffold
- When use the drag handle on top of the sheet from the native component then it will be announced as “collapse drag handle” with actions available (three fingers tapping to view). With no drag handle designed UI, a close button is required to be displayed and have the init focus whenever the sheet is triggered and showing.
Groupings
-
Group elements together logically to better understand the context of the screen
- Android Views
ViewGroup
- Set the container object’s
android:screenReaderFocusable
attribute to true, and each inner object’sandroid:focusable
attribute to false. In doing so, accessibility services can present the inner elements’contentDescription
or names, one after the other, in a single announcement.
- Jetpack Compose
Modifier.semantics(mergeDescendants = true) {}
is equivalent toimportantForAccessibility
when compared to android viewsFocusRequester.createRefs()
helps to request focus to inner elements with in the group
State
- Android Views
- Grabber, if any, announces as expands/collapses
- Active:
android:enabled=true
- Disabled:
android:enabled=false
. Announcement: disabled
- Jetpack Compose
- Partial expanded bottom sheet: Drag handle, if any, announces as “collapse drag handle” with actions available in taleback menu (Expand/Dismiss)
- Full expanded bottom sheet: Drag handle, if any, announces as “expanded drag handle” with actions available in talkback menu (Collapse/Dismiss)
- Full expanded bottom sheet with no partial expanded state: Drag handle, if any, announces as “drag handle” with actions available in talkback menu (Dismiss)
- Fixed bottom sheet with Close button and no drag handle: Close button state:
- Active: default state is active and enabled. Use
Button(enabled = true)
to specify explicitly - Disabled:
Button(enabled = false)
announces as disabled - Alternatively can use
modifier = Modifier.semantics { disabled() }
to announce as disabled
- Active: default state is active and enabled. Use
Focus
- Only manage focus when needed. Primarily, let the device manage default focus
- Consider how focus should be managed between child elements and their parent views
- Moving focus into the sheet when an action opens it makes it clear to the screen reader user that there is a sheet available
- When the sheet is closed, the focus should return to the triggering element.
- External keyboard tab order often follows the screen reader focus, but sometimes needs focus management
-
Initial focus on the sheet should land on the grabber
- Android Views
importantForAccessibility
makes the element visible to the Accessibility APIandroid:focusable
android=clickable
- Implement an
onClick( )
event handler for keyboard, as well asonTouch( )
nextFocusDown
nextFocusUp
nextFocusRight
nextFocusLeft
accessibilityTraversalBefore
(or after)- To move screen reader focus to newly revealed content:
Type_View_Focused
- To NOT move focus, but dynamically announce new content:
accessibilityLiveRegion
(set to polite or assertive) - To hide controls:
importantForAccessibility=false
- For a
ViewGroup
, setscreenReaderFocusable=true
and each inner object’s attribute to keyboard focus (focusable=false
)
- Jetpack Compose
Modifier.focusTarget()
makes the component focusableModifier.focusOrder()
needs to be used in combination with FocusRequesters to define focus orderModifier.onFocusEvent()
,Modifier.onFocusChanged()
can be used to observe the changes to focus stateFocusRequester
allows to request focus to individual elements with in a group of merged descendant views- Example: To customize the focus events
- step 1: define the focus requester prior.
val (first, second) = FocusRequester.createRefs()
- step 2: update the modifier to set the order.
modifier = Modifier.focusOrder(first) { this.down = second }
- focus order accepts following values: up, down, left, right, previous, next, start, end
- step 3: use
second.requestFocus()
to gain focus
- step 1: define the focus requester prior.
Code example
- Jetpack Compose
Announcement examples
-
Note: When the user has hints turned on in settings, “double tap to activate” will announce at the end of most interactive controls. Testing should be done with hints turned on to ensure the user understands a control is interactive by hearing either “button” or “double tap to activate” or both. Announcements on Android devices vary slightly due to manufacturer.
- “Sheet grabber, collapsed, double tap to activate” (In minimized state)
- “Sheet grabber, expanded, double tap to activate” (In expanded state)