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 = [[NSMutableArray alloc] initWithCapacity:newChildCount];
103 for (NSUInteger
i = 0;
i < newChildCount; ++
i) {
105 [newChildren addObject:child];
107 NSMutableArray* newChildrenInHitTestOrder =
108 [[NSMutableArray alloc] initWithCapacity:newChildCount];
109 for (NSUInteger
i = 0;
i < newChildCount; ++
i) {
111 [newChildrenInHitTestOrder addObject:child];
113 object.children = newChildren;
114 object.childrenInHitTestOrder = newChildrenInHitTestOrder;
116 NSMutableArray<FlutterCustomAccessibilityAction*>* accessibilityCustomActions =
117 [[NSMutableArray alloc] init];
118 for (int32_t action_id : node.customAccessibilityActions) {
120 if (
action.overrideId != -1) {
125 NSString* label = @(
action.label.data());
126 SEL selector =
@selector(onCustomAccessibilityAction:);
131 customAction.
uid = action_id;
132 [accessibilityCustomActions addObject:customAction];
134 object.accessibilityCustomActions = accessibilityCustomActions;
137 if (needsAnnouncement) {
143 NSString* announcement = [[NSString alloc] initWithUTF8String:
object.node.label.c_str()];
144 UIAccessibilityPostNotification(
145 UIAccessibilityAnnouncementNotification,
146 [[NSAttributedString alloc] initWithString:announcement
148 UIAccessibilitySpeechAttributeQueueAnnouncement : @YES
155 bool routeChanged =
false;
159 if (!view_controller_.view.accessibilityElements) {
160 view_controller_.view.accessibilityElements =
161 @[ [root accessibilityContainer] ?: [NSNull null] ];
163 NSMutableArray<SemanticsObject*>* newRoutes = [[NSMutableArray alloc] init];
164 [root collectRoutes:newRoutes];
167 if (std::find(previous_routes_.begin(), previous_routes_.end(), [route uid]) ==
168 previous_routes_.end()) {
173 if (lastAdded == nil && [newRoutes count] > 0) {
174 int index = [newRoutes count] - 1;
175 lastAdded = [newRoutes objectAtIndex:index];
184 if (lastAdded != nil &&
185 ([lastAdded uid] != previous_route_id_ || [newRoutes count] != previous_routes_.size())) {
186 previous_route_id_ = [lastAdded uid];
189 previous_routes_.clear();
191 previous_routes_.push_back([route uid]);
194 view_controller_.viewIfLoaded.accessibilityElements = nil;
197 NSMutableArray<NSNumber*>* doomed_uids = [NSMutableArray arrayWithArray:objects_.allKeys];
199 VisitObjectsRecursivelyAndRemove(root, doomed_uids);
201 [objects_ removeObjectsForKeys:doomed_uids];
204 [
object accessibilityBridgeDidFinishUpdate];
207 if (!ios_delegate_->IsFlutterViewControllerPresentingModalViewController(view_controller_)) {
208 layoutChanged = layoutChanged || [doomed_uids count] > 0;
211 NSString* routeName = [lastAdded routeName];
212 ios_delegate_->PostAccessibilityNotification(UIAccessibilityScreenChangedNotification,
218 SemanticsObject* lastFocused = [objects_ objectForKey:@(last_focused_semantics_object_id_)];
222 ios_delegate_->PostAccessibilityNotification(
223 UIAccessibilityLayoutChangedNotification,
225 }
else if (scrollOccured) {
229 ios_delegate_->PostAccessibilityNotification(
230 UIAccessibilityPageScrolledNotification,
231 FindNextFocusableIfNecessary().nativeAccessibility);
236void AccessibilityBridge::DispatchSemanticsAction(int32_t node_uid,
240 platform_view_->DispatchSemanticsAction(kFlutterImplicitViewId, node_uid,
action, {});
243void AccessibilityBridge::DispatchSemanticsAction(int32_t node_uid,
248 platform_view_->DispatchSemanticsAction(kFlutterImplicitViewId, node_uid,
action,
254 NSMutableDictionary<NSNumber*, SemanticsObject*>* objects) {
257 NSNumber* nodeId = @(oldObject.
node.
id);
258 NSUInteger positionInChildlist = [oldObject.
parent.
children indexOfObject:oldObject];
260 [oldObject.
parent replaceChildAtIndex:positionInChildlist withChild:newObject];
261 [objects removeObjectForKey:nodeId];
262 objects[nodeId] = newObject;
278 weak_ptr->GetPlatformViewsController();
280 [platformViewsController flutterTouchInterceptingViewForId:node.
platformViewId];
283 platformView:touchInterceptingView];
293 object = CreateObject(updates[uid], GetWeakPtr());
294 objects_[@(uid)] =
object;
297 auto nodeEntry = updates.find(
object.node.
id);
298 if (nodeEntry != updates.end()) {
313 SemanticsObject* newSemanticsObject = CreateObject(node, GetWeakPtr());
314 ReplaceSemanticsObject(
object, newSemanticsObject, objects_);
315 object = newSemanticsObject;
322void AccessibilityBridge::VisitObjectsRecursivelyAndRemove(
SemanticsObject*
object,
323 NSMutableArray<NSNumber*>* doomed_uids) {
324 [doomed_uids removeObject:@(
object.uid)];
326 VisitObjectsRecursivelyAndRemove(child, doomed_uids);
332 if (last_focused_semantics_object_id_ == kSemanticObjectIdInvalid) {
337 return FindFirstFocusable(objects_[@(last_focused_semantics_object_id_)]);
342 if (!currentObject) {
345 if (currentObject.isAccessibilityElement) {
346 return currentObject;
358void AccessibilityBridge::HandleEvent(NSDictionary<NSString*, id>* annotatedEvent) {
359 NSString*
type = annotatedEvent[
@"type"];
360 if ([
type isEqualToString:
@"announce"]) {
361 NSString*
message = annotatedEvent[
@"data"][
@"message"];
362 ios_delegate_->PostAccessibilityNotification(UIAccessibilityAnnouncementNotification,
message);
364 if ([
type isEqualToString:
@"focus"]) {
366 ios_delegate_->PostAccessibilityNotification(UIAccessibilityLayoutChangedNotification, node);
371 return weak_factory_.GetWeakPtr();
374void AccessibilityBridge::clearState() {
375 [objects_ removeAllObjects];
376 previous_route_id_ = 0;
377 previous_routes_.clear();
378 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