1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #ifndef UI_ACCESSIBILITY_AX_TREE_SERIALIZER_H_
6 #define UI_ACCESSIBILITY_AX_TREE_SERIALIZER_H_
7
8 #include <set>
9
10 #include "base/containers/hash_tables.h"
11 #include "base/logging.h"
12 #include "base/stl_util.h"
13 #include "ui/accessibility/ax_tree_source.h"
14 #include "ui/accessibility/ax_tree_update.h"
15
16 namespace ui {
17
18 struct ClientTreeNode;
19
20 // AXTreeSerializer is a helper class that serializes incremental
21 // updates to an AXTreeSource as a AXTreeUpdate struct.
22 // These structs can be unserialized by a client object such as an
23 // AXTree. An AXTreeSerializer keeps track of the tree of node ids that its
24 // client is aware of so that it will never generate an AXTreeUpdate that
25 // results in an invalid tree.
26 //
27 // Every node in the source tree must have an id that's a unique positive
28 // integer, the same node must not appear twice.
29 //
30 // Usage:
31 //
32 // You must call SerializeChanges() every time a node in the tree changes,
33 // and send the generated AXTreeUpdate to the client.
34 //
35 // If a node is added, call SerializeChanges on its parent.
36 // If a node is removed, call SerializeChanges on its parent.
37 // If a whole new subtree is added, just call SerializeChanges on its root.
38 // If the root of the tree changes, call SerializeChanges on the new root.
39 //
40 // AXTreeSerializer will avoid re-serializing nodes that do not change.
41 // For example, if node 1 has children 2, 3, 4, 5 and then child 2 is
42 // removed and a new child 6 is added, the AXTreeSerializer will only
43 // update nodes 1 and 6 (and any children of node 6 recursively). It will
44 // assume that nodes 3, 4, and 5 are not modified unless you explicitly
45 // call SerializeChanges() on them.
46 //
47 // As long as the source tree has unique ids for every node and no loops,
48 // and as long as every update is applied to the client tree, AXTreeSerializer
49 // will continue to work. If the source tree makes a change but fails to
50 // call SerializeChanges properly, the trees may get out of sync - but
51 // because AXTreeSerializer always keeps track of what updates it's sent,
52 // it will never send an invalid update and the client tree will not break,
53 // it just may not contain all of the changes.
54 template<typename AXSourceNode>
55 class AXTreeSerializer {
56 public:
57 explicit AXTreeSerializer(AXTreeSource<AXSourceNode>* tree);
58 ~AXTreeSerializer();
59
60 // Throw out the internal state that keeps track of the nodes the client
61 // knows about. This has the effect that the next update will send the
62 // entire tree over because it assumes the client knows nothing.
63 void Reset();
64
65 // Serialize all changes to |node| and append them to |out_update|.
66 void SerializeChanges(AXSourceNode node,
67 AXTreeUpdate* out_update);
68
69 // Delete the client subtree for this node, ensuring that the subtree
70 // is re-serialized.
71 void DeleteClientSubtree(AXSourceNode node);
72
73 // Only for unit testing. Normally this class relies on getting a call
74 // to SerializeChanges() every time the source tree changes. For unit
75 // testing, it's convenient to create a static AXTree for the initial
76 // state and then call ChangeTreeSourceForTesting and then SerializeChanges
77 // to simulate the changes you'd get if a tree changed from the initial
78 // state to the second tree's state.
79 void ChangeTreeSourceForTesting(AXTreeSource<AXSourceNode>* new_tree);
80
81 private:
82 // Return the least common ancestor of a node in the source tree
83 // and a node in the client tree, or NULL if there is no such node.
84 // The least common ancestor is the closest ancestor to |node| (which
85 // may be |node| itself) that's in both the source tree and client tree,
86 // and for which both the source and client tree agree on their ancestor
87 // chain up to the root.
88 //
89 // Example 1:
90 //
91 // Client Tree Source tree |
92 // 1 1 |
93 // / \ / \ |
94 // 2 3 2 4 |
95 //
96 // LCA(source node 2, client node 2) is node 2.
97 // LCA(source node 3, client node 4) is node 1.
98 //
99 // Example 2:
100 //
101 // Client Tree Source tree |
102 // 1 1 |
103 // / \ / \ |
104 // 2 3 2 3 |
105 // / \ / / |
106 // 4 7 8 4 |
107 // / \ / \ |
108 // 5 6 5 6 |
109 //
110 // LCA(source node 8, client node 7) is node 2.
111 // LCA(source node 5, client node 5) is node 1.
112 // It's not node 5, because the two trees disagree on the parent of
113 // node 4, so the LCA is the first ancestor both trees agree on.
114 AXSourceNode LeastCommonAncestor(AXSourceNode node,
115 ClientTreeNode* client_node);
116
117 // Return the least common ancestor of |node| that's in the client tree.
118 // This just walks up the ancestors of |node| until it finds a node that's
119 // also in the client tree, and then calls LeastCommonAncestor on the
120 // source node and client node.
121 AXSourceNode LeastCommonAncestor(AXSourceNode node);
122
123 // Walk the subtree rooted at |node| and return true if any nodes that
124 // would be updated are being reparented. If so, update |out_lca| to point
125 // to the least common ancestor of the previous LCA and the previous
126 // parent of the node being reparented.
127 bool AnyDescendantWasReparented(AXSourceNode node,
128 AXSourceNode* out_lca);
129
130 ClientTreeNode* ClientTreeNodeById(int32 id);
131
132 // Delete the given client tree node and recursively delete all of its
133 // descendants.
134 void DeleteClientSubtree(ClientTreeNode* client_node);
135
136 // Helper function, called recursively with each new node to serialize.
137 void SerializeChangedNodes(AXSourceNode node,
138 AXTreeUpdate* out_update);
139
140 // The tree source.
141 AXTreeSource<AXSourceNode>* tree_;
142
143 // Our representation of the client tree.
144 ClientTreeNode* client_root_;
145
146 // A map from IDs to nodes in the client tree.
147 base::hash_map<int32, ClientTreeNode*> client_id_map_;
148 };
149
150 // In order to keep track of what nodes the client knows about, we keep a
151 // representation of the client tree - just IDs and parent/child
152 // relationships.
153 struct AX_EXPORT ClientTreeNode {
154 ClientTreeNode();
155 virtual ~ClientTreeNode();
156 int32 id;
157 ClientTreeNode* parent;
158 std::vector<ClientTreeNode*> children;
159 };
160
161 template<typename AXSourceNode>
AXTreeSerializer(AXTreeSource<AXSourceNode> * tree)162 AXTreeSerializer<AXSourceNode>::AXTreeSerializer(
163 AXTreeSource<AXSourceNode>* tree)
164 : tree_(tree),
165 client_root_(NULL) {
166 }
167
168 template<typename AXSourceNode>
~AXTreeSerializer()169 AXTreeSerializer<AXSourceNode>::~AXTreeSerializer() {
170 Reset();
171 }
172
173 template<typename AXSourceNode>
Reset()174 void AXTreeSerializer<AXSourceNode>::Reset() {
175 if (!client_root_)
176 return;
177
178 DeleteClientSubtree(client_root_);
179 client_id_map_.erase(client_root_->id);
180 delete client_root_;
181 client_root_ = NULL;
182 }
183
184 template<typename AXSourceNode>
ChangeTreeSourceForTesting(AXTreeSource<AXSourceNode> * new_tree)185 void AXTreeSerializer<AXSourceNode>::ChangeTreeSourceForTesting(
186 AXTreeSource<AXSourceNode>* new_tree) {
187 tree_ = new_tree;
188 }
189
190 template<typename AXSourceNode>
LeastCommonAncestor(AXSourceNode node,ClientTreeNode * client_node)191 AXSourceNode AXTreeSerializer<AXSourceNode>::LeastCommonAncestor(
192 AXSourceNode node, ClientTreeNode* client_node) {
193 if (!tree_->IsValid(node) || client_node == NULL)
194 return tree_->GetNull();
195
196 std::vector<AXSourceNode> ancestors;
197 while (tree_->IsValid(node)) {
198 ancestors.push_back(node);
199 node = tree_->GetParent(node);
200 }
201
202 std::vector<ClientTreeNode*> client_ancestors;
203 while (client_node) {
204 client_ancestors.push_back(client_node);
205 client_node = client_node->parent;
206 }
207
208 // Start at the root. Keep going until the source ancestor chain and
209 // client ancestor chain disagree. The last node before they disagree
210 // is the LCA.
211 AXSourceNode lca = tree_->GetNull();
212 int source_index = static_cast<int>(ancestors.size() - 1);
213 int client_index = static_cast<int>(client_ancestors.size() - 1);
214 while (source_index >= 0 && client_index >= 0) {
215 if (tree_->GetId(ancestors[source_index]) !=
216 client_ancestors[client_index]->id) {
217 return lca;
218 }
219 lca = ancestors[source_index];
220 source_index--;
221 client_index--;
222 }
223 return lca;
224 }
225
226 template<typename AXSourceNode>
LeastCommonAncestor(AXSourceNode node)227 AXSourceNode AXTreeSerializer<AXSourceNode>::LeastCommonAncestor(
228 AXSourceNode node) {
229 // Walk up the tree until the source node's id also exists in the
230 // client tree, then call LeastCommonAncestor on those two nodes.
231 ClientTreeNode* client_node = ClientTreeNodeById(tree_->GetId(node));
232 while (tree_->IsValid(node) && !client_node) {
233 node = tree_->GetParent(node);
234 if (tree_->IsValid(node))
235 client_node = ClientTreeNodeById(tree_->GetId(node));
236 }
237 return LeastCommonAncestor(node, client_node);
238 }
239
240 template<typename AXSourceNode>
AnyDescendantWasReparented(AXSourceNode node,AXSourceNode * out_lca)241 bool AXTreeSerializer<AXSourceNode>::AnyDescendantWasReparented(
242 AXSourceNode node, AXSourceNode* out_lca) {
243 bool result = false;
244 int id = tree_->GetId(node);
245 std::vector<AXSourceNode> children;
246 tree_->GetChildren(node, &children);
247 for (size_t i = 0; i < children.size(); ++i) {
248 AXSourceNode& child = children[i];
249 int child_id = tree_->GetId(child);
250 ClientTreeNode* client_child = ClientTreeNodeById(child_id);
251 if (client_child) {
252 if (!client_child->parent) {
253 // If the client child has no parent, it must have been the
254 // previous root node, so there is no LCA and we can exit early.
255 *out_lca = tree_->GetNull();
256 return true;
257 } else if (client_child->parent->id != id) {
258 // If the client child's parent is not this node, update the LCA
259 // and return true (reparenting was found).
260 *out_lca = LeastCommonAncestor(*out_lca, client_child);
261 result = true;
262 } else {
263 // This child is already in the client tree, we won't
264 // recursively serialize it so we don't need to check this
265 // subtree recursively for reparenting.
266 continue;
267 }
268 }
269
270 // This is a new child or reparented child, check it recursively.
271 if (AnyDescendantWasReparented(child, out_lca))
272 result = true;
273 }
274 return result;
275 }
276
277 template<typename AXSourceNode>
ClientTreeNodeById(int32 id)278 ClientTreeNode* AXTreeSerializer<AXSourceNode>::ClientTreeNodeById(int32 id) {
279 base::hash_map<int32, ClientTreeNode*>::iterator iter =
280 client_id_map_.find(id);
281 if (iter != client_id_map_.end())
282 return iter->second;
283 else
284 return NULL;
285 }
286
287 template<typename AXSourceNode>
SerializeChanges(AXSourceNode node,AXTreeUpdate * out_update)288 void AXTreeSerializer<AXSourceNode>::SerializeChanges(
289 AXSourceNode node,
290 AXTreeUpdate* out_update) {
291 // If the node isn't in the client tree, we need to serialize starting
292 // with the LCA.
293 AXSourceNode lca = LeastCommonAncestor(node);
294
295 if (client_root_) {
296 bool need_delete = false;
297 if (tree_->IsValid(lca)) {
298 // Check for any reparenting within this subtree - if there is
299 // any, we need to delete and reserialize the whole subtree
300 // that contains the old and new parents of the reparented node.
301 if (AnyDescendantWasReparented(lca, &lca))
302 need_delete = true;
303 }
304
305 if (!tree_->IsValid(lca)) {
306 // If there's no LCA, just tell the client to destroy the whole
307 // tree and then we'll serialize everything from the new root.
308 out_update->node_id_to_clear = client_root_->id;
309 Reset();
310 } else if (need_delete) {
311 // Otherwise, if we need to reserialize a subtree, first we need
312 // to delete those nodes in our client tree so that
313 // SerializeChangedNodes() will be sure to send them again.
314 out_update->node_id_to_clear = tree_->GetId(lca);
315 ClientTreeNode* client_lca = ClientTreeNodeById(tree_->GetId(lca));
316 CHECK(client_lca);
317 for (size_t i = 0; i < client_lca->children.size(); ++i) {
318 client_id_map_.erase(client_lca->children[i]->id);
319 DeleteClientSubtree(client_lca->children[i]);
320 delete client_lca->children[i];
321 }
322 client_lca->children.clear();
323 }
324 }
325
326 // Serialize from the LCA, or from the root if there isn't one.
327 if (!tree_->IsValid(lca))
328 lca = tree_->GetRoot();
329 SerializeChangedNodes(lca, out_update);
330 }
331
332 template<typename AXSourceNode>
DeleteClientSubtree(AXSourceNode node)333 void AXTreeSerializer<AXSourceNode>::DeleteClientSubtree(AXSourceNode node) {
334 ClientTreeNode* client_node = ClientTreeNodeById(tree_->GetId(node));
335 if (client_node)
336 DeleteClientSubtree(client_node);
337 }
338
339 template<typename AXSourceNode>
DeleteClientSubtree(ClientTreeNode * client_node)340 void AXTreeSerializer<AXSourceNode>::DeleteClientSubtree(
341 ClientTreeNode* client_node) {
342 for (size_t i = 0; i < client_node->children.size(); ++i) {
343 client_id_map_.erase(client_node->children[i]->id);
344 DeleteClientSubtree(client_node->children[i]);
345 delete client_node->children[i];
346 }
347 client_node->children.clear();
348 }
349
350 template<typename AXSourceNode>
SerializeChangedNodes(AXSourceNode node,AXTreeUpdate * out_update)351 void AXTreeSerializer<AXSourceNode>::SerializeChangedNodes(
352 AXSourceNode node,
353 AXTreeUpdate* out_update) {
354 // This method has three responsibilities:
355 // 1. Serialize |node| into an AXNodeData, and append it to
356 // the AXTreeUpdate to be sent to the client.
357 // 2. Determine if |node| has any new children that the client doesn't
358 // know about yet, and call SerializeChangedNodes recursively on those.
359 // 3. Update our internal data structure that keeps track of what nodes
360 // the client knows about.
361
362 // First, find the ClientTreeNode for this id in our data structure where
363 // we keep track of what accessibility objects the client already knows
364 // about. If we don't find it, then this must be the new root of the
365 // accessibility tree.
366 int id = tree_->GetId(node);
367 ClientTreeNode* client_node = ClientTreeNodeById(id);
368 if (!client_node) {
369 Reset();
370 client_root_ = new ClientTreeNode();
371 client_node = client_root_;
372 client_node->id = id;
373 client_node->parent = NULL;
374 client_id_map_[client_node->id] = client_node;
375 }
376
377 // Iterate over the ids of the children of |node|.
378 // Create a set of the child ids so we can quickly look
379 // up which children are new and which ones were there before.
380 base::hash_set<int32> new_child_ids;
381 std::vector<AXSourceNode> children;
382 tree_->GetChildren(node, &children);
383 for (size_t i = 0; i < children.size(); ++i) {
384 AXSourceNode& child = children[i];
385 int new_child_id = tree_->GetId(child);
386 new_child_ids.insert(new_child_id);
387
388 // This is a sanity check - there shouldn't be any reparenting
389 // because we've already handled it above.
390 ClientTreeNode* client_child = client_id_map_[new_child_id];
391 CHECK(!client_child || client_child->parent == client_node);
392 }
393
394 // Go through the old children and delete subtrees for child
395 // ids that are no longer present, and create a map from
396 // id to ClientTreeNode for the rest. It's important to delete
397 // first in a separate pass so that nodes that are reparented
398 // don't end up children of two different parents in the middle
399 // of an update, which can lead to a double-free.
400 base::hash_map<int32, ClientTreeNode*> client_child_id_map;
401 std::vector<ClientTreeNode*> old_children;
402 old_children.swap(client_node->children);
403 for (size_t i = 0; i < old_children.size(); ++i) {
404 ClientTreeNode* old_child = old_children[i];
405 int old_child_id = old_child->id;
406 if (new_child_ids.find(old_child_id) == new_child_ids.end()) {
407 client_id_map_.erase(old_child_id);
408 DeleteClientSubtree(old_child);
409 delete old_child;
410 } else {
411 client_child_id_map[old_child_id] = old_child;
412 }
413 }
414
415 // Serialize this node. This fills in all of the fields in
416 // AXNodeData except child_ids, which we handle below.
417 out_update->nodes.push_back(AXNodeData());
418 AXNodeData* serialized_node = &out_update->nodes.back();
419 tree_->SerializeNode(node, serialized_node);
420 // TODO(dmazzoni/dtseng): Make the serializer not depend on roles to identify
421 // the root.
422 if (serialized_node->id == client_root_->id &&
423 (serialized_node->role != AX_ROLE_ROOT_WEB_AREA &&
424 serialized_node->role != AX_ROLE_DESKTOP)) {
425 serialized_node->role = AX_ROLE_ROOT_WEB_AREA;
426 }
427 serialized_node->child_ids.clear();
428
429 // Iterate over the children, make note of the ones that are new
430 // and need to be serialized, and update the ClientTreeNode
431 // data structure to reflect the new tree.
432 std::vector<AXSourceNode> children_to_serialize;
433 client_node->children.reserve(children.size());
434 for (size_t i = 0; i < children.size(); ++i) {
435 AXSourceNode& child = children[i];
436 int child_id = tree_->GetId(child);
437
438 // No need to do anything more with children that aren't new;
439 // the client will reuse its existing object.
440 if (new_child_ids.find(child_id) == new_child_ids.end())
441 continue;
442
443 new_child_ids.erase(child_id);
444 serialized_node->child_ids.push_back(child_id);
445 if (client_child_id_map.find(child_id) != client_child_id_map.end()) {
446 ClientTreeNode* reused_child = client_child_id_map[child_id];
447 client_node->children.push_back(reused_child);
448 } else {
449 ClientTreeNode* new_child = new ClientTreeNode();
450 new_child->id = child_id;
451 new_child->parent = client_node;
452 client_node->children.push_back(new_child);
453 client_id_map_[child_id] = new_child;
454 children_to_serialize.push_back(child);
455 }
456 }
457
458 // Serialize all of the new children, recursively.
459 for (size_t i = 0; i < children_to_serialize.size(); ++i)
460 SerializeChangedNodes(children_to_serialize[i], out_update);
461 }
462
463 } // namespace ui
464
465 #endif // UI_ACCESSIBILITY_AX_TREE_SERIALIZER_H_
466