From 465d41d02852a005e92197ff73c13b3091fa5803 Mon Sep 17 00:00:00 2001 From: Paul Grenier Date: Tue, 24 Oct 2017 21:22:01 -0400 Subject: [PATCH] fix(aria-errormessage): adds support for aria-errormessage (#517) * fix(aria-errormessage): adds support for aria-errormessage * fix(aria-valid-attr-value): switches from `any` mode to `all` --- lib/checks/aria/errormessage.js | 25 ++++++++++ lib/checks/aria/errormessage.json | 11 +++++ lib/checks/aria/valid-attr-value.js | 11 +++-- lib/commons/aria/index.js | 3 ++ lib/rules/aria-valid-attr-value.json | 7 +-- test/checks/aria/errormessage.js | 49 +++++++++++++++++++ .../rules/aria-allowed-attr/passes.json | 4 +- .../aria-valid-attr-value.html | 5 ++ .../aria-valid-attr-value.json | 5 +- 9 files changed, 110 insertions(+), 10 deletions(-) create mode 100644 lib/checks/aria/errormessage.js create mode 100644 lib/checks/aria/errormessage.json create mode 100644 test/checks/aria/errormessage.js diff --git a/lib/checks/aria/errormessage.js b/lib/checks/aria/errormessage.js new file mode 100644 index 0000000000..82dd6ce1e5 --- /dev/null +++ b/lib/checks/aria/errormessage.js @@ -0,0 +1,25 @@ +options = Array.isArray(options) ? options : []; + +var attr = node.getAttribute('aria-errormessage'), + hasAttr = node.hasAttribute('aria-errormessage'); + +var doc = axe.commons.dom.getRootNode(node); + +function validateAttrValue() { + var idref = attr && doc.getElementById(attr); + if (idref) { + return idref.getAttribute('role') === 'alert' || + idref.getAttribute('aria-live') === 'assertive' || + axe.utils.tokenList(node.getAttribute('aria-describedby') || '').indexOf(attr) > -1; + } +} + +// limit results to elements that actually have this attribute +if (options.indexOf(attr) === -1 && hasAttr) { + if (!validateAttrValue()) { + this.data(attr); + return false; + } +} + +return true; diff --git a/lib/checks/aria/errormessage.json b/lib/checks/aria/errormessage.json new file mode 100644 index 0000000000..5bf3563ce3 --- /dev/null +++ b/lib/checks/aria/errormessage.json @@ -0,0 +1,11 @@ +{ + "id": "aria-errormessage", + "evaluate": "errormessage.js", + "metadata": { + "impact": "critical", + "messages": { + "pass": "Uses a supported aria-errormessage technique", + "fail": "aria-errormessage value{{=it.data && it.data.length > 1 ? 's' : ''}} {{~it.data:value}} `{{=value}}{{~}}` must use a technique to announce the message (e.g., aria-live, aria-describedby, role=alert, etc.)" + } + } +} diff --git a/lib/checks/aria/valid-attr-value.js b/lib/checks/aria/valid-attr-value.js index 02ffad9ac0..b5a7c58757 100644 --- a/lib/checks/aria/valid-attr-value.js +++ b/lib/checks/aria/valid-attr-value.js @@ -6,13 +6,18 @@ var invalid = [], var attr, attrName, attrs = node.attributes; +var skipAttrs = ['aria-errormessage']; + for (var i = 0, l = attrs.length; i < l; i++) { attr = attrs[i]; attrName = attr.name; - if (options.indexOf(attrName) === -1 && aria.test(attrName) && - !axe.commons.aria.validateAttrValue(node, attrName)) { + // skip any attributes handled elsewhere + if (!skipAttrs.includes(attrName)) { + if (options.indexOf(attrName) === -1 && aria.test(attrName) && + !axe.commons.aria.validateAttrValue(node, attrName)) { - invalid.push(attrName + '="' + attr.nodeValue + '"'); + invalid.push(attrName + '="' + attr.nodeValue + '"'); + } } } diff --git a/lib/commons/aria/index.js b/lib/commons/aria/index.js index d6fbb9983b..d0752ac083 100644 --- a/lib/commons/aria/index.js +++ b/lib/commons/aria/index.js @@ -48,6 +48,9 @@ lookupTables.attributes = { type: 'nmtokens', values: ['copy', 'move', 'reference', 'execute', 'popup', 'none'] }, + 'aria-errormessage': { + type: 'idref' + }, 'aria-expanded': { type: 'nmtoken', values: ['true', 'false', 'undefined'] diff --git a/lib/rules/aria-valid-attr-value.json b/lib/rules/aria-valid-attr-value.json index b009bb5914..dc1afa1e4e 100644 --- a/lib/rules/aria-valid-attr-value.json +++ b/lib/rules/aria-valid-attr-value.json @@ -12,9 +12,10 @@ "description": "Ensures all ARIA attributes have valid values", "help": "ARIA attributes must conform to valid values" }, - "all": [], - "any": [ - "aria-valid-attr-value" + "all": [ + "aria-valid-attr-value", + "aria-errormessage" ], + "any": [], "none": [] } diff --git a/test/checks/aria/errormessage.js b/test/checks/aria/errormessage.js new file mode 100644 index 0000000000..819cd875b8 --- /dev/null +++ b/test/checks/aria/errormessage.js @@ -0,0 +1,49 @@ +describe('aria-errormessage', function () { + 'use strict'; + + var fixture = document.getElementById('fixture'); + + var checkContext = axe.testUtils.MockCheckContext(); + + afterEach(function () { + fixture.innerHTML = ''; + checkContext._data = null; + }); + + it('should return false if aria-errormessage value is invalid', function () { + var testHTML = '
'; + testHTML += '
'; + fixture.innerHTML = testHTML; + var target = fixture.children[0]; + target.setAttribute('aria-errormessage', 'plain'); + assert.isFalse(checks['aria-errormessage'].evaluate.call(checkContext, target)); + }); + + it('should return true if aria-errormessage id is alert', function () { + var testHTML = '
'; + testHTML += ''; + fixture.innerHTML = testHTML; + var target = fixture.children[0]; + target.setAttribute('aria-errormessage', 'alert'); + assert.isTrue(checks['aria-errormessage'].evaluate.call(checkContext, target)); + }); + + it('should return true if aria-errormessage id is aria-live=assertive', function () { + var testHTML = '
'; + testHTML += '
'; + fixture.innerHTML = testHTML; + var target = fixture.children[0]; + target.setAttribute('aria-errormessage', 'live'); + assert.isTrue(checks['aria-errormessage'].evaluate.call(checkContext, target)); + }); + + it('should return true if aria-errormessage id is aria-describedby', function () { + var testHTML = '
'; + testHTML += '
'; + fixture.innerHTML = testHTML; + var target = fixture.children[0]; + target.setAttribute('aria-errormessage', 'plain'); + target.setAttribute('aria-describedby', 'plain'); + assert.isTrue(checks['aria-errormessage'].evaluate.call(checkContext, target)); + }); +}); diff --git a/test/integration/rules/aria-allowed-attr/passes.json b/test/integration/rules/aria-allowed-attr/passes.json index 13ad56f7b0..d6e9ea5ece 100644 --- a/test/integration/rules/aria-allowed-attr/passes.json +++ b/test/integration/rules/aria-allowed-attr/passes.json @@ -11,6 +11,6 @@ ["#pass43"], ["#pass44"], ["#pass45"], ["#pass46"], ["#pass47"], ["#pass48"], ["#pass49"], ["#pass50"], ["#pass51"], ["#pass52"], ["#pass53"], ["#pass54"], ["#pass55"], ["#pass56"], ["#pass57"], ["#pass58"], ["#pass59"], ["#pass60"], ["#pass61"], ["#pass62"], ["#pass63"], - ["#pass64"], ["#pass65"], ["#pass66"], ["#pass67"], ["#pass68"] + ["#pass64"], ["#pass65"], ["#pass66"], ["#pass67"], ["#pass68"] ] -} \ No newline at end of file +} diff --git a/test/integration/rules/aria-valid-attr-value/aria-valid-attr-value.html b/test/integration/rules/aria-valid-attr-value/aria-valid-attr-value.html index 54be03617a..85f62e5d72 100644 --- a/test/integration/rules/aria-valid-attr-value/aria-valid-attr-value.html +++ b/test/integration/rules/aria-valid-attr-value/aria-valid-attr-value.html @@ -40,6 +40,7 @@

Violations

hi
hi
hi
+
hi

Possible False Positives

@@ -245,6 +246,10 @@

Possible False Positives

hi
hi
hi
+
hi
+
hi
+
hi
+
Hi
Hi2
diff --git a/test/integration/rules/aria-valid-attr-value/aria-valid-attr-value.json b/test/integration/rules/aria-valid-attr-value/aria-valid-attr-value.json index 3bbbfbe1d4..e5c1a67369 100644 --- a/test/integration/rules/aria-valid-attr-value/aria-valid-attr-value.json +++ b/test/integration/rules/aria-valid-attr-value/aria-valid-attr-value.json @@ -7,7 +7,7 @@ ["#violation12"], ["#violation13"], ["#violation14"], ["#violation15"], ["#violation16"], ["#violation17"], ["#violation18"], ["#violation19"], ["#violation20"], ["#violation21"], ["#violation22"], ["#violation23"], ["#violation24"], ["#violation25"], ["#violation26"], ["#violation27"], ["#violation28"], ["#violation29"], - ["#violation30"], ["#violation31"], ["#violation32"], ["#violation33"], ["#violation34"] + ["#violation30"], ["#violation31"], ["#violation32"], ["#violation33"], ["#violation34"],["#violation35"] ], "passes": [ ["#pass1"], ["#pass2"], ["#pass3"], ["#pass4"], ["#pass5"], ["#pass6"], ["#pass7"], ["#pass8"], ["#pass9"], @@ -31,6 +31,7 @@ ["#pass140"], ["#pass141"], ["#pass142"], ["#pass143"], ["#pass144"], ["#pass145"], ["#pass146"], ["#pass147"], ["#pass148"], ["#pass149"], ["#pass150"], ["#pass151"], ["#pass152"], ["#pass153"], ["#pass154"], ["#pass155"], ["#pass156"], ["#pass157"], ["#pass158"], ["#pass159"], ["#pass160"], - ["#pass161"], ["#pass162"], ["#pass163"], ["#pass164"], ["#pass165"], ["#pass166"], ["#pass167"] + ["#pass161"], ["#pass162"], ["#pass163"], ["#pass164"], ["#pass165"], ["#pass166"], ["#pass167"], + ["#pass168"], ["#pass169"], ["#pass170"] ] }