-
Notifications
You must be signed in to change notification settings - Fork 41
/
GTXAccessibilityTree.m
281 lines (254 loc) · 11.2 KB
/
GTXAccessibilityTree.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
//
// Copyright 2018 Google Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
#import <UIKit/UIKit.h>
#import "GTXAccessibilityTree.h"
#import "GTXAssertions.h"
#import "GTXTreeIteratorContext.h"
/**
* There seems to be errors in accessibility children reported by some UIKit classes especially
* UITextEffectsWindow which reports 9223372036854775807 possibly due to internal type conversions
* with -1, we use this bounds value to detect that case..
*/
static const NSInteger kAccessibilityChildrenUpperBound = 50000;
/**
* The class name for @c UIPickerTableView elements. Must be accessed via @c NSClassFromString
* because it is a private class.
*/
static NSString *const kUIPickerTableViewClassName = @"UIPickerTableView";
/**
* The class name for accessibility children of @c UIPickerTableView elements. Must be accessed via
* @c NSClassFromString because it is a private class.
*/
static NSString *const kUIPickerTableViewAccessibilityElementClassName =
@"UITableViewCellAccessibilityElement";
@implementation GTXAccessibilityTree {
// Context for this object's NSEnumerator.
GTXTreeIteratorContext *_enumeratorContext;
// Root elements that this object handles.
NSArray *_rootElements;
}
- (instancetype)initWithRootElements:(NSArray *)rootElements {
self = [super init];
if (self) {
for (id element in rootElements) {
if ([element isKindOfClass:[UIViewController class]]) {
GTX_ASSERT(NO,
@"Invalid root element %@ found. Check GTXToolkit docs to learn more about "
@"valid root elements. Did you mean to use the .view property instead?",
element);
}
}
_enumeratorContext = [[GTXTreeIteratorContext alloc] initWithElements:rootElements];
_rootElements = rootElements;
}
return self;
}
- (void)iterateAllElementsWithBlock:(GTXTreeIterationBlock)block {
// Create a new tree object for iteration since the current object may be in the middle of a
// for-in loop.
GTXAccessibilityTree *tree = [[GTXAccessibilityTree alloc] initWithRootElements:_rootElements];
GTXTreeIteratorElement *iteratorElement;
while ((iteratorElement = [tree gtx_nextObject])) {
block(iteratorElement);
}
}
#pragma mark - NSEnumerator
- (id)nextObject {
id nextObject = [self gtx_nextObject].current;
if (!nextObject) {
// Allow the tree object to be re-enumerated once through.
_enumeratorContext = [[GTXTreeIteratorContext alloc] initWithElements:_rootElements];
}
return nextObject;
}
#pragma mark - NSExtendedEnumerator
- (NSArray *)allObjects {
NSMutableArray *allObjects = [[NSMutableArray alloc] init];
[self iterateAllElementsWithBlock:^(GTXTreeIteratorElement *_Nonnull iteratorElement) {
[allObjects addObject:iteratorElement.current];
}];
return allObjects;
}
#pragma mark - Private
/**
* @return The next @c GTXTreeIteratorElement for the current @c GTXTreeIteratorContext.
*/
- (GTXTreeIteratorElement *)gtx_nextObject {
if (![_enumeratorContext hasElementsInQueue]) {
return nil;
}
id nextInQueue;
GTXTreeIteratorElement *nextIterationElementInQueue;
// Get the next "unvisited" element.
do {
GTXTreeIteratorElement *nextIterationElementCandidate = [_enumeratorContext peekNextElement];
id nextCandidate = nextIterationElementCandidate.current;
[_enumeratorContext dequeueNextElement];
if (![_enumeratorContext didVisitElement:nextCandidate]) {
if (![self gtx_isAccessibilityHiddenElement:nextCandidate]) {
nextInQueue = nextCandidate;
nextIterationElementInQueue = nextIterationElementCandidate;
}
}
} while ([_enumeratorContext hasElementsInQueue] && !nextInQueue);
if (!nextInQueue) {
return nil;
}
[_enumeratorContext visitElement:nextInQueue];
if ([nextInQueue respondsToSelector:@selector(isAccessibilityElement)]) {
if (![nextInQueue isAccessibilityElement]) {
// nextInQueue could be an accessibility container, if so enqueue its children.
// There are two ways of getting the children of an accessibility container:
// First, using @selector(accessibilityElements)
NSArray *axElements = [self gtx_accessibilityElementsOfElement:nextInQueue];
// Second, using @selector(accessibilityElementAtIndex:)
NSArray *axElementsFromIndices =
[self gtx_accessibilityElementsFromIndicesOfElement:nextInQueue];
// Ensure that either the children are available only through one method or elements via both
// are the same. Otherwise we must fail as the the accessibility tree is inconsistent.
if (axElements && axElementsFromIndices) {
NSSet *accessibilityElementsSet = [NSSet setWithArray:axElements];
NSSet *accessibilityElementsFromIndicesSet = [NSSet setWithArray:axElementsFromIndices];
NSAssert([accessibilityElementsSet isEqualToSet:accessibilityElementsFromIndicesSet],
@"Accessibility elements obtained from -accessibilityElements and"
@" -accessibilityElementAtIndex: are different - they must not be. Either provide"
@" elements via one method or provide the same elements.\nDetails:\nElements via"
@" accessibilityElements:%@\nElements via accessibilityElementAtIndex:\n"
@"accessibilityElementCount:%@\nElements:%@",
accessibilityElementsSet, @([nextInQueue accessibilityElementCount]),
accessibilityElementsFromIndicesSet);
// Ensure accessibilityElements* are marked as used even if NSAssert is removed.
(void)accessibilityElementsSet;
(void)accessibilityElementsFromIndicesSet;
} else {
// Set accessibilityElements to whichever is non nil or leave it as is.
axElements = axElementsFromIndices ? axElementsFromIndices : axElements;
}
if (![nextInQueue respondsToSelector:@selector(accessibilityElementsHidden)] ||
![nextInQueue accessibilityElementsHidden]) {
for (id element in axElements) {
if (![_enumeratorContext didVisitElement:element]) {
[_enumeratorContext
queueElement:[[GTXTreeIteratorElement alloc] initWithElement:element
inContainer:nextInQueue]];
}
}
}
// nextInQueue could be a UIView subclass, if so enqueue its subviews.
NSArray *subViews;
if ([nextInQueue isKindOfClass:[UITableViewCell class]] ||
[nextInQueue isKindOfClass:[UICollectionViewCell class]]) {
subViews = [nextInQueue contentView].subviews;
} else if ([nextInQueue respondsToSelector:@selector(subviews)]) {
subViews = [nextInQueue subviews];
}
if ([nextInQueue respondsToSelector:@selector(isHidden)] && ![nextInQueue isHidden]) {
for (id child in subViews) {
if (![_enumeratorContext didVisitElement:child]) {
[_enumeratorContext
queueElement:[[GTXTreeIteratorElement alloc] initWithElement:child
inContainer:nextInQueue]];
}
}
}
}
}
return nextIterationElementInQueue;
}
/**
* @return An array of accessible children of the given @c element as reported by the selector
* -[NSObject(UIAccessibility) accessibilityElements].
*/
- (NSArray *)gtx_accessibilityElementsOfElement:(id)element {
if ([element respondsToSelector:@selector(accessibilityElements)]) {
return [element accessibilityElements];
}
return nil;
}
/**
* @return An array of accessible children of the given @c element as reported by the selector
* -[NSObject(UIAccessibility) accessibilityElementAtIndex:].
*/
- (NSArray *)gtx_accessibilityElementsFromIndicesOfElement:(id)element {
NSMutableArray *axElementsFromIndices;
if ([element respondsToSelector:@selector(accessibilityElementAtIndex:)] &&
[element respondsToSelector:@selector(accessibilityElementCount)]) {
NSInteger childrenCount = [element accessibilityElementCount];
// This is a workaround to deal with UIKit classes that are reporting incorrect
// accessibilityElementCount, see kAccessibilityChildrenUpperBound.
if (childrenCount > 0 && childrenCount < kAccessibilityChildrenUpperBound) {
axElementsFromIndices = [[NSMutableArray alloc] initWithCapacity:(NSUInteger)childrenCount];
for (NSInteger index = 0; index < childrenCount; index++) {
[axElementsFromIndices addObject:[element accessibilityElementAtIndex:index]];
}
}
}
return axElementsFromIndices;
}
/**
* Elements are hidden from accessibility trees
*
* @return @c YES if the element is hidden from accessibility tree @c NO otherwise.
*/
- (BOOL)gtx_isAccessibilityHiddenElement:(id)element {
BOOL isHidden = NO;
BOOL isAccessibilityHidden = NO;
BOOL isHiddenDueToAccessibilityFrame = NO;
BOOL isHiddenDueToFrame = NO;
if ([element respondsToSelector:@selector(isHidden)]) {
isHidden = [element isHidden];
}
if ([element respondsToSelector:@selector(accessibilityElementsHidden)]) {
isAccessibilityHidden = [element accessibilityElementsHidden];
}
if ([element respondsToSelector:@selector(accessibilityFrame)]) {
CGRect accessibilityFrame = [element accessibilityFrame];
isHiddenDueToAccessibilityFrame =
(accessibilityFrame.size.width == 0 || accessibilityFrame.size.height == 0);
}
if ([element respondsToSelector:@selector(frame)]) {
CGRect frame = [element frame];
isHiddenDueToFrame = frame.size.width == 0 || frame.size.height == 0;
}
return (isHidden || isAccessibilityHidden ||
(isHiddenDueToFrame && isHiddenDueToAccessibilityFrame) ||
[self gtx_isElementOffscreenPickerViewElement:element]);
}
/**
* Determines if the element represents an accessibility element in a @c UIPickerTableView, and the
* element is offscreen.
*
* @param element The accessibility element to check.
* @return @c YES if the element is an offscreen accessibility element whose container is a
* @c UIPickerTableView, @c NO otherwise.
*/
- (BOOL)gtx_isElementOffscreenPickerViewElement:(id)element {
if (![element respondsToSelector:@selector(accessibilityFrame)] ||
![element respondsToSelector:@selector(accessibilityContainer)]) {
return NO;
}
id accessibilityContainer = [element accessibilityContainer];
if ([accessibilityContainer isKindOfClass:NSClassFromString(kUIPickerTableViewClassName)] &&
[element isKindOfClass:NSClassFromString(kUIPickerTableViewAccessibilityElementClassName)]) {
CGRect containerAccessibilityFrame = [accessibilityContainer accessibilityFrame];
CGRect childAccessibilityFrame = [element accessibilityFrame];
if (!CGRectIntersectsRect(childAccessibilityFrame, containerAccessibilityFrame)) {
return YES;
}
}
return NO;
}
@end