#include "ui/accessibility/ax_tree_manager.h"
#include "base/lazy_instance.h"
#include "base/no_destructor.h"
#include "ui/accessibility/ax_common.h"
#include "ui/accessibility/ax_export.h"
#include "ui/accessibility/ax_node.h"
#include "ui/accessibility/ax_tree_id.h"
#include "ui/accessibility/ax_tree_manager_map.h"
#include "ui/accessibility/ax_tree_observer.h"
namespace ui {
namespace {
base::LazyInstance<base::RepeatingClosure>::DestructorAtExit
g_focus_change_callback_for_testing = LAZY_INSTANCE_INITIALIZER;
}
AXTreeManagerMap& AXTreeManager::GetMap() {
static base::NoDestructor<AXTreeManagerMap> map;
return *map;
}
AXTreeManager* AXTreeManager::FromID(const AXTreeID& ax_tree_id) {
return ax_tree_id != AXTreeIDUnknown() ? GetMap().GetManager(ax_tree_id)
: nullptr;
}
AXTreeManager* AXTreeManager::ForChildTree(const AXNode& parent_node) {
if (!parent_node.HasStringAttribute(
ax::mojom::StringAttribute::kChildTreeId)) {
return nullptr;
}
AXTreeID child_tree_id = AXTreeID::FromString(
parent_node.GetStringAttribute(ax::mojom::StringAttribute::kChildTreeId));
AXTreeManager* child_tree_manager = GetMap().GetManager(child_tree_id);
DCHECK(!child_tree_manager ||
!child_tree_manager->GetParentNodeFromParentTree() ||
child_tree_manager->GetParentNodeFromParentTree()->id() ==
parent_node.id());
return child_tree_manager;
}
void AXTreeManager::SetFocusChangeCallbackForTesting(
base::RepeatingClosure callback) {
g_focus_change_callback_for_testing.Get() = std::move(callback);
}
AXTreeManager::AXTreeManager()
: connected_to_parent_tree_node_(false),
ax_tree_(nullptr),
event_generator_(ax_tree()) {}
AXTreeManager::AXTreeManager(std::unique_ptr<AXTree> tree)
: connected_to_parent_tree_node_(false),
ax_tree_(std::move(tree)),
event_generator_(ax_tree()) {
DCHECK(ax_tree_);
if (HasValidTreeID()) {
GetMap().AddTreeManager(GetTreeID(), this);
}
tree_observation_.Observe(ax_tree());
}
void AXTreeManager::FireFocusEvent(AXNode* node) {
if (g_focus_change_callback_for_testing.Get())
g_focus_change_callback_for_testing.Get().Run();
}
AXNode* AXTreeManager::RetargetForEvents(AXNode* node,
RetargetEventType type) const {
return node;
}
bool AXTreeManager::CanFireEvents() const {
if (!HasValidTreeID()) {
return false;
}
AXTreeManager* root_manager = GetRootManager();
if (!root_manager)
return false;
const AXTreeManager* ancestor_manager = this;
while (!ancestor_manager->IsRoot()) {
AXNode* host_node = ancestor_manager->GetParentNodeFromParentTree();
if (!host_node)
return false;
ancestor_manager = host_node->GetManager();
}
return true;
}
AXNode* AXTreeManager::GetNodeFromTree(const AXTreeID& tree_id,
const AXNodeID node_id) const {
auto* manager = AXTreeManager::FromID(tree_id);
return manager ? manager->GetNode(node_id) : nullptr;
}
void AXTreeManager::Initialize(const ui::AXTreeUpdate& initial_tree) {
if (!ax_tree()->Unserialize(initial_tree)) {
LOG(FATAL) << "No recovery is possible if the initial tree is broken: "
<< ax_tree()->error();
}
}
AXNode* AXTreeManager::GetNode(const AXNodeID node_id) const {
return ax_tree_ ? ax_tree_->GetFromId(node_id) : nullptr;
}
const AXTreeData& AXTreeManager::GetTreeData() const {
return ax_tree_ ? ax_tree_->data() : AXTreeDataUnknown();
}
AXTreeID AXTreeManager::GetParentTreeID() const {
return ax_tree_ ? ax_tree_->data().parent_tree_id : AXTreeIDUnknown();
}
AXNode* AXTreeManager::GetRoot() const {
return ax_tree_ ? ax_tree_->root() : nullptr;
}
void AXTreeManager::WillBeRemovedFromMap() {
if (HasValidTreeID()) {
ax_tree_->NotifyTreeManagerWillBeRemoved(GetTreeID());
}
}
absl::optional<AXNodeID> AXTreeManager::last_focused_node_id_ = {};
absl::optional<AXTreeID> AXTreeManager::last_focused_node_tree_id_ = {};
void AXTreeManager::SetLastFocusedNode(AXNode* node) {
#if defined(AX_FAIL_FAST_BUILD)
static auto* const ax_crash_key_focus = base::debug::AllocateCrashKeyString(
"ax_focus", base::debug::CrashKeySize::Size256);
#endif
static auto* const ax_crash_key_focus_top_frame =
base::debug::AllocateCrashKeyString("ax_focus_top_frame",
base::debug::CrashKeySize::Size256);
static auto* const ax_crash_key_focus_frame =
base::debug::AllocateCrashKeyString("ax_focus_frame",
base::debug::CrashKeySize::Size256);
if (node) {
std::ostringstream node_info_focus, node_info_top_frame, node_info_frame;
#if defined(AX_FAIL_FAST_BUILD)
node_info_focus << node;
base::debug::SetCrashKeyString(ax_crash_key_focus, node_info_focus.str());
#endif
DCHECK(node->GetManager());
if (node->GetManager()->GetTreeID() != last_focused_node_tree_id_) {
if (node->GetManager() && node->GetManager()->GetRootManager()) {
node_info_top_frame << node->GetManager()->GetRootManager()->GetRoot();
base::debug::SetCrashKeyString(ax_crash_key_focus_top_frame,
node_info_top_frame.str());
if (!node->GetManager()->IsRoot()) {
node_info_frame << node->GetManager()->GetRoot();
base::debug::SetCrashKeyString(ax_crash_key_focus_frame,
node_info_frame.str());
}
}
}
last_focused_node_id_ = node->id();
last_focused_node_tree_id_ = node->GetManager()->GetTreeID();
DCHECK(last_focused_node_tree_id_);
DCHECK(last_focused_node_tree_id_ != AXTreeIDUnknown());
} else {
#if defined(AX_FAIL_FAST_BUILD)
base::debug::ClearCrashKeyString(ax_crash_key_focus);
#endif
base::debug::ClearCrashKeyString(ax_crash_key_focus_top_frame);
base::debug::ClearCrashKeyString(ax_crash_key_focus_frame);
last_focused_node_id_.reset();
last_focused_node_tree_id_.reset();
}
}
AXNode* AXTreeManager::GetLastFocusedNode() {
if (last_focused_node_id_) {
DCHECK(last_focused_node_tree_id_);
DCHECK(last_focused_node_tree_id_ != AXTreeIDUnknown());
if (AXTreeManager* last_focused_manager =
FromID(last_focused_node_tree_id_.value())) {
return last_focused_manager->GetNode(last_focused_node_id_.value());
}
}
return nullptr;
}
AXTreeManager::~AXTreeManager() {
AXNode* parent = nullptr;
if (connected_to_parent_tree_node_)
parent = GetParentNodeFromParentTree();
if (ax_tree_)
ax_tree_->Destroy();
CleanUp();
event_generator_.ReleaseTree();
if (HasValidTreeID()) {
GetMap().RemoveTreeManager(GetTreeID());
if (last_focused_node_tree_id_ &&
GetTreeID() == *last_focused_node_tree_id_) {
SetLastFocusedNode(nullptr);
}
}
ParentConnectionChanged(parent);
}
void AXTreeManager::OnTreeDataChanged(AXTree* tree,
const AXTreeData& old_data,
const AXTreeData& new_data) {
DCHECK_NE(ax_tree(), nullptr);
DCHECK_EQ(ax_tree(), tree);
DCHECK_EQ(GetTreeID(), new_data.tree_id);
if (new_data.tree_id == old_data.tree_id) {
return;
}
connected_to_parent_tree_node_ = false;
if (last_focused_node_tree_id_ != AXTreeIDUnknown() &&
last_focused_node_tree_id_ == old_data.tree_id) {
SetLastFocusedNode(nullptr);
}
if (old_data.tree_id != AXTreeIDUnknown()) {
GetMap().RemoveTreeManager(old_data.tree_id);
}
if (new_data.tree_id != AXTreeIDUnknown()) {
GetMap().AddTreeManager(GetTreeID(), this);
}
}
void AXTreeManager::OnNodeWillBeDeleted(AXTree* tree, AXNode* node) {
DCHECK(node);
if (node == GetLastFocusedNode())
SetLastFocusedNode(nullptr);
if (node->GetRole() == ax::mojom::Role::kMenu)
FireGeneratedEvent(AXEventGenerator::Event::MENU_POPUP_END, node);
}
void AXTreeManager::OnAtomicUpdateFinished(
AXTree* tree,
bool root_changed,
const std::vector<AXTreeObserver::Change>& changes) {
DCHECK_EQ(ax_tree(), tree);
if (root_changed)
connected_to_parent_tree_node_ = false;
}
AXTreeManager* AXTreeManager::GetParentManager() const {
AXTreeID parent_tree_id = GetParentTreeID();
if (parent_tree_id == AXTreeIDUnknown()) {
return nullptr;
}
return FromID(parent_tree_id);
}
bool AXTreeManager::IsRoot() const {
return GetParentTreeID() == AXTreeIDUnknown();
}
AXTreeManager* AXTreeManager::GetRootManager() const {
if (IsRoot())
return const_cast<AXTreeManager*>(this);
AXTreeManager* parent = GetParentManager();
if (!parent) {
return nullptr;
}
return parent->GetRootManager();
}
AXNode* AXTreeManager::GetParentNodeFromParentTree() const {
AXTreeManager* parent_manager = GetParentManager();
if (!parent_manager)
return nullptr;
DCHECK(GetRoot());
std::set<int32_t> host_node_ids =
parent_manager->ax_tree()->GetNodeIdsForChildTreeId(GetTreeID());
if (host_node_ids.empty()) {
return nullptr;
}
CHECK_EQ(host_node_ids.size(), 1U)
<< "Multiple nodes cannot claim the same child tree ID.";
AXNode* parent_node = parent_manager->GetNode(*(host_node_ids.begin()));
DCHECK(parent_node);
DCHECK_EQ(GetTreeID(), AXTreeID::FromString(parent_node->GetStringAttribute(
ax::mojom::StringAttribute::kChildTreeId)))
<< "A node that hosts a child tree should expose its tree ID in its "
"`kChildTreeId` attribute.";
return parent_node;
}
void AXTreeManager::ParentConnectionChanged(AXNode* parent) {
if (!parent) {
connected_to_parent_tree_node_ = false;
return;
}
connected_to_parent_tree_node_ = true;
parent->tree()->NotifyChildTreeConnectionChanged(parent, ax_tree_.get());
UpdateAttributesOnParent(parent);
AXTreeManager* parent_manager = parent->GetManager();
parent = parent_manager->RetargetForEvents(
parent, RetargetEventType::RetargetEventTypeGenerated);
DCHECK(parent) << "RetargetForEvents shouldn't return a "
"null pointer when |parent| is not null.";
parent_manager->FireGeneratedEvent(
ui::AXEventGenerator::Event::CHILDREN_CHANGED, parent);
}
void AXTreeManager::EnsureParentConnectionIfNotRootManager() {
AXNode* parent = GetParentNodeFromParentTree();
if (parent) {
if (!connected_to_parent_tree_node_)
ParentConnectionChanged(parent);
SANITIZER_CHECK(!IsRoot());
return;
}
if (connected_to_parent_tree_node_) {
connected_to_parent_tree_node_ = false;
DCHECK(IsRoot() || !CanFireEvents());
}
}
}