react-native-accessibility-scanner 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (79) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +501 -0
  3. package/dist/cli/index.d.ts +3 -0
  4. package/dist/cli/index.d.ts.map +1 -0
  5. package/dist/cli/index.js +131 -0
  6. package/dist/cli/index.js.map +1 -0
  7. package/dist/eslint/index.d.ts +59 -0
  8. package/dist/eslint/index.d.ts.map +1 -0
  9. package/dist/eslint/index.js +181 -0
  10. package/dist/eslint/index.js.map +1 -0
  11. package/dist/index.d.ts +12 -0
  12. package/dist/index.d.ts.map +1 -0
  13. package/dist/index.js +36 -0
  14. package/dist/index.js.map +1 -0
  15. package/dist/reporters/console-reporter.d.ts +5 -0
  16. package/dist/reporters/console-reporter.d.ts.map +1 -0
  17. package/dist/reporters/console-reporter.js +67 -0
  18. package/dist/reporters/console-reporter.js.map +1 -0
  19. package/dist/reporters/json-reporter.d.ts +5 -0
  20. package/dist/reporters/json-reporter.d.ts.map +1 -0
  21. package/dist/reporters/json-reporter.js +13 -0
  22. package/dist/reporters/json-reporter.js.map +1 -0
  23. package/dist/reporters/markdown-reporter.d.ts +5 -0
  24. package/dist/reporters/markdown-reporter.d.ts.map +1 -0
  25. package/dist/reporters/markdown-reporter.js +66 -0
  26. package/dist/reporters/markdown-reporter.js.map +1 -0
  27. package/dist/rules/duplicate-labels.d.ts +25 -0
  28. package/dist/rules/duplicate-labels.d.ts.map +1 -0
  29. package/dist/rules/duplicate-labels.js +107 -0
  30. package/dist/rules/duplicate-labels.js.map +1 -0
  31. package/dist/rules/index.d.ts +13 -0
  32. package/dist/rules/index.d.ts.map +1 -0
  33. package/dist/rules/index.js +38 -0
  34. package/dist/rules/index.js.map +1 -0
  35. package/dist/rules/missing-hint.d.ts +14 -0
  36. package/dist/rules/missing-hint.d.ts.map +1 -0
  37. package/dist/rules/missing-hint.js +94 -0
  38. package/dist/rules/missing-hint.js.map +1 -0
  39. package/dist/rules/missing-label.d.ts +10 -0
  40. package/dist/rules/missing-label.d.ts.map +1 -0
  41. package/dist/rules/missing-label.js +75 -0
  42. package/dist/rules/missing-label.js.map +1 -0
  43. package/dist/rules/missing-role.d.ts +10 -0
  44. package/dist/rules/missing-role.d.ts.map +1 -0
  45. package/dist/rules/missing-role.js +82 -0
  46. package/dist/rules/missing-role.js.map +1 -0
  47. package/dist/rules/small-touch-target.d.ts +10 -0
  48. package/dist/rules/small-touch-target.d.ts.map +1 -0
  49. package/dist/rules/small-touch-target.js +90 -0
  50. package/dist/rules/small-touch-target.js.map +1 -0
  51. package/dist/rules/touchable-without-label.d.ts +17 -0
  52. package/dist/rules/touchable-without-label.d.ts.map +1 -0
  53. package/dist/rules/touchable-without-label.js +81 -0
  54. package/dist/rules/touchable-without-label.js.map +1 -0
  55. package/dist/scanner/config-loader.d.ts +3 -0
  56. package/dist/scanner/config-loader.d.ts.map +1 -0
  57. package/dist/scanner/config-loader.js +46 -0
  58. package/dist/scanner/config-loader.js.map +1 -0
  59. package/dist/scanner/file-scanner.d.ts +6 -0
  60. package/dist/scanner/file-scanner.d.ts.map +1 -0
  61. package/dist/scanner/file-scanner.js +82 -0
  62. package/dist/scanner/file-scanner.js.map +1 -0
  63. package/dist/scanner/index.d.ts +12 -0
  64. package/dist/scanner/index.d.ts.map +1 -0
  65. package/dist/scanner/index.js +64 -0
  66. package/dist/scanner/index.js.map +1 -0
  67. package/dist/scanner/scorer.d.ts +26 -0
  68. package/dist/scanner/scorer.d.ts.map +1 -0
  69. package/dist/scanner/scorer.js +65 -0
  70. package/dist/scanner/scorer.js.map +1 -0
  71. package/dist/types/index.d.ts +88 -0
  72. package/dist/types/index.d.ts.map +1 -0
  73. package/dist/types/index.js +13 -0
  74. package/dist/types/index.js.map +1 -0
  75. package/dist/utils/ast.d.ts +24 -0
  76. package/dist/utils/ast.d.ts.map +1 -0
  77. package/dist/utils/ast.js +170 -0
  78. package/dist/utils/ast.js.map +1 -0
  79. package/package.json +67 -0
@@ -0,0 +1,59 @@
1
+ /**
2
+ * eslint-plugin-react-native-accessibility-scanner
3
+ */
4
+ type ESLintNode = {
5
+ type: string;
6
+ [key: string]: unknown;
7
+ };
8
+ type RuleContext = {
9
+ report(descriptor: {
10
+ node: ESLintNode;
11
+ messageId: string;
12
+ data?: Record<string, string>;
13
+ }): void;
14
+ options: Array<{
15
+ minWidth?: number;
16
+ minHeight?: number;
17
+ }>;
18
+ };
19
+ type RuleListener = Record<string, (node: ESLintNode) => void>;
20
+ type RuleModule = {
21
+ meta: {
22
+ type: string;
23
+ docs: {
24
+ description: string;
25
+ recommended: boolean;
26
+ url: string;
27
+ };
28
+ schema: unknown[];
29
+ messages: Record<string, string>;
30
+ };
31
+ create(context: RuleContext): RuleListener;
32
+ };
33
+ declare const missingLabelRule: RuleModule;
34
+ declare const missingRoleRule: RuleModule;
35
+ declare const smallTouchTargetRule: RuleModule;
36
+ declare const plugin: {
37
+ meta: {
38
+ name: string;
39
+ version: string;
40
+ };
41
+ rules: {
42
+ "missing-label": RuleModule;
43
+ "missing-role": RuleModule;
44
+ "small-touch-target": RuleModule;
45
+ };
46
+ configs: {
47
+ recommended: {
48
+ plugins: string[];
49
+ rules: {
50
+ "react-native-accessibility-scanner/missing-label": string;
51
+ "react-native-accessibility-scanner/missing-role": string;
52
+ "react-native-accessibility-scanner/small-touch-target": string;
53
+ };
54
+ };
55
+ };
56
+ };
57
+ export default plugin;
58
+ export { missingLabelRule, missingRoleRule, smallTouchTargetRule };
59
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/eslint/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAYH,KAAK,UAAU,GAAG;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAA;CAAE,CAAC;AAC3D,KAAK,WAAW,GAAG;IACjB,MAAM,CAAC,UAAU,EAAE;QAAE,IAAI,EAAE,UAAU,CAAC;QAAC,SAAS,EAAE,MAAM,CAAC;QAAC,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;KAAE,GAAG,IAAI,CAAC;IACjG,OAAO,EAAE,KAAK,CAAC;QAAE,QAAQ,CAAC,EAAE,MAAM,CAAC;QAAC,SAAS,CAAC,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;CAC3D,CAAC;AACF,KAAK,YAAY,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC,IAAI,EAAE,UAAU,KAAK,IAAI,CAAC,CAAC;AAC/D,KAAK,UAAU,GAAG;IAChB,IAAI,EAAE;QACJ,IAAI,EAAE,MAAM,CAAC;QACb,IAAI,EAAE;YAAE,WAAW,EAAE,MAAM,CAAC;YAAC,WAAW,EAAE,OAAO,CAAC;YAAC,GAAG,EAAE,MAAM,CAAA;SAAE,CAAC;QACjE,MAAM,EAAE,OAAO,EAAE,CAAC;QAClB,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;KAClC,CAAC;IACF,MAAM,CAAC,OAAO,EAAE,WAAW,GAAG,YAAY,CAAC;CAC5C,CAAC;AAQF,QAAA,MAAM,gBAAgB,EAAE,UA4BvB,CAAC;AAIF,QAAA,MAAM,eAAe,EAAE,UA2BtB,CAAC;AAIF,QAAA,MAAM,oBAAoB,EAAE,UA6C3B,CAAC;AAIF,QAAA,MAAM,MAAM;;;;;;;;;;;;;;;;;;;;CAiBX,CAAC;AAEF,eAAe,MAAM,CAAC;AACtB,OAAO,EAAE,gBAAgB,EAAE,eAAe,EAAE,oBAAoB,EAAE,CAAC"}
@@ -0,0 +1,181 @@
1
+ "use strict";
2
+ /**
3
+ * eslint-plugin-react-native-accessibility-scanner
4
+ */
5
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
6
+ if (k2 === undefined) k2 = k;
7
+ var desc = Object.getOwnPropertyDescriptor(m, k);
8
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
9
+ desc = { enumerable: true, get: function() { return m[k]; } };
10
+ }
11
+ Object.defineProperty(o, k2, desc);
12
+ }) : (function(o, m, k, k2) {
13
+ if (k2 === undefined) k2 = k;
14
+ o[k2] = m[k];
15
+ }));
16
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
17
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
18
+ }) : function(o, v) {
19
+ o["default"] = v;
20
+ });
21
+ var __importStar = (this && this.__importStar) || (function () {
22
+ var ownKeys = function(o) {
23
+ ownKeys = Object.getOwnPropertyNames || function (o) {
24
+ var ar = [];
25
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
26
+ return ar;
27
+ };
28
+ return ownKeys(o);
29
+ };
30
+ return function (mod) {
31
+ if (mod && mod.__esModule) return mod;
32
+ var result = {};
33
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
34
+ __setModuleDefault(result, mod);
35
+ return result;
36
+ };
37
+ })();
38
+ Object.defineProperty(exports, "__esModule", { value: true });
39
+ exports.smallTouchTargetRule = exports.missingRoleRule = exports.missingLabelRule = void 0;
40
+ const ast_1 = require("../utils/ast");
41
+ const t = __importStar(require("@babel/types"));
42
+ function toBabelNode(node) {
43
+ return node;
44
+ }
45
+ // ─── Rule: missing-label ──────────────────────────────────────────────────────
46
+ const missingLabelRule = {
47
+ meta: {
48
+ type: "problem",
49
+ docs: {
50
+ description: "Interactive React Native elements must have an accessibilityLabel or Text child.",
51
+ recommended: true,
52
+ url: "https://github.com/YOUR_USERNAME/react-native-accessibility-scanner#missing-label",
53
+ },
54
+ schema: [],
55
+ messages: {
56
+ missingLabel: "<{{name}}> is interactive but has no accessibilityLabel or Text child. Add accessibilityLabel=\"Describe the action\".",
57
+ },
58
+ },
59
+ create(context) {
60
+ return {
61
+ JSXElement(node) {
62
+ const bNode = toBabelNode(node);
63
+ if (!t.isJSXElement(bNode))
64
+ return;
65
+ const opening = bNode.openingElement;
66
+ const name = (0, ast_1.getComponentName)(opening);
67
+ if (!name || !ast_1.INTERACTIVE_COMPONENTS.has(name))
68
+ return;
69
+ if ((0, ast_1.hasNonEmptyAttribute)(opening, "accessibilityLabel"))
70
+ return;
71
+ if ((0, ast_1.hasTextChild)(bNode))
72
+ return;
73
+ context.report({ node, messageId: "missingLabel", data: { name } });
74
+ },
75
+ };
76
+ },
77
+ };
78
+ exports.missingLabelRule = missingLabelRule;
79
+ // ─── Rule: missing-role ───────────────────────────────────────────────────────
80
+ const missingRoleRule = {
81
+ meta: {
82
+ type: "suggestion",
83
+ docs: {
84
+ description: "Interactive React Native elements should declare their accessibilityRole.",
85
+ recommended: true,
86
+ url: "https://github.com/YOUR_USERNAME/react-native-accessibility-scanner#missing-role",
87
+ },
88
+ schema: [],
89
+ messages: {
90
+ missingRole: "<{{name}}> is missing accessibilityRole. Add accessibilityRole=\"button\" (or appropriate role).",
91
+ },
92
+ },
93
+ create(context) {
94
+ return {
95
+ JSXElement(node) {
96
+ const bNode = toBabelNode(node);
97
+ if (!t.isJSXElement(bNode))
98
+ return;
99
+ const opening = bNode.openingElement;
100
+ const name = (0, ast_1.getComponentName)(opening);
101
+ if (!name || !ast_1.INTERACTIVE_COMPONENTS.has(name))
102
+ return;
103
+ if ((0, ast_1.hasNonEmptyAttribute)(opening, "accessibilityRole"))
104
+ return;
105
+ context.report({ node, messageId: "missingRole", data: { name } });
106
+ },
107
+ };
108
+ },
109
+ };
110
+ exports.missingRoleRule = missingRoleRule;
111
+ // ─── Rule: small-touch-target ─────────────────────────────────────────────────
112
+ const smallTouchTargetRule = {
113
+ meta: {
114
+ type: "suggestion",
115
+ docs: {
116
+ description: "Touchable elements should meet minimum touch target size (44×44pt).",
117
+ recommended: true,
118
+ url: "https://github.com/YOUR_USERNAME/react-native-accessibility-scanner#small-touch-target",
119
+ },
120
+ schema: [
121
+ {
122
+ type: "object",
123
+ properties: {
124
+ minWidth: { type: "number" },
125
+ minHeight: { type: "number" },
126
+ },
127
+ additionalProperties: false,
128
+ },
129
+ ],
130
+ messages: {
131
+ smallWidth: "<{{name}}> width={{width}} is below minimum {{min}}pt.",
132
+ smallHeight: "<{{name}}> height={{height}} is below minimum {{min}}pt.",
133
+ },
134
+ },
135
+ create(context) {
136
+ const options = context.options[0] ?? {};
137
+ const minWidth = options.minWidth ?? 44;
138
+ const minHeight = options.minHeight ?? 44;
139
+ return {
140
+ JSXElement(node) {
141
+ const bNode = toBabelNode(node);
142
+ if (!t.isJSXElement(bNode))
143
+ return;
144
+ const opening = bNode.openingElement;
145
+ const name = (0, ast_1.getComponentName)(opening);
146
+ if (!name || !ast_1.TOUCHABLE_COMPONENTS.has(name))
147
+ return;
148
+ const width = (0, ast_1.extractStyleDimension)(opening, "width");
149
+ const height = (0, ast_1.extractStyleDimension)(opening, "height");
150
+ if (width !== null && width < minWidth) {
151
+ context.report({ node, messageId: "smallWidth", data: { name, width: String(width), min: String(minWidth) } });
152
+ }
153
+ if (height !== null && height < minHeight) {
154
+ context.report({ node, messageId: "smallHeight", data: { name, height: String(height), min: String(minHeight) } });
155
+ }
156
+ },
157
+ };
158
+ },
159
+ };
160
+ exports.smallTouchTargetRule = smallTouchTargetRule;
161
+ // ─── Plugin export ────────────────────────────────────────────────────────────
162
+ const plugin = {
163
+ meta: { name: "react-native-accessibility-scanner", version: "1.0.0" },
164
+ rules: {
165
+ "missing-label": missingLabelRule,
166
+ "missing-role": missingRoleRule,
167
+ "small-touch-target": smallTouchTargetRule,
168
+ },
169
+ configs: {
170
+ recommended: {
171
+ plugins: ["react-native-accessibility-scanner"],
172
+ rules: {
173
+ "react-native-accessibility-scanner/missing-label": "error",
174
+ "react-native-accessibility-scanner/missing-role": "warn",
175
+ "react-native-accessibility-scanner/small-touch-target": "warn",
176
+ },
177
+ },
178
+ },
179
+ };
180
+ exports.default = plugin;
181
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/eslint/index.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,12 @@
1
+ export { AccessibilityScanner } from "./scanner";
2
+ export { loadConfig } from "./scanner/config-loader";
3
+ export { scanFile } from "./scanner/file-scanner";
4
+ export { computeScore, scoreFile, scoreEmoji } from "./scanner/scorer";
5
+ export { registerRule, getRule, getAllRules } from "./rules";
6
+ export { ConsoleReporter } from "./reporters/console-reporter";
7
+ export { JsonReporter } from "./reporters/json-reporter";
8
+ export { MarkdownReporter } from "./reporters/markdown-reporter";
9
+ export { MissingLabelRule, MissingRoleRule, SmallTouchTargetRule, DuplicateLabelsRule, MissingHintRule, TouchableWithoutLabelRule, } from "./rules";
10
+ export type { AccessibilityRule, AccessibilityIssue, ScanReport, ScanSummary, FileScanResult, ScanOptions, ScannerConfig, Severity, Reporter, TouchTargetConfig, RuleContext, } from "./types";
11
+ export { DEFAULT_CONFIG } from "./types";
12
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,oBAAoB,EAAE,MAAM,WAAW,CAAC;AACjD,OAAO,EAAE,UAAU,EAAE,MAAM,yBAAyB,CAAC;AACrD,OAAO,EAAE,QAAQ,EAAE,MAAM,wBAAwB,CAAC;AAClD,OAAO,EAAE,YAAY,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AACvE,OAAO,EAAE,YAAY,EAAE,OAAO,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AAG7D,OAAO,EAAE,eAAe,EAAE,MAAM,8BAA8B,CAAC;AAC/D,OAAO,EAAE,YAAY,EAAE,MAAM,2BAA2B,CAAC;AACzD,OAAO,EAAE,gBAAgB,EAAE,MAAM,+BAA+B,CAAC;AAGjE,OAAO,EACL,gBAAgB,EAChB,eAAe,EACf,oBAAoB,EACpB,mBAAmB,EACnB,eAAe,EACf,yBAAyB,GAC1B,MAAM,SAAS,CAAC;AAGjB,YAAY,EACV,iBAAiB,EACjB,kBAAkB,EAClB,UAAU,EACV,WAAW,EACX,cAAc,EACd,WAAW,EACX,aAAa,EACb,QAAQ,EACR,QAAQ,EACR,iBAAiB,EACjB,WAAW,GACZ,MAAM,SAAS,CAAC;AAEjB,OAAO,EAAE,cAAc,EAAE,MAAM,SAAS,CAAC"}
package/dist/index.js ADDED
@@ -0,0 +1,36 @@
1
+ "use strict";
2
+ // ─── Public API ───────────────────────────────────────────────────────────────
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.DEFAULT_CONFIG = exports.TouchableWithoutLabelRule = exports.MissingHintRule = exports.DuplicateLabelsRule = exports.SmallTouchTargetRule = exports.MissingRoleRule = exports.MissingLabelRule = exports.MarkdownReporter = exports.JsonReporter = exports.ConsoleReporter = exports.getAllRules = exports.getRule = exports.registerRule = exports.scoreEmoji = exports.scoreFile = exports.computeScore = exports.scanFile = exports.loadConfig = exports.AccessibilityScanner = void 0;
5
+ var scanner_1 = require("./scanner");
6
+ Object.defineProperty(exports, "AccessibilityScanner", { enumerable: true, get: function () { return scanner_1.AccessibilityScanner; } });
7
+ var config_loader_1 = require("./scanner/config-loader");
8
+ Object.defineProperty(exports, "loadConfig", { enumerable: true, get: function () { return config_loader_1.loadConfig; } });
9
+ var file_scanner_1 = require("./scanner/file-scanner");
10
+ Object.defineProperty(exports, "scanFile", { enumerable: true, get: function () { return file_scanner_1.scanFile; } });
11
+ var scorer_1 = require("./scanner/scorer");
12
+ Object.defineProperty(exports, "computeScore", { enumerable: true, get: function () { return scorer_1.computeScore; } });
13
+ Object.defineProperty(exports, "scoreFile", { enumerable: true, get: function () { return scorer_1.scoreFile; } });
14
+ Object.defineProperty(exports, "scoreEmoji", { enumerable: true, get: function () { return scorer_1.scoreEmoji; } });
15
+ var rules_1 = require("./rules");
16
+ Object.defineProperty(exports, "registerRule", { enumerable: true, get: function () { return rules_1.registerRule; } });
17
+ Object.defineProperty(exports, "getRule", { enumerable: true, get: function () { return rules_1.getRule; } });
18
+ Object.defineProperty(exports, "getAllRules", { enumerable: true, get: function () { return rules_1.getAllRules; } });
19
+ // Reporters
20
+ var console_reporter_1 = require("./reporters/console-reporter");
21
+ Object.defineProperty(exports, "ConsoleReporter", { enumerable: true, get: function () { return console_reporter_1.ConsoleReporter; } });
22
+ var json_reporter_1 = require("./reporters/json-reporter");
23
+ Object.defineProperty(exports, "JsonReporter", { enumerable: true, get: function () { return json_reporter_1.JsonReporter; } });
24
+ var markdown_reporter_1 = require("./reporters/markdown-reporter");
25
+ Object.defineProperty(exports, "MarkdownReporter", { enumerable: true, get: function () { return markdown_reporter_1.MarkdownReporter; } });
26
+ // Built-in rule classes (for extension / custom configs)
27
+ var rules_2 = require("./rules");
28
+ Object.defineProperty(exports, "MissingLabelRule", { enumerable: true, get: function () { return rules_2.MissingLabelRule; } });
29
+ Object.defineProperty(exports, "MissingRoleRule", { enumerable: true, get: function () { return rules_2.MissingRoleRule; } });
30
+ Object.defineProperty(exports, "SmallTouchTargetRule", { enumerable: true, get: function () { return rules_2.SmallTouchTargetRule; } });
31
+ Object.defineProperty(exports, "DuplicateLabelsRule", { enumerable: true, get: function () { return rules_2.DuplicateLabelsRule; } });
32
+ Object.defineProperty(exports, "MissingHintRule", { enumerable: true, get: function () { return rules_2.MissingHintRule; } });
33
+ Object.defineProperty(exports, "TouchableWithoutLabelRule", { enumerable: true, get: function () { return rules_2.TouchableWithoutLabelRule; } });
34
+ var types_1 = require("./types");
35
+ Object.defineProperty(exports, "DEFAULT_CONFIG", { enumerable: true, get: function () { return types_1.DEFAULT_CONFIG; } });
36
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AAAA,iFAAiF;;;AAEjF,qCAAiD;AAAxC,+GAAA,oBAAoB,OAAA;AAC7B,yDAAqD;AAA5C,2GAAA,UAAU,OAAA;AACnB,uDAAkD;AAAzC,wGAAA,QAAQ,OAAA;AACjB,2CAAuE;AAA9D,sGAAA,YAAY,OAAA;AAAE,mGAAA,SAAS,OAAA;AAAE,oGAAA,UAAU,OAAA;AAC5C,iCAA6D;AAApD,qGAAA,YAAY,OAAA;AAAE,gGAAA,OAAO,OAAA;AAAE,oGAAA,WAAW,OAAA;AAE3C,YAAY;AACZ,iEAA+D;AAAtD,mHAAA,eAAe,OAAA;AACxB,2DAAyD;AAAhD,6GAAA,YAAY,OAAA;AACrB,mEAAiE;AAAxD,qHAAA,gBAAgB,OAAA;AAEzB,yDAAyD;AACzD,iCAOiB;AANf,yGAAA,gBAAgB,OAAA;AAChB,wGAAA,eAAe,OAAA;AACf,6GAAA,oBAAoB,OAAA;AACpB,4GAAA,mBAAmB,OAAA;AACnB,wGAAA,eAAe,OAAA;AACf,kHAAA,yBAAyB,OAAA;AAkB3B,iCAAyC;AAAhC,uGAAA,cAAc,OAAA"}
@@ -0,0 +1,5 @@
1
+ import type { Reporter, ScanReport } from "../types";
2
+ export declare class ConsoleReporter implements Reporter {
3
+ report(scan: ScanReport): void;
4
+ }
5
+ //# sourceMappingURL=console-reporter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"console-reporter.d.ts","sourceRoot":"","sources":["../../src/reporters/console-reporter.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,QAAQ,EAAE,UAAU,EAAgC,MAAM,UAAU,CAAC;AAcnF,qBAAa,eAAgB,YAAW,QAAQ;IAC9C,MAAM,CAAC,IAAI,EAAE,UAAU,GAAG,IAAI;CAwD/B"}
@@ -0,0 +1,67 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.ConsoleReporter = void 0;
7
+ const chalk_1 = __importDefault(require("chalk"));
8
+ const SEVERITY_LABEL = {
9
+ high: chalk_1.default.bgRed.white.bold(" HIGH "),
10
+ medium: chalk_1.default.bgYellow.black.bold(" MED "),
11
+ low: chalk_1.default.bgBlue.white.bold(" LOW "),
12
+ };
13
+ const SEVERITY_COLOR = {
14
+ high: chalk_1.default.red,
15
+ medium: chalk_1.default.yellow,
16
+ low: chalk_1.default.blue,
17
+ };
18
+ class ConsoleReporter {
19
+ report(scan) {
20
+ const { summary, issues } = scan;
21
+ console.log("\n" + chalk_1.default.bold.cyan("═══════════════════════════════════════════════════════"));
22
+ console.log(chalk_1.default.bold.cyan(" React Native Accessibility Scanner"));
23
+ console.log(chalk_1.default.bold.cyan("═══════════════════════════════════════════════════════\n"));
24
+ // Summary
25
+ console.log(chalk_1.default.bold("📊 Summary"));
26
+ console.log(` Scanned : ${chalk_1.default.white(summary.scannedFiles)} files`);
27
+ console.log(` Issues : ${chalk_1.default.white(summary.totalIssues)} total`);
28
+ console.log(` ${chalk_1.default.red(`■ HIGH ${summary.highIssues}`)} ${chalk_1.default.yellow(`■ MEDIUM ${summary.mediumIssues}`)} ${chalk_1.default.blue(`■ LOW ${summary.lowIssues}`)}`);
29
+ console.log(` Duration: ${summary.durationMs}ms\n`);
30
+ if (issues.length === 0) {
31
+ console.log(chalk_1.default.green("✅ No accessibility issues found! Great work.\n"));
32
+ return;
33
+ }
34
+ // Group by severity
35
+ const grouped = {
36
+ high: issues.filter((i) => i.severity === "high"),
37
+ medium: issues.filter((i) => i.severity === "medium"),
38
+ low: issues.filter((i) => i.severity === "low"),
39
+ };
40
+ for (const severity of ["high", "medium", "low"]) {
41
+ const group = grouped[severity];
42
+ if (group.length === 0)
43
+ continue;
44
+ console.log(chalk_1.default.bold(`\n${SEVERITY_LABEL[severity]} ${group.length} issue${group.length > 1 ? "s" : ""}`));
45
+ console.log(chalk_1.default.dim("─".repeat(55)));
46
+ for (const issue of group) {
47
+ const color = SEVERITY_COLOR[severity];
48
+ const location = `${issue.file}:${issue.line}:${issue.column}`;
49
+ console.log(`\n ${color.bold(issue.ruleName)}`);
50
+ console.log(` ${chalk_1.default.dim(location)}`);
51
+ console.log(` ${chalk_1.default.white(issue.message)}`);
52
+ if (issue.codeSnippet) {
53
+ console.log(`\n ${chalk_1.default.dim("Code:")} ${chalk_1.default.gray(issue.codeSnippet)}`);
54
+ }
55
+ console.log(` ${chalk_1.default.green("Fix:")} ${chalk_1.default.greenBright(issue.suggestion)}`);
56
+ }
57
+ }
58
+ console.log("\n" + chalk_1.default.dim("─".repeat(55)));
59
+ // Exit hint
60
+ if (summary.highIssues > 0) {
61
+ console.log(chalk_1.default.red(`\n⚠️ ${summary.highIssues} HIGH severity issue${summary.highIssues > 1 ? "s" : ""} found.`));
62
+ }
63
+ console.log(chalk_1.default.dim("\nTip: Run with --json to output machine-readable results.\n"));
64
+ }
65
+ }
66
+ exports.ConsoleReporter = ConsoleReporter;
67
+ //# sourceMappingURL=console-reporter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"console-reporter.js","sourceRoot":"","sources":["../../src/reporters/console-reporter.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ import type { Reporter, ScanReport } from "../types";
2
+ export declare class JsonReporter implements Reporter {
3
+ report(scan: ScanReport): string;
4
+ }
5
+ //# sourceMappingURL=json-reporter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"json-reporter.d.ts","sourceRoot":"","sources":["../../src/reporters/json-reporter.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,UAAU,EAAE,MAAM,UAAU,CAAC;AAErD,qBAAa,YAAa,YAAW,QAAQ;IAC3C,MAAM,CAAC,IAAI,EAAE,UAAU,GAAG,MAAM;CAUjC"}
@@ -0,0 +1,13 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.JsonReporter = void 0;
4
+ class JsonReporter {
5
+ report(scan) {
6
+ return JSON.stringify({
7
+ summary: scan.summary,
8
+ issues: scan.issues,
9
+ }, null, 2);
10
+ }
11
+ }
12
+ exports.JsonReporter = JsonReporter;
13
+ //# sourceMappingURL=json-reporter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"json-reporter.js","sourceRoot":"","sources":["../../src/reporters/json-reporter.ts"],"names":[],"mappings":";;;AAEA,MAAa,YAAY;IACvB,MAAM,CAAC,IAAgB;QACrB,OAAO,IAAI,CAAC,SAAS,CACnB;YACE,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,MAAM,EAAE,IAAI,CAAC,MAAM;SACpB,EACD,IAAI,EACJ,CAAC,CACF,CAAC;IACJ,CAAC;CACF;AAXD,oCAWC"}
@@ -0,0 +1,5 @@
1
+ import type { Reporter, ScanReport } from "../types";
2
+ export declare class MarkdownReporter implements Reporter {
3
+ report(scan: ScanReport): string;
4
+ }
5
+ //# sourceMappingURL=markdown-reporter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"markdown-reporter.d.ts","sourceRoot":"","sources":["../../src/reporters/markdown-reporter.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,UAAU,EAAgC,MAAM,UAAU,CAAC;AAQnF,qBAAa,gBAAiB,YAAW,QAAQ;IAC/C,MAAM,CAAC,IAAI,EAAE,UAAU,GAAG,MAAM;CA6DjC"}
@@ -0,0 +1,66 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.MarkdownReporter = void 0;
4
+ const SEVERITY_EMOJI = {
5
+ high: "🔴",
6
+ medium: "🟡",
7
+ low: "🔵",
8
+ };
9
+ class MarkdownReporter {
10
+ report(scan) {
11
+ const { summary, issues } = scan;
12
+ const date = new Date().toISOString().split("T")[0];
13
+ const lines = [];
14
+ lines.push("# ♿ React Native Accessibility Report");
15
+ lines.push(`\n> Generated on ${date}`);
16
+ lines.push("\n---\n");
17
+ // Summary table
18
+ lines.push("## 📊 Summary\n");
19
+ lines.push("| Metric | Value |");
20
+ lines.push("| ------ | ----- |");
21
+ lines.push(`| Files Scanned | ${summary.scannedFiles} |`);
22
+ lines.push(`| Total Issues | ${summary.totalIssues} |`);
23
+ lines.push(`| 🔴 High | ${summary.highIssues} |`);
24
+ lines.push(`| 🟡 Medium | ${summary.mediumIssues} |`);
25
+ lines.push(`| 🔵 Low | ${summary.lowIssues} |`);
26
+ lines.push(`| Duration | ${summary.durationMs}ms |`);
27
+ if (issues.length === 0) {
28
+ lines.push("\n---\n");
29
+ lines.push("## ✅ No Issues Found\n");
30
+ lines.push("Great work! Your React Native app passes all accessibility checks.");
31
+ return lines.join("\n");
32
+ }
33
+ // Issues grouped by severity
34
+ for (const severity of ["high", "medium", "low"]) {
35
+ const group = issues.filter((i) => i.severity === severity);
36
+ if (group.length === 0)
37
+ continue;
38
+ const label = severity.charAt(0).toUpperCase() + severity.slice(1);
39
+ lines.push(`\n---\n\n## ${SEVERITY_EMOJI[severity]} ${label} Severity (${group.length})\n`);
40
+ // Group by rule within severity
41
+ const byRule = new Map();
42
+ for (const issue of group) {
43
+ if (!byRule.has(issue.ruleId))
44
+ byRule.set(issue.ruleId, []);
45
+ byRule.get(issue.ruleId).push(issue);
46
+ }
47
+ for (const [, ruleIssues] of byRule.entries()) {
48
+ const first = ruleIssues[0];
49
+ lines.push(`### ${first.ruleName}`);
50
+ lines.push(`\n> ${first.message.split(".")[0]}.\n`);
51
+ for (const issue of ruleIssues) {
52
+ lines.push(`**📄 File:** \`${issue.file}\``);
53
+ lines.push(`**📍 Location:** Line ${issue.line}, Column ${issue.column}`);
54
+ if (issue.codeSnippet) {
55
+ lines.push(`\n**Code:**\n\`\`\`tsx\n${issue.codeSnippet}\n\`\`\``);
56
+ }
57
+ lines.push(`\n**💡 Fix:** ${issue.suggestion}\n`);
58
+ lines.push("---\n");
59
+ }
60
+ }
61
+ }
62
+ return lines.join("\n");
63
+ }
64
+ }
65
+ exports.MarkdownReporter = MarkdownReporter;
66
+ //# sourceMappingURL=markdown-reporter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"markdown-reporter.js","sourceRoot":"","sources":["../../src/reporters/markdown-reporter.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,25 @@
1
+ import * as t from "@babel/types";
2
+ import type { AccessibilityRule, AccessibilityIssue, RuleContext } from "../types";
3
+ /**
4
+ * Duplicate Labels Rule
5
+ *
6
+ * Unlike other rules which run per-node, this rule is designed to be run
7
+ * once per file: call it with the Program node (root) after all other nodes
8
+ * are processed, passing the collected label map.
9
+ *
10
+ * The scanner calls this rule's `runOnFile` method after traversal.
11
+ * The `run` method handles per-node label collection (returns empty array
12
+ * since duplicate detection requires the full file scan).
13
+ */
14
+ export declare class DuplicateLabelsRule implements AccessibilityRule {
15
+ id: string;
16
+ name: string;
17
+ description: string;
18
+ severity: "low";
19
+ private labelMap;
20
+ reset(): void;
21
+ run(node: t.Node, context: RuleContext): AccessibilityIssue[];
22
+ /** Call after traversal to collect duplicate label issues */
23
+ collectIssues(context: RuleContext): AccessibilityIssue[];
24
+ }
25
+ //# sourceMappingURL=duplicate-labels.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"duplicate-labels.d.ts","sourceRoot":"","sources":["../../src/rules/duplicate-labels.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,CAAC,MAAM,cAAc,CAAC;AAClC,OAAO,KAAK,EAAE,iBAAiB,EAAE,kBAAkB,EAAE,WAAW,EAAE,MAAM,UAAU,CAAC;AAQnF;;;;;;;;;;GAUG;AACH,qBAAa,mBAAoB,YAAW,iBAAiB;IAC3D,EAAE,SAAsB;IACxB,IAAI,SAAoC;IACxC,WAAW,SACuH;IAClI,QAAQ,EAAG,KAAK,CAAU;IAG1B,OAAO,CAAC,QAAQ,CAA4E;IAE5F,KAAK;IAIL,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,OAAO,EAAE,WAAW,GAAG,kBAAkB,EAAE;IAsB7D,6DAA6D;IAC7D,aAAa,CAAC,OAAO,EAAE,WAAW,GAAG,kBAAkB,EAAE;CAsB1D"}