|
1 |
| -// Regex to match alt text that is the same as the default image filename |
2 |
| -// e.g. "Screen Shot 2020-10-20 at 2 52 27 PM" |
3 |
| -// e.g. "Screenshot 2020-10-20 at 2 52 27 PM" |
4 |
| -// e.g. "Clean Shot 2020-10-20 @45x" |
5 |
| -// e.g. "image" |
6 |
| -const defaultMacOsScreenshotMarkdownRegex = |
7 |
| - /^(Screen|Clean) ?[S|s]hot \d{4}-\d{2}-\d{2}/gi; |
8 |
| -const imageMarkdownRegex = /^image$/i; |
| 1 | +/** |
| 2 | + * Examples: |
| 3 | + * * "Screen Shot 2020-10-20 at 2 52 27 PM" |
| 4 | + * * "Screenshot 2020-10-20 at 2 52 27 PM" |
| 5 | + * * "Clean Shot 2020-10-20 @45x" |
| 6 | + */ |
| 7 | +const defaultScreenshotRegex = |
| 8 | + "(?:screen|clean) ?shot \\d{4}-\\d{2}-\\d{2}[^'\"\\]]*"; |
9 | 9 |
|
10 |
| -const defaultMacOsScreenshotHtmlRegex = |
11 |
| - /alt="(Screen|Clean) ?[S|s]hot \d{4}-\d{2}-\d{2}/gi; |
12 |
| -const imageHtmlRegex = /alt="image"/i; |
| 10 | +const imageRegex = "image"; |
| 11 | +const combinedRegex = `(${[defaultScreenshotRegex, imageRegex].join("|")})`; |
| 12 | + |
| 13 | +const markdownAltRegex = new RegExp(`!\\[${combinedRegex}\\]\\(.*\\)`, "gid"); |
| 14 | +const htmlAltRegex = new RegExp(`alt=["']${combinedRegex}["']`, "gid"); |
13 | 15 |
|
14 | 16 | module.exports = {
|
15 | 17 | names: ["GH001", "no-default-alt-text"],
|
16 |
| - description: |
17 |
| - "Images should set meaningful alternative text (alt text), and not use the macOS default screenshot filename or `Image`.", |
| 18 | + description: "Images should have meaningful alternative text (alt text)", |
18 | 19 | information: new URL(
|
19 | 20 | "https://github.com/github/markdownlint-github/blob/main/docs/rules/GH001-no-default-alt-text.md"
|
20 | 21 | ),
|
21 | 22 | tags: ["accessibility", "images"],
|
22 | 23 | function: function GH001(params, onError) {
|
23 |
| - // markdown syntax |
24 |
| - const inlineTokens = params.tokens.filter((t) => t.type === "inline"); |
25 |
| - for (const token of inlineTokens) { |
26 |
| - const imageTokens = token.children.filter((t) => t.type === "image"); |
27 |
| - for (const image of imageTokens) { |
28 |
| - if ( |
29 |
| - image.content.match(defaultMacOsScreenshotMarkdownRegex) || |
30 |
| - image.content.match(imageMarkdownRegex) |
31 |
| - ) { |
32 |
| - onError({ |
33 |
| - lineNumber: image.lineNumber, |
34 |
| - detail: `For image: ${image.content}`, |
35 |
| - }); |
36 |
| - } |
37 |
| - } |
38 |
| - } |
| 24 | + for (const [lineIndex, line] of params.lines.entries()) { |
| 25 | + for (const match of [ |
| 26 | + ...line.matchAll(markdownAltRegex), |
| 27 | + ...line.matchAll(htmlAltRegex), |
| 28 | + ]) { |
| 29 | + // The alt text is contained in the first capture group |
| 30 | + const altText = match[1]; |
| 31 | + const [startIndex] = match.indices[1]; |
39 | 32 |
|
40 |
| - // html syntax |
41 |
| - let lineNumber = 1; |
42 |
| - for (const line of params.lines) { |
43 |
| - if ( |
44 |
| - line.match(defaultMacOsScreenshotHtmlRegex) || |
45 |
| - line.match(imageHtmlRegex) |
46 |
| - ) { |
47 | 33 | onError({
|
48 |
| - lineNumber, |
49 |
| - detail: `For image: ${line}`, |
| 34 | + lineNumber: lineIndex + 1, |
| 35 | + range: [startIndex + 1, altText.length], |
50 | 36 | });
|
51 | 37 | }
|
52 |
| - lineNumber++; |
53 | 38 | }
|
54 | 39 | },
|
55 | 40 | };
|
0 commit comments