17#pragma GCC diagnostic error "-Wundeclared-selector"
24class DefaultIosDelegate :
public AccessibilityBridge::IosDelegate {
26 bool IsFlutterViewControllerPresentingModalViewController(
28 if (view_controller) {
29 return view_controller.isPresentingViewController;
35 void PostAccessibilityNotification(UIAccessibilityNotifications notification,
36 id argument)
override {
37 UIAccessibilityPostNotification(notification, argument);
46 std::unique_ptr<IosDelegate> ios_delegate)
47 : view_controller_(view_controller),
49 platform_views_controller_(platform_views_controller),
50 objects_([[NSMutableDictionary alloc] init]),
52 ios_delegate_(ios_delegate ? std::move(ios_delegate)
53 :
std::make_unique<DefaultIosDelegate>()),
56 initWithName:
@"flutter/accessibility"
57 binaryMessenger:
platform_view->GetOwnerViewController().engine.binaryMessenger
60 HandleEvent((NSDictionary*)
message);
64AccessibilityBridge::~AccessibilityBridge() {
65 [accessibility_channel_ setMessageHandler:nil];
69UIView<UITextInput>* AccessibilityBridge::textInputView() {
70 return [[platform_view_->GetOwnerViewController().engine
textInputPlugin] textInputView];
73void AccessibilityBridge::AccessibilityObjectDidBecomeFocused(int32_t
id) {
74 last_focused_semantics_object_id_ =
id;
75 [accessibility_channel_ sendMessage:@{
@"type" :
@"didGainFocus",
@"nodeId" : @(
id)}];
78void AccessibilityBridge::AccessibilityObjectDidLoseFocus(int32_t
id) {
79 if (last_focused_semantics_object_id_ ==
id) {
80 last_focused_semantics_object_id_ = kSemanticObjectIdInvalid;
84void AccessibilityBridge::UpdateSemantics(
87 BOOL layoutChanged = NO;
88 BOOL scrollOccured = NO;
89 BOOL needsAnnouncement = NO;
90 for (
const auto& entry : actions) {
94 for (
const auto& entry : nodes) {
97 layoutChanged = layoutChanged || [
object nodeWillCauseLayoutChange:&node];
98 scrollOccured = scrollOccured || [
object nodeWillCauseScroll:&node];
99 needsAnnouncement = [
object nodeShouldTriggerAnnouncement:&node];
100 [
object setSemanticsNode:&node];
102 NSMutableArray* newChildren =
103 [[NSMutableArray alloc] initWithCapacity:newChildCountInTraversalOrder];
104 for (NSUInteger
i = 0;
i < newChildCountInTraversalOrder; ++
i) {
106 [newChildren addObject:child];
109 NSMutableArray* newChildrenInHitTestOrder =
110 [[NSMutableArray alloc] initWithCapacity:newChildCountInHitTestOrder];
111 for (NSUInteger
i = 0;
i < newChildCountInHitTestOrder; ++
i) {
113 [newChildrenInHitTestOrder addObject:child];
115 object.children = newChildren;
116 object.childrenInHitTestOrder = newChildrenInHitTestOrder;
118 NSMutableArray<FlutterCustomAccessibilityAction*>* accessibilityCustomActions =
119 [[NSMutableArray alloc] init];
120 for (int32_t action_id : node.customAccessibilityActions) {
122 if (
action.overrideId != -1) {
127 NSString* label = @(
action.label.data());
128 SEL selector =
@selector(onCustomAccessibilityAction:);
133 customAction.
uid = action_id;
134 [accessibilityCustomActions addObject:customAction];
136 object.accessibilityCustomActions = accessibilityCustomActions;
139 if (needsAnnouncement) {
145 NSString* announcement = [[NSString alloc] initWithUTF8String:
object.node.label.c_str()];
146 UIAccessibilityPostNotification(
147 UIAccessibilityAnnouncementNotification,
148 [[NSAttributedString alloc] initWithString:announcement
150 UIAccessibilitySpeechAttributeQueueAnnouncement : @YES
157 bool routeChanged =
false;
161 if (!view_controller_.view.accessibilityElements) {
162 view_controller_.view.accessibilityElements =
163 @[ [root accessibilityContainer] ?: [NSNull null] ];
165 NSMutableArray<SemanticsObject*>* newRoutes = [[NSMutableArray alloc] init];
166 [root collectRoutes:newRoutes];
169 if (std::find(previous_routes_.begin(), previous_routes_.end(), [route uid]) ==
170 previous_routes_.end()) {
175 if (lastAdded == nil && [newRoutes count] > 0) {
176 int index = [newRoutes count] - 1;
177 lastAdded = [newRoutes objectAtIndex:index];
186 if (lastAdded != nil &&
187 ([lastAdded uid] != previous_route_id_ || [newRoutes count] != previous_routes_.size())) {
188 previous_route_id_ = [lastAdded uid];
191 previous_routes_.clear();
193 previous_routes_.push_back([route uid]);
196 view_controller_.viewIfLoaded.accessibilityElements = nil;
199 NSMutableArray<NSNumber*>* doomed_uids = [NSMutableArray arrayWithArray:objects_.allKeys];
201 VisitObjectsRecursivelyAndRemove(root, doomed_uids);
203 [objects_ removeObjectsForKeys:doomed_uids];
206 [
object accessibilityBridgeDidFinishUpdate];
209 if (!ios_delegate_->IsFlutterViewControllerPresentingModalViewController(view_controller_)) {
210 layoutChanged = layoutChanged || [doomed_uids count] > 0;
213 NSString* routeName = [lastAdded routeName];
214 ios_delegate_->PostAccessibilityNotification(UIAccessibilityScreenChangedNotification,
220 SemanticsObject* lastFocused = [objects_ objectForKey:@(last_focused_semantics_object_id_)];
224 ios_delegate_->PostAccessibilityNotification(
225 UIAccessibilityLayoutChangedNotification,
227 }
else if (scrollOccured) {
231 ios_delegate_->PostAccessibilityNotification(
232 UIAccessibilityPageScrolledNotification,
233 FindNextFocusableIfNecessary().nativeAccessibility);
238void AccessibilityBridge::DispatchSemanticsAction(int32_t node_uid,
242 platform_view_->DispatchSemanticsAction(kFlutterImplicitViewId, node_uid,
action, {});
245void AccessibilityBridge::DispatchSemanticsAction(int32_t node_uid,
250 platform_view_->DispatchSemanticsAction(kFlutterImplicitViewId, node_uid,
action,
256 NSMutableDictionary<NSNumber*, SemanticsObject*>* objects) {
259 NSNumber* nodeId = @(oldObject.
node.
id);
260 NSUInteger positionInChildlist = [oldObject.
parent.
children indexOfObject:oldObject];
262 [oldObject.
parent replaceChildAtIndex:positionInChildlist withChild:newObject];
263 [objects removeObjectForKey:nodeId];
264 objects[nodeId] = newObject;
280 weak_ptr->GetPlatformViewsController();
282 [platformViewsController flutterTouchInterceptingViewForId:node.
platformViewId];
285 platformView:touchInterceptingView];
295 object = CreateObject(updates[uid], GetWeakPtr());
296 objects_[@(uid)] =
object;
299 auto nodeEntry = updates.find(
object.node.
id);
300 if (nodeEntry != updates.end()) {
315 SemanticsObject* newSemanticsObject = CreateObject(node, GetWeakPtr());
316 ReplaceSemanticsObject(
object, newSemanticsObject, objects_);
317 object = newSemanticsObject;
324void AccessibilityBridge::VisitObjectsRecursivelyAndRemove(
SemanticsObject*
object,
325 NSMutableArray<NSNumber*>* doomed_uids) {
326 [doomed_uids removeObject:@(
object.uid)];
328 VisitObjectsRecursivelyAndRemove(child, doomed_uids);
334 if (last_focused_semantics_object_id_ == kSemanticObjectIdInvalid) {
339 return FindFirstFocusable(objects_[@(last_focused_semantics_object_id_)]);
344 if (!currentObject) {
347 if (currentObject.isAccessibilityElement) {
348 return currentObject;
360void AccessibilityBridge::HandleEvent(NSDictionary<NSString*, id>* annotatedEvent) {
361 NSString*
type = annotatedEvent[
@"type"];
362 if ([
type isEqualToString:
@"announce"]) {
363 NSString*
message = annotatedEvent[
@"data"][
@"message"];
364 ios_delegate_->PostAccessibilityNotification(UIAccessibilityAnnouncementNotification,
message);
366 if ([
type isEqualToString:
@"focus"]) {
368 ios_delegate_->PostAccessibilityNotification(UIAccessibilityLayoutChangedNotification, node);
373 return weak_factory_.GetWeakPtr();
376void AccessibilityBridge::clearState() {
377 [objects_ removeAllObjects];
378 previous_route_id_ = 0;
379 previous_routes_.clear();
380 view_controller_.viewIfLoaded.accessibilityElements = nil;
NS_ASSUME_NONNULL_BEGIN typedef void(^ FlutterReply)(id _Nullable reply)
constexpr int32_t kRootNodeId
AccessibilityBridge()
Creates a new instance of a accessibility bridge.
A Mapping like NonOwnedMapping, but uses Free as its release proc.
G_BEGIN_DECLS G_MODULE_EXPORT FlValue * args
G_BEGIN_DECLS GBytes * message
#define FML_DCHECK(condition)
flutter::SemanticsNode node
NSArray< SemanticsObject * > * children
FlutterTextInputPlugin * textInputPlugin
std::unordered_map< int32_t, SemanticsNode > SemanticsNodeUpdates
std::unordered_map< int32_t, CustomAccessibilityAction > CustomAccessibilityActionUpdates
SemanticsCheckState isChecked
SemanticsTristate isToggled
bool isInMutuallyExclusiveGroup
bool hasImplicitScrolling
std::vector< int32_t > childrenInHitTestOrder
std::vector< int32_t > customAccessibilityActions
bool IsPlatformViewNode() const
std::vector< int32_t > childrenInTraversalOrder