blob: 077eb10b89f7f376cbe3a795416065e7cbaa1a14 [file] [log] [blame]
// Copyright (c) 2012 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 "content/browser/accessibility/accessibility_tree_formatter.h"
#include <math.h>
#include <oleacc.h>
#include <stddef.h>
#include <stdint.h>
#include <wrl/client.h>
#include <iostream>
#include <string>
#include <utility>
#include "base/command_line.h"
#include "base/files/file_path.h"
#include "base/strings/string_number_conversions.h"
#include "base/strings/string_piece.h"
#include "base/strings/string_util.h"
#include "base/strings/stringprintf.h"
#include "base/strings/utf_string_conversions.h"
#include "base/values.h"
#include "base/win/com_init_util.h"
#include "base/win/scoped_bstr.h"
#include "base/win/scoped_variant.h"
#include "content/browser/accessibility/accessibility_tree_formatter_blink.h"
#include "content/browser/accessibility/accessibility_tree_formatter_uia_win.h"
#include "content/browser/accessibility/accessibility_tree_formatter_utils_win.h"
#include "content/browser/accessibility/browser_accessibility_manager.h"
#include "content/browser/accessibility/browser_accessibility_win.h"
#include "third_party/iaccessible2/ia2_api_all.h"
#include "ui/accessibility/accessibility_switches.h"
#include "ui/base/win/atl_module.h"
#include "ui/gfx/win/hwnd_util.h"
namespace content {
class AccessibilityTreeFormatterWin : public AccessibilityTreeFormatter {
public:
AccessibilityTreeFormatterWin();
~AccessibilityTreeFormatterWin() override;
std::unique_ptr<base::DictionaryValue> BuildAccessibilityTree(
BrowserAccessibility* start) override;
std::unique_ptr<base::DictionaryValue> BuildAccessibilityTreeForWindow(
gfx::AcceleratedWidget hwnd) override;
std::unique_ptr<base::DictionaryValue> BuildAccessibilityTreeForProcess(
base::ProcessId pid) override;
std::unique_ptr<base::DictionaryValue> BuildAccessibilityTreeForPattern(
const base::StringPiece& pattern) override;
std::unique_ptr<base::DictionaryValue> BuildAccessibilityTree(
Microsoft::WRL::ComPtr<IAccessible> start,
LONG window_x = 0,
LONG window_y = 0);
static void SetUpCommandLineForTestPass(base::CommandLine* command_line);
void AddDefaultFilters(
std::vector<PropertyFilter>* property_filters) override;
private:
void RecursiveBuildAccessibilityTree(
const Microsoft::WRL::ComPtr<IAccessible> node,
base::DictionaryValue* dict,
LONG root_x,
LONG root_y);
const base::FilePath::StringType GetExpectedFileSuffix() override;
const std::string GetAllowEmptyString() override;
const std::string GetAllowString() override;
const std::string GetDenyString() override;
const std::string GetDenyNodeString() override;
void AddProperties(const Microsoft::WRL::ComPtr<IAccessible>,
base::DictionaryValue* dict,
LONG root_x,
LONG root_y);
void AddMSAAProperties(const Microsoft::WRL::ComPtr<IAccessible>,
base::DictionaryValue* dict,
LONG root_x,
LONG root_y);
void AddSimpleDOMNodeProperties(const Microsoft::WRL::ComPtr<IAccessible>,
base::DictionaryValue* dict);
bool AddIA2Properties(const Microsoft::WRL::ComPtr<IAccessible>,
base::DictionaryValue* dict);
void AddIA2ActionProperties(const Microsoft::WRL::ComPtr<IAccessible>,
base::DictionaryValue* dict);
void AddIA2HypertextProperties(const Microsoft::WRL::ComPtr<IAccessible>,
base::DictionaryValue* dict);
void AddIA2TextProperties(const Microsoft::WRL::ComPtr<IAccessible>,
base::DictionaryValue* dict);
void AddIA2TableProperties(const Microsoft::WRL::ComPtr<IAccessible>,
base::DictionaryValue* dict);
void AddIA2TableCellProperties(const Microsoft::WRL::ComPtr<IAccessible>,
base::DictionaryValue* dict);
void AddIA2ValueProperties(const Microsoft::WRL::ComPtr<IAccessible>,
base::DictionaryValue* dict);
base::string16 ProcessTreeForOutput(
const base::DictionaryValue& node,
base::DictionaryValue* filtered_dict_result = nullptr) override;
};
// static
std::unique_ptr<AccessibilityTreeFormatter>
AccessibilityTreeFormatter::Create() {
base::win::AssertComInitialized();
return std::make_unique<AccessibilityTreeFormatterWin>();
}
// static
std::vector<AccessibilityTreeFormatter::TestPass>
AccessibilityTreeFormatter::GetTestPasses() {
// In addition to the 'Blink' pass, Windows includes two accessibility APIs
// that need to be tested independently (MSAA & UIA).
return {
{"blink", &AccessibilityTreeFormatterBlink::CreateBlink, nullptr},
{"win", &AccessibilityTreeFormatter::Create,
&AccessibilityTreeFormatterWin::SetUpCommandLineForTestPass},
{"uia", &AccessibilityTreeFormatterUia::CreateUia,
&AccessibilityTreeFormatterUia::SetUpCommandLineForTestPass},
};
}
void AccessibilityTreeFormatterWin::SetUpCommandLineForTestPass(
base::CommandLine* command_line) {
command_line->RemoveSwitch(::switches::kEnableExperimentalUIAutomation);
}
void AccessibilityTreeFormatterWin::AddDefaultFilters(
std::vector<PropertyFilter>* property_filters) {
// Too noisy: HOTTRACKED, LINKED, SELECTABLE, IA2_STATE_EDITABLE,
// IA2_STATE_OPAQUE, IA2_STATE_SELECTAbLE_TEXT,
// IA2_STATE_SINGLE_LINE, IA2_STATE_VERTICAL.
// Too unpredictiable: OFFSCREEN
// Windows states to log by default:
AddPropertyFilter(property_filters, "ALERT*");
AddPropertyFilter(property_filters, "ANIMATED*");
AddPropertyFilter(property_filters, "BUSY");
AddPropertyFilter(property_filters, "CHECKED");
AddPropertyFilter(property_filters, "COLLAPSED");
AddPropertyFilter(property_filters, "EXPANDED");
AddPropertyFilter(property_filters, "FLOATING");
AddPropertyFilter(property_filters, "FOCUSABLE");
AddPropertyFilter(property_filters, "HASPOPUP");
AddPropertyFilter(property_filters, "INVISIBLE");
AddPropertyFilter(property_filters, "MARQUEED");
AddPropertyFilter(property_filters, "MIXED");
AddPropertyFilter(property_filters, "MOVEABLE");
AddPropertyFilter(property_filters, "MULTISELECTABLE");
AddPropertyFilter(property_filters, "PRESSED");
AddPropertyFilter(property_filters, "PROTECTED");
AddPropertyFilter(property_filters, "READONLY");
AddPropertyFilter(property_filters, "SELECTED");
AddPropertyFilter(property_filters, "SIZEABLE");
AddPropertyFilter(property_filters, "TRAVERSED");
AddPropertyFilter(property_filters, "UNAVAILABLE");
AddPropertyFilter(property_filters, "IA2_STATE_ACTIVE");
AddPropertyFilter(property_filters, "IA2_STATE_ARMED");
AddPropertyFilter(property_filters, "IA2_STATE_CHECKABLE");
AddPropertyFilter(property_filters, "IA2_STATE_DEFUNCT");
AddPropertyFilter(property_filters, "IA2_STATE_HORIZONTAL");
AddPropertyFilter(property_filters, "IA2_STATE_ICONIFIED");
AddPropertyFilter(property_filters, "IA2_STATE_INVALID_ENTRY");
AddPropertyFilter(property_filters, "IA2_STATE_MODAL");
AddPropertyFilter(property_filters, "IA2_STATE_MULTI_LINE");
AddPropertyFilter(property_filters, "IA2_STATE_PINNED");
AddPropertyFilter(property_filters, "IA2_STATE_REQUIRED");
AddPropertyFilter(property_filters, "IA2_STATE_STALE");
AddPropertyFilter(property_filters, "IA2_STATE_TRANSIENT");
// Reduce flakiness.
AddPropertyFilter(property_filters, "FOCUSED", PropertyFilter::DENY);
AddPropertyFilter(property_filters, "HOTTRACKED", PropertyFilter::DENY);
AddPropertyFilter(property_filters, "OFFSCREEN", PropertyFilter::DENY);
}
AccessibilityTreeFormatterWin::AccessibilityTreeFormatterWin() {
ui::win::CreateATLModuleIfNeeded();
}
AccessibilityTreeFormatterWin::~AccessibilityTreeFormatterWin() {}
static HRESULT QuerySimpleDOMNode(IAccessible* accessible,
ISimpleDOMNode** simple_dom_node) {
// IA2 Spec dictates that IServiceProvider should be used instead of
// QueryInterface when retrieving IAccessible2.
Microsoft::WRL::ComPtr<IServiceProvider> service_provider;
HRESULT hr = accessible->QueryInterface(service_provider.GetAddressOf());
if (FAILED(hr))
return hr;
return service_provider->QueryService(IID_ISimpleDOMNode, simple_dom_node);
}
static HRESULT QueryIAccessible2(IAccessible* accessible,
IAccessible2** accessible2) {
// IA2 Spec dictates that IServiceProvider should be used instead of
// QueryInterface when retrieving IAccessible2.
Microsoft::WRL::ComPtr<IServiceProvider> service_provider;
HRESULT hr = accessible->QueryInterface(service_provider.GetAddressOf());
if (FAILED(hr))
return hr;
return service_provider->QueryService(IID_IAccessible2, accessible2);
}
static HRESULT QueryIAccessibleAction(IAccessible* accessible,
IAccessibleAction** accessibleAction) {
// IA2 Spec dictates that IServiceProvider should be used instead of
// QueryInterface when retrieving alternate interfaces.
Microsoft::WRL::ComPtr<IServiceProvider> service_provider;
HRESULT hr = accessible->QueryInterface(service_provider.GetAddressOf());
if (FAILED(hr))
return hr;
return service_provider->QueryService(IID_IAccessibleAction,
accessibleAction);
}
static HRESULT QueryIAccessibleHypertext(
IAccessible* accessible,
IAccessibleHypertext** accessibleHypertext) {
// IA2 Spec dictates that IServiceProvider should be used instead of
// QueryInterface when retrieving alternate interfaces.
Microsoft::WRL::ComPtr<IServiceProvider> service_provider;
HRESULT hr = accessible->QueryInterface(service_provider.GetAddressOf());
if (FAILED(hr))
return hr;
return service_provider->QueryService(IID_IAccessibleHypertext,
accessibleHypertext);
}
static HRESULT QueryIAccessibleTable(IAccessible* accessible,
IAccessibleTable** accessibleTable) {
// IA2 Spec dictates that IServiceProvider should be used instead of
// QueryInterface when retrieving alternate interfaces.
Microsoft::WRL::ComPtr<IServiceProvider> service_provider;
HRESULT hr = accessible->QueryInterface(service_provider.GetAddressOf());
if (FAILED(hr))
return hr;
return service_provider->QueryService(IID_IAccessibleTable, accessibleTable);
}
static HRESULT QueryIAccessibleTableCell(
IAccessible* accessible,
IAccessibleTableCell** accessibleTableCell) {
// IA2 Spec dictates that IServiceProvider should be used instead of
// QueryInterface when retrieving alternate interfaces.
Microsoft::WRL::ComPtr<IServiceProvider> service_provider;
HRESULT hr = accessible->QueryInterface(service_provider.GetAddressOf());
if (FAILED(hr))
return hr;
return service_provider->QueryService(IID_IAccessibleTableCell,
accessibleTableCell);
}
static HRESULT QueryIAccessibleText(IAccessible* accessible,
IAccessibleText** accessibleText) {
// IA2 Spec dictates that IServiceProvider should be used instead of
// QueryInterface when retrieving alternate interfaces.
Microsoft::WRL::ComPtr<IServiceProvider> service_provider;
HRESULT hr = accessible->QueryInterface(service_provider.GetAddressOf());
if (FAILED(hr))
return hr;
return service_provider->QueryService(IID_IAccessibleText, accessibleText);
}
static HRESULT QueryIAccessibleValue(IAccessible* accessible,
IAccessibleValue** accessibleValue) {
// IA2 Spec dictates that IServiceProvider should be used instead of
// QueryInterface when retrieving alternate interfaces.
Microsoft::WRL::ComPtr<IServiceProvider> service_provider;
HRESULT hr = accessible->QueryInterface(service_provider.GetAddressOf());
if (FAILED(hr))
return hr;
return service_provider->QueryService(IID_IAccessibleValue, accessibleValue);
}
std::unique_ptr<base::DictionaryValue>
AccessibilityTreeFormatterWin::BuildAccessibilityTree(
BrowserAccessibility* start_node) {
DCHECK(start_node);
base::win::ScopedVariant variant_self(CHILDID_SELF);
LONG root_x, root_y, root_width, root_height;
BrowserAccessibility* root =
start_node->manager()->GetRootManager()->GetRoot();
HRESULT hr = ToBrowserAccessibilityWin(root)->GetCOM()->accLocation(
&root_x, &root_y, &root_width, &root_height, variant_self);
DCHECK(SUCCEEDED(hr));
Microsoft::WRL::ComPtr<IAccessible> start_ia =
ToBrowserAccessibilityComWin(start_node);
return BuildAccessibilityTree(start_ia, root_x, root_y);
}
std::unique_ptr<base::DictionaryValue>
AccessibilityTreeFormatterWin::BuildAccessibilityTree(
Microsoft::WRL::ComPtr<IAccessible> start,
LONG root_x,
LONG root_y) {
CHECK(start);
std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue);
RecursiveBuildAccessibilityTree(start, dict.get(), root_x, root_y);
return dict;
}
std::unique_ptr<base::DictionaryValue>
AccessibilityTreeFormatterWin::BuildAccessibilityTreeForWindow(
gfx::AcceleratedWidget hwnd) {
if (!hwnd)
return nullptr;
// Get IAccessible* for window
Microsoft::WRL::ComPtr<IAccessible> start;
HRESULT hr = ::AccessibleObjectFromWindow(
hwnd, static_cast<DWORD>(OBJID_CLIENT), IID_PPV_ARGS(&start));
if (FAILED(hr))
return nullptr;
auto dict(std::make_unique<base::DictionaryValue>());
RecursiveBuildAccessibilityTree(start, dict.get(), 0, 0);
return dict;
}
std::unique_ptr<base::DictionaryValue>
AccessibilityTreeFormatterWin::BuildAccessibilityTreeForProcess(
base::ProcessId pid) {
// Get HWND for process id.
HWND hwnd = GetHwndForProcess(pid);
return BuildAccessibilityTreeForWindow(hwnd);
}
std::unique_ptr<base::DictionaryValue>
AccessibilityTreeFormatterWin::BuildAccessibilityTreeForPattern(
const base::StringPiece& pattern) {
LOG(ERROR) << "Windows does not yet support building accessibility trees for "
"patterns";
return nullptr;
}
void AccessibilityTreeFormatterWin::RecursiveBuildAccessibilityTree(
const Microsoft::WRL::ComPtr<IAccessible> node,
base::DictionaryValue* dict,
LONG root_x,
LONG root_y) {
AddProperties(node, dict, root_x, root_y);
auto children = std::make_unique<base::ListValue>();
LONG child_count;
if (S_OK != node->get_accChildCount(&child_count))
return;
std::unique_ptr<VARIANT[]> children_array(new VARIANT[child_count]);
LONG obtained_count = 0;
HRESULT hr = AccessibleChildren(node.Get(), 0, child_count,
children_array.get(), &obtained_count);
if (hr != S_OK)
return;
for (LONG index = 0; index < obtained_count; index++) {
base::win::ScopedVariant child_variant;
child_variant.Reset(
children_array[index]); // Sets without adding another reference.
std::unique_ptr<base::DictionaryValue> child_dict(
new base::DictionaryValue);
Microsoft::WRL::ComPtr<IDispatch> dispatch;
if (child_variant.type() == VT_DISPATCH) {
dispatch = V_DISPATCH(child_variant.ptr());
} else if (child_variant.type() == VT_I4) {
hr = node->get_accChild(child_variant, dispatch.GetAddressOf());
if (FAILED(hr)) {
child_dict->SetString("error",
base::ASCIIToUTF16("[Error retrieving child]"));
} else if (!dispatch) {
// Partial child does not have its own object.
// Add minimal info -- role and name.
base::win::ScopedVariant role_variant;
if (SUCCEEDED(
node->get_accRole(child_variant, role_variant.Receive()))) {
if (role_variant.type() == VT_I4) {
child_dict->SetString("role",
base::ASCIIToUTF16(" [partial child]"));
}
}
base::win::ScopedBstr temp_bstr;
if (S_OK == node->get_accName(child_variant, temp_bstr.Receive())) {
base::string16 name = base::string16(temp_bstr, temp_bstr.Length());
child_dict->SetString("name", name);
}
}
} else {
child_dict->SetString("error",
base::ASCIIToUTF16("[Unknown child type]"));
}
if (dispatch) {
Microsoft::WRL::ComPtr<IAccessible> accessible;
if (SUCCEEDED(dispatch.As(&accessible)))
RecursiveBuildAccessibilityTree(accessible, child_dict.get(), root_x,
root_y);
}
children->Append(std::move(child_dict));
}
dict->Set(kChildrenDictAttr, std::move(children));
}
const char* const ALL_ATTRIBUTES[] = {
"name",
"parent",
"window_class",
"value",
"states",
"attributes",
"text_attributes",
"ia2_hypertext",
"currentValue",
"minimumValue",
"maximumValue",
"description",
"default_action",
"action_name",
"keyboard_shortcut",
"location",
"size",
"index_in_parent",
"n_relations",
"group_level",
"similar_items_in_group",
"position_in_group",
"table_rows",
"table_columns",
"row_index",
"column_index",
"row_headers",
"column_headers",
"n_characters",
"caret_offset",
"n_selections",
"selection_start",
"selection_end",
"localized_extended_role",
"inner_html",
};
void AccessibilityTreeFormatterWin::AddProperties(
const Microsoft::WRL::ComPtr<IAccessible> node,
base::DictionaryValue* dict,
LONG root_x,
LONG root_y) {
AddMSAAProperties(node, dict, root_x, root_y);
AddSimpleDOMNodeProperties(node, dict);
if (AddIA2Properties(node, dict)) {
AddIA2ActionProperties(node, dict);
AddIA2HypertextProperties(node, dict);
AddIA2TableProperties(node, dict);
AddIA2TableCellProperties(node, dict);
AddIA2TextProperties(node, dict);
AddIA2ValueProperties(node, dict);
}
}
base::string16 RoleVariantToString(const base::win::ScopedVariant& role) {
if (role.type() == VT_I4) {
return IAccessible2RoleToString(V_I4(role.ptr()));
} else if (role.type() == VT_BSTR) {
BSTR bstr_role = V_BSTR(role.ptr());
return base::string16(bstr_role, SysStringLen(bstr_role));
}
return base::string16();
}
void AccessibilityTreeFormatterWin::AddMSAAProperties(
const Microsoft::WRL::ComPtr<IAccessible> node,
base::DictionaryValue* dict,
LONG root_x,
LONG root_y) {
base::win::ScopedVariant variant_self(CHILDID_SELF);
base::win::ScopedBstr temp_bstr;
base::win::ScopedVariant ia_role_variant;
LONG ia_role = 0;
if (SUCCEEDED(node->get_accRole(variant_self, ia_role_variant.Receive()))) {
dict->SetString("role", RoleVariantToString(ia_role_variant));
ia_role = V_I4(ia_role_variant.ptr());
}
// If S_FALSE it means there is no name
if (S_OK == node->get_accName(variant_self, temp_bstr.Receive())) {
base::string16 name = base::string16(temp_bstr, temp_bstr.Length());
dict->SetString("name", name);
}
temp_bstr.Reset();
Microsoft::WRL::ComPtr<IDispatch> parent_dispatch;
if (SUCCEEDED(node->get_accParent(parent_dispatch.GetAddressOf()))) {
Microsoft::WRL::ComPtr<IAccessible> parent_accessible;
if (!parent_dispatch) {
dict->SetString("parent", "[null]");
} else if (SUCCEEDED(
parent_dispatch.CopyTo(parent_accessible.GetAddressOf()))) {
base::win::ScopedVariant parent_ia_role_variant;
if (SUCCEEDED(parent_accessible->get_accRole(
variant_self, parent_ia_role_variant.Receive())))
dict->SetString("parent", RoleVariantToString(parent_ia_role_variant));
else
dict->SetString("parent", L"[Error retrieving role from parent]");
} else {
dict->SetString("parent", L"[Error getting IAccessible* for parent]");
}
} else {
dict->SetString("parent", L"[Error retrieving parent]");
}
HWND hwnd;
if (SUCCEEDED(::WindowFromAccessibleObject(node.Get(), &hwnd)) && hwnd) {
dict->SetString("window_class", gfx::GetClassName(hwnd));
} else {
// This method is implemented by oleacc.dll and uses get_accParent,
// therefore it Will fail if get_accParent from root fails.
dict->SetString("window_class", L"[Error]");
}
if (SUCCEEDED(node->get_accValue(variant_self, temp_bstr.Receive())))
dict->SetString("value", base::string16(temp_bstr, temp_bstr.Length()));
temp_bstr.Reset();
int32_t ia_state = 0;
base::win::ScopedVariant ia_state_variant;
if (node->get_accState(variant_self, ia_state_variant.Receive()) == S_OK &&
ia_state_variant.type() == VT_I4) {
ia_state = ia_state_variant.ptr()->intVal;
std::vector<base::string16> state_strings;
IAccessibleStateToStringVector(ia_state, &state_strings);
std::unique_ptr<base::ListValue> states(new base::ListValue());
states->AppendStrings(state_strings);
dict->Set("states", std::move(states));
}
if (SUCCEEDED(node->get_accDescription(variant_self, temp_bstr.Receive()))) {
dict->SetString("description",
base::string16(temp_bstr, temp_bstr.Length()));
}
temp_bstr.Reset();
// |get_accDefaultAction| returns a localized string.
if (SUCCEEDED(
node->get_accDefaultAction(variant_self, temp_bstr.Receive()))) {
dict->SetString("default_action",
base::string16(temp_bstr, temp_bstr.Length()));
}
temp_bstr.Reset();
if (SUCCEEDED(
node->get_accKeyboardShortcut(variant_self, temp_bstr.Receive()))) {
dict->SetString("keyboard_shortcut",
base::string16(temp_bstr, temp_bstr.Length()));
}
temp_bstr.Reset();
if (SUCCEEDED(node->get_accHelp(variant_self, temp_bstr.Receive())))
dict->SetString("help", base::string16(temp_bstr, temp_bstr.Length()));
temp_bstr.Reset();
LONG x, y, width, height;
if (SUCCEEDED(node->accLocation(&x, &y, &width, &height, variant_self))) {
auto location = std::make_unique<base::DictionaryValue>();
location->SetInteger("x", x - root_x);
location->SetInteger("y", y - root_y);
dict->Set("location", std::move(location));
auto size = std::make_unique<base::DictionaryValue>();
size->SetInteger("width", width);
size->SetInteger("height", height);
dict->Set("size", std::move(size));
}
}
void AccessibilityTreeFormatterWin::AddSimpleDOMNodeProperties(
const Microsoft::WRL::ComPtr<IAccessible> node,
base::DictionaryValue* dict) {
Microsoft::WRL::ComPtr<ISimpleDOMNode> simple_dom_node;
if (S_OK != QuerySimpleDOMNode(node.Get(), simple_dom_node.GetAddressOf()))
return; // No IA2Value, we are finished with this node.
base::win::ScopedBstr temp_bstr;
if (SUCCEEDED(simple_dom_node->get_innerHTML(temp_bstr.Receive()))) {
dict->SetString("inner_html",
base::string16(temp_bstr, temp_bstr.Length()));
}
temp_bstr.Reset();
}
bool AccessibilityTreeFormatterWin::AddIA2Properties(
const Microsoft::WRL::ComPtr<IAccessible> node,
base::DictionaryValue* dict) {
Microsoft::WRL::ComPtr<IAccessible2> ia2;
if (S_OK != QueryIAccessible2(node.Get(), ia2.GetAddressOf()))
return false; // No IA2, we are finished with this node.
LONG ia2_role = 0;
if (SUCCEEDED(ia2->role(&ia2_role))) {
std::string legacy_role;
dict->GetString("role", &legacy_role);
dict->SetString("msaa_legacy_role", legacy_role);
// Overwrite MSAA role which is more limited.
dict->SetString("role", IAccessible2RoleToString(ia2_role));
}
std::vector<base::string16> state_strings;
AccessibleStates states;
if (ia2->get_states(&states) == S_OK) {
IAccessible2StateToStringVector(states, &state_strings);
// Append IA2 state list to MSAA state
base::ListValue* states_list;
if (dict->GetList("states", &states_list))
states_list->AppendStrings(state_strings);
}
base::win::ScopedBstr temp_bstr;
if (ia2->get_attributes(temp_bstr.Receive()) == S_OK) {
// get_attributes() returns a semicolon delimited string. Turn it into a
// ListValue
std::vector<base::string16> ia2_attributes = base::SplitString(
base::string16(temp_bstr, temp_bstr.Length()), base::string16(1, ';'),
base::KEEP_WHITESPACE, base::SPLIT_WANT_ALL);
std::unique_ptr<base::ListValue> attributes(new base::ListValue());
attributes->AppendStrings(ia2_attributes);
dict->Set("attributes", std::move(attributes));
}
temp_bstr.Reset();
LONG index_in_parent;
if (SUCCEEDED(ia2->get_indexInParent(&index_in_parent)))
dict->SetInteger("index_in_parent", index_in_parent);
LONG n_relations;
if (SUCCEEDED(ia2->get_nRelations(&n_relations)))
dict->SetInteger("n_relations", n_relations);
LONG group_level, similar_items_in_group, position_in_group;
// |GetGroupPosition| returns S_FALSE when no grouping information is
// available so avoid using |SUCCEEDED|.
if (ia2->get_groupPosition(&group_level, &similar_items_in_group,
&position_in_group) == S_OK) {
dict->SetInteger("group_level", group_level);
dict->SetInteger("similar_items_in_group", similar_items_in_group);
dict->SetInteger("position_in_group", position_in_group);
}
if (SUCCEEDED(ia2->get_localizedExtendedRole(temp_bstr.Receive()))) {
dict->SetString("localized_extended_role",
base::string16(temp_bstr, temp_bstr.Length()));
}
temp_bstr.Reset();
return true;
}
void AccessibilityTreeFormatterWin::AddIA2ActionProperties(
const Microsoft::WRL::ComPtr<IAccessible> node,
base::DictionaryValue* dict) {
Microsoft::WRL::ComPtr<IAccessibleAction> ia2action;
if (S_OK != QueryIAccessibleAction(node.Get(), ia2action.GetAddressOf()))
return; // No IA2Value, we are finished with this node.
base::win::ScopedBstr temp_bstr;
// |IAccessibleAction::get_name| returns a localized string.
if (SUCCEEDED(
ia2action->get_name(0 /* action_index */, temp_bstr.Receive()))) {
dict->SetString("action_name",
base::string16(temp_bstr, temp_bstr.Length()));
}
}
void AccessibilityTreeFormatterWin::AddIA2HypertextProperties(
Microsoft::WRL::ComPtr<IAccessible> node,
base::DictionaryValue* dict) {
Microsoft::WRL::ComPtr<IAccessibleHypertext> ia2hyper;
if (S_OK != QueryIAccessibleHypertext(node.Get(), ia2hyper.GetAddressOf()))
return; // No IA2, we are finished with this node
base::win::ScopedBstr text_bstr;
HRESULT hr;
hr = ia2hyper->get_text(0, IA2_TEXT_OFFSET_LENGTH, text_bstr.Receive());
if (FAILED(hr))
return;
base::string16 ia2_hypertext(text_bstr, text_bstr.Length());
// IA2 Spec calls embedded objects hyperlinks. We stick to embeds for clarity.
LONG number_of_embeds;
hr = ia2hyper->get_nHyperlinks(&number_of_embeds);
if (SUCCEEDED(hr) && number_of_embeds > 0) {
// Replace all embedded characters with the child indices of the
// accessibility objects they refer to.
base::string16 embedded_character(
1, BrowserAccessibilityComWin::kEmbeddedCharacter);
size_t character_index = 0;
size_t hypertext_index = 0;
while (hypertext_index < ia2_hypertext.length()) {
if (ia2_hypertext[hypertext_index] !=
BrowserAccessibilityComWin::kEmbeddedCharacter) {
++character_index;
++hypertext_index;
continue;
}
LONG index_of_embed;
hr = ia2hyper->get_hyperlinkIndex(character_index, &index_of_embed);
// S_FALSE will be returned if no embedded object is found at the given
// embedded character offset. Exclude child index from such cases.
LONG child_index = -1;
if (hr == S_OK) {
DCHECK_GE(index_of_embed, 0);
Microsoft::WRL::ComPtr<IAccessibleHyperlink> embedded_object;
hr = ia2hyper->get_hyperlink(index_of_embed,
embedded_object.GetAddressOf());
DCHECK(SUCCEEDED(hr));
Microsoft::WRL::ComPtr<IAccessible2> ax_embed;
hr = embedded_object.CopyTo(ax_embed.GetAddressOf());
DCHECK(SUCCEEDED(hr));
hr = ax_embed->get_indexInParent(&child_index);
DCHECK(SUCCEEDED(hr));
}
base::string16 child_index_str(L"<obj");
if (child_index >= 0) {
base::StringAppendF(&child_index_str, L"%d>", child_index);
} else {
base::StringAppendF(&child_index_str, L">");
}
base::ReplaceFirstSubstringAfterOffset(
&ia2_hypertext, hypertext_index, embedded_character, child_index_str);
++character_index;
hypertext_index += child_index_str.length();
--number_of_embeds;
}
}
DCHECK_EQ(number_of_embeds, 0);
dict->SetString("ia2_hypertext", ia2_hypertext);
}
void AccessibilityTreeFormatterWin::AddIA2TableProperties(
const Microsoft::WRL::ComPtr<IAccessible> node,
base::DictionaryValue* dict) {
Microsoft::WRL::ComPtr<IAccessibleTable> ia2table;
if (S_OK != QueryIAccessibleTable(node.Get(), ia2table.GetAddressOf()))
return; // No IA2Text, we are finished with this node.
LONG table_rows;
if (SUCCEEDED(ia2table->get_nRows(&table_rows)))
dict->SetInteger("table_rows", table_rows);
LONG table_columns;
if (SUCCEEDED(ia2table->get_nColumns(&table_columns)))
dict->SetInteger("table_columns", table_columns);
}
static base::string16 ProcessAccessiblesArray(IUnknown** accessibles,
LONG num_accessibles) {
base::string16 related_accessibles_string;
if (num_accessibles <= 0)
return related_accessibles_string;
base::win::ScopedVariant variant_self(CHILDID_SELF);
for (int index = 0; index < num_accessibles; index++) {
related_accessibles_string += index > 0 ? L"," : L"<";
Microsoft::WRL::ComPtr<IUnknown> unknown = accessibles[index];
Microsoft::WRL::ComPtr<IAccessible> accessible;
if (SUCCEEDED(unknown.CopyTo(accessible.GetAddressOf()))) {
base::win::ScopedBstr temp_bstr;
if (S_OK == accessible->get_accName(variant_self, temp_bstr.Receive()))
related_accessibles_string += temp_bstr;
else
related_accessibles_string += L"no name";
}
}
return related_accessibles_string + L">";
}
void AccessibilityTreeFormatterWin::AddIA2TableCellProperties(
const Microsoft::WRL::ComPtr<IAccessible> node,
base::DictionaryValue* dict) {
Microsoft::WRL::ComPtr<IAccessibleTableCell> ia2cell;
if (S_OK != QueryIAccessibleTableCell(node.Get(), ia2cell.GetAddressOf()))
return; // No IA2Text, we are finished with this node.
LONG n_row_header_cells;
IUnknown** row_headers;
if (SUCCEEDED(
ia2cell->get_rowHeaderCells(&row_headers, &n_row_header_cells)) &&
n_row_header_cells > 0) {
base::string16 accessibles_desc =
ProcessAccessiblesArray(row_headers, n_row_header_cells);
CoTaskMemFree(row_headers); // Free the array manually.
dict->SetString("row_headers", accessibles_desc);
}
LONG n_column_header_cells;
IUnknown** column_headers;
if (SUCCEEDED(ia2cell->get_columnHeaderCells(&column_headers,
&n_column_header_cells)) &&
n_column_header_cells > 0) {
base::string16 accessibles_desc =
ProcessAccessiblesArray(column_headers, n_column_header_cells);
CoTaskMemFree(column_headers); // Free the array manually.
dict->SetString("column_headers", accessibles_desc);
}
}
void AccessibilityTreeFormatterWin::AddIA2TextProperties(
const Microsoft::WRL::ComPtr<IAccessible> node,
base::DictionaryValue* dict) {
Microsoft::WRL::ComPtr<IAccessibleText> ia2text;
if (S_OK != QueryIAccessibleText(node.Get(), ia2text.GetAddressOf()))
return; // No IA2Text, we are finished with this node.
LONG n_characters;
if (SUCCEEDED(ia2text->get_nCharacters(&n_characters)))
dict->SetInteger("n_characters", n_characters);
LONG caret_offset;
if (ia2text->get_caretOffset(&caret_offset) == S_OK)
dict->SetInteger("caret_offset", caret_offset);
LONG n_selections;
if (SUCCEEDED(ia2text->get_nSelections(&n_selections))) {
dict->SetInteger("n_selections", n_selections);
if (n_selections > 0) {
LONG start, end;
if (SUCCEEDED(ia2text->get_selection(0, &start, &end))) {
dict->SetInteger("selection_start", start);
dict->SetInteger("selection_end", end);
}
}
}
// Handle IA2 text attributes, adding them as a list.
// IA2 text attributes comes formatted as a single string, as follows:
// https://wiki.linuxfoundation.org/accessibility/iaccessible2/textattributes
std::unique_ptr<base::ListValue> text_attributes(new base::ListValue());
LONG current_offset = 0, start_offset, end_offset;
while (current_offset < n_characters) {
// TODO(aleventhal) n_characters is not actually useful for ending the
// loop, because it counts embedded object characters as more than 1,
// meaning that it counts all the text in the subtree. However, the
// offsets used in other IAText methods only count the embedded object
// characters as 1.
base::win::ScopedBstr temp_bstr;
HRESULT hr = ia2text->get_attributes(current_offset, &start_offset,
&end_offset, temp_bstr.Receive());
// The below start_offset < current_offset check is needed because
// nCharacters is not helpful as described above.
// When asking for a range past the end of the string, this will occur,
// although it's not clear whether that's desired or whether
// S_FALSE or an error should be returned when the offset is out of range.
if (FAILED(hr) || start_offset < current_offset)
break;
// DCHECK(start_offset == current_offset); // Always at text range start.
if (hr == S_OK && temp_bstr && wcslen(temp_bstr)) {
// Append offset:<number>.
base::string16 offset_str =
base::ASCIIToUTF16("offset:") + base::NumberToString16(start_offset);
text_attributes->AppendString(offset_str);
// Append name:value pairs.
std::vector<base::string16> name_val_pairs =
SplitString(base::string16(temp_bstr), base::ASCIIToUTF16(";"),
base::KEEP_WHITESPACE, base::SPLIT_WANT_ALL);
text_attributes->AppendStrings(name_val_pairs);
}
current_offset = end_offset;
}
dict->Set("text_attributes", std::move(text_attributes));
}
void AccessibilityTreeFormatterWin::AddIA2ValueProperties(
const Microsoft::WRL::ComPtr<IAccessible> node,
base::DictionaryValue* dict) {
Microsoft::WRL::ComPtr<IAccessibleValue> ia2value;
if (S_OK != QueryIAccessibleValue(node.Get(), ia2value.GetAddressOf()))
return; // No IA2Value, we are finished with this node.
base::win::ScopedVariant current_value;
if (ia2value->get_currentValue(current_value.Receive()) == S_OK &&
isfinite(V_R8(current_value.ptr()))) {
dict->SetDouble("currentValue", V_R8(current_value.ptr()));
}
base::win::ScopedVariant minimum_value;
if (ia2value->get_minimumValue(minimum_value.Receive()) == S_OK &&
isfinite(V_R8(minimum_value.ptr()))) {
dict->SetDouble("minimumValue", V_R8(minimum_value.ptr()));
}
base::win::ScopedVariant maximum_value;
if (ia2value->get_maximumValue(maximum_value.Receive()) == S_OK &&
isfinite(V_R8(maximum_value.ptr()))) {
dict->SetDouble("maximumValue", V_R8(maximum_value.ptr()));
}
}
base::string16 AccessibilityTreeFormatterWin::ProcessTreeForOutput(
const base::DictionaryValue& dict,
base::DictionaryValue* filtered_dict_result) {
base::string16 line;
// Always show role, and show it first.
base::string16 role_value;
dict.GetString("role", &role_value);
WriteAttribute(true, base::UTF16ToUTF8(role_value), &line);
if (filtered_dict_result)
filtered_dict_result->SetString("role", role_value);
for (const char* attribute_name : ALL_ATTRIBUTES) {
const base::Value* value;
if (!dict.Get(attribute_name, &value))
continue;
switch (value->type()) {
case base::Value::Type::STRING: {
base::string16 string_value;
value->GetAsString(&string_value);
bool did_pass_filters = WriteAttribute(
false,
base::StringPrintf(L"%ls='%ls'",
base::UTF8ToUTF16(attribute_name).c_str(),
string_value.c_str()),
&line);
if (filtered_dict_result && did_pass_filters)
filtered_dict_result->SetString(attribute_name, string_value);
break;
}
case base::Value::Type::INTEGER: {
int int_value = 0;
value->GetAsInteger(&int_value);
bool did_pass_filters = WriteAttribute(
false,
base::StringPrintf(L"%ls=%d",
base::UTF8ToUTF16(attribute_name).c_str(),
int_value),
&line);
if (filtered_dict_result && did_pass_filters)
filtered_dict_result->SetInteger(attribute_name, int_value);
break;
}
case base::Value::Type::DOUBLE: {
double double_value = 0.0;
value->GetAsDouble(&double_value);
bool did_pass_filters = WriteAttribute(
false,
base::StringPrintf(L"%ls=%.2f",
base::UTF8ToUTF16(attribute_name).c_str(),
double_value),
&line);
if (filtered_dict_result && did_pass_filters)
filtered_dict_result->SetDouble(attribute_name, double_value);
break;
}
case base::Value::Type::LIST: {
// Currently all list values are string and are written without
// attribute names.
const base::ListValue* list_value;
value->GetAsList(&list_value);
std::unique_ptr<base::ListValue> filtered_list(new base::ListValue());
for (base::ListValue::const_iterator it = list_value->begin();
it != list_value->end(); ++it) {
base::string16 string_value;
if (it->GetAsString(&string_value))
if (WriteAttribute(false, string_value, &line))
filtered_list->AppendString(string_value);
}
if (filtered_dict_result && !filtered_list->empty())
filtered_dict_result->Set(attribute_name, std::move(filtered_list));
break;
}
case base::Value::Type::DICTIONARY: {
// Currently all dictionary values are coordinates.
// Revisit this if that changes.
const base::DictionaryValue* dict_value;
value->GetAsDictionary(&dict_value);
bool did_pass_filters = false;
if (strcmp(attribute_name, "size") == 0) {
did_pass_filters = WriteAttribute(
false, FormatCoordinates("size", "width", "height", *dict_value),
&line);
} else if (strcmp(attribute_name, "location") == 0) {
did_pass_filters = WriteAttribute(
false, FormatCoordinates("location", "x", "y", *dict_value),
&line);
}
if (filtered_dict_result && did_pass_filters)
filtered_dict_result->SetKey(attribute_name, dict_value->Clone());
break;
}
default:
NOTREACHED();
break;
}
}
return line;
}
const base::FilePath::StringType
AccessibilityTreeFormatterWin::GetExpectedFileSuffix() {
return FILE_PATH_LITERAL("-expected-win.txt");
}
const std::string AccessibilityTreeFormatterWin::GetAllowEmptyString() {
return "@WIN-ALLOW-EMPTY:";
}
const std::string AccessibilityTreeFormatterWin::GetAllowString() {
return "@WIN-ALLOW:";
}
const std::string AccessibilityTreeFormatterWin::GetDenyString() {
return "@WIN-DENY:";
}
const std::string AccessibilityTreeFormatterWin::GetDenyNodeString() {
return "@WIN-DENY-NODE:";
}
} // namespace content