Skip to content

Commit 630116b

Browse files
mattxwangljharb
authored andcommittedJul 9, 2022
[New] add getAccessibleChildText util
1 parent d678a98 commit 630116b

File tree

2 files changed

+151
-0
lines changed

2 files changed

+151
-0
lines changed
 
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,96 @@
1+
import expect from 'expect';
2+
import { elementType } from 'jsx-ast-utils';
3+
import getAccessibleChildText from '../../../src/util/getAccessibleChildText';
4+
import JSXAttributeMock from '../../../__mocks__/JSXAttributeMock';
5+
import JSXElementMock from '../../../__mocks__/JSXElementMock';
6+
7+
describe('getAccessibleChildText', () => {
8+
it('returns the aria-label when present', () => {
9+
expect(getAccessibleChildText(JSXElementMock(
10+
'a',
11+
[JSXAttributeMock('aria-label', 'foo')],
12+
), elementType)).toBe('foo');
13+
});
14+
15+
it('returns the aria-label instead of children', () => {
16+
expect(getAccessibleChildText(JSXElementMock(
17+
'a',
18+
[JSXAttributeMock('aria-label', 'foo')],
19+
[{ type: 'JSXText', value: 'bar' }],
20+
), elementType)).toBe('foo');
21+
});
22+
23+
it('skips elements with aria-hidden=true', () => {
24+
expect(getAccessibleChildText(JSXElementMock(
25+
'a',
26+
[JSXAttributeMock('aria-hidden', 'true')],
27+
), elementType)).toBe('');
28+
});
29+
30+
it('returns literal value for JSXText child', () => {
31+
expect(getAccessibleChildText(JSXElementMock(
32+
'a',
33+
[],
34+
[{ type: 'JSXText', value: 'bar' }],
35+
), elementType)).toBe('bar');
36+
});
37+
38+
it('returns literal value for JSXText child', () => {
39+
expect(getAccessibleChildText(JSXElementMock(
40+
'a',
41+
[],
42+
[{ type: 'Literal', value: 'bar' }],
43+
), elementType)).toBe('bar');
44+
});
45+
46+
it('returns recursive value for JSXElement child', () => {
47+
expect(getAccessibleChildText(JSXElementMock(
48+
'a',
49+
[],
50+
[JSXElementMock(
51+
'span',
52+
[],
53+
[{ type: 'Literal', value: 'bar' }],
54+
)],
55+
), elementType)).toBe('bar');
56+
});
57+
58+
it('skips children with aria-hidden-true', () => {
59+
expect(getAccessibleChildText(JSXElementMock(
60+
'a',
61+
[],
62+
[JSXElementMock(
63+
'span',
64+
[],
65+
[JSXElementMock(
66+
'span',
67+
[JSXAttributeMock('aria-hidden', 'true')],
68+
)],
69+
)],
70+
), elementType)).toBe('');
71+
});
72+
73+
it('joins multiple children properly - no spacing', () => {
74+
expect(getAccessibleChildText(JSXElementMock(
75+
'a',
76+
[],
77+
[{ type: 'Literal', value: 'foo' }, { type: 'Literal', value: 'bar' }],
78+
), elementType)).toBe('foo bar');
79+
});
80+
81+
it('joins multiple children properly - with spacing', () => {
82+
expect(getAccessibleChildText(JSXElementMock(
83+
'a',
84+
[],
85+
[{ type: 'Literal', value: ' foo ' }, { type: 'Literal', value: ' bar ' }],
86+
), elementType)).toBe('foo bar');
87+
});
88+
89+
it('skips unknown elements', () => {
90+
expect(getAccessibleChildText(JSXElementMock(
91+
'a',
92+
[],
93+
[{ type: 'Literal', value: 'foo' }, { type: 'Unknown' }, { type: 'Literal', value: 'bar' }],
94+
), elementType)).toBe('foo bar');
95+
});
96+
});

‎src/util/getAccessibleChildText.js

+55
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
// @flow
2+
3+
import type { JSXElement, JSXOpeningElement, Node } from 'ast-types-flow';
4+
5+
import { getProp, getLiteralPropValue } from 'jsx-ast-utils';
6+
7+
import isHiddenFromScreenReader from './isHiddenFromScreenReader';
8+
9+
/**
10+
* Returns a new "standardized" string: all whitespace is collapsed to one space,
11+
* and the string is lowercase
12+
* @param {string} input
13+
* @returns lowercase, single-spaced, trimmed string
14+
*/
15+
function standardizeSpaceAndCase(input: string): string {
16+
return input.trim().replace(/\s\s+/g, ' ').toLowerCase();
17+
}
18+
19+
/**
20+
* Returns the (recursively-defined) accessible child text of a node, which (in-order) is:
21+
* 1. The element's aria-label
22+
* 2. If the element is a direct literal, the literal value
23+
* 3. Otherwise, merge all of its children
24+
* @param {JSXElement} node - node to traverse
25+
* @returns child text as a string
26+
*/
27+
export default function getAccessibleChildText(node: JSXElement, elementType: (JSXOpeningElement) => string): string {
28+
const ariaLabel = getLiteralPropValue(getProp(node.openingElement.attributes, 'aria-label'));
29+
// early escape-hatch when aria-label is applied
30+
if (ariaLabel) return standardizeSpaceAndCase(ariaLabel);
31+
32+
// skip if aria-hidden is true
33+
if (
34+
isHiddenFromScreenReader(
35+
elementType(node.openingElement),
36+
node.openingElement.attributes,
37+
)
38+
) {
39+
return '';
40+
}
41+
42+
const rawChildText = node.children
43+
.map((currentNode: Node): string => {
44+
// $FlowFixMe JSXText is missing in ast-types-flow
45+
if (currentNode.type === 'Literal' || currentNode.type === 'JSXText') {
46+
return String(currentNode.value);
47+
}
48+
if (currentNode.type === 'JSXElement') {
49+
return getAccessibleChildText(currentNode, elementType);
50+
}
51+
return '';
52+
}).join(' ');
53+
54+
return standardizeSpaceAndCase(rawChildText);
55+
}

0 commit comments

Comments
 (0)
Please sign in to comment.