mirror of
https://github.com/Kpa-clawbot/meshcore-analyzer.git
synced 2026-06-05 01:41:18 +00:00
933ef4e6ef
Red commit: d48c1add88 (CI:
https://github.com/Kpa-clawbot/CoreScope/actions/runs/26411462973)
Green commit CI:
https://github.com/Kpa-clawbot/CoreScope/actions/runs/26411699037
## What
Brings the map's three visual surfaces — cluster bubbles, role pills
inside cluster bubbles, and multi-byte hash labels on repeater markers —
up to WCAG 2.2 AA. Replaces the prior color-only signaling with
structural carriers (size, border-style, glyph, letter prefix) so color
is no longer the only channel.
## How
Locked design = Tufte's structural framing ([issue
comment](https://github.com/Kpa-clawbot/CoreScope/issues/1356#issuecomment-4535244400))
WITH the WCAG audit's "Minimal patch to reach AA" applied as overrides
([issue
comment](https://github.com/Kpa-clawbot/CoreScope/issues/1356#issuecomment-4535849354)).
Where the audit and the original proposal disagreed (border color, pill
text color, V3 accent palette, font sizes), the audit's values won.
## V1 cluster bubbles
- Neutral fill `rgba(33,41,54,0.92)` via new `--mc-cluster-fill` (was
per-bucket `--info / --warning / --accent`).
- Border-style ramp as the redundant non-color carrier of the count
bucket: `mc-sm` `1.5px solid`, `mc-md` `2.5px solid`, `mc-lg` `2px
double`.
- Border color `#666` + dark halo `box-shadow: 0 0 0 1px
rgba(0,0,0,0.5), 0 1px 2px rgba(0,0,0,0.35)` so the border edge is
visible against both Carto Positron (`#f8f9fa`) and Carto Dark Matter
(`#262626`).
- `<div role="img" aria-label="<n> nodes — <breakdown>">` with the count
+ pills wrapped `aria-hidden="true"` so the AT announcement is the
summary, not the literal glyphs.
## V2 role pills
- `ROLE_LETTERS` map (`R` / `C` / `M` / `S` / `O`) is the primary
carrier — visible inside every pill, so protanopes/deuteranopes can read
the role without depending on hue.
- Wong (2011) palette as the secondary carrier, declared as
`--mc-role-repeater/companion/room/sensor/observer` — does NOT touch the
reserved `--info / --warning / --accent` system vars.
- `color: #1a1a1a` on **all five** pills (CSS rule + inline
defense-in-depth). Passes SC 1.4.3 small-text (≥4.5:1) against every
Wong hue.
- Font now `0.625rem/1.1 ui-monospace` (was `9px`, audit bumped to
`10px`, this PR converts to `rem` so user font-size preferences scale
the pill).
- Per-pill `aria-label="<n> <role>s"`, `overflow: visible` so a user
`letter-spacing` override doesn't clip (SC 1.4.12).
## V3 multi-byte hash labels
- `MB_GLYPHS` prefix (`✓` / `?` / `✗`) is the primary non-color status
carrier; the hash text is the data.
- Neutral dark fill `--mc-mb-fill` + colored 3px left border via
per-status `--mc-mb-confirmed/suspected/unknown` (high-luminance set
`#56F0A0` / `#FFD966` / `#FF8888` — audit override of original Tol
"vibrant" set, which failed border-stripe SC 1.4.11).
- Font now `0.75rem/1.2 ui-monospace` (was `11px`, audit bumped to
`12px`, this PR converts to `rem` for SC 1.4.4 robustness).
- `<div role="img" aria-label="multi-byte <status>, hash <ID>"><span
aria-hidden="true">` so AT reads the meaningful label (not the literal
`✓ 3E`). Observer-overlay `★` carries `aria-hidden="true"` for the same
reason. Null `mbStatus` falls through to `"repeater hash <ID>"` cleanly
— no `"multi-byte undefined"`.
- Forced-colors graceful degradation via `@media (forced-colors:
active)` block mapping all three surfaces to `Canvas` / `CanvasText`
with `forced-color-adjust: auto` (NOT `none`).
## TDD red→green
| Commit | Files | CI |
|---|---|---|
| `d48c1add` (red) | `test-issue-1356-map-a11y.js`,
`.github/workflows/deploy.yml` (test + wiring only) | [**failure** — 27
assertion ✗, exit
1](https://github.com/Kpa-clawbot/CoreScope/actions/runs/26411462973) |
| `b94755e6` (green) | `public/map.js`, `public/style.css`,
`test-issue-1356-map-a11y.js` (impl) |
[**success**](https://github.com/Kpa-clawbot/CoreScope/actions/runs/26411699037)
|
| `ac63e6ab` | refactor: drop `MB_COLORS` alias, hoist `MB_MARKER_TINT`
(round-1 #3 + #4) | (round-2) |
| `8aad60cb` | style: font sizes to `rem` for SC 1.4.4 (round-1 #2) |
(round-2) |
| `50a1aab1` | test: round-1 coverage adds + de-tautologise V2.c / V3.h
(round-1 #5) | (round-2) |
Red commit failed on **assertions** (not compile error) — the harness
loaded `public/map.js` + `public/style.css` end-to-end and exhausted all
27 string-presence checks. Green commit lands the audit-overridden
design and clears 32/32. Round-2 commits extend coverage to 40/40
without altering the original red→green gate.
## WCAG SC addressed
- **SC 1.4.1 Use of Color (A)**: cluster size + border-style ramp; pill
capital-letter prefix; MB label glyph prefix. Every visual is now
carried by at least one non-color channel.
- **SC 1.4.3 Contrast Minimum (AA)**: cluster `#fff` count on composited
fill = 10.12:1 vs Positron / 14.64:1 vs Dark Matter. MB label text =
11.48:1 / 14.65:1. Pill `#1a1a1a` on Wong hues: R 5.43, C 9.10, M 6.14,
S 13.16, O 6.86 — all ≥4.5:1.
- **SC 1.4.11 Non-text Contrast (AA)**: cluster border `#666` = 4.83:1
vs Positron, 3.30:1 vs Dark Matter; MB stripes vs `--mc-mb-fill`:
`#56F0A0` 5.13, `#FFD966` 8.66, `#FF8888` 4.62. Stripe-vs-basemap edge
is mitigated by the 1px dark halo box-shadow on `.mc-mb-label`.
- **SC 1.3.1 Info & Relationships (A)**: every divIcon now has
`role="img"` + a descriptive `aria-label`; visible glyph spans are
`aria-hidden="true"` so AT reads the meaning, not the typography.
- **SC 1.4.5 Images of Text (AA)**: implemented surfaces use live text
(`<span>` + `<div>` with CSS font), not rasterised glyphs — user
font-size / zoom scale them. Where SVG markers are used (non-label
path), the textual information is also exposed via `marker.alt` + popup,
satisfying the "essential" exception.
## Manual verification
1. **Both Carto themes on staging.** Open https://analyzer.00id.net and
switch the basemap (Positron and Dark Matter) — cluster bubbles, pills,
and MB labels must remain legible on both. Border edge of cluster bubble
visible on Positron (was the original bug).
2. **Screen-reader (NVDA / VoiceOver) test.**
- Focus a cluster bubble → expect `"<n> nodes — <role breakdown>"` and
NO literal letter/number announce per pill.
- Focus a MB label on a repeater marker → expect `"multi-byte confirmed,
hash 3E"` (or whatever status/hash applies) and NO `"check mark thin
space 3 E"`.
- Observer-also-repeater label → still announces the meaningful label
only; ★ is silent.
3. **Coblis simulation** (or equivalent). Run cluster + pills + MB
labels through deuteranopia / protanopia / tritanopia simulation.
Cluster bucket must be distinguishable by size + border-style (without
hue). Pill role must be distinguishable by the letter (without hue). MB
status must be distinguishable by glyph (without hue).
4. **Windows High Contrast / forced-colors.** Toggle on; all three
surfaces should fall back to `Canvas` / `CanvasText` (no invisible
elements, no `forced-color-adjust: none` regression).
## Out of scope
Filed for separate follow-up issues (audit explicitly tagged these as
either pre-existing or modern-interpretation non-blockers):
1. **SC 2.1.1 Keyboard (A)** — cluster click-to-zoom is mouse-only today
(Leaflet markercluster limitation). Needs `role="button"` + `tabindex=0`
+ `keydown` handler. Pre-existing, not introduced by this PR.
2. **SC 2.4.7 Focus Visible (AA)** — moot until #1 is addressed (no
focusable target). When the cluster becomes focusable, a
`:focus-visible` outline must be added.
3. **`prefers-reduced-motion` gate** — `.mc-cluster:hover { transform:
scale(1.06) }` and the 120ms transition are untouched from pre-PR.
Should be gated on `@media (prefers-reduced-motion: reduce)` in a
follow-up hygiene pass.
4. **px → rem for non-font sizes** — this PR converts font sizes (the SC
1.4.4 sensitive surface). Border widths and small paddings are kept in
px because physical-pixel snapping matters more for borders than user
font-zoom.
Fixes #1356
---------
Co-authored-by: Kpa-clawbot <bot@kpa-clawbot.local>
201 lines
11 KiB
JavaScript
201 lines
11 KiB
JavaScript
/**
|
|
* #1356 — WCAG 2.2 AA accessibility for map cluster bubbles, role pills,
|
|
* and multi-byte hash labels.
|
|
*
|
|
* Locked design = Tufte's structural framing (drop color as primary signal,
|
|
* use shape / glyph / border-style as carriers) WITH the audit's "Minimal
|
|
* patch to Tufte's proposal to reach AA" applied.
|
|
*
|
|
* Design sources:
|
|
* - https://github.com/Kpa-clawbot/CoreScope/issues/1356#issuecomment-4535244400
|
|
* - https://github.com/Kpa-clawbot/CoreScope/issues/1356#issuecomment-4535849354
|
|
*
|
|
* Pure-string assertions (mirrors test-issue-1293-marker-shapes.js pattern)
|
|
* so this runs in the JS-unit-tests CI step without a browser.
|
|
*/
|
|
'use strict';
|
|
|
|
const fs = require('fs');
|
|
const path = require('path');
|
|
|
|
let passed = 0, failed = 0;
|
|
function assert(cond, msg) {
|
|
if (cond) { passed++; console.log(' ✓ ' + msg); }
|
|
else { failed++; console.error(' ✗ ' + msg); }
|
|
}
|
|
|
|
const mapSrc = fs.readFileSync(path.join(__dirname, 'public', 'map.js'), 'utf8');
|
|
const cssSrc = fs.readFileSync(path.join(__dirname, 'public', 'style.css'), 'utf8');
|
|
|
|
console.log('\n=== #1356 V1: cluster bubble — neutral fill, border-style ramp, ARIA ===');
|
|
|
|
// V1.a — CSS must define a neutral cluster fill constant (not the bucket color).
|
|
assert(/--mc-cluster-fill\s*:/.test(cssSrc),
|
|
'style.css declares --mc-cluster-fill CSS variable');
|
|
|
|
// V1.b — Per-bucket background MUST NOT be the old --info/--warning/--accent system colors.
|
|
// (Those system vars are reserved per AGENTS.md / issue scope.)
|
|
const clusterBlock = cssSrc.match(/\.mc-cluster\.mc-sm[\s\S]{0,400}\.mc-cluster\.mc-lg[^}]*\}/);
|
|
assert(clusterBlock && !/var\(--info|var\(--warning|var\(--accent/.test(clusterBlock[0]),
|
|
'cluster sm/md/lg no longer use --info / --warning / --accent for fill');
|
|
|
|
// V1.c — Border-style ramp (solid → heavier → double) is the redundant carrier.
|
|
assert(/\.mc-cluster\.mc-lg[^}]*double/.test(cssSrc),
|
|
'cluster lg uses "double" border-style as a non-color carrier');
|
|
|
|
// V1.d — Audit override: border color must be #666 (NOT white) plus a dark halo via box-shadow.
|
|
assert(/--mc-cluster-border\s*:\s*#666/i.test(cssSrc),
|
|
'--mc-cluster-border is #666 (audit fix for SC 1.4.11 vs Carto-light)');
|
|
assert(/\.mc-cluster[^{]*\{[\s\S]*?box-shadow[^;]*rgba\(0\s*,\s*0\s*,\s*0/i.test(cssSrc),
|
|
'.mc-cluster has a dark halo box-shadow (audit fix for border visibility)');
|
|
|
|
// V1.e — ARIA on the cluster div (rendered in makeClusterIcon).
|
|
assert(/role=["']img["']/.test(mapSrc) && /aria-label[^=]*=[^>]*nodes/.test(mapSrc),
|
|
'makeClusterIcon emits role="img" + aria-label summarising count + role breakdown');
|
|
assert(/' nodes — '/.test(mapSrc) || /\d+ nodes — /.test(mapSrc) ||
|
|
/total\s*\+\s*' nodes — '/.test(mapSrc),
|
|
'cluster aria-label matches /\\d+ nodes — / pattern (summary + breakdown)');
|
|
|
|
console.log('\n=== #1356 V2: role pills — letter primary, Wong palette, dark text ===');
|
|
|
|
// V2.a — A ROLE_LETTERS map is defined for the 5 roles.
|
|
assert(/ROLE_LETTERS\s*=\s*\{[\s\S]*?repeater[\s\S]*?['"]R['"][\s\S]*?companion[\s\S]*?['"]C['"][\s\S]*?room[\s\S]*?['"]M['"][\s\S]*?sensor[\s\S]*?['"]S['"][\s\S]*?observer[\s\S]*?['"]O['"]/.test(mapSrc),
|
|
'map.js defines ROLE_LETTERS with R/C/M/S/O for the five roles');
|
|
|
|
// V2.b — makeClusterIcon emits the letter (not just a count) inside the pill.
|
|
const pillEmitRe = /<span class="mc-pill[^>]*>[^<]*' \+\s*ROLE_LETTERS\[/;
|
|
assert(pillEmitRe.test(mapSrc) || /ROLE_LETTERS\[role\][\s\S]{0,200}mc-pill/.test(mapSrc) ||
|
|
/mc-pill[\s\S]{0,200}ROLE_LETTERS\[role\]/.test(mapSrc),
|
|
'pill HTML embeds ROLE_LETTERS[role] as the primary content');
|
|
|
|
// V2.c — Dark text on ALL five pills (audit override of Tufte's per-pill switch).
|
|
// Require the CSS rule `.mc-pill { color: #1a1a1a }` (authoritative).
|
|
// The inline-style fallback alone is NOT enough: a regression that drops the
|
|
// CSS rule but keeps a stray inline style would still green, masking the
|
|
// theming-illusion bug (round-1 adversarial #5 short-circuit).
|
|
assert(/\.mc-pill\b[^{]*\{[^}]*color\s*:\s*#1a1a1a/i.test(cssSrc),
|
|
'.mc-pill CSS rule sets color #1a1a1a (authoritative, not just inline-style fallback)');
|
|
assert(/class="mc-pill[^"]*"[^>]*style="[^"]*color:\s*#1a1a1a/i.test(mapSrc),
|
|
'.mc-pill render-site also emits inline color #1a1a1a (defense-in-depth for divIcon)');
|
|
|
|
// V2.d — font-size ≥ 10px (audit bumped from 9px).
|
|
const pillFontMatch = cssSrc.match(/\.mc-pill\b[^{]*\{[^}]*font[^;]*;/);
|
|
assert(pillFontMatch && /1[0-9]px|0\.625rem|0\.6875rem|0\.75rem/.test(pillFontMatch[0]),
|
|
'.mc-pill font-size is ≥ 10px (audit fix for SC 1.4.3 / 1.4.4)');
|
|
|
|
// V2.e — Wong palette declared as --mc-role-* constants.
|
|
['repeater','companion','room','sensor','observer'].forEach(function(r){
|
|
assert(new RegExp('--mc-role-' + r + '\\s*:').test(cssSrc),
|
|
'--mc-role-' + r + ' CSS variable declared');
|
|
});
|
|
|
|
// V2.f — per-pill aria-label "<N> <role>s".
|
|
assert(/aria-label="'\s*\+\s*n\s*\+\s*' '\s*\+\s*role/.test(mapSrc) ||
|
|
/aria-label=("|')[\s\S]{0,80}\+\s*n\s*\+[\s\S]{0,80}\+\s*role/.test(mapSrc),
|
|
'pill HTML emits aria-label with count + role');
|
|
|
|
// V2.g — DO NOT touch --info / --warning / --accent (out of scope hard rule).
|
|
const mcRoleBlock = cssSrc.match(/--mc-role-[\s\S]{0,1500}/);
|
|
assert(mcRoleBlock && !/--info\s*:|--warning\s*:|--accent\s*:/.test(mcRoleBlock[0]),
|
|
'role pill constants are --mc-* namespaced (do not redefine --info/--warning/--accent)');
|
|
|
|
console.log('\n=== #1356 V3: multi-byte hash labels — glyph + neutral fill + colored border-left ===');
|
|
|
|
// V3.a — MB_GLYPHS map for ✓ / ? / ✗.
|
|
assert(/MB_GLYPHS\s*=\s*\{[\s\S]*?confirmed[\s\S]*?['"\\]u2713|MB_GLYPHS\s*=\s*\{[\s\S]*?confirmed[\s\S]*?['"]\u2713['"]/.test(mapSrc) ||
|
|
/MB_GLYPHS\s*=\s*\{[\s\S]*?confirmed[\s\S]*?['"]✓['"]/.test(mapSrc),
|
|
'map.js defines MB_GLYPHS with ✓ for confirmed');
|
|
assert(/MB_GLYPHS[\s\S]*?suspected[\s\S]*?['"]\?['"]/.test(mapSrc),
|
|
'MB_GLYPHS.suspected === "?"');
|
|
assert(/MB_GLYPHS[\s\S]*?unknown[\s\S]*?['"\\]u2717|MB_GLYPHS[\s\S]*?unknown[\s\S]*?['"]✗['"]/.test(mapSrc),
|
|
'MB_GLYPHS.unknown === ✗ (u2717)');
|
|
|
|
// V3.b — Neutral fill constant for multi-byte label.
|
|
assert(/--mc-mb-fill\s*:/.test(cssSrc),
|
|
'--mc-mb-fill CSS variable declared (neutral fill, not status color)');
|
|
|
|
// V3.c — High-luminance accent set (audit override of Tol "vibrant").
|
|
// Confirmed #56F0A0 / suspected #FFD966 / unknown #FF8888.
|
|
assert(/--mc-mb-confirmed\s*:\s*#56F0A0/i.test(cssSrc),
|
|
'--mc-mb-confirmed is #56F0A0 (audit high-luminance set, not #117733)');
|
|
assert(/--mc-mb-suspected\s*:\s*#FFD966/i.test(cssSrc),
|
|
'--mc-mb-suspected is #FFD966');
|
|
assert(/--mc-mb-unknown\s*:\s*#FF8888/i.test(cssSrc),
|
|
'--mc-mb-unknown is #FF8888');
|
|
|
|
// V3.d — 3px colored left border in style.
|
|
assert(/border-left\s*:\s*3px solid/.test(cssSrc),
|
|
'.mc-mb-label has 3px solid border-left (colored accent stripe)');
|
|
|
|
// V3.e — makeRepeaterLabelIcon prepends MB_GLYPHS[status].
|
|
assert(/MB_GLYPHS\[[^\]]+\][\s\S]{0,200}shortHash|shortHash[\s\S]{0,200}MB_GLYPHS\[/.test(mapSrc),
|
|
'makeRepeaterLabelIcon prepends MB_GLYPHS glyph to the hash text');
|
|
|
|
// V3.f — aria-label "multi-byte <status>, hash <ID>".
|
|
assert(/aria-label="'\s*\+\s*ariaStatus\s*\+\s*'"/.test(mapSrc) ||
|
|
/'multi-byte '\s*\+\s*status\s*\+\s*', hash '\s*\+\s*shortHash/.test(mapSrc) ||
|
|
/aria-label="multi-byte \$\{[^}]+\}, hash \$\{shortHash\}"/.test(mapSrc),
|
|
'makeRepeaterLabelIcon emits aria-label "multi-byte <status>, hash <ID>"');
|
|
|
|
// V3.g — Glyph span must be aria-hidden so AT does not read "check mark 3 E".
|
|
assert(/<span aria-hidden="true">[\s\S]{0,100}shortHash|<span aria-hidden="true">'\s*\+\s*(?:glyph|visible)/.test(mapSrc) ||
|
|
/aria-hidden="true">'\s*\+\s*visible/.test(mapSrc),
|
|
'visible glyph+hash span is aria-hidden="true" (AT reads aria-label only)');
|
|
|
|
// V3.h — repeater label MUST use the neutral fill via var(--mc-mb-fill); MUST
|
|
// NOT paint background per-status (that would re-enable the pre-#1356
|
|
// color-only signal). Affirmative check on the neutral-fill rule AND
|
|
// negative check on the per-status bgColor pattern (round-1 adversarial #5:
|
|
// the prior `!removal || affirmative` form short-circuited to a tautology).
|
|
assert(/\.mc-mb-label\b[^{]*\{[^}]*background\s*:\s*var\(--mc-mb-fill\)/.test(cssSrc),
|
|
'.mc-mb-label background uses var(--mc-mb-fill) — neutral fill, not status color');
|
|
assert(!/bgColor\s*=\s*colorOverride\s*\|\|\s*s\.color/.test(mapSrc),
|
|
'old per-status bgColor pattern is gone (no per-status background painting)');
|
|
|
|
console.log('\n=== #1356 Round-1 coverage adds: dual-marker star, null mbStatus, forced-colors ===');
|
|
|
|
// COV-1 — Observer-also-repeater dual marker: the ★ star glyph inside
|
|
// makeRepeaterLabelIcon's obsIndicator branch MUST carry aria-hidden="true",
|
|
// otherwise the AT announcement is polluted with "black star" / "star" on
|
|
// top of the meaningful aria-label. Round-1 (Kent + adversarial) flagged.
|
|
// Match the exact obsIndicator construction shape: `isAlsoObserver ? ' <span aria-hidden="true" ... ★`.
|
|
assert(/isAlsoObserver[\s\S]{0,40}\?\s*['"][^'"]*<span\s+aria-hidden="true"[^>]*>[^<]*★/.test(mapSrc),
|
|
'observer-also-repeater star span carries aria-hidden="true" (no AT pollution)');
|
|
|
|
// COV-2 — makeRepeaterLabelIcon with no multi_byte_status field must NOT emit
|
|
// an aria-label containing "multi-byte undefined" (the obvious bug if the
|
|
// null-fallback branch is dropped). Verify the source has the explicit
|
|
// `mbStatus || null` + truthy-check structure that prevents this.
|
|
assert(/var\s+status\s*=\s*mbStatus\s*\|\|\s*null\s*;/.test(mapSrc),
|
|
'makeRepeaterLabelIcon normalises missing mbStatus to null (not "undefined")');
|
|
assert(/ariaStatus\s*=\s*status\s*\?\s*\(\s*['"]multi-byte\s/.test(mapSrc),
|
|
'ariaStatus uses ternary on truthy `status` — null falls through to "repeater hash <ID>" branch');
|
|
// Negative regression: no template/concat that would ever produce "multi-byte undefined".
|
|
assert(!/['"]multi-byte\s*['"]\s*\+\s*mbStatus(?![^,]*\?)/.test(mapSrc),
|
|
'no unconditional concat of "multi-byte " + mbStatus (would emit "multi-byte undefined" on null)');
|
|
|
|
// COV-3 — @media (forced-colors: active) block MUST exist in style.css AND
|
|
// MUST NOT contain `forced-color-adjust: none` anywhere within its body
|
|
// (audit explicitly warned against `none`; degrades High Contrast Mode).
|
|
const fcMatch = cssSrc.match(/@media\s*\(\s*forced-colors\s*:\s*active\s*\)\s*\{[\s\S]*?\n\}/);
|
|
assert(fcMatch, '@media (forced-colors: active) block present in style.css');
|
|
assert(fcMatch && !/forced-color-adjust\s*:\s*none/i.test(fcMatch[0]),
|
|
'@media (forced-colors: active) block does NOT use forced-color-adjust: none (audit regression guard)');
|
|
|
|
|
|
console.log('\n=== #1356 Hard rules: --info / --warning / --accent untouched ===');
|
|
|
|
// Sanity: ensure new --mc-* constants don't redefine the reserved system vars.
|
|
// (--info and --warning are only used via var(..., fallback) — they may not be declared
|
|
// at all; --accent IS declared.)
|
|
const newConstantsBlock = (cssSrc.match(/\/\*[^*]*#1356[\s\S]*?\*\/[\s\S]*?(?=\/\*|$)/) || ['', ''])[0];
|
|
assert(!/--info\s*:|--warning\s*:|--accent\s*:/.test(newConstantsBlock),
|
|
'#1356 CSS block does not redefine --info / --warning / --accent');
|
|
assert(/--accent\s*:/.test(cssSrc), '--accent CSS variable still defined');
|
|
|
|
console.log('\n=== Summary ===');
|
|
console.log(` Passed: ${passed}`);
|
|
console.log(` Failed: ${failed}`);
|
|
if (failed > 0) { console.error('\n#1356 FAIL'); process.exit(1); }
|
|
console.log('\n#1356 PASS');
|