Shad 3ffd0e68a7 Mission Control demo v2
Polished command-center for FlowMaster with two data modes:
- SNAPSHOT: bundled src/scenarios.json from demo.flow-master.ai
- LIVE: in-browser fetch via src/lib/api.ts (dev-login + bearer)

Scenarios:
- procurement, extra-1, extra-2 (live from EA2)
- ar, hcm, gl, service (industry blueprints, same typed shell)

Honesty pass after Oracle review:
- No invented numbers (Telemetry derives SLA + agent acceptance from real data)
- Preview-only actions fire toasts naming the endpoint to wire them
- Blueprint tours framed as 'industry blueprint', not 'we don't have this yet'
- Mode pill + last-fetch age + refresh in topbar
- Dev CORS dodged via vite proxy; production deploys same-origin

18 vitest tests + 26 playwright smoke assertions + DOM layout audit.

Constraint: cross-origin live mode rejected by browser → fall back to snapshot
Rejected: hardcoded SLA % | dishonest demo metrics
Directive: wire preview-only action handlers to /api/runtime/transactions/{id}/actions to ship them for real
Confidence: high
Scope-risk: narrow
Not-tested: production deployment via flowmaster-ops overlay
2026-06-14 00:09:32 +04:00

133 lines
5.6 KiB
TypeScript

// Cinematic landing scene: brand, scenarios pick, start tour CTA.
import { motion } from "framer-motion";
import { useApp } from "../state/store";
import { liveMeta } from "../data/scenarios";
import { Sparkles, Arrow, Cmd, Bot, Pulse } from "../components/icons";
export default function Landing() {
const setScene = useApp((s) => s.setScene);
const setScenarioId = useApp((s) => s.setScenarioId);
const startTour = useApp((s) => s.startTour);
const setCmdOpen = useApp((s) => s.setCmdOpen);
const scenarios = useApp((s) => s.scenarios);
const mode = useApp((s) => s.mode);
const setMode = useApp((s) => s.setMode);
const liveLoading = useApp((s) => s.liveLoading);
const liveTotals = useApp((s) => s.liveTotals);
return (
<div className="landing">
<div className="landing-bg" aria-hidden />
<div className="landing-grid" aria-hidden />
<header className="landing-top">
<div className="brand-lock">
<span className="brand-mark" />
<span className="brand-name">FlowMaster</span>
<span className="brand-divider" />
<span className="brand-sub">Mission Control</span>
</div>
<button className="link-btn" onClick={() => setCmdOpen(true)}>
<Cmd size={13} /> Command <kbd>K</kbd>
</button>
</header>
<main className="landing-main">
<motion.div
className="landing-hero"
initial={{ opacity: 0, y: 16 }}
animate={{ opacity: 1, y: 0 }}
transition={{ duration: 0.45 }}
>
<span className="hero-eyebrow"><Sparkles size={13} /> Business-as-code · Mission Control</span>
<h1 className="hero-title">
Every process. <span className="hl">One control surface.</span>
</h1>
<p className="hero-sub">
FlowMaster turns the operational map of a company into living,
typed processes backed by humans, agents, and rules and gives you a
single command-center to drive them. Procurement scenarios are real,
backed by EA2 on{" "}
<span className="mono">{liveMeta.fetchedFrom?.replace("https://", "") ?? "demo"}</span>;
AR, HCM, GL, and Service are industry blueprints showing how this
same shell extends to any process family.
</p>
<div className="hero-actions">
<button className="btn btn-primary btn-lg" onClick={startTour}>
<Sparkles size={14} /> Start guided tour
</button>
<button className="btn btn-ghost btn-lg" onClick={() => setScene("mission")}>
Skip to Mission Control <Arrow size={13} />
</button>
<button
className={`btn btn-ghost btn-lg${mode === "live" ? " is-on" : ""}`}
onClick={() => setMode(mode === "live" ? "snapshot" : "live")}
disabled={liveLoading}
title="Toggle between bundled snapshot and a live in-browser fetch from demo.flow-master.ai"
>
{liveLoading ? <span className="spin" /> : <Pulse size={13} />}
{mode === "live" ? "Live mode · on" : "Go live"}
</button>
</div>
<div className="hero-stats">
<div className="stat">
<Pulse size={12} />
<span className="stat-v mono">{liveTotals?.workItems ?? liveMeta.workItems}</span>
<span className="stat-l">{mode === "live" ? "live work items (now)" : "snapshot work items"}</span>
</div>
<div className="stat">
<span className="stat-v mono">{liveTotals?.distinctDefs ?? liveMeta.distinctDefs}</span>
<span className="stat-l">process definitions</span>
</div>
<div className="stat">
<Bot size={12} />
<span className="stat-v mono">{scenarios.length}</span>
<span className="stat-l">scenarios in catalog</span>
</div>
<div className="stat">
<span className={`mode-pill mode-${mode}`}>{mode === "live" ? "LIVE" : "SNAPSHOT"}</span>
<span className="stat-l">data mode</span>
</div>
</div>
</motion.div>
<motion.div
className="landing-cards"
initial={{ opacity: 0, y: 18 }}
animate={{ opacity: 1, y: 0 }}
transition={{ delay: 0.12, duration: 0.45 }}
>
{scenarios.map((s) => (
<button
key={s.id}
className="sc-card"
style={{ ["--sc-accent" as string]: s.family.accent }}
onClick={() => { setScenarioId(s.id); setScene("mission"); }}
>
<div className="sc-card-top">
<span className="sc-card-mark" />
<span className={`tag ${s.live ? "tag-live" : "tag-syn"}`}>{s.live ? "live" : "blueprint"}</span>
</div>
<h3 className="sc-card-title">{s.family.label}</h3>
<p className="sc-card-sub">{s.family.subtitle}</p>
<div className="sc-card-meta">
<span>{s.steps.length} steps</span>
<span>·</span>
<span>{s.queue.length} cases</span>
<span>·</span>
<span className="mono">{s.version}</span>
</div>
<div className="sc-card-cta">Open <Arrow size={12} /></div>
</button>
))}
</motion.div>
</main>
<footer className="landing-foot">
<span className="foot-eyebrow">FlowMaster · Mission Control demo · synthesised on top of demo.flow-master.ai</span>
</footer>
</div>
);
}