blob: 92b2166cdffc62923a6095d062891ef969c77954 [file] [log] [blame]
// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "ui/accessibility/tree_generator.h"
#include "ui/accessibility/ax_serializable_tree.h"
#include "ui/accessibility/ax_tree.h"
namespace ui {
static int UniqueTreeCountForNodeCount(int node_count,
bool permutations) {
int unique_tree_count = 1;
// (n-1)! for the possible trees.
for (int i = 2; i < node_count; ++i)
unique_tree_count *= i;
// n! for the permutations of ids.
if (permutations)
unique_tree_count = unique_tree_count * unique_tree_count * node_count;
return unique_tree_count;
}
TreeGenerator::TreeGenerator(int max_node_count, bool permutations)
: max_node_count_(max_node_count),
permutations_(permutations),
total_unique_tree_count_(0) {
unique_tree_count_by_size_.push_back(0);
for (int i = 1; i <= max_node_count; ++i) {
int unique_tree_count = UniqueTreeCountForNodeCount(i, permutations);
unique_tree_count_by_size_.push_back(unique_tree_count);
total_unique_tree_count_ += unique_tree_count;
}
}
TreeGenerator::~TreeGenerator() {
}
int TreeGenerator::UniqueTreeCount() const {
return total_unique_tree_count_;
}
void TreeGenerator::BuildUniqueTree(int tree_index, AXTree* out_tree) const {
AXTreeUpdate update;
BuildUniqueTreeUpdate(tree_index, &update);
CHECK(out_tree->Unserialize(update)) << out_tree->error();
}
int TreeGenerator::IgnoredPermutationCountPerUniqueTree(int tree_index) const {
int unique_tree_count_so_far = 0;
for (int node_count = 1; node_count <= max_node_count_; ++node_count) {
int unique_tree_count = unique_tree_count_by_size_[node_count];
if (tree_index - unique_tree_count_so_far < unique_tree_count) {
// Each node other than the root can be either ignored or not,
// so return 2 ^ (node_count - 1)
return 1 << (node_count - 1);
}
unique_tree_count_so_far += unique_tree_count;
}
NOTREACHED();
return 0;
}
void TreeGenerator::BuildUniqueTreeWithIgnoredNodes(int tree_index,
int ignored_index,
AXTree* out_tree) const {
AXTreeUpdate update;
BuildUniqueTreeUpdate(tree_index, &update);
int node_count = int{update.nodes.size()};
CHECK_GE(ignored_index, 0);
CHECK_LT(ignored_index, 1 << (node_count - 1));
for (int i = 0; i < node_count - 1; i++) {
if (ignored_index & (1 << i))
update.nodes[i + 1].AddState(ax::mojom::State::kIgnored);
}
CHECK(out_tree->Unserialize(update)) << out_tree->error();
}
void TreeGenerator::BuildUniqueTreeUpdate(int tree_index,
AXTreeUpdate* out_update) const {
CHECK_LT(tree_index, total_unique_tree_count_);
int unique_tree_count_so_far = 0;
for (int node_count = 1; node_count <= max_node_count_; ++node_count) {
int unique_tree_count = unique_tree_count_by_size_[node_count];
if (tree_index - unique_tree_count_so_far < unique_tree_count) {
BuildUniqueTreeUpdateWithSize(
node_count, tree_index - unique_tree_count_so_far, out_update);
return;
}
unique_tree_count_so_far += unique_tree_count;
}
}
void TreeGenerator::BuildUniqueTreeUpdateWithSize(
int node_count,
int tree_index,
AXTreeUpdate* out_update) const {
std::vector<int> indices;
std::vector<int> permuted;
int unique_tree_count = unique_tree_count_by_size_[node_count];
CHECK_LT(tree_index, unique_tree_count);
if (permutations_) {
// Use the first few bits of |tree_index| to permute the indices.
for (int i = 0; i < node_count; ++i)
indices.push_back(i + 1);
for (int i = 0; i < node_count; ++i) {
int p = (node_count - i);
int index = tree_index % p;
tree_index /= p;
permuted.push_back(indices[index]);
indices.erase(indices.begin() + index);
}
} else {
for (int i = 0; i < node_count; ++i)
permuted.push_back(i + 1);
}
// Build an AXTreeUpdate. The first two nodes of the tree always
// go in the same place.
out_update->root_id = permuted[0];
out_update->nodes.resize(node_count);
out_update->nodes[0].id = permuted[0];
if (node_count > 1) {
out_update->nodes[0].child_ids.push_back(permuted[1]);
out_update->nodes[1].id = permuted[1];
}
// The remaining nodes are assigned based on their parent
// selected from the next bits from |tree_index|.
for (int i = 2; i < node_count; ++i) {
out_update->nodes[i].id = permuted[i];
int parent_index = (tree_index % i);
tree_index /= i;
out_update->nodes[parent_index].child_ids.push_back(permuted[i]);
}
}
} // namespace ui