Files
vyndr/tests/unit/vyndrMobile.test.js
T
builtbykev f88961885c Session 37: Design system Phase F — mobile parity: 5-tab bar, More sheet, PWA polish (1872 tests)
VYNDR 2.0 conversion, Phase F (mobile is the PWA we launch first). Frontend-only;
zero backend changes.

- BottomTabBar rewritten to the §6 5-tab spec: Slate/Terminal/Scan/Ledger/More,
  with Scan as the prominent raised grade-green action. Shown for anon too (only
  mobile nav). Integrated More bottom sheet (sheet-up, backdrop dismiss, 48px mono
  rows). iOS safe-area + 44px touch targets.
- Nav hamburger retired on mobile (tab bar owns nav).
- globals.css mobile section: tab-bar hidden >=768, main bottom padding,
  grade-hero 80px, terminal-grid stacks, game-lines horizontal scroll.
- PWA: manifest shortcuts (Slate/Scan/Terminal) + categories; viewport-fit=cover.

Gotcha: `as const` on the TABS array broke type-check (distinct literal types);
fixed with a shared TabDef interface.

19 new tests. Backend 1853 -> 1872, 145 suites, zero regressions. Web build clean.

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-06-16 10:11:52 -04:00

107 lines
4.2 KiB
JavaScript

// VYNDR 2.0 — Phase F mobile parity (Session 37): 5-tab bar, More sheet,
// mobile CSS, PWA manifest/viewport. The manifest is required as JSON; the
// .tsx/.css are asserted against source text (plain-JS Jest, no transform).
const fs = require('fs');
const path = require('path');
const WEB = path.join(__dirname, '..', '..', 'web', 'src');
const read = (rel) => fs.readFileSync(path.join(WEB, rel), 'utf8');
const manifest = require('../../web/public/manifest.json');
describe('Phase F — BottomTabBar (5-tab spec)', () => {
const src = read('components/BottomTabBar.tsx');
it('renders the five spec tabs', () => {
['Slate', 'Terminal', 'Scan', 'Ledger', 'More'].forEach((label) => {
expect(src).toContain(`'${label}'`);
});
});
it('routes the primary tabs to real pages', () => {
['/dashboard', '/terminal', '/scan', '/ledger'].forEach((href) => expect(src).toContain(href));
});
it('makes Scan the prominent (primary) action in grade-green', () => {
expect(src).toContain('primary: true');
expect(src).toContain("background: 'var(--g-a)'");
});
it('marks the active tab grade-green, inactive faint', () => {
expect(src).toContain("'var(--g-a)'");
expect(src).toContain("'var(--text-2)'");
});
it('respects the iOS safe area', () => {
expect(src).toContain('env(safe-area-inset-bottom');
});
it('uses the mobile-only class so it hides on desktop', () => {
expect(src).toContain("className=\"mobile-tab-bar\"");
});
});
describe('Phase F — More bottom sheet', () => {
const src = read('components/BottomTabBar.tsx');
it('lists the secondary routes', () => {
['/compare', '/tracker', '/blog', '/invite', '/pricing', '/account', '/help', '/about', '/responsible-gambling'].forEach((href) =>
expect(src).toContain(href),
);
});
it('slides up with a dismissible backdrop', () => {
expect(src).toContain('sheet-up');
expect(src).toContain('rgba(6,6,11,.6)'); // backdrop
expect(src).toContain('onClick={() => setMoreOpen(false)}'); // tap-to-dismiss
});
it('gives sheet items a ≥44px touch target', () => {
expect(src).toContain('minHeight: 48');
});
});
describe('Phase F — mobile CSS (globals.css)', () => {
const css = read('app/globals.css');
it('hides the tab bar at the desktop breakpoint', () => {
expect(css).toMatch(/@media \(min-width: 768px\)[\s\S]*?\.mobile-tab-bar \{ display: none/);
});
it('pads main for the bottom bar on mobile with safe-area', () => {
expect(css).toMatch(/@media \(max-width: 767px\)/);
expect(css).toContain('env(safe-area-inset-bottom, 0px)');
});
it('shrinks the grade hero and stacks the terminal grid on mobile', () => {
expect(css).toContain('.grade-hero { font-size: 80px');
expect(css).toContain('.terminal-grid { grid-template-columns: 1fr');
});
it('enables horizontal scroll for book tables', () => {
expect(css).toContain('.game-lines-grid');
expect(css).toContain('overflow-x: auto');
});
});
describe('Phase F — Nav mobile header', () => {
it('retires the hamburger on mobile (tab bar owns nav)', () => {
const src = read('components/Nav.tsx');
expect(src).toMatch(/nav-mobile-toggle[\s\S]*?display: 'none'/);
});
});
describe('Phase F — GradeResultCard mobile hook', () => {
it('tags the grade letter with the grade-hero class', () => {
expect(read('components/vyndr/GradeResultCard.tsx')).toContain('grade-hero');
});
});
describe('Phase F — PWA manifest + viewport (§11)', () => {
it('is a standalone PWA themed to the void', () => {
expect(manifest.display).toBe('standalone');
expect(manifest.theme_color.toLowerCase()).toBe('#06060b');
expect(manifest.background_color.toLowerCase()).toBe('#06060b');
});
it('ships deep-link shortcuts for Slate / Scan / Terminal', () => {
const urls = (manifest.shortcuts || []).map((s) => s.url);
expect(urls).toContain('/dashboard');
expect(urls).toContain('/scan');
expect(urls).toContain('/terminal');
});
it('declares its categories', () => {
expect(manifest.categories).toContain('sports');
expect(manifest.categories).toContain('productivity');
});
it('sets viewport-fit=cover for the notch', () => {
expect(read('app/layout.tsx')).toContain("viewportFit: 'cover'");
});
});