diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-19 00:47:55 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-19 00:47:55 +0000 |
commit | 26a029d407be480d791972afb5975cf62c9360a6 (patch) | |
tree | f435a8308119effd964b339f76abb83a57c29483 /accessible/tests/browser/pivot | |
parent | Initial commit. (diff) | |
download | firefox-26a029d407be480d791972afb5975cf62c9360a6.tar.xz firefox-26a029d407be480d791972afb5975cf62c9360a6.zip |
Adding upstream version 124.0.1.upstream/124.0.1
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'accessible/tests/browser/pivot')
-rw-r--r-- | accessible/tests/browser/pivot/browser.toml | 10 | ||||
-rw-r--r-- | accessible/tests/browser/pivot/browser_pivot.js | 103 | ||||
-rw-r--r-- | accessible/tests/browser/pivot/head.js | 122 |
3 files changed, 235 insertions, 0 deletions
diff --git a/accessible/tests/browser/pivot/browser.toml b/accessible/tests/browser/pivot/browser.toml new file mode 100644 index 0000000000..f6552741e5 --- /dev/null +++ b/accessible/tests/browser/pivot/browser.toml @@ -0,0 +1,10 @@ +[DEFAULT] +subsuite = "a11y" +support-files = [ + "!/accessible/tests/browser/shared-head.js", + "head.js", + "!/accessible/tests/mochitest/*.js", +] +prefs = ["javascript.options.asyncstack_capture_debuggee_only=false"] + +["browser_pivot.js"] diff --git a/accessible/tests/browser/pivot/browser_pivot.js b/accessible/tests/browser/pivot/browser_pivot.js new file mode 100644 index 0000000000..bd46ae4933 --- /dev/null +++ b/accessible/tests/browser/pivot/browser_pivot.js @@ -0,0 +1,103 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +"use strict"; + +/** + * Tests Pivot API + */ +addAccessibleTask( + ` + <h1 id="heading-1-1">Main Title</h1> + <h2 id="heading-2-1" aria-hidden="true">First Section Title</h2> + <p id="paragraph-1"> + Lorem ipsum <strong>dolor</strong> sit amet. Integer vitae urna + leo, id <a href="#">semper</a> nulla. + </p> + <h2 id="heading-2-2" aria-hidden="undefined">Second Section Title</h2> + <p id="paragraph-2" aria-hidden=""> + Sed accumsan luctus lacus, vitae mollis arcu tristique vulputate.</p> + <p id="paragraph-3" aria-hidden="true"> + <a href="#" id="hidden-link">Maybe</a> it was the other <i>George Michael</i>. + You know, the <a href="#">singer-songwriter</a>. + </p> + <p style="opacity: 0;" id="paragraph-4"> + This is completely transparent + </p> + <iframe + src="data:text/html,<html><body>An <i>embedded</i> document.</body></html>"> + </iframe> + <div id="hide-me">Hide me</div> + <p id="links" aria-hidden="false"> + <a href="http://mozilla.org" title="Link 1 title">Link 1</a> + <a href="http://mozilla.org" title="Link 2 title">Link 2</a> + <a href="http://mozilla.org" title="Link 3 title">Link 3</a> + </p> + <ul> + <li>Hello<span> </span></li> + <li>World</li> + </ul> + `, + async function (browser, docAcc) { + let pivot = gAccService.createAccessiblePivot(docAcc); + testPivotSequence(pivot, HeadersTraversalRule, [ + "heading-1-1", + "heading-2-2", + ]); + + testPivotSequence(pivot, ObjectTraversalRule, [ + "Main Title", + "Lorem ipsum ", + "dolor", + " sit amet. Integer vitae urna leo, id ", + "semper", + " nulla. ", + "Second Section Title", + "Sed accumsan luctus lacus, vitae mollis arcu tristique vulputate.", + "An ", + "embedded", + " document.", + "Hide me", + "Link 1", + "Link 2", + "Link 3", + "Hello", + "World", + ]); + + let hideMeAcc = findAccessibleChildByID(docAcc, "hide-me"); + let onHide = waitForEvent(EVENT_HIDE, hideMeAcc); + invokeContentTask(browser, [], () => { + content.document.getElementById("hide-me").remove(); + }); + + await onHide; + testFailsWithNotInTree( + () => pivot.next(hideMeAcc, ObjectTraversalRule), + "moveNext from defunct accessible should fail" + ); + + let linksAcc = findAccessibleChildByID(docAcc, "links"); + + let removedRootPivot = gAccService.createAccessiblePivot(linksAcc); + onHide = waitForEvent(EVENT_HIDE, linksAcc); + invokeContentTask(browser, [], () => { + content.document.getElementById("links").remove(); + }); + + await onHide; + testFailsWithNotInTree( + () => removedRootPivot.last(ObjectTraversalRule), + "moveLast with pivot with defunct root should fail" + ); + + let [x, y] = getBounds(findAccessibleChildByID(docAcc, "heading-1-1")); + let hitacc = pivot.atPoint(x + 1, y + 1, HeadersTraversalRule); + is(getIdOrName(hitacc), "heading-1-1", "Matching accessible at point"); + + hitacc = pivot.atPoint(x - 1, y - 1, HeadersTraversalRule); + ok(!hitacc, "No heading at given point"); + }, + { iframe: true, remoteIframe: true, topLevel: true, chrome: true } +); diff --git a/accessible/tests/browser/pivot/head.js b/accessible/tests/browser/pivot/head.js new file mode 100644 index 0000000000..8389190a69 --- /dev/null +++ b/accessible/tests/browser/pivot/head.js @@ -0,0 +1,122 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +"use strict"; + +/* exported HeadersTraversalRule, ObjectTraversalRule, testPivotSequence, testFailsWithNotInTree */ + +// Load the shared-head file first. +Services.scriptloader.loadSubScript( + "chrome://mochitests/content/browser/accessible/tests/browser/shared-head.js", + this +); + +/* import-globals-from ../../mochitest/layout.js */ +/* import-globals-from ../../mochitest/role.js */ +/* import-globals-from ../../mochitest/states.js */ +loadScripts( + { name: "common.js", dir: MOCHITESTS_DIR }, + { name: "promisified-events.js", dir: MOCHITESTS_DIR }, + { name: "states.js", dir: MOCHITESTS_DIR }, + { name: "role.js", dir: MOCHITESTS_DIR }, + { name: "layout.js", dir: MOCHITESTS_DIR } +); + +const FILTER_MATCH = nsIAccessibleTraversalRule.FILTER_MATCH; +const FILTER_IGNORE = nsIAccessibleTraversalRule.FILTER_IGNORE; +const FILTER_IGNORE_SUBTREE = nsIAccessibleTraversalRule.FILTER_IGNORE_SUBTREE; + +const NS_ERROR_NOT_IN_TREE = 0x80780026; + +// ////////////////////////////////////////////////////////////////////////////// +// Traversal rules + +/** + * Rule object to traverse all focusable nodes and text nodes. + */ +const HeadersTraversalRule = { + match(acc) { + return acc.role == ROLE_HEADING ? FILTER_MATCH : FILTER_IGNORE; + }, + + QueryInterface: ChromeUtils.generateQI([nsIAccessibleTraversalRule]), +}; + +/** + * Traversal rule for all focusable nodes or leafs. + */ +const ObjectTraversalRule = { + match(acc) { + let [state, extstate] = getStates(acc); + if (state & STATE_INVISIBLE) { + return FILTER_IGNORE; + } + + if ((extstate & EXT_STATE_OPAQUE) == 0) { + return FILTER_IGNORE | FILTER_IGNORE_SUBTREE; + } + + let rv = FILTER_IGNORE; + let role = acc.role; + if ( + hasState(acc, STATE_FOCUSABLE) && + role != ROLE_DOCUMENT && + role != ROLE_INTERNAL_FRAME + ) { + rv = FILTER_IGNORE_SUBTREE | FILTER_MATCH; + } else if ( + acc.childCount == 0 && + role != ROLE_LISTITEM_MARKER && + acc.name.trim() + ) { + rv = FILTER_MATCH; + } + + return rv; + }, + + QueryInterface: ChromeUtils.generateQI([nsIAccessibleTraversalRule]), +}; + +function getIdOrName(acc) { + let id = getAccessibleDOMNodeID(acc); + if (id) { + return id; + } + return acc.name; +} + +function* pivotNextGenerator(pivot, rule) { + for (let acc = pivot.first(rule); acc; acc = pivot.next(acc, rule)) { + yield acc; + } +} + +function* pivotPreviousGenerator(pivot, rule) { + for (let acc = pivot.last(rule); acc; acc = pivot.prev(acc, rule)) { + yield acc; + } +} + +function testPivotSequence(pivot, rule, expectedSequence) { + is( + JSON.stringify([...pivotNextGenerator(pivot, rule)].map(getIdOrName)), + JSON.stringify(expectedSequence), + "Forward pivot sequence is correct" + ); + is( + JSON.stringify([...pivotPreviousGenerator(pivot, rule)].map(getIdOrName)), + JSON.stringify([...expectedSequence].reverse()), + "Reverse pivot sequence is correct" + ); +} + +function testFailsWithNotInTree(func, msg) { + try { + func(); + ok(false, msg); + } catch (x) { + is(x.result, NS_ERROR_NOT_IN_TREE, `Expecting NOT_IN_TREE: ${msg}`); + } +} |