-
Notifications
You must be signed in to change notification settings - Fork 128
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Proposal for various cross platform a11y improvements. #56
Proposal for various cross platform a11y improvements. #56
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Happy to see others thinking about how to add more accessibility features. I've been coming at this same problem from the perspective of web, where there are perhaps greater constraints on what is possible in some cases, and clearer standards in others.
Properties only available on iOS include: | ||
|
||
* accessibilityViewIsModal | ||
* accessibilityElementsHidden |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is similar to Android importantForAccessibility
prop. It would be good to consolidate these together and learn from how web handles these cases
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Perhaps something like ARIA hiden would be adequate. Android's importantForAccessibility is a bad model to emulate-- it includes yes, no, and auto as values. Auto is the default I believe, and from the developer's point of view, is hard to use, since the rules for when something is automatically marked as unimportant for accessibility aren't clear. I think a better model is to assume everything is important for accessibility, unless explicitly marked otherwise. So it seems like some notion of hidden is adequate.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do you think we should make accessibilityElementsHidden
cross-platform? Mapping to aria-hidden
on web and equivalent on Android?
|
||
Properties only available on Android include: | ||
|
||
* accessibilityLiveRegion |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is also available on web
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We could standardize on ARIA's notion of a live region, and then dumb down the implementation to the various platforms. Android's notion of a live region is pretty basic (polite or assertive). iOS has no notion of live regions as far as I know, so we'd have to use iOS primitives to implement a subset of live region functionality.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
accessibilityLiveRegion
is essentially already identical to ARIA live region, there's just a different in what the none
state is called. So you could extend support for the existing API to iOS and then we'll have the 3 main platforms covered.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
+1 on the accessibilityLiveRegion property being more universal. We just implemented this for Windows (see PR here).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
+1 for accessibilityLiveRegion on iOS
Several properties are missing from both platforms: | ||
|
||
* accessibilityCompoundComponent: replace accessible-- indicates to assistive technologies that all the components inside this one should be treated as a single component. | ||
* accessibilityRelationships: describes how this component is related to others |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I have a proposal internally that is based on Web's ARIA properties: accessibilityLabelID
(equivalent to aria-labelledby
), accessibilityDescriptionID
(equivalent to aria-describedby
), and accessibilityFocusID
(equivalent to aria-activedescendent
). All these props would take the value of the nativeID
of the element they are associated with.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Makes sense. From naming point of view, are you in favor of using the ARIA namespace, or creating our own properties based on ARIA?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What do you mean by "ARIA namespace"? Something like accessibilityLabelledBy
vs accessibilityLabelID
?
|
||
* accessibilityCompoundComponent: replace accessible-- indicates to assistive technologies that all the components inside this one should be treated as a single component. | ||
* accessibilityRelationships: describes how this component is related to others | ||
* accessibilityRange: expose information about range-based components such as sliders |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think we can probably learn from web here and expose the range props to the underlying accessibility layer of the platform. Or model this off ARIA props for progressbar etc
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We've found that in practice, exposing range information in a consumable way is fairly difficult. Exposing a minimum, current, and maximum value allow you to speak a percentage, which is fine for some types of progress bars. But for others, where the actual value is relevant (say a color contrast setting, stereo balance control, movie timeline slider, etc.) it would be nice to include some information that could cause the screen reader to speak either a string or the current value instead of a percentage. Slider's also imply programatic actions to adjust them.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
But for others, where the actual value is relevant (say a color contrast setting, stereo balance control, movie timeline slider, etc.) it would be nice to include some information that could cause the screen reader to speak either a string or the current value instead of a percentage.
ARIA has a property for that aria-valuetext
- https://www.w3.org/TR/wai-aria-practices/examples/slider/slider-2.html
I do like how understandable and extensible the accessibilityRelationships
and accessibilityRange
props names are vs having many more individual props ARIA.
Are you thinking it would look something like this (property names tbh for relationships)?
<View
accessibilityRange={{ min, max, current, text }}
accessibilityRelationships={{ labelID, descriptionID, focusID }}
/>
|
||
### accessibilityCompoundComponent | ||
|
||
We think the current implementation, which is called "accessible", should be renamed to accessibilityCompoundComponent, to more accurately represent what it does. The functionality should remain the same. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The behaviour of accessible
has always felt a bit ambiguous to me, as on Android and Web it's equivalent to focusable
. I don't think accessibilityCompoundComponent
is any clearer a name than accessible
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think the whole notion of compound component handling needs quite a bit more thought. My main opposition to calling this accessible is that it's misleading-- it has little to do with whether the component is accessible, and almost everything to do with how assistive technologies should treat the component.
Maybe I should drop this from this proposal, and make a separate proposal covering compound component handling? I'm thinking as a start about containers with multiple children, which we want to treat as a single item, but depending on the container layout, perhaps only one child is clickable and should be manipulated when the screen reader double taps, etc.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe I should drop this from this proposal, and make a separate proposal covering compound component handling?
I think that sounds good. The idea of an accessibility group is something that's a little complicated on web too, because you have both screen-reader accessibility and keyboard accessibility.
* labeledBy: this component is labeled by the specified component | ||
* describedBy: this component is described by the specified components. | ||
|
||
When referring to other components, we may need to add an additional property to view-- namely accessibilityId. We need to find the best forward-looking approach for referencing other components. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We already have nativeID
that we can use for this
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Great, when was this nativeID property introduced? Trying to understand backward compatibility.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks like it was introduced for both platforms here in June 2017 facebook/react-native@70e0455
|
||
We think the current implementation, which is called "accessible", should be renamed to accessibilityCompoundComponent, to more accurately represent what it does. The functionality should remain the same. | ||
|
||
### accessibilityRelationships |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What does this API look like in practice? Is the value an object?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If you're referring to compound component handling, I think the answer is yes. perhaps a separate proposal would let us discuss and flesh out exactly how to do this.
Relationships could be an object, or each relationship could be a separate property. We've found it sometimes useful for a relationship to include multiple targets-- for example, cases where multiple components describe a single component.
For context, Twitter uses React Native to implement their PWA. The RN abstractions (which have web DNA) are considerably better for building modern web apps that straight DOM abstractions. However, Twitter relies on an undocumented feature of React Native for Web that passes |
Properties only available on iOS include: | ||
|
||
* accessibilityViewIsModal | ||
* accessibilityElementsHidden |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do you think we should make accessibilityElementsHidden
cross-platform? Mapping to aria-hidden
on web and equivalent on Android?
|
||
Properties only available on Android include: | ||
|
||
* accessibilityLiveRegion |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
accessibilityLiveRegion
is essentially already identical to ARIA live region, there's just a different in what the none
state is called. So you could extend support for the existing API to iOS and then we'll have the 3 main platforms covered.
Several properties are missing from both platforms: | ||
|
||
* accessibilityCompoundComponent: replace accessible-- indicates to assistive technologies that all the components inside this one should be treated as a single component. | ||
* accessibilityRelationships: describes how this component is related to others |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What do you mean by "ARIA namespace"? Something like accessibilityLabelledBy
vs accessibilityLabelID
?
|
||
* accessibilityCompoundComponent: replace accessible-- indicates to assistive technologies that all the components inside this one should be treated as a single component. | ||
* accessibilityRelationships: describes how this component is related to others | ||
* accessibilityRange: expose information about range-based components such as sliders |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
But for others, where the actual value is relevant (say a color contrast setting, stereo balance control, movie timeline slider, etc.) it would be nice to include some information that could cause the screen reader to speak either a string or the current value instead of a percentage.
ARIA has a property for that aria-valuetext
- https://www.w3.org/TR/wai-aria-practices/examples/slider/slider-2.html
I do like how understandable and extensible the accessibilityRelationships
and accessibilityRange
props names are vs having many more individual props ARIA.
Are you thinking it would look something like this (property names tbh for relationships)?
<View
accessibilityRange={{ min, max, current, text }}
accessibilityRelationships={{ labelID, descriptionID, focusID }}
/>
|
||
## Motivation | ||
|
||
All accessibility properties should be available on all supported platforms. This reduces the burden on developers to understand the nuances of accessibility support on various platforms. It also allows JavaScript components to expose rich accessibility information, rather than relying on the accessibility support in native components. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
👍
|
||
### accessibilityCompoundComponent | ||
|
||
We think the current implementation, which is called "accessible", should be renamed to accessibilityCompoundComponent, to more accurately represent what it does. The functionality should remain the same. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe I should drop this from this proposal, and make a separate proposal covering compound component handling?
I think that sounds good. The idea of an accessibility group is something that's a little complicated on web too, because you have both screen-reader accessibility and keyboard accessibility.
* labeledBy: this component is labeled by the specified component | ||
* describedBy: this component is described by the specified components. | ||
|
||
When referring to other components, we may need to add an additional property to view-- namely accessibilityId. We need to find the best forward-looking approach for referencing other components. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks like it was introduced for both platforms here in June 2017 facebook/react-native@70e0455
|
||
This property describes how this component is related to others. Relationship types include: | ||
|
||
* labelFor: This component labels the specified component |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is there a use-case you had in mind for having the relationship be defined in either direction like this?
|
||
The same documentation that is used to describe the properties on either iOS or Android can be marked as cross-platform for use on either. For the accessibilityRelationships property, labeledBy and labelFor are common paradigms across accessibility support on multiple platforms including web. | ||
|
||
## Unresolved questions |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I have a related question (and issue) about content that is hidden visually but visible to screen readers. Typically it will be hidden headings so they show up in the screen reader headings list (e.g., mobile.twitter.com does this) and it is visually hidden with a particular CSS incantation. Would you / how would you do something like this in native apps?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe something like this?
import React, { useEffect, useState } from 'react';
import { AccessibilityInfo, Text } from 'react-native';
const useIsScreenReaderEnabled = () => {
const [isScreenReaderEnabled, setIsScreenReaderEnabled] = useState(false);
useEffect(() => {
AccessibilityInfo.addEventListener('screenReaderChanged', setIsScreenReaderEnabled);
AccessibilityInfo.isScreenReaderEnabled().then(setIsScreenReaderEnabled);
return () => AccessibilityInfo.removeEventListener('screenReaderChanged', setIsScreenReaderEnabled);
}, []);
return isScreenReaderEnabled;
};
const Example = () => {
const isScreenReaderEnabled = useIsScreenReaderEnabled();
return <Text>I {isScreenReaderEnabled ? 'love' : '❤️'} React Native for Web</Text>;
};
I'm going to close this proposal as we've subsequently focused on cross-platform APIs based on the existing ARIA spec as per #496 |
This proposal describes various cross platform improvements supported by most platforms.