[a11y] `role` prop not fully supported on React Native
Describe the issue
React Native requires accessible={true} prop to be set on View elements to announce role prop. However, the accessible prop is not exposed by RSD.
The only way in where role works correctly in RN with RSD is:
Textelements behave as ifaccessible={true}by default- When using
Pressable, it setsaccessible={true}on the rendererViewelement. UsingonPresshandler with RSD triggers usingPressableinstead of plainView.
These two cases are frequent enough that they avoid the problem in most cases, but unfortunately, not all.
A simple solution would be to have the role prop (perhaps excluding none/presentation) set accessible to true. However, that creates an issue in the iOS environment, which generally does not support nested accessibility elements. If there are two nested Views with accessible prop, only one is recognized (the ancestor one).
Expected behavior
There should be a way of controlling accessible prop. Perhaps a reasonable default would be to set it to true when role is not none/presentation. However, given the iOS issue described above, there should be a way to disable the status of accessibility element (on a parent), as it might prevent children from receiving a11y focus.
Steps to reproduce
- Render
<html.div role="heading">element, withoutonClickhandler. - Compare with the case when you set
onClickhandler, with rendersaccessibleprop throughPressablecomponent - Use a screen reader to navigate to the button
- Role is not being announced
https://github.com/user-attachments/assets/be386fe5-3a61-46fb-bd3c-00659801e6df
Notice the header role announced on elements with accessible prop, and not announced otherwise.
Test case
https://github.com/mdjastrzebski/rsd-role-repro
Additional comments
No response
Thanks! The fix for nested accessible elements in iOS should probably live in React Native, since it sounds like the same problem can occur there
The source of problem with nested accessibility elements is the default behavior of iOS API itself. There seems to be solution for that called Accessibility Containers (seems to be the same as recent accessibilityOrder prop in RN).