-
Notifications
You must be signed in to change notification settings - Fork 16
Expand file tree
/
Copy pathPhabricatorA11yFixes.user.js
More file actions
134 lines (119 loc) · 4.21 KB
/
PhabricatorA11yFixes.user.js
File metadata and controls
134 lines (119 loc) · 4.21 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
// ==UserScript==
// @name Phabricator Accessibility Fixes
// @namespace http://axSgrease.nvaccess.org/
// @description Improves the accessibility of Phabricator.
// @author James Teh <jteh@mozilla.com>
// @copyright 2018-2025 Mozilla Corporation
// @license Mozilla Public License version 2.0
// @version 2025.1
// @grant GM_log
// @include https://phabricator.services.mozilla.com/D*
// ==/UserScript==
/*** Functions for common tweaks. ***/
function makeHeading(el, level) {
el.setAttribute("role", "heading");
el.setAttribute("aria-level", level);
}
function makeRegion(el, label) {
el.setAttribute("role", "region");
el.setAttribute("aria-label", label);
}
function makeButton(el, label) {
el.setAttribute("role", "button");
el.setAttribute("aria-label", label);
}
function makePresentational(el) {
el.setAttribute("role", "presentation");
}
function setLabel(el, label) {
el.setAttribute("aria-label", label);
}
function labelHiddenStatusIcon(el, label) {
el.removeAttribute("aria-hidden");
el.setAttribute("role", "image");
setLabel(el, label);
}
/*** Code to apply the tweaks when appropriate. ***/
function applyTweaks(root, tweaks) {
for (let tweak of tweaks) {
for (let el of root.querySelectorAll(tweak.selector)) {
if (Array.isArray(tweak.tweak)) {
let [func, ...args] = tweak.tweak;
func(el, ...args);
} else {
tweak.tweak(el);
}
}
}
}
function init() {
applyTweaks(document, LOAD_TWEAKS);
applyTweaks(document, DYNAMIC_TWEAKS);
}
let observer = new MutationObserver(function(mutations) {
for (let mutation of mutations) {
try {
if (mutation.type === "childList") {
for (let node of mutation.addedNodes) {
if (node.nodeType != Node.ELEMENT_NODE) {
continue;
}
applyTweaks(node, DYNAMIC_TWEAKS);
}
}/* else if (mutation.type === "attributes") {
if (mutation.attributeName == "class") {
onClassModified(mutation.target);
}
}*/
} catch (e) {
// Catch exceptions for individual mutations so other mutations are still handled.
GM_log("Exception while handling mutation: " + e);
}
}
});
observer.observe(document, {childList: true,/* attributes: true,*/
subtree: true/*, attributeFilter: ["class"]*/});
/*** Define the actual tweaks. ***/
// Tweaks that only need to be applied on load.
const LOAD_TWEAKS = [
// There are some off-screen headings to denote various sections, but they
// are h3 instead of h1 as they should be.
{selector: '.phui-main-column .phui-timeline-view h3.aural-only, .phui-comment-preview-view .phui-timeline-view h3.aural-only, .phui-comment-form-view h3.aural-only',
tweak: [makeHeading, 1]},
// The diff is an h1, so the files inside the diff should be an h2, not an h1.
{selector: '.differential-file-icon-header',
tweak: [makeHeading, 2]},
// Reviewer status icons.
{selector: '.phui-status-item-target .fa-circle-o',
tweak: [labelHiddenStatusIcon, "pending"]},
{selector: '.phui-status-item-target .fa-minus-circle',
tweak: [labelHiddenStatusIcon, "pending blocking"]},
{selector: '.phui-status-item-target .fa-check-circle',
tweak: [labelHiddenStatusIcon, "accepted"]},
{selector: '.phui-status-item-target .fa-check-circle-o',
tweak: [labelHiddenStatusIcon, "accepted prior"]},
{selector: '.phui-status-item-target .fa-times-circle',
tweak: [labelHiddenStatusIcon, "requested changes"]},
{selector: '.phui-status-item-target .fa-times-circle-o',
tweak: [labelHiddenStatusIcon, "requested changes to prior"]},
{selector: '.phui-status-item-target .fa-comment',
tweak: [labelHiddenStatusIcon, "comment"]},
]
// Tweaks that must be applied whenever a node is added.
const DYNAMIC_TWEAKS = [
// Timeline headings, "Summary" heading.
{selector: '.phui-timeline-title, .phui-property-list-section-header',
tweak: [makeHeading, 2]},
// Inline comment headings.
{selector: '.differential-inline-comment-head .inline-head-left',
tweak: [makeHeading, 3]},
{selector: '.phui-timeline-image, .phui-head-thing-image',
tweak: makePresentational},
// Code line numbers.
{selector: '.remarkup-code th',
// We don't want these to be header cells, as this causes a heap of spurious
// verbosity.
tweak: el => el.setAttribute("role", "cell")},
]
/*** Lights, camera, action! ***/
init();