Skip to content
This repository was archived by the owner on Jul 29, 2024. It is now read-only.

Commit 54163dc

Browse files
Marcy Suttonjuliemr
Marcy Sutton
authored andcommitted
feat(a11yPlugin): plugin for integrating with Chrome Accessibility Developer Tools
Also includes missing Angular map files. See plugins/accessibility/index.js for usage.
1 parent 64704e0 commit 54163dc

File tree

11 files changed

+230
-0
lines changed

11 files changed

+230
-0
lines changed

plugins/accessibility/index.js

+109
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,109 @@
1+
var q = require('q'),
2+
fs = require('fs'),
3+
path = require('path'),
4+
_ = require('lodash');
5+
6+
/**
7+
* You can enable this plugin in your config file:
8+
*
9+
* // The Chrome Accessibility Developer Tools are currently
10+
* // the only integration option.
11+
*
12+
* exports.config = {
13+
* ...
14+
* plugins: [{
15+
* chromeA11YDevTools: true,
16+
* path: 'node_modules/protractor.plugins/accessiblity'
17+
* }]
18+
* }
19+
*
20+
*/
21+
22+
var AUDIT_FILE = path.join(__dirname, '../../node_modules/accessibility-developer-tools/dist/js/axs_testing.js');
23+
24+
/**
25+
* Checks the information returned by the accessibility audit and
26+
* displays passed/failed results as console output.
27+
*
28+
* @param {Object} config The configuration file for the accessibility plugin
29+
* @return {q.Promise} A promise which resolves to the results of any passed or
30+
* failed tests
31+
* @public
32+
*/
33+
function teardown(config) {
34+
35+
if (config.chromeA11YDevTools) {
36+
37+
var data = fs.readFileSync(AUDIT_FILE, 'utf-8');
38+
data = data + ' return axs.Audit.run();';
39+
40+
var testOut = {failedCount: 0, specResults: []},
41+
elementPromises = [];
42+
43+
return browser.executeScript_(data, 'a11y developer tool rules').then(function(results) {
44+
45+
var audit = results.map(function(result) {
46+
var DOMElements = result.elements;
47+
if (DOMElements !== undefined) {
48+
49+
DOMElements.forEach(function(elem) {
50+
// get elements from WebDriver, add to promises array
51+
elementPromises.push(
52+
elem.getOuterHtml().then(function(text) {
53+
return {
54+
code: result.rule.code,
55+
list: text
56+
};
57+
})
58+
);
59+
});
60+
result.elementCount = DOMElements.length;
61+
}
62+
return result;
63+
});
64+
65+
// Wait for element names to be fetched
66+
return q.all(elementPromises).then(function(elementFailures) {
67+
68+
audit.forEach(function(result, index) {
69+
if (result.result === 'FAIL') {
70+
result.passed = false;
71+
testOut.failedCount++;
72+
73+
var label = result.elementCount === 1 ? ' element ' : ' elements ';
74+
result.output = '\n\t\t' + result.elementCount + label + 'failed:';
75+
76+
// match elements returned via promises
77+
// by their failure codes
78+
elementFailures.forEach(function(element, index) {
79+
if (element.code === result.rule.code) {
80+
result.output += '\n\t\t' + elementFailures[index].list;
81+
}
82+
});
83+
result.output += '\n\n\t\t' + result.rule.url;
84+
}
85+
else {
86+
result.passed = true;
87+
result.output = '';
88+
}
89+
90+
testOut.specResults.push({
91+
description: result.rule.heading,
92+
assertions: [{
93+
passed: result.passed,
94+
errorMsg: result.output
95+
}],
96+
duration: 1
97+
});
98+
});
99+
100+
if ((testOut.failedCount > 0) || (testOut.specResults.length > 0)) {
101+
return testOut;
102+
}
103+
});
104+
});
105+
}
106+
}
107+
108+
// Export
109+
exports.teardown = teardown;
+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
describe('check if accessibility plugin works on bad apps', function() {
2+
it('should have accessibility problems on markup', function() {
3+
browser.get('accessibility/badMarkup.html');
4+
});
5+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
var env = require('../../../spec/environment.js');
2+
3+
exports.config = {
4+
seleniumAddress: env.seleniumAddress,
5+
framework: 'jasmine2',
6+
specs: ['fail_spec.js'],
7+
baseUrl: env.baseUrl,
8+
plugins: [{
9+
chromeA11YDevTools: true,
10+
path: '../index.js'
11+
}]
12+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
var env = require('../../../spec/environment.js');
2+
3+
exports.config = {
4+
seleniumAddress: env.seleniumAddress,
5+
framework: 'jasmine2',
6+
specs: ['success_spec.js'],
7+
baseUrl: env.baseUrl,
8+
plugins: [{
9+
chromeA11YDevTools: true,
10+
path: "../index.js"
11+
}]
12+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
describe('accessibility', function() {
2+
it('should get a file to test', function() {
3+
browser.get('accessibility/index.html');
4+
5+
element.all(by.css('input')).then(function(inputs) {
6+
expect(inputs.length).toEqual(2);
7+
});
8+
});
9+
});

scripts/test.js

+15
Original file line numberDiff line numberDiff line change
@@ -112,4 +112,19 @@ executor.addCommandlineTest(
112112
message: 'warning -- Module "xApp" was created but never loaded.'
113113
}]);
114114

115+
// Check accessibility plugin
116+
117+
executor.addCommandlineTest(
118+
'node lib/cli.js plugins/accessibility/spec/failureConfig.js')
119+
.expectExitCode(1)
120+
.expectErrors([{
121+
message: '2 elements failed:'+
122+
'\n\t\t<input ng-model="firstName" type="text" class="ng-pristine ng-valid ng-touched">'+
123+
'\n\t\t<input ng-model="lastName" type="text" class="ng-pristine ng-untouched ng-valid">'
124+
},
125+
{
126+
message: '1 element failed:'+
127+
'\n\t\t<img src="http://example.com/img.jpg">'
128+
}]);
129+
115130
executor.execute();

testapp/accessibility/badMarkup.html

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
<!DOCTYPE html>
2+
3+
<html ng-app ng-hint>
4+
<head>
5+
<meta charset="utf-8">
6+
<title>Angular.js Example</title>
7+
<script src="../lib/angular/angular.min.js"></script>
8+
<script src="../lib/angular/angular-aria.min.js"></script>
9+
<script>
10+
angular.module('xApp', []);
11+
</script>
12+
</head>
13+
<body>
14+
First name:<input ng-model="firstName" type="text"/>
15+
<br>
16+
Last name:<input ng-model="lastName" type="text"/>
17+
<br>
18+
Hello {{firstName}} {{lastName}}
19+
<img src="http://example.com/img.jpg">
20+
</body>
21+
</html>

testapp/accessibility/index.html

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
<!DOCTYPE html>
2+
3+
<html ng-app="xApp">
4+
<head>
5+
<meta charset="utf-8">
6+
<title>Angular.js Example</title>
7+
<script src="../../lib/angular/angular.min.js"></script>
8+
<script src="../../lib/angular/angular-aria.min.js"></script>
9+
<script>
10+
angular.module('xApp', []);
11+
</script>
12+
</head>
13+
<body>
14+
<label for="firstName">First name:</label>
15+
<input ng-model="firstName" type="text" id="firstName" />
16+
<br>
17+
<label for="lastName">Last name:</label>
18+
<input ng-model="lastName" type="text" id="lastName" />
19+
<br>
20+
Hello {{firstName}} {{lastName}}
21+
<img src="http://example.com/img.jpg" alt="{{firstName}} {{lastName}}">
22+
</body>
23+
</html>

testapp/lib/angular_v1.3.13/angular-animate.min.js.map

+8
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

testapp/lib/angular_v1.3.13/angular-route.min.js.map

+8
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

testapp/lib/angular_v1.3.13/angular.min.js.map

+8
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)