-
Notifications
You must be signed in to change notification settings - Fork 1
/
get-requirements-map.ts
142 lines (127 loc) · 4.3 KB
/
get-requirements-map.ts
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
135
136
137
138
139
140
141
142
import assert from "assert";
import { outdent } from "outdent";
import { joinStrings, indent } from "../../utils";
import { getAccessibilityRequirement } from "../../act/get-accessibility-requirement";
import { RuleFrontMatter, AccessibilityRequirement } from "../../types";
type RequirementEntry = [string, AccessibilityRequirement];
type Args = { frontmatter: RuleFrontMatter };
export const headingText = "Accessibility Requirements Mapping";
export const secondaryReqText = outdent`
This rule is related to the following accessibility requirements, but was
not designed to test this requirements directly. These
[secondary requirements](https://w3c.github.io/wcag-act/act-rules-format.html#secondary-requirements)
can either be stricter than the rule requires, or may be satisfied in ways
not tested by the rule:
`;
export function getRequirementsMap({ frontmatter }: Args): string {
const requirementText: string[] = [`## ${headingText}`];
const accMapping: RequirementEntry[] = Object.entries(
frontmatter.accessibility_requirements || {}
);
const secondaryReq: RequirementEntry[] = [];
const conformanceReq: RequirementEntry[] = [];
for (const requirement of accMapping) {
if ("secondary" in requirement[1]) {
secondaryReq.push(requirement);
} else {
conformanceReq.push(requirement);
}
}
if (conformanceReq.length > 0) {
requirementText.push(outdent`
<ul class="act-requirements-list">
${indent(conformanceReq.map(conformanceReqItem).join("\n"))}
</ul>
`);
}
if (secondaryReq.length > 0) {
requirementText.push(
"### Secondary Requirements",
secondaryReqText,
secondaryReq.map(secondaryReqItem).join("\n")
);
}
if (requirementText.length === 1) {
requirementText.push("This rule is not required for conformance.");
}
return joinStrings(...requirementText);
}
function conformanceReqItem([
requirementId,
mapping,
]: RequirementEntry): string {
assert(!("secondary" in mapping), "Must be conformance requirement");
const accRequirement = getAccessibilityRequirement({
requirementId,
title: mapping.title,
});
if (!accRequirement) {
return `<li>This rule is <strong>not required</strong> for conformance to WCAG 2.1 at any level.</li>`;
}
const { conformanceLevel, url, requirementType, shortTitle } = accRequirement;
const title = accRequirement.title || mapping.title;
const requiredText = conformanceLevel
? `<strong>Required for conformance</strong> to ${conformanceLevel}.`
: `Not required for conformance to any W3C accessibility recommendation.`;
return outdent`
<li><details>
<summary><span>${title}</span></summary>
<ul>
<li><a href="${url}">Learn more about ${shortTitle}</a></li>
<li>${requiredText}</li>
<li>Outcome mapping: ${indent(
outcomeMapping(requirementType, mapping),
" ",
4
).trim()}</li>
</ul>
</details></li>
`;
}
function outcomeMapping(
requirementType = "success criterion",
{
failed = "not satisfied",
passed = "further testing is needed",
inapplicable = "further testing is needed",
}
) {
return outdent`
<ul>
<li>Any <code>failed</code> outcomes: ${requirementType} ${getConformanceText(
failed
)}</li>
<li>All <code>passed</code> outcomes: ${requirementType} ${getConformanceText(
passed
)}</li>
<li>An <code>inapplicable</code> outcome: ${requirementType} ${getConformanceText(
inapplicable
)}</li>
</ul>
`;
}
const outcomeMap: Record<string, string> = {
satisfied: "is satisfied",
"not satisfied": "is not satisfied",
"further testing needed": "needs further testing",
};
function getConformanceText(outcome: string): string {
let outcomeValue = ``;
if (outcomeMap[outcome]) {
outcomeValue = outcomeMap[outcome];
}
return outcomeValue;
}
function secondaryReqItem([
requirementId,
requirement,
]: RequirementEntry): string {
assert("secondary" in requirement, "secondary mus be defined");
const { title, secondary } = requirement;
const accRequirement = getAccessibilityRequirement({ requirementId, title });
if (!accRequirement) {
return `- ${title}: ${secondary.trim()}`;
}
const label = accRequirement.title || title;
return `- [${label}](${accRequirement.url}): ${secondary.trim()}`;
}