f88961885c
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>
107 lines
4.2 KiB
JavaScript
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'");
|
|
});
|
|
});
|