2019-12-19 16:53:24 -08:00
|
|
|
/**
|
|
|
|
* Copyright 2018 Google Inc. All rights reserved.
|
|
|
|
* Modifications copyright (c) Microsoft Corporation.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2020-01-03 11:15:43 -08:00
|
|
|
import { CRSession } from './crConnection';
|
|
|
|
import { Protocol } from './protocol';
|
2020-08-24 06:51:51 -07:00
|
|
|
import * as dom from '../dom';
|
|
|
|
import * as accessibility from '../accessibility';
|
|
|
|
import * as types from '../types';
|
2020-01-03 11:15:43 -08:00
|
|
|
|
2020-02-07 13:36:49 -08:00
|
|
|
export async function getAccessibilityTree(client: CRSession, needle?: dom.ElementHandle): Promise<{tree: accessibility.AXNode, needle: accessibility.AXNode | null}> {
|
2021-09-27 18:58:08 +02:00
|
|
|
const { nodes } = await client.send('Accessibility.getFullAXTree');
|
2020-01-14 16:54:50 -08:00
|
|
|
const tree = CRAXNode.createTree(client, nodes);
|
|
|
|
return {
|
|
|
|
tree,
|
|
|
|
needle: needle ? await tree._findElement(needle) : null
|
|
|
|
};
|
2019-12-19 16:53:24 -08:00
|
|
|
}
|
|
|
|
|
2020-01-03 11:15:43 -08:00
|
|
|
class CRAXNode implements accessibility.AXNode {
|
2019-12-19 16:53:24 -08:00
|
|
|
_payload: Protocol.Accessibility.AXNode;
|
|
|
|
_children: CRAXNode[] = [];
|
|
|
|
private _richlyEditable = false;
|
|
|
|
private _editable = false;
|
|
|
|
private _focusable = false;
|
|
|
|
private _expanded = false;
|
|
|
|
private _hidden = false;
|
|
|
|
private _name: string;
|
|
|
|
private _role: string;
|
|
|
|
private _cachedHasFocusableChild: boolean | undefined;
|
2020-01-03 11:15:43 -08:00
|
|
|
private _client: CRSession;
|
2019-12-19 16:53:24 -08:00
|
|
|
|
2020-01-03 11:15:43 -08:00
|
|
|
constructor(client: CRSession, payload: Protocol.Accessibility.AXNode) {
|
|
|
|
this._client = client;
|
2019-12-19 16:53:24 -08:00
|
|
|
this._payload = payload;
|
|
|
|
|
|
|
|
this._name = this._payload.name ? this._payload.name.value : '';
|
|
|
|
this._role = this._payload.role ? this._payload.role.value : 'Unknown';
|
|
|
|
|
|
|
|
for (const property of this._payload.properties || []) {
|
|
|
|
if (property.name === 'editable') {
|
|
|
|
this._richlyEditable = property.value.value === 'richtext';
|
|
|
|
this._editable = true;
|
|
|
|
}
|
|
|
|
if (property.name === 'focusable')
|
|
|
|
this._focusable = property.value.value;
|
|
|
|
if (property.name === 'expanded')
|
|
|
|
this._expanded = property.value.value;
|
|
|
|
if (property.name === 'hidden')
|
|
|
|
this._hidden = property.value.value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private _isPlainTextField(): boolean {
|
|
|
|
if (this._richlyEditable)
|
|
|
|
return false;
|
|
|
|
if (this._editable)
|
|
|
|
return true;
|
|
|
|
return this._role === 'textbox' || this._role === 'ComboBox' || this._role === 'searchbox';
|
|
|
|
}
|
|
|
|
|
|
|
|
private _isTextOnlyObject(): boolean {
|
|
|
|
const role = this._role;
|
|
|
|
return (role === 'LineBreak' || role === 'text' ||
|
2021-04-26 12:02:54 -07:00
|
|
|
role === 'InlineTextBox' || role === 'StaticText');
|
2019-12-19 16:53:24 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
private _hasFocusableChild(): boolean {
|
|
|
|
if (this._cachedHasFocusableChild === undefined) {
|
|
|
|
this._cachedHasFocusableChild = false;
|
|
|
|
for (const child of this._children) {
|
|
|
|
if (child._focusable || child._hasFocusableChild()) {
|
|
|
|
this._cachedHasFocusableChild = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return this._cachedHasFocusableChild;
|
|
|
|
}
|
|
|
|
|
2020-01-03 11:15:43 -08:00
|
|
|
children() {
|
|
|
|
return this._children;
|
|
|
|
}
|
|
|
|
|
2020-01-14 16:54:50 -08:00
|
|
|
async _findElement(element: dom.ElementHandle): Promise<CRAXNode | null> {
|
2020-05-27 22:19:05 -07:00
|
|
|
const objectId = element._objectId;
|
2021-09-27 18:58:08 +02:00
|
|
|
const { node: { backendNodeId } } = await this._client.send('DOM.describeNode', { objectId });
|
2020-01-03 11:15:43 -08:00
|
|
|
const needle = this.find(node => node._payload.backendDOMNodeId === backendNodeId);
|
|
|
|
return needle || null;
|
|
|
|
}
|
|
|
|
|
2019-12-19 16:53:24 -08:00
|
|
|
find(predicate: (arg0: CRAXNode) => boolean): CRAXNode | null {
|
|
|
|
if (predicate(this))
|
|
|
|
return this;
|
|
|
|
for (const child of this._children) {
|
|
|
|
const result = child.find(predicate);
|
|
|
|
if (result)
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
isLeafNode(): boolean {
|
|
|
|
if (!this._children.length)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// These types of objects may have children that we use as internal
|
|
|
|
// implementation details, but we want to expose them as leaves to platform
|
|
|
|
// accessibility APIs because screen readers might be confused if they find
|
|
|
|
// any children.
|
|
|
|
if (this._isPlainTextField() || this._isTextOnlyObject())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Roles whose children are only presentational according to the ARIA and
|
|
|
|
// HTML5 Specs should be hidden from screen readers.
|
|
|
|
// (Note that whilst ARIA buttons can have only presentational children, HTML5
|
|
|
|
// buttons are allowed to have content.)
|
|
|
|
switch (this._role) {
|
|
|
|
case 'doc-cover':
|
|
|
|
case 'graphics-symbol':
|
|
|
|
case 'img':
|
|
|
|
case 'Meter':
|
|
|
|
case 'scrollbar':
|
|
|
|
case 'slider':
|
|
|
|
case 'separator':
|
|
|
|
case 'progressbar':
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Here and below: Android heuristics
|
|
|
|
if (this._hasFocusableChild())
|
|
|
|
return false;
|
2021-04-26 12:02:54 -07:00
|
|
|
if (this._focusable && this._role !== 'WebArea' && this._role !== 'RootWebArea' && this._name)
|
2019-12-19 16:53:24 -08:00
|
|
|
return true;
|
|
|
|
if (this._role === 'heading' && this._name)
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
isControl(): boolean {
|
|
|
|
switch (this._role) {
|
|
|
|
case 'button':
|
|
|
|
case 'checkbox':
|
|
|
|
case 'ColorWell':
|
|
|
|
case 'combobox':
|
|
|
|
case 'DisclosureTriangle':
|
|
|
|
case 'listbox':
|
|
|
|
case 'menu':
|
|
|
|
case 'menubar':
|
|
|
|
case 'menuitem':
|
|
|
|
case 'menuitemcheckbox':
|
|
|
|
case 'menuitemradio':
|
|
|
|
case 'radio':
|
|
|
|
case 'scrollbar':
|
|
|
|
case 'searchbox':
|
|
|
|
case 'slider':
|
|
|
|
case 'spinbutton':
|
|
|
|
case 'switch':
|
|
|
|
case 'tab':
|
|
|
|
case 'textbox':
|
|
|
|
case 'tree':
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
isInteresting(insideControl: boolean): boolean {
|
|
|
|
const role = this._role;
|
|
|
|
if (role === 'Ignored' || this._hidden)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (this._focusable || this._richlyEditable)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// If it's not focusable but has a control role, then it's interesting.
|
|
|
|
if (this.isControl())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// A non focusable child of a control is not interesting
|
|
|
|
if (insideControl)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return this.isLeafNode() && !!this._name;
|
|
|
|
}
|
|
|
|
|
2021-04-26 12:02:54 -07:00
|
|
|
normalizedRole() {
|
|
|
|
switch (this._role) {
|
|
|
|
case 'RootWebArea':
|
|
|
|
return 'WebArea';
|
|
|
|
case 'StaticText':
|
|
|
|
return 'text';
|
|
|
|
default:
|
|
|
|
return this._role;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-25 18:01:18 -07:00
|
|
|
serialize(): types.SerializedAXNode {
|
2019-12-19 16:53:24 -08:00
|
|
|
const properties: Map<string, number | string | boolean> = new Map();
|
|
|
|
for (const property of this._payload.properties || [])
|
|
|
|
properties.set(property.name.toLowerCase(), property.value.value);
|
|
|
|
if (this._payload.description)
|
|
|
|
properties.set('description', this._payload.description.value);
|
|
|
|
|
2020-06-25 18:01:18 -07:00
|
|
|
const node: {[x in keyof types.SerializedAXNode]: any} = {
|
2021-04-26 12:02:54 -07:00
|
|
|
role: this.normalizedRole(),
|
2020-08-18 20:25:03 -07:00
|
|
|
name: this._payload.name ? (this._payload.name.value || '') : '',
|
2019-12-19 16:53:24 -08:00
|
|
|
};
|
|
|
|
|
2020-06-25 18:01:18 -07:00
|
|
|
const userStringProperties: Array<keyof types.SerializedAXNode> = [
|
2019-12-19 16:53:24 -08:00
|
|
|
'description',
|
|
|
|
'keyshortcuts',
|
|
|
|
'roledescription',
|
|
|
|
'valuetext',
|
|
|
|
];
|
|
|
|
for (const userStringProperty of userStringProperties) {
|
|
|
|
if (!properties.has(userStringProperty))
|
|
|
|
continue;
|
|
|
|
node[userStringProperty] = properties.get(userStringProperty);
|
|
|
|
}
|
2020-06-25 18:01:18 -07:00
|
|
|
const booleanProperties: Array<keyof types.SerializedAXNode> = [
|
2019-12-19 16:53:24 -08:00
|
|
|
'disabled',
|
|
|
|
'expanded',
|
|
|
|
'focused',
|
|
|
|
'modal',
|
|
|
|
'multiline',
|
|
|
|
'multiselectable',
|
|
|
|
'readonly',
|
|
|
|
'required',
|
|
|
|
'selected',
|
|
|
|
];
|
|
|
|
for (const booleanProperty of booleanProperties) {
|
|
|
|
// WebArea's treat focus differently than other nodes. They report whether their frame has focus,
|
|
|
|
// not whether focus is specifically on the root node.
|
2021-04-26 12:02:54 -07:00
|
|
|
if (booleanProperty === 'focused' && (this._role === 'WebArea' || this._role === 'RootWebArea'))
|
2019-12-19 16:53:24 -08:00
|
|
|
continue;
|
|
|
|
const value = properties.get(booleanProperty);
|
|
|
|
if (!value)
|
|
|
|
continue;
|
|
|
|
node[booleanProperty] = value;
|
|
|
|
}
|
2020-06-25 18:01:18 -07:00
|
|
|
const numericalProperties: Array<keyof types.SerializedAXNode> = [
|
2019-12-19 16:53:24 -08:00
|
|
|
'level',
|
|
|
|
'valuemax',
|
|
|
|
'valuemin',
|
|
|
|
];
|
|
|
|
for (const numericalProperty of numericalProperties) {
|
|
|
|
if (!properties.has(numericalProperty))
|
|
|
|
continue;
|
|
|
|
node[numericalProperty] = properties.get(numericalProperty);
|
|
|
|
}
|
2020-06-25 18:01:18 -07:00
|
|
|
const tokenProperties: Array<keyof types.SerializedAXNode> = [
|
2019-12-19 16:53:24 -08:00
|
|
|
'autocomplete',
|
|
|
|
'haspopup',
|
|
|
|
'invalid',
|
|
|
|
'orientation',
|
|
|
|
];
|
|
|
|
for (const tokenProperty of tokenProperties) {
|
|
|
|
const value = properties.get(tokenProperty);
|
|
|
|
if (!value || value === 'false')
|
|
|
|
continue;
|
|
|
|
node[tokenProperty] = value;
|
|
|
|
}
|
2020-08-18 20:25:03 -07:00
|
|
|
|
|
|
|
const axNode = node as types.SerializedAXNode;
|
|
|
|
if (this._payload.value) {
|
|
|
|
if (typeof this._payload.value.value === 'string')
|
|
|
|
axNode.valueString = this._payload.value.value;
|
|
|
|
if (typeof this._payload.value.value === 'number')
|
|
|
|
axNode.valueNumber = this._payload.value.value;
|
|
|
|
}
|
|
|
|
if (properties.has('checked'))
|
|
|
|
axNode.checked = properties.get('checked') === 'true' ? 'checked' : properties.get('checked') === 'false' ? 'unchecked' : 'mixed';
|
|
|
|
if (properties.has('pressed'))
|
|
|
|
axNode.pressed = properties.get('pressed') === 'true' ? 'pressed' : properties.get('pressed') === 'false' ? 'released' : 'mixed';
|
|
|
|
return axNode;
|
2019-12-19 16:53:24 -08:00
|
|
|
}
|
|
|
|
|
2020-01-03 11:15:43 -08:00
|
|
|
static createTree(client: CRSession, payloads: Protocol.Accessibility.AXNode[]): CRAXNode {
|
2019-12-19 16:53:24 -08:00
|
|
|
const nodeById: Map<string, CRAXNode> = new Map();
|
|
|
|
for (const payload of payloads)
|
2020-01-03 11:15:43 -08:00
|
|
|
nodeById.set(payload.nodeId, new CRAXNode(client, payload));
|
2019-12-19 16:53:24 -08:00
|
|
|
for (const node of nodeById.values()) {
|
|
|
|
for (const childId of node._payload.childIds || [])
|
2020-01-13 13:33:25 -08:00
|
|
|
node._children.push(nodeById.get(childId)!);
|
2019-12-19 16:53:24 -08:00
|
|
|
}
|
|
|
|
return nodeById.values().next().value;
|
|
|
|
}
|
|
|
|
}
|