-
Notifications
You must be signed in to change notification settings - Fork 791
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(landmark-one-main): add rule ensuring one main landmark in docum…
…ent (#498) * feat(landmark-one-main): add rule ensuring one main landmark in document * fix: rename integration tests * fix: move checks from 'any' to 'all' * fix: add missing 'after' check * fix: update to pass virtualnode in to check * fix: change incorrect rule name in integration tests * fix: remove line for debugging * fix: correct faulty check tests * test: add shadowCheckSetup util * test: fix main tests for shadowdom * fix: resolve timeout issues * fix: add event listener * fix: change where to check for passes * style: comment code for comprehension * test: add test for second violation node
- Loading branch information
1 parent
63040bd
commit dfc6069
Showing
18 changed files
with
408 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,16 @@ | ||
var hasMain = false; | ||
|
||
//iterate through results from each document | ||
//stops if any document contains a main landmark | ||
for (var i = 0; i < results.length && !hasMain; i++) { | ||
hasMain = results[i].data; | ||
} | ||
|
||
//if any document contains a main landmark, set all documents to pass the check | ||
//otherwise, fail all documents | ||
//since this is a page level rule, all documents either pass or fail the requirement | ||
for (var i = 0; i < results.length; i++) { | ||
results[i].result = hasMain; | ||
} | ||
|
||
return results; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
const mains = axe.utils.querySelectorAll(virtualNode, 'main,[role=main]'); | ||
this.data(!!mains[0]); | ||
return !!mains[0]; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
{ | ||
"id": "has-at-least-one-main", | ||
"evaluate": "has-at-least-one-main.js", | ||
"after": "has-at-least-one-main-after.js", | ||
"metadata": { | ||
"impact": "moderate", | ||
"messages": { | ||
"pass": "Document has at least one main landmark", | ||
"fail": "Document has no main landmarks" | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,2 @@ | ||
const mains = axe.utils.querySelectorAll(virtualNode, 'main,[role=main]'); | ||
return mains.length<=1; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
{ | ||
"id": "has-no-more-than-one-main", | ||
"evaluate": "has-no-more-than-one-main.js", | ||
"metadata": { | ||
"impact": "moderate", | ||
"messages": { | ||
"pass": "Document has no more than one main landmark", | ||
"fail": "Document has more than one main landmark" | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,17 @@ | ||
{ | ||
"id": "landmark-one-main", | ||
"selector": "html", | ||
"tags": [ | ||
"best-practice" | ||
], | ||
"metadata": { | ||
"description": "Ensures a navigation point to the primary content of the page. If the page contains iframes, each iframe should contain either no main landmarks or just one.", | ||
"help": "Page must contain one main landmark." | ||
}, | ||
"all": [ | ||
"has-at-least-one-main", | ||
"has-no-more-than-one-main" | ||
], | ||
"any": [], | ||
"none": [] | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,58 @@ | ||
describe('has-at-least-one-main', function () { | ||
'use strict'; | ||
|
||
var fixture = document.getElementById('fixture'); | ||
var checkContext = new axe.testUtils.MockCheckContext(); | ||
var checkSetup = axe.testUtils.checkSetup; | ||
|
||
afterEach(function () { | ||
fixture.innerHTML = ''; | ||
checkContext.reset(); | ||
}); | ||
|
||
it('should return false if no div has role property', function() { | ||
var params = checkSetup('<div id = "target">No role</div>'); | ||
var mainIsFound = checks['has-at-least-one-main'].evaluate.apply(checkContext, params); | ||
assert.isFalse(mainIsFound); | ||
assert.deepEqual(checkContext._data, mainIsFound); | ||
}); | ||
|
||
it('should return false if div has empty role', function() { | ||
var params = checkSetup('<div id = "target" role = "">Empty role</div>'); | ||
var mainIsFound = checks['has-at-least-one-main'].evaluate.apply(checkContext, params); | ||
assert.isFalse(mainIsFound); | ||
assert.equal(checkContext._data, mainIsFound); | ||
}); | ||
|
||
it('should return false if div has role not equal to main', function() { | ||
var params = checkSetup('<div id = "target" role = "bananas">Wrong role</div>'); | ||
var mainIsFound = checks['has-at-least-one-main'].evaluate.apply(checkContext, params); | ||
assert.isFalse(mainIsFound); | ||
assert.equal(checkContext._data, mainIsFound); | ||
}); | ||
|
||
it('should return true if main landmark exists', function(){ | ||
var params = checkSetup('<main id = "target">main landmark</main>'); | ||
var mainIsFound = checks['has-at-least-one-main'].evaluate.apply(checkContext, params); | ||
assert.isTrue(mainIsFound); | ||
assert.equal(checkContext._data, mainIsFound); | ||
}); | ||
|
||
it('should return true if one div has role equal to main', function() { | ||
var params = checkSetup('<div id = "target" role = "main">Div with role main</div>'); | ||
var mainIsFound = checks['has-at-least-one-main'].evaluate.apply(checkContext, params); | ||
assert.isTrue(mainIsFound); | ||
assert.equal(checkContext._data, mainIsFound); | ||
}); | ||
|
||
it('should return true if any document has a main landmark', function() { | ||
var results = [{data: false, result: false}, {data: true, result: true}]; | ||
assert.isTrue(checks['has-at-least-one-main'].after(results)[0].result && checks['has-at-least-one-main'].after(results)[1].result); | ||
}); | ||
|
||
it('should return false if no document has a main landmark', function() { | ||
var results = [{data: false, result: false}, {data: false, result: false}]; | ||
assert.isFalse(checks['has-at-least-one-main'].after(results)[0].result && checks['has-at-least-one-main'].after(results)[1].result); | ||
}); | ||
|
||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,45 @@ | ||
describe('has-no-more-than-one-main', function () { | ||
'use strict'; | ||
|
||
var fixture = document.getElementById('fixture'); | ||
var checkContext = new axe.testUtils.MockCheckContext(); | ||
var checkSetup = axe.testUtils.checkSetup; | ||
var shadowCheckSetup = axe.testUtils.shadowCheckSetup; | ||
var shadowSupported = axe.testUtils.shadowSupport.v1; | ||
|
||
afterEach(function () { | ||
fixture.innerHTML = ''; | ||
checkContext.reset(); | ||
}); | ||
|
||
it('should return false if there is more than one element with role main', function () { | ||
var params = checkSetup('<div id="target"><div role="main"></div><div role="main"></div></div>'); | ||
assert.isFalse(checks['has-no-more-than-one-main'].evaluate.apply(checkContext, params)); | ||
|
||
}); | ||
|
||
it('should return false if there is more than one main element', function () { | ||
var params = checkSetup('<div id="target"><main></main><main></main></div>'); | ||
assert.isFalse(checks['has-no-more-than-one-main'].evaluate.apply(checkContext, params)); | ||
}); | ||
|
||
it('should return true if there is only one element with role main', function(){ | ||
var params = checkSetup('<div role="main" id="target"></div>'); | ||
assert.isTrue(checks['has-no-more-than-one-main'].evaluate.apply(checkContext, params)); | ||
}); | ||
|
||
it('should return true if there is only one main element', function(){ | ||
var params = checkSetup('<main id="target"></main>'); | ||
assert.isTrue(checks['has-no-more-than-one-main'].evaluate.apply(checkContext, params)); | ||
}); | ||
|
||
(shadowSupported ? it : xit) | ||
('should return false if there is a second main element inside the shadow dom', function () { | ||
var params = shadowCheckSetup( | ||
'<div role="main" id="target"></div>', | ||
'<div role="main"></div>' | ||
); | ||
assert.isFalse(checks['has-no-more-than-one-main'].evaluate.apply(checkContext, params)); | ||
}); | ||
|
||
}); |
10 changes: 10 additions & 0 deletions
10
test/integration/full/landmark-one-main/frames/level1-fail.html
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
<!doctype html> | ||
<html lang="en" id="violation2"> | ||
<head> | ||
<meta charset="utf8"> | ||
<script src="/axe.js"></script> | ||
</head> | ||
<body> | ||
<p>No main content here either</p> | ||
</body> | ||
</html> |
12 changes: 12 additions & 0 deletions
12
test/integration/full/landmark-one-main/frames/level1.html
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
<!doctype html> | ||
<html lang="en" id="pass2"> | ||
<head> | ||
<meta charset="utf8"> | ||
<script src="/axe.js"></script> | ||
</head> | ||
<body> | ||
<p>No main content here either</p> | ||
<iframe id="frame2" src="level2-a.html"></iframe> | ||
<iframe id="frame3" src="level2.html"></iframe> | ||
</body> | ||
</html> |
12 changes: 12 additions & 0 deletions
12
test/integration/full/landmark-one-main/frames/level2-a.html
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
<!doctype html> | ||
<html lang="en" id="pass3"> | ||
<head> | ||
<meta charset="utf8"> | ||
<script src="/axe.js"></script> | ||
</head> | ||
<body> | ||
<main> | ||
<p>Main landmark created with main tag</p> | ||
</main> | ||
</body> | ||
</html> |
10 changes: 10 additions & 0 deletions
10
test/integration/full/landmark-one-main/frames/level2.html
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
<!doctype html> | ||
<html lang="en" id="pass4"> | ||
<head> | ||
<meta charset="utf8"> | ||
<script src="/axe.js"></script> | ||
</head> | ||
<body> | ||
<p>No main content in this iframe</p> | ||
</body> | ||
</html> |
24 changes: 24 additions & 0 deletions
24
test/integration/full/landmark-one-main/landmark-one-main-fail.html
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
<!doctype html> | ||
<html lang="en" id="fail1"> | ||
<head> | ||
<meta charset="utf8"> | ||
<link rel="stylesheet" type="text/css" href="/node_modules/mocha/mocha.css" /> | ||
<script src="/node_modules/mocha/mocha.js"></script> | ||
<script src="/node_modules/chai/chai.js"></script> | ||
<script src="/axe.js"></script> | ||
<script> | ||
mocha.setup({ | ||
timeout: 10000, | ||
ui: 'bdd' | ||
}); | ||
var assert = chai.assert; | ||
</script> | ||
</head> | ||
<body> | ||
<p>No main content here</p> | ||
<iframe id="frame1" src="frames/level1-fail.html"></iframe> | ||
<div id="mocha"></div> | ||
<script src="landmark-one-main-fail.js"></script> | ||
<script src="/test/integration/adapter.js"></script> | ||
</body> | ||
</html> |
46 changes: 46 additions & 0 deletions
46
test/integration/full/landmark-one-main/landmark-one-main-fail.js
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,46 @@ | ||
describe('landmark-one-main test failure', function () { | ||
'use strict'; | ||
var results; | ||
before(function (done) { | ||
function start() { | ||
axe.run({ runOnly: { type: 'rule', values: ['landmark-one-main'] } }, function (err, r) { | ||
assert.isNull(err); | ||
results = r; | ||
done(); | ||
}); | ||
} | ||
if (document.readyState !== 'complete') { | ||
window.addEventListener('load', start); | ||
} else { | ||
start(); | ||
} | ||
}); | ||
|
||
describe('violations', function () { | ||
it('should find 1', function () { | ||
assert.lengthOf(results.violations[0].nodes, 2); | ||
}); | ||
|
||
it('should find #frame1', function () { | ||
assert.deepEqual(results.violations[0].nodes[0].target, ['#fail1']); | ||
}); | ||
|
||
it('should find #frame1, #violation2', function () { | ||
assert.deepEqual(results.violations[0].nodes[1].target, ['#frame1', '#violation2']); | ||
}); | ||
}); | ||
|
||
describe('passes', function () { | ||
it('should find 0', function () { | ||
assert.lengthOf(results.passes, 0); | ||
}); | ||
}); | ||
|
||
it('should find 0 inapplicable', function () { | ||
assert.lengthOf(results.inapplicable, 0); | ||
}); | ||
|
||
it('should find 0 incomplete', function () { | ||
assert.lengthOf(results.incomplete, 0); | ||
}); | ||
}); |
24 changes: 24 additions & 0 deletions
24
test/integration/full/landmark-one-main/landmark-one-main-pass.html
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
<!doctype html> | ||
<html lang="en" id="pass1"> | ||
<head> | ||
<meta charset="utf8"> | ||
<link rel="stylesheet" type="text/css" href="/node_modules/mocha/mocha.css" /> | ||
<script src="/node_modules/mocha/mocha.js"></script> | ||
<script src="/node_modules/chai/chai.js"></script> | ||
<script src="/axe.js"></script> | ||
<script> | ||
mocha.setup({ | ||
timeout: 10000, | ||
ui: 'bdd' | ||
}); | ||
var assert = chai.assert; | ||
</script> | ||
</head> | ||
<body> | ||
<p>No main content</p> | ||
<iframe id="frame1" src="frames/level1.html"></iframe> | ||
<div id="mocha"></div> | ||
<script src="landmark-one-main-pass.js"></script> | ||
<script src="/test/integration/adapter.js"></script> | ||
</body> | ||
</html> |
55 changes: 55 additions & 0 deletions
55
test/integration/full/landmark-one-main/landmark-one-main-pass.js
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,55 @@ | ||
describe('landmark-one-main test pass', function () { | ||
'use strict'; | ||
var results; | ||
before(function (done) { | ||
function start() { | ||
axe.run({ runOnly: { type: 'rule', values: ['landmark-one-main'] } }, function (err, r) { | ||
assert.isNull(err); | ||
results = r; | ||
done(); | ||
}); | ||
} | ||
if (document.readyState !== 'complete') { | ||
window.addEventListener('load', start); | ||
} else { | ||
start(); | ||
} | ||
}); | ||
|
||
describe('violations', function () { | ||
it('should find 0', function () { | ||
assert.lengthOf(results.violations, 0); | ||
}); | ||
}); | ||
|
||
describe('passes', function () { | ||
it('should find 4', function () { | ||
assert.lengthOf(results.passes[0].nodes, 4); | ||
}); | ||
|
||
it('should find #pass1', function () { | ||
assert.deepEqual(results.passes[0].nodes[0].target, ['#pass1']); | ||
}); | ||
|
||
it('should find #frame1, #pass2', function () { | ||
assert.deepEqual(results.passes[0].nodes[1].target, ['#frame1', '#pass2']); | ||
}); | ||
|
||
it('should find #frame1, #frame2, #pass3', function () { | ||
assert.deepEqual(results.passes[0].nodes[2].target, ['#frame1', '#frame2', '#pass3']); | ||
}); | ||
|
||
it('should find #frame1, #frame3, #pass4', function () { | ||
assert.deepEqual(results.passes[0].nodes[3].target, ['#frame1', '#frame3', '#pass4']); | ||
}); | ||
}); | ||
|
||
it('should find 0 inapplicable', function () { | ||
assert.lengthOf(results.inapplicable, 0); | ||
}); | ||
|
||
it('should find 0 incomplete', function () { | ||
assert.lengthOf(results.incomplete, 0); | ||
}); | ||
|
||
}); |
Oops, something went wrong.