Skip to content

Commit

Permalink
feat: Initial test suite, based on the Landmarks extension
Browse files Browse the repository at this point in the history
This is a basic version of the test suite from Landmarks that has been designed to work with other tools, such as a11y-outline/aria-api. The fixtures and expectations come from the Landmarks browser extension, but the code was written from scratch using TDD.

There are lots more features to come, though this initial version is used as a drop-in replacement for the Landmarks extension test suite.

For more information, consult matatk/landmarks#303
  • Loading branch information
matatk committed Apr 9, 2019
1 parent 643bb47 commit 0662921
Show file tree
Hide file tree
Showing 93 changed files with 5,346 additions and 0 deletions.
16 changes: 16 additions & 0 deletions .editorconfig
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
# http://EditorConfig.org
root = true

[*]
charset = utf-8
end_of_line = lf
indent_style = tab
insert_final_newline = true
trim_trailing_whitespace = true

[*.{json,yml}]
indent_style = space
indent_size = 2

[*.md]
trim_trailing_whitespace = false
89 changes: 89 additions & 0 deletions .eslintrc.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,89 @@
{
"env": {
"es6": true,
"node": true
},
"extends": "eslint:recommended",
"parserOptions": {
"ecmaVersion": 2017
},
"root": true,
"rules": {
"block-scoped-var": "error",
"brace-style": "error",
"camelcase": "warn",
"curly": [
"error",
"multi-line"
],
"eqeqeq": "error",
"func-call-spacing": "error",
"indent": [
"error",
"tab",
{
"SwitchCase": 1
}
],
"new-cap": "error",
"no-bitwise": "error",
"no-caller": "error",
"no-console": "off",
"no-else-return": "error",
"no-eq-null": "error",
"no-labels": "error",
"no-lone-blocks": "error",
"no-lonely-if": "error",
"no-loop-func": "warn",
"no-multi-assign": "error",
"no-throw-literal": "error",
"no-undef": "error",
"no-unexpected-multiline": "error",
"no-unused-expressions": "error",
"no-unused-vars": "error",
"no-use-before-define": [
"error",
{
"functions": false
}
],
"no-var": "error",
"one-var": [
"error",
"never"
],
"prefer-const": "error",
"quotes": [
"error",
"single",
{
"avoidEscape": true
}
],
"semi": [
"error",
"never"
],
"space-before-blocks": "error",
"space-before-function-paren": [
"error",
"never"
],
"space-infix-ops": "error",
"space-unary-ops": "error",
"spaced-comment": "error",
"strict": [
"error",
"global"
],
"use-isnan": "error",
"valid-jsdoc": "warn",
"valid-typeof": "warn",
"vars-on-top": "warn",
"wrap-iife": [
"error",
"inside"
],
"yoda": "error"
}
}
4 changes: 4 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
.DS_Store
*.swp
node_modules
.nyc_output
2 changes: 2 additions & 0 deletions .npmrc
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
tag-version-prefix=
message=chore(release): %s
19 changes: 19 additions & 0 deletions LICENCE
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
Copyright © 2019 Matthew Tylee Atkinson

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
102 changes: 102 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,102 @@
Page Structural Semantics Scanner Tests
=======================================

This is a test and benchmarking suite that can be used to check tools that scan for semantic structural information on web pages, such as landmarks \[with headings and articles in the works\]. Such information is often used to afford or improve the accessibility of pages, allowing people using screen-readers and alternative browsers to navigate and understand the content.

You can use this tool as:

- [A test suite for code that checks for landmarks](#use-as-a-test-suite)
- [A performance checker for the code](#use-as-a-benchmarking-tool)
- [A place to find results of tests carried out on tools that scan for landmarks](#use-as-a-comparison-between-accessible-structural-semantics-scanners)

You may find the following information helpful:

- [Support for landmarks](#support-for-landmarks)
- [Development](#development)

Use as a test suite
-------------------

There are three main components that come together in order to test structure-scanning code:

- **Fixtures** are HTML files, provided here.
- **Expectations** contain the correct set of landmarks for a given fixture, also provided here.
- A **scanner** function is the code under test. It runs on the DOM created from the fixture.

In order to test a tool that finds landmarks on a page, a `runner(converter, scanner)` function is provided. This runs the tool's code on DOMs created from all fixtures and compares the results to the expectations, then reports if they matched. The runner takes two arguments, both of which are functions:

- `converter(<object>) -> <object>` should convert the format of the expectations in this test suite to the format that the tool being tested outputs.
- `scanner(window, document) -> object` runs the tool's code on the DOM constructed from the existing HTML fixture and returns the results (i.e. landmarks found).

### Predefined runners for known tools

For known tools, such as the [Landmarks browser extension](http://matatk.agrip.org.uk/landmarks/) a converter, and custom runner function, are already supplied. Consult [the Landmarks extension's test code](https://github.com/matatk/landmarks/blob/master/test/test-landmarks.js) for an example of their use.

The `runners.landmarks(scanner)` function runs the given function `scanner(window, document)` on DOMs created from all test fixtures, with the expectations converted to [Landmarks](http://matatk.agrip.org.uk/landmarks/)' format.

Use as a benchmarking tool
--------------------------

FIXME

Use as a comparison between accessible structural semantics scanners
--------------------------------------------------------------------

FIXME

Support for landmarks
---------------------

All of the core [WAI-ARIA landmark roles](https://www.w3.org/TR/wai-aria-1.1/#landmark_roles), both as supplied via the `role` attribute and as [implicit landmarks via HTML 5 elements](https://www.w3.org/TR/html-aam-1.0/#html-element-role-mappings) are supported, with some caveats, as described below.

- banner<sup>1</sup>
- complementary
- contentinfo<sup>1</sup>
- form<sup>2</sup>
- main
- navigation
- region<sup>2</sup>
- search

### Caveats

1. Both `<header>` (`banner`) and `<footer>` (`contentinfo`) elements are not considered landmarks unless they are the page-wide header/footer elements. (As per the [HTML element role mappings](https://www.w3.org/TR/html-aam-1.0/#html-element-role-mappings).)

2. [`form`](https://www.w3.org/TR/wai-aria-1.1/#form) and [`region`](https://www.w3.org/TR/wai-aria-1.1/#region) landmarks are intended to be labelled. Ideally, this should be done with a visual label and an `aria-labelledby` attribute, so that all users can perceive the label. However, if a label is only provided by the non-visual `aria-label` attribute, this extension will recognise it.

The HTML Accessibility API Mapping is clear that both [unlabelled `<form>`](https://www.w3.org/TR/html-aam-1.0/#details-id-42) and [unlabelled `<section>` (`region`)](https://www.w3.org/TR/html-aam-1.0/#details-id-119) elements are *not* to be counted as landmark regions. This extension discounts *any* unlabelled element with a role of `form` or `region` too, which is in line with most assistive technologies, and is intended to reduce noise in landmark navigation.

### Labelling landmarks

If a landmark label is present (via the `aria-labelledby` or `aria-label` attributes), it'll be shown in the pop-up. As per the [accessible name calculation algorithm](https://www.w3.org/TR/accname-aam-1.1/#mapping_additional_nd_te) used by browsers, the `aria-labelledby` attribute takes precedence over `aria-label`.

If an `aria-labelledby` attribute references multiple elements, all of those elements' text content will be joined to form the label for the landmark. However, it's not recommended that you label landmark regions with more than one element (usually referring to a single HTML heading element is sufficient). Using more than one labelling element could be a sign that your landmark structure is too complicated. [Referencing multiple labelling elements is more suited for labelling `<input>` elements with information from multiple sources.](https://www.w3.org/WAI/GL/wiki/Using_aria-labelledby_to_concatenate_a_label_from_several_text_nodes#Example_1:_A_time-out_input_field_with_concatenated_label)

### Digital publishing ARIA landmarks

The following additional landmark roles defined in the [Digital Publishing WAI-ARIA Module 1.0](https://www.w3.org/TR/dpub-aria-1.0/) are also supported.

- `doc-acknowledgements`
- `doc-afterword`
- `doc-appendix`
- `doc-bibliography`
- `doc-chapter`
- `doc-conclusion`
- `doc-credits`
- `doc-endnotes`
- `doc-epilogue`
- `doc-errata`
- `doc-foreword`
- `doc-glossary`
- `doc-index` (is a landmark via `navigation`)
- `doc-introduction`
- `doc-pagelist` (is a landmark via `navigation`)
- `doc-part`
- `doc-preface`
- `doc-prologue`
- `doc-toc` (is a landmark via `navigation`)

Development
-----------

Development is test-driven; use `npm test` to run the tests on the suite itself.
31 changes: 31 additions & 0 deletions bin/htmlLandmarksResults.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
'use strict'
const path = require('path')
const iterator = require('../lib/iterator')
const converter = require('../lib/converters').landmarks
const scan = require('../lib/scan').scanAndReturnPatch
const tableEntryMaker = require('../lib/tableEntryMaker')

const fixtures = path.join(__dirname, '..', 'fixtures')
const expectations = path.join(__dirname, '..', 'expectations')

const style = `
table {
border-collapse: collapse;
}
th, td {
border: 1px solid gray;
padding: 0.5em;
}`

console.log(`<html><head><style>${style}</style>
<body><h1>Landmarks Tests</h1><table>`)
console.log('<tr><th>Fixture</th><th>Result</th></tr>')

iterator(fixtures, expectations, (testName, html, expectation) => {
const landmarksFormatExpectation = converter(expectation)
const result = scan(() => 42, html, landmarksFormatExpectation)
console.log(tableEntryMaker(testName, result.patch))
})

console.log('</table></body></html>')
8 changes: 8 additions & 0 deletions bin/runLandmarksTests.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
'use strict'
const path = require('path')
const tap = require('tap')
const runner = require('../lib/runner')
const converter = require('../lib/converters').landmarks
const fixtures = path.join(__dirname, '..', 'fixtures')
const expectations = path.join(__dirname, '..', 'expectations')
runner(tap, fixtures, expectations, converter, () => 42)
3 changes: 3 additions & 0 deletions expectations/application-alone-on-body-is-ignored.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
{
"expected": []
}
38 changes: 38 additions & 0 deletions expectations/aria-labelledby-multiple-idrefs.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
{
"expected": [
{
"type": "landmark",
"role": "banner",
"label": null,
"selector": "body > header",
"contains": [
{
"type": "landmark",
"role": "navigation",
"label": "World of wombats",
"selector": "body > header > nav"
}
]
},
{
"type": "landmark",
"role": "main",
"label": "Looking after your wombat",
"selector": "body > main",
"contains": [
{
"type": "landmark",
"role": "navigation",
"label": "Looking after your wombat Topics",
"selector": "body > main > nav:nth-child(2)"
}
]
},
{
"type": "landmark",
"role": "contentinfo",
"label": null,
"selector": "body > footer"
}
]
}
3 changes: 3 additions & 0 deletions expectations/article-is-ignored.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
{
"expected": []
}
10 changes: 10 additions & 0 deletions expectations/aside-alone-is-recognised.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
{
"expected": [
{
"type": "landmark",
"role": "complementary",
"label": null,
"selector": "body > aside"
}
]
}
10 changes: 10 additions & 0 deletions expectations/aside-aria-labelledby.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
{
"expected": [
{
"type": "landmark",
"role": "complementary",
"label": "And another thing...",
"selector": "body > aside"
}
]
}
Loading

0 comments on commit 0662921

Please sign in to comment.