Files
bellsystems-cp/strategies/Flash Page Rework/bellsystems-console-template/project/Flashing.html

1144 lines
52 KiB
HTML

<!DOCTYPE html><html lang="en" style="--accent: #22e07a;"><head>
<meta charset="utf-8">
<title>Provisioning Wizard — Flash · Bellsystems Console</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="preconnect" href="https://fonts.googleapis.com">
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin="">
<link href="https://fonts.googleapis.com/css2?family=Geist:wght@300;400;500;600;700&amp;family=JetBrains+Mono:wght@400;500;600;700&amp;display=swap" rel="stylesheet">
<style>
:root{
--bg-0:#07090b;
--bg-1:#0b0f13;
--bg-2:#10161c;
--bg-3:#151c24;
--line:#1e2730;
--line-strong:#2a3642;
--text-0:#eef3f8;
--text-1:#c8d2dc;
--text-2:#8a97a4;
--text-3:#5a6775;
--accent:#22e07a; /* primary green */
--accent-ink:#062211;
--accent-dim:#0f3a23;
--accent-glow: 0 0 0 1px rgba(34,224,122,.35), 0 0 22px -6px rgba(34,224,122,.6);
--warn:#ffb347;
--danger:#ff5a6a;
--blue:#5ab0ff;
--violet:#b48bff;
--radius:12px;
--radius-sm:8px;
--mono:"JetBrains Mono", ui-monospace, Menlo, monospace;
--sans:"Geist", ui-sans-serif, system-ui, -apple-system, "Segoe UI", sans-serif;
}
*{box-sizing:border-box}
html,body{margin:0;padding:0}
body{
font-family:var(--sans);
color:var(--text-0);
background:
radial-gradient(1200px 600px at 85% -10%, rgba(34,224,122,.06), transparent 60%),
radial-gradient(900px 500px at -10% 110%, rgba(90,176,255,.05), transparent 60%),
var(--bg-0);
min-height:100vh;
-webkit-font-smoothing:antialiased;
font-feature-settings:"ss01","cv11";
letter-spacing:-0.005em;
}
/* Subtle noise */
body::before{
content:"";
position:fixed;inset:0;
pointer-events:none;
opacity:.035;
background-image:url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' width='160' height='160'><filter id='n'><feTurbulence type='fractalNoise' baseFrequency='0.9' numOctaves='2' stitchTiles='stitch'/></filter><rect width='100%' height='100%' filter='url(%23n)'/></svg>");
mix-blend-mode:overlay;
z-index:0;
}
.shell{position:relative;z-index:1;max-width:1440px;margin:0 auto;padding:28px 40px 48px}
/* -------- Top bar -------- */
.topbar{display:flex;align-items:flex-start;justify-content:space-between;gap:24px;margin-bottom:28px}
.brand{display:flex;align-items:center;gap:14px}
.logo{
width:40px;height:40px;border-radius:10px;
background:linear-gradient(135deg,#0c1117,#161e27);
border:1px solid var(--line-strong);
display:grid;place-items:center;
box-shadow: inset 0 0 0 1px rgba(255,255,255,.02);
position:relative;
}
.logo svg{width:22px;height:22px}
.title h1{font-size:20px;font-weight:600;margin:0;letter-spacing:-0.01em}
.title p{margin:2px 0 0;color:var(--text-2);font-size:13px}
.crumbs{display:flex;align-items:center;gap:10px;color:var(--text-2);font-size:12.5px;font-family:var(--mono)}
.crumbs a{color:var(--text-2);text-decoration:none}
.crumbs a:hover{color:var(--text-0)}
.crumbs .sep{color:var(--text-3)}
.top-actions{display:flex;gap:10px;align-items:center}
/* -------- Stepper -------- */
.stepper{
display:grid;grid-template-columns:repeat(5, 1fr);
gap:10px;
padding:14px;
border:1px solid var(--line);
border-radius:14px;
background:linear-gradient(180deg, rgba(255,255,255,.02), rgba(255,255,255,0));
margin-bottom:24px;
position:relative;
}
.step{
display:flex;align-items:center;gap:12px;
padding:12px 14px;
border-radius:10px;
position:relative;
min-width:0;
}
.step .num{
width:26px;height:26px;border-radius:50%;
display:grid;place-items:center;
font-family:var(--mono);font-size:12px;font-weight:600;
background:var(--bg-3);color:var(--text-2);
border:1px solid var(--line-strong);
flex-shrink:0;
}
.step.done .num{background:rgba(34,224,122,.12);color:var(--accent);border-color:rgba(34,224,122,.4)}
.step.active .num{background:var(--accent);color:var(--accent-ink);border-color:transparent;box-shadow:var(--accent-glow)}
.step .label{font-size:12.5px;color:var(--text-2);text-transform:uppercase;letter-spacing:.08em;font-weight:500}
.step.active .label{color:var(--text-0)}
.step.done .label{color:var(--text-1)}
.step .sub{font-size:11px;color:var(--text-3);font-family:var(--mono);margin-top:1px}
.step .meta{display:flex;flex-direction:column;min-width:0}
.step-divider{
position:absolute;top:50%;right:-6px;width:12px;height:1px;
background:var(--line-strong);
}
.step:last-child .step-divider{display:none}
/* -------- Main grid -------- */
.main{
display:grid;
grid-template-columns: minmax(0, 1.1fr) minmax(0, 1fr);
gap:20px;
}
/* Layout: stacked */
body.layout-stacked .main{grid-template-columns:1fr}
body.layout-stacked .console-body{min-height:260px;max-height:360px}
/* Layout: drawer */
body.layout-drawer .main{grid-template-columns:1fr}
body.layout-drawer .console-card{
position:fixed;top:0;right:0;bottom:0;width:min(640px, 92vw);
z-index:40;border-radius:0;border-left:1px solid var(--line-strong);
transform:translateX(100%);transition:transform .25s ease;
box-shadow:-20px 0 60px -10px rgba(0,0,0,.7);
}
body.layout-drawer.drawer-open .console-card{transform:translateX(0)}
body.layout-drawer .console-body{min-height:0;max-height:none;flex:1}
.drawer-toggle{
display:none;position:fixed;bottom:20px;right:20px;z-index:35;
padding:12px 16px;border-radius:999px;
background:var(--accent);color:var(--accent-ink);font-weight:600;
border:0;cursor:pointer;font-size:13px;
box-shadow:0 10px 30px -8px rgba(34,224,122,.5);
align-items:center;gap:8px;font-family:var(--sans);
}
body.layout-drawer .drawer-toggle{display:inline-flex}
body.layout-drawer.drawer-open .drawer-toggle{display:none}
.card{
background:linear-gradient(180deg, var(--bg-1), var(--bg-2) 60%);
border:1px solid var(--line);
border-radius:var(--radius);
overflow:hidden;
position:relative;
}
.card::before{
content:"";position:absolute;inset:0 0 auto 0;height:1px;
background:linear-gradient(90deg, transparent, rgba(255,255,255,.05), transparent);
}
.card-head{
display:flex;align-items:center;justify-content:space-between;
padding:14px 18px;
border-bottom:1px solid var(--line);
background:rgba(255,255,255,.015);
}
.card-head h2{
margin:0;font-size:11px;font-weight:600;letter-spacing:.14em;
text-transform:uppercase;color:var(--text-2);
display:flex;align-items:center;gap:10px;
}
.card-head h2 .dot{width:6px;height:6px;border-radius:50%;background:var(--accent);box-shadow:0 0 10px var(--accent)}
/* -------- Device section -------- */
.device{
padding:20px 22px 18px;
display:flex;flex-direction:column;gap:20px;
border-bottom:1px solid var(--line);
}
.device-head{display:flex;align-items:flex-start;justify-content:space-between;gap:16px}
.serial-label{font-size:10.5px;letter-spacing:.14em;color:var(--text-3);text-transform:uppercase;font-weight:500;margin-bottom:6px}
.serial{
font-family:var(--mono);font-size:22px;font-weight:600;
letter-spacing:-0.01em;color:var(--text-0);
display:flex;align-items:center;gap:10px;
}
.serial .copy{
width:26px;height:26px;display:grid;place-items:center;
border-radius:6px;background:transparent;border:1px solid var(--line);
color:var(--text-3);cursor:pointer;transition:.15s;
}
.serial .copy:hover{color:var(--text-0);border-color:var(--line-strong);background:var(--bg-3)}
.manufactured{
display:inline-flex;align-items:center;gap:8px;
padding:5px 10px;border-radius:999px;
background:rgba(34,224,122,.08);
border:1px solid rgba(34,224,122,.22);
color:var(--accent);
font-family:var(--mono);font-size:10.5px;font-weight:600;
letter-spacing:.1em;text-transform:uppercase;
margin-top:8px;
}
.manufactured .pulse{
width:6px;height:6px;border-radius:50%;background:var(--accent);
box-shadow:0 0 8px var(--accent);
animation:pulse 2s infinite;
}
@keyframes pulse{
0%,100%{opacity:1;transform:scale(1)}
50%{opacity:.6;transform:scale(.85)}
}
.board-grid{
display:grid;grid-template-columns:repeat(3,minmax(0,1fr));
gap:10px;
}
.board-cell{
padding:12px 14px;
border:1px solid var(--line);
border-radius:10px;
background:rgba(255,255,255,.015);
}
.board-cell .k{font-size:10.5px;letter-spacing:.14em;color:var(--text-3);text-transform:uppercase;font-weight:500;margin-bottom:6px}
.board-cell .v{font-family:var(--mono);font-size:14px;color:var(--text-0);font-weight:500;display:flex;align-items:center;gap:8px}
.board-cell .v .sub{color:var(--text-3);font-weight:400;font-size:12.5px}
.board-cell .pill{
display:inline-block;padding:1px 7px;border-radius:4px;
background:rgba(180,139,255,.12);color:var(--violet);
font-size:10.5px;font-weight:600;letter-spacing:.05em;
border:1px solid rgba(180,139,255,.22);
text-transform:uppercase;
}
/* -------- Port status -------- */
.port-bar{
display:flex;align-items:center;gap:12px;
padding:10px 14px;border-radius:10px;
border:1px solid var(--line);
background:var(--bg-2);
}
.port-bar .status-dot{
width:8px;height:8px;border-radius:50%;
background:var(--text-3);
flex-shrink:0;
}
.port-bar.connected .status-dot{background:var(--accent);box-shadow:0 0 10px var(--accent)}
.port-bar.flashing .status-dot{background:var(--warn);box-shadow:0 0 10px var(--warn);animation:pulse 1s infinite}
.port-bar .txt{font-family:var(--mono);font-size:12.5px;color:var(--text-1);flex:1;min-width:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
.port-bar .txt strong{color:var(--text-0);font-weight:600}
.port-bar .baud{
font-family:var(--mono);font-size:11px;color:var(--text-3);
padding:3px 8px;border-radius:4px;background:var(--bg-3);
border:1px solid var(--line);
}
/* -------- Settings section -------- */
.settings{
padding:18px 22px 20px;
display:flex;flex-direction:column;gap:18px;
border-bottom:1px solid var(--line);
}
.section-label{
font-size:10.5px;letter-spacing:.14em;color:var(--text-3);
text-transform:uppercase;font-weight:600;
display:flex;align-items:center;gap:10px;
}
.section-label::after{content:"";flex:1;height:1px;background:var(--line)}
/* Segmented NVS control */
.segmented{
display:inline-grid;grid-auto-flow:column;
padding:3px;gap:3px;
background:var(--bg-3);border:1px solid var(--line);
border-radius:8px;
position:relative;
}
.segmented button{
border:0;background:transparent;
color:var(--text-2);font-family:var(--sans);font-size:12.5px;font-weight:500;
padding:7px 14px;border-radius:6px;cursor:pointer;
display:flex;align-items:center;gap:8px;
transition:.18s;
letter-spacing:.01em;
}
.segmented button .tag{
font-family:var(--mono);font-size:9.5px;
padding:1px 5px;border-radius:3px;
background:var(--bg-1);color:var(--text-3);
border:1px solid var(--line-strong);
}
.segmented button:hover{color:var(--text-0)}
.segmented button.active{
background:var(--bg-1);color:var(--text-0);
box-shadow: inset 0 0 0 1px var(--line-strong), 0 1px 0 rgba(0,0,0,.3);
}
.segmented button.active .tag{background:var(--accent);color:var(--accent-ink);border-color:transparent}
.segmented button.active.legacy .tag{background:var(--warn);color:#2a1800}
.row{display:flex;align-items:center;justify-content:space-between;gap:16px}
/* Toggle */
.toggle-row{
display:flex;align-items:center;justify-content:space-between;gap:16px;
padding:12px 14px;border-radius:10px;
border:1px solid var(--line);
background:rgba(255,255,255,.01);
}
.toggle-row.danger-row{
border-color:rgba(255,90,106,.2);
background:rgba(255,90,106,.04);
}
.toggle-text .t{font-size:13.5px;color:var(--text-0);font-weight:500}
.toggle-text .d{font-size:12px;color:var(--text-2);margin-top:2px}
.toggle-row.danger-row .toggle-text .t{color:#ffb9c0}
.toggle-row.danger-row .toggle-text .d{color:#d98d95}
.switch{position:relative;display:inline-block;width:38px;height:22px;flex-shrink:0}
.switch input{opacity:0;width:0;height:0}
.switch .slider{
position:absolute;inset:0;cursor:pointer;
background:var(--bg-3);border:1px solid var(--line-strong);
border-radius:999px;transition:.2s;
}
.switch .slider::before{
content:"";position:absolute;height:14px;width:14px;left:3px;top:50%;transform:translateY(-50%);
background:var(--text-2);border-radius:50%;transition:.2s;
}
.switch input:checked + .slider{background:var(--accent-dim);border-color:rgba(34,224,122,.4)}
.switch input:checked + .slider::before{transform:translate(16px, -50%);background:var(--accent)}
.toggle-row.danger-row .switch input:checked + .slider{background:rgba(255,90,106,.2);border-color:rgba(255,90,106,.4)}
.toggle-row.danger-row .switch input:checked + .slider::before{background:var(--danger)}
/* -------- Flash map -------- */
.flash-map{
padding:18px 22px 20px;
display:flex;flex-direction:column;gap:10px;
}
.partition{
display:grid;
grid-template-columns: 22px minmax(0,1fr) auto;
grid-template-rows: auto auto;
align-items:center;
column-gap:14px;
row-gap:8px;
padding:12px 14px;
border:1px solid var(--line);
border-radius:10px;
background:rgba(255,255,255,.012);
position:relative;
transition:.2s;
}
.partition .icon{grid-column:1;grid-row:1}
.partition .info{grid-column:2;grid-row:1;min-width:0}
.partition .pct{grid-column:3;grid-row:1;min-width:48px}
.partition .bar{grid-column:1 / -1;grid-row:2;width:100%}
.partition.active{
border-color:rgba(34,224,122,.25);
background:rgba(34,224,122,.03);
}
.partition.complete{
border-color:rgba(34,224,122,.18);
}
.partition .icon{
width:22px;height:22px;display:grid;place-items:center;
color:var(--text-3);
}
.partition.active .icon{color:var(--accent)}
.partition.complete .icon{color:var(--accent)}
.partition .info .name{font-size:13px;color:var(--text-0);font-weight:500}
.partition .info .addr{font-family:var(--mono);font-size:11px;color:var(--text-3);margin-top:2px}
.partition.active .info .addr{color:var(--accent)}
.bar{
height:6px;background:var(--bg-3);border-radius:3px;overflow:hidden;
position:relative;border:1px solid var(--line);
}
.bar-fill{
height:100%;background:var(--accent);border-radius:3px;
transition:width .3s ease;
position:relative;
box-shadow:0 0 8px rgba(34,224,122,.4);
}
.bar-fill::after{
content:"";position:absolute;inset:0;
background:linear-gradient(90deg, transparent, rgba(255,255,255,.25), transparent);
animation:shimmer 1.4s infinite;
}
.partition.complete .bar-fill::after{display:none}
@keyframes shimmer{0%{transform:translateX(-100%)}100%{transform:translateX(100%)}}
.partition .pct{
font-family:var(--mono);font-size:12px;color:var(--text-2);
text-align:right;font-weight:500;
font-variant-numeric:tabular-nums;
}
.partition.active .pct{color:var(--accent)}
.partition.complete .pct{color:var(--accent)}
/* -------- Footer / Action bar -------- */
.action-bar{
display:flex;align-items:center;justify-content:space-between;gap:14px;
padding:14px 22px;
background:rgba(0,0,0,.25);
border-top:1px solid var(--line);
}
.meta-inline{
font-family:var(--mono);font-size:11px;color:var(--text-3);
display:flex;align-items:center;gap:10px;
flex-wrap:wrap;
}
.meta-inline .sep{color:var(--line-strong)}
.meta-inline .k{color:var(--text-3)}
.meta-inline .v{color:var(--text-1)}
/* Buttons */
.btn{
display:inline-flex;align-items:center;justify-content:center;gap:9px;
padding:10px 16px;
font-family:var(--sans);font-size:13.5px;font-weight:600;
border-radius:8px;border:1px solid var(--line-strong);
background:var(--bg-3);color:var(--text-0);
cursor:pointer;transition:.15s;
letter-spacing:.005em;
}
.btn:hover{background:var(--bg-2);border-color:#3a4756}
.btn svg{width:15px;height:15px}
.btn-ghost{background:transparent;border-color:var(--line)}
.btn-ghost:hover{background:var(--bg-3)}
.btn-primary{
background:var(--accent);color:var(--accent-ink);
border-color:transparent;
box-shadow:0 1px 0 rgba(255,255,255,.18) inset, 0 8px 24px -10px rgba(34,224,122,.6);
}
.btn-primary:hover{background:#2df088;box-shadow:0 1px 0 rgba(255,255,255,.2) inset, 0 10px 28px -8px rgba(34,224,122,.75)}
.btn-danger{background:rgba(255,90,106,.1);color:#ff8690;border-color:rgba(255,90,106,.25)}
.btn-danger:hover{background:rgba(255,90,106,.15)}
.btn-lg{padding:13px 22px;font-size:14.5px}
.btn .kbd{font-family:var(--mono);font-size:10px;padding:1px 5px;border-radius:3px;background:rgba(0,0,0,.2);color:inherit;border:1px solid rgba(0,0,0,.3);opacity:.7}
.btn-primary .kbd{background:rgba(0,0,0,.15);border-color:rgba(0,0,0,.25)}
/* -------- Console / Log panel -------- */
.console-card{
display:flex;flex-direction:column;
min-height:0;
background:linear-gradient(180deg, #080b0e, #05070a);
}
.console-head{
display:flex;align-items:center;justify-content:space-between;
padding:14px 18px;
border-bottom:1px solid var(--line);
background:rgba(0,0,0,.35);
}
.console-head .tabs{display:flex;gap:2px;font-family:var(--mono);font-size:11px}
.console-head .tabs button{
background:transparent;border:0;color:var(--text-3);cursor:pointer;
padding:6px 10px;border-radius:5px;
font-family:inherit;font-size:inherit;letter-spacing:.05em;text-transform:uppercase;
}
.console-head .tabs button.active{color:var(--text-0);background:var(--bg-3)}
.console-head .tools{display:flex;gap:6px}
.console-head .tools button{
width:28px;height:28px;border-radius:6px;
display:grid;place-items:center;
background:transparent;border:1px solid var(--line);
color:var(--text-3);cursor:pointer;
}
.console-head .tools button:hover{color:var(--text-0);border-color:var(--line-strong);background:var(--bg-3)}
.console-head .tools button svg{width:13px;height:13px}
.console-body{
flex:1;
font-family:var(--mono);font-size:12px;line-height:1.65;
padding:14px 18px 12px;
overflow:auto;
color:#b8c4d0;
min-height:440px;
max-height:560px;
scrollbar-width:thin;scrollbar-color: var(--line-strong) transparent;
background:
repeating-linear-gradient(180deg, rgba(255,255,255,.012) 0 1px, transparent 1px 24px);
}
.console-body::-webkit-scrollbar{width:8px;height:8px}
.console-body::-webkit-scrollbar-thumb{background:var(--line-strong);border-radius:4px}
.log-line{display:grid;grid-template-columns: 100px 56px 1fr;gap:14px;padding:1px 0;white-space:pre-wrap;word-break:break-word}
.log-time{color:#4a5562;font-variant-numeric:tabular-nums;white-space:nowrap}
.log-level{font-weight:600;letter-spacing:.04em}
.log-level.info{color:#6aa9ff}
.log-level.ok{color:var(--accent)}
.log-level.warn{color:var(--warn)}
.log-level.err{color:var(--danger)}
.log-level.dbg{color:#8a97a4}
.log-msg .tok-hex{color:#b48bff}
.log-msg .tok-num{color:#5ab0ff}
.log-msg .tok-file{color:var(--accent)}
.log-msg .tok-em{color:#eef3f8;font-weight:600}
.log-msg .tok-mut{color:#6a7683}
.caret{display:inline-block;width:7px;height:13px;background:var(--accent);vertical-align:middle;animation:blink 1s steps(2) infinite;margin-left:4px}
@keyframes blink{50%{opacity:0}}
.console-empty{
display:flex;flex-direction:column;align-items:center;justify-content:center;
height:100%;min-height:420px;
color:var(--text-3);font-family:var(--mono);font-size:12px;
gap:14px;
text-align:center;
}
.console-empty .bigdot{
width:54px;height:54px;border-radius:50%;
background:radial-gradient(circle at 30% 30%, #1a2430, #0c1319);
border:1px solid var(--line);
display:grid;place-items:center;
}
.console-empty .bigdot svg{width:22px;height:22px;color:var(--text-3)}
.console-foot{
display:flex;align-items:center;justify-content:space-between;
padding:6px 18px;border-top:1px solid var(--line);
font-family:var(--mono);font-size:10.5px;color:var(--text-3);
letter-spacing:.05em;text-transform:uppercase;
background:rgba(0,0,0,.25);
line-height:1.4;
}
.console-foot .stats{display:flex;gap:16px}
.console-foot strong{color:var(--text-1);font-weight:600}
/* -------- Tweaks panel -------- */
.tweaks{
position:fixed;bottom:20px;right:20px;z-index:30;
width:260px;
background:var(--bg-1);border:1px solid var(--line-strong);
border-radius:12px;
padding:14px;
display:none;
box-shadow:0 20px 60px -20px rgba(0,0,0,.8);
font-size:12.5px;
}
.tweaks.on{display:block}
.tweaks h3{margin:0 0 10px;font-size:11px;letter-spacing:.14em;text-transform:uppercase;color:var(--text-2)}
.tweaks .tw-row{display:flex;align-items:center;justify-content:space-between;gap:10px;padding:6px 0}
.tweaks .tw-row label{color:var(--text-1)}
.tweaks select{
background:var(--bg-3);color:var(--text-0);
border:1px solid var(--line-strong);border-radius:6px;
padding:5px 8px;font-family:var(--mono);font-size:11px;
}
/* Small helpers */
.hide{display:none !important}
@media (max-width: 1100px){
.main{grid-template-columns:1fr}
.stepper{grid-template-columns:repeat(5, minmax(0,1fr));font-size:11px}
}
</style>
</head>
<body class="layout-split" style="cursor: crosshair; background-color: rgb(13, 17, 20); font-family: Geist;" data-cc-id="cc-1">
<div class="shell">
<!-- TOP BAR -->
<div class="topbar">
<div class="brand">
<div class="logo" aria-hidden="true">
<svg viewBox="0 0 24 24" fill="none" stroke="#22e07a" stroke-width="1.8" stroke-linecap="round" stroke-linejoin="round">
<path d="M6 3h8l4 4v14H6z"></path>
<path d="M14 3v4h4"></path>
<path d="M10 12l-2 3h3l-1 3 3-4h-2l1-2z" fill="#22e07a" stroke="none"></path>
</svg>
</div>
<div class="title">
<h1>Provisioning Wizard</h1>
<p>Flash firmware to a Bell Systems board via WebSerial</p>
</div>
</div>
<div class="top-actions" style="margin-top:2px">
<button class="btn btn-ghost">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M19 12H5"></path><path d="M12 19l-7-7 7-7"></path></svg>
Back to Inventory
</button>
</div>
</div>
<!-- STEPPER -->
<div class="stepper" role="list">
<div class="step done" role="listitem">
<div class="num">
<svg viewBox="0 0 24 24" width="12" height="12" fill="none" stroke="currentColor" stroke-width="3" stroke-linecap="round" stroke-linejoin="round"><polyline points="20 6 9 17 4 12"></polyline></svg>
</div>
<div class="meta"><span class="label">Mode</span><span class="sub">Provision</span></div>
<div class="step-divider"></div>
</div>
<div class="step done" role="listitem">
<div class="num">
<svg viewBox="0 0 24 24" width="12" height="12" fill="none" stroke="currentColor" stroke-width="3" stroke-linecap="round" stroke-linejoin="round"><polyline points="20 6 9 17 4 12"></polyline></svg>
</div>
<div class="meta"><span class="label">Device</span><span class="sub">BSVSPR-26D19J</span></div>
<div class="step-divider"></div>
</div>
<div class="step active" role="listitem">
<div class="num">3</div>
<div class="meta"><span class="label">Flash</span><span class="sub" id="step-sub-flash">complete</span></div>
<div class="step-divider"></div>
</div>
<div class="step" role="listitem">
<div class="num">4</div>
<div class="meta"><span class="label">Verify</span><span class="sub">pending</span></div>
<div class="step-divider"></div>
</div>
<div class="step" role="listitem">
<div class="num">
<svg viewBox="0 0 24 24" width="12" height="12" fill="none" stroke="currentColor" stroke-width="2.2" stroke-linecap="round" stroke-linejoin="round"><path d="M5 13l4 4L19 7"></path></svg>
</div>
<div class="meta"><span class="label">Done</span><span class="sub">pending</span></div>
</div>
</div>
<!-- MAIN -->
<div class="main">
<!-- LEFT: Settings + Info -->
<section class="card" aria-label="Device to flash">
<div class="card-head">
<h2><span class="dot"></span> Device to Flash</h2>
<div class="port-bar connected" id="portBarTop" style="padding:6px 10px">
<span class="status-dot"></span>
<span class="txt" id="portTxtTop"><strong>/dev/cu.usbserial-0142A91B</strong> &nbsp;·&nbsp; complete</span>
</div>
</div>
<!-- Device identity -->
<div class="device">
<div class="device-head">
<div>
<div class="serial-label">Serial number</div>
<div class="serial">
BSVSPR-26D19J-STD10R-XSNBBP
<button class="copy" title="Copy serial" aria-label="Copy serial">
<svg viewBox="0 0 24 24" width="13" height="13" fill="none" stroke="currentColor" stroke-width="1.8" stroke-linecap="round" stroke-linejoin="round"><rect x="9" y="9" width="13" height="13" rx="2"></rect><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"></path></svg>
</button>
</div>
<span class="manufactured"><span class="pulse"></span>Manufactured · ready to flash</span>
</div>
</div>
<div class="board-grid">
<div class="board-cell" style="padding:12px 14px">
<div class="k">Board type</div>
<div class="v">Vesper <span class="pill">V2</span></div>
</div>
<div class="board-cell" style="padding:12px 14px">
<div class="k">Code name</div>
<div class="v" style="font-family:var(--mono);color:var(--text-1)">vesper-basic</div>
</div>
<div class="board-cell" style="padding:12px 14px">
<div class="k">Revision</div>
<div class="v">Rev&nbsp;1.0 <span class="sub">· ESP32-S3</span></div>
</div>
</div>
</div>
<!-- Settings -->
<div class="settings">
<div class="section-label" style="margin-bottom:4px">Settings</div>
<div style="display:grid;grid-template-columns:auto 1fr;gap:14px;align-items:stretch">
<div style="display:flex;flex-direction:column">
<div style="font-size:11px;color:var(--text-3);margin-bottom:6px;text-transform:uppercase;letter-spacing:.08em;display:flex;align-items:center;gap:8px;height:16px">NVS Variant <span style="font-family:var(--mono);font-size:9.5px;color:var(--warn);letter-spacing:.1em">TEMPORARY</span></div>
<div class="segmented" role="tablist" style="flex:1;align-items:center">
<button class="active" data-nvs="current" role="tab">
Current Gen NVS <span class="tag">v15x</span>
</button>
<button class="legacy" data-nvs="legacy" role="tab">
Legacy NVS <span class="tag">v1.x</span>
</button>
</div>
</div>
<div style="display:flex;flex-direction:column;min-width:0">
<div style="font-size:11px;color:var(--text-3);margin-bottom:6px;text-transform:uppercase;letter-spacing:.08em;height:16px">Wipe Flash</div>
<div class="toggle-row danger-row" style="flex:1">
<div class="toggle-text">
<div class="t">Wipe flash before writing</div>
</div>
<label class="switch">
<input type="checkbox" id="wipe">
<span class="slider"></span>
</label>
</div>
</div>
</div>
</div>
<!-- Flash map / progress -->
<div class="flash-map">
<div class="section-label" style="margin-bottom:4px">Flash Map</div>
<div class="partition complete" data-part="bootloader">
<span class="icon">
<svg viewBox="0 0 24 24" width="16" height="16" fill="none" stroke="currentColor" stroke-width="1.7" stroke-linecap="round" stroke-linejoin="round"><polygon points="13 2 3 14 12 14 11 22 21 10 12 10 13 2"></polygon></svg>
</span>
<div class="info">
<div class="name">Bootloader</div>
<div class="addr">0x1000 · bootloader.bin · 24 KB</div>
</div>
<div class="bar" style="border-radius:3px"><div class="bar-fill" style="width: 100%;"></div></div>
<div class="pct">100%</div>
</div>
<div class="partition complete" data-part="partition">
<span class="icon">
<svg viewBox="0 0 24 24" width="16" height="16" fill="none" stroke="currentColor" stroke-width="1.7" stroke-linecap="round" stroke-linejoin="round"><rect x="3" y="4" width="18" height="4" rx="1"></rect><rect x="3" y="10" width="10" height="4" rx="1"></rect><rect x="3" y="16" width="18" height="4" rx="1"></rect></svg>
</span>
<div class="info">
<div class="name">Partition Table</div>
<div class="addr">0x8000 · partitions.bin · 3 KB</div>
</div>
<div class="bar" style="border-radius:3px"><div class="bar-fill" style="width: 100%;"></div></div>
<div class="pct">100%</div>
</div>
<div class="partition complete" data-part="nvs">
<span class="icon">
<svg viewBox="0 0 24 24" width="16" height="16" fill="none" stroke="currentColor" stroke-width="1.7" stroke-linecap="round" stroke-linejoin="round"><ellipse cx="12" cy="5" rx="9" ry="3"></ellipse><path d="M3 5v6c0 1.66 4.03 3 9 3s9-1.34 9-3V5"></path><path d="M3 11v6c0 1.66 4.03 3 9 3s9-1.34 9-3v-6"></path></svg>
</span>
<div class="info">
<div class="name">NVS (Current Gen)</div>
<div class="addr">0x9000 · nvs_vesper.bin · 20 KB</div>
</div>
<div class="bar" style="border-radius:3px"><div class="bar-fill" style="width: 100%;"></div></div>
<div class="pct">100%</div>
</div>
<div class="partition complete" data-part="firmware">
<span class="icon">
<svg viewBox="0 0 24 24" width="16" height="16" fill="none" stroke="currentColor" stroke-width="1.7" stroke-linecap="round" stroke-linejoin="round"><rect x="4" y="4" width="16" height="16" rx="2"></rect><rect x="9" y="9" width="6" height="6"></rect><path d="M9 2v2M15 2v2M9 20v2M15 20v2M2 9h2M2 15h2M20 9h2M20 15h2"></path></svg>
</span>
<div class="info">
<div class="name">Firmware</div>
<div class="addr">0x10000 · bellsys-vesper-2.7.3.bin · 1.4 MB</div>
</div>
<div class="bar" style="border-radius:3px"><div class="bar-fill" style="width: 100%;"></div></div>
<div class="pct">100%</div>
</div>
</div>
<!-- Action bar -->
<div class="action-bar">
<div class="meta-inline">
<span><span class="k">BAUD</span> <span class="v" id="baudMeta">921600</span></span>
<span class="sep">·</span>
<span><span class="k">NVS</span> <span class="v">0x9000</span></span>
<span class="sep">·</span>
<span><span class="k">FW</span> <span class="v">0x10000</span></span>
<span class="sep">·</span>
<span><span class="k">STUB</span> <span class="v">yes</span></span>
</div>
<button class="btn btn-primary btn-lg" id="mainAction">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M4 4h4l2 3h10v11a2 2 0 0 1-2 2H4z"></path><circle cx="14" cy="14" r="2.5"></circle></svg>
<a id="verifyLink" href="Verify.html" style="color:inherit;text-decoration:none;display:inline-flex;align-items:center;gap:9px"><span id="mainActionTxt">Continue to Verify →</span></a>
<span class="kbd"></span>
</button>
</div>
</section>
<!-- RIGHT: Console / Flash output -->
<section class="card console-card" aria-label="Flash output">
<div class="console-head">
<div class="tabs">
<button class="active">Output</button>
<button>esptool.py</button>
<button>Serial Monitor</button>
</div>
<div class="tools">
<button title="Pause autoscroll">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><rect x="6" y="5" width="4" height="14"></rect><rect x="14" y="5" width="4" height="14"></rect></svg>
</button>
<button title="Copy">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect x="9" y="9" width="13" height="13" rx="2"></rect><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"></path></svg>
</button>
<button title="Download log">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"></path><polyline points="7 10 12 15 17 10"></polyline><line x1="12" y1="15" x2="12" y2="3"></line></svg>
</button>
<button title="Clear">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="3 6 5 6 21 6"></polyline><path d="M8 6V4a2 2 0 0 1 2-2h4a2 2 0 0 1 2 2v2"></path><path d="M19 6l-1 14a2 2 0 0 1-2 2H8a2 2 0 0 1-2-2L5 6"></path></svg>
</button>
</div>
</div>
<div class="console-body" id="consoleBody"><div class="console-empty" id="consoleEmpty" style="display: none;">
<div class="bigdot">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.6" stroke-linecap="round" stroke-linejoin="round"><polyline points="4 17 10 11 4 5"></polyline><line x1="12" y1="19" x2="20" y2="19"></line></svg>
</div>
<div>Output will appear here once flashing starts.<br><span style="color:var(--text-3);opacity:.7">Select a COM port to begin.</span></div>
</div><div class="log-line"><span class="log-time">16:17:41.337</span><span class="log-level info">INFO</span><span class="log-msg">WebSerial: requesting user-gated port…</span></div><div class="log-line"><span class="log-time">16:17:41.338</span><span class="log-level ok">OK</span><span class="log-msg">Port granted → <span class="tok-em">/dev/cu.usbserial-0142A91B</span></span></div><div class="log-line"><span class="log-time">16:17:41.339</span><span class="log-level info">INFO</span><span class="log-msg">Opening @ <span class="tok-num">115200</span> baud (handshake)</span></div><div class="log-line"><span class="log-time">16:17:41.339</span><span class="log-level dbg">DBG</span><span class="log-msg">Detecting chip… <span class="tok-mut">(up to 3s)</span></span></div><div class="log-line"><span class="log-time">16:17:41.339</span><span class="log-level ok">OK</span><span class="log-msg">Chip detected: <span class="tok-em">ESP32-S3</span> · rev <span class="tok-num">0.2</span> · flash <span class="tok-num">8MB</span></span></div><div class="log-line"><span class="log-time">16:17:41.339</span><span class="log-level info">INFO</span><span class="log-msg">MAC: <span class="tok-hex">84:F7:03:1A:2E:D0</span></span></div><div class="log-line"><span class="log-time">16:17:41.339</span><span class="log-level info">INFO</span><span class="log-msg">Uploading stub flasher → <span class="tok-file">stub_flasher_s3.bin</span></span></div><div class="log-line"><span class="log-time">16:17:41.339</span><span class="log-level ok">OK</span><span class="log-msg">Stub running. Changing baud → <span class="tok-num">460800</span></span></div><div class="log-line"><span class="log-time">16:17:41.340</span><span class="log-level info">INFO</span><span class="log-msg">Ready. Waiting for FLASH DEVICE.</span></div><div class="log-line"><span class="log-time">16:17:41.340</span><span class="log-level info">INFO</span><span class="log-msg">Starting flash sequence…</span></div><div class="log-line"><span class="log-time">16:17:41.340</span><span class="log-level info">INFO</span><span class="log-msg">Writing <span class="tok-file">bootloader.bin @ 0x1000</span></span></div><div class="log-line"><span class="log-time">16:17:43.143</span><span class="log-level ok">OK</span><span class="log-msg">Wrote <span class="tok-file">bootloader.bin @ 0x1000</span> (<span class="tok-num">24,576</span> B) in <span class="tok-num">1.80s</span></span></div><div class="log-line"><span class="log-time">16:17:43.143</span><span class="log-level info">INFO</span><span class="log-msg">Writing <span class="tok-file">partitions.bin @ 0x8000</span></span></div><div class="log-line"><span class="log-time">16:17:43.744</span><span class="log-level ok">OK</span><span class="log-msg">Wrote <span class="tok-file">partitions.bin @ 0x8000</span> (<span class="tok-num">3,072</span> B) in <span class="tok-num">0.60s</span></span></div><div class="log-line"><span class="log-time">16:17:43.744</span><span class="log-level info">INFO</span><span class="log-msg">Writing <span class="tok-file">nvs_vesper.bin @ 0x9000</span></span></div><div class="log-line"><span class="log-time">16:17:44.949</span><span class="log-level ok">OK</span><span class="log-msg">Wrote <span class="tok-file">nvs_vesper.bin @ 0x9000</span> (<span class="tok-num">20,480</span> B) in <span class="tok-num">1.20s</span></span></div><div class="log-line"><span class="log-time">16:17:44.950</span><span class="log-level info">INFO</span><span class="log-msg">Writing <span class="tok-file">bellsys-vesper-2.7.3.bin @ 0x10000</span></span></div><div class="log-line"><span class="log-time">16:17:50.161</span><span class="log-level ok">OK</span><span class="log-msg">Wrote <span class="tok-file">bellsys-vesper-2.7.3.bin @ 0x10000</span> (<span class="tok-num">1,468,400</span> B) in <span class="tok-num">5.20s</span></span></div><div class="log-line"><span class="log-time">16:17:50.564</span><span class="log-level ok">OK</span><span class="log-msg">Bootloader verified (<span class="tok-hex">md5 d4c3…a912</span>)</span></div><div class="log-line"><span class="log-time">16:17:50.565</span><span class="log-level ok">OK</span><span class="log-msg">Partition Table verified (<span class="tok-hex">md5 01fe…88b1</span>)</span></div><div class="log-line"><span class="log-time">16:17:50.565</span><span class="log-level ok">OK</span><span class="log-msg">NVS verified (<span class="tok-hex">md5 7a22…e0d4</span>)</span></div><div class="log-line"><span class="log-time">16:17:50.565</span><span class="log-level ok">OK</span><span class="log-msg">Firmware verified (<span class="tok-hex">md5 5b9c…72a1</span>)</span></div><div class="log-line"><span class="log-time">16:17:50.566</span><span class="log-level info">INFO</span><span class="log-msg">Hard-resetting device…</span></div><div class="log-line"><span class="log-time">16:17:50.566</span><span class="log-level ok">OK</span><span class="log-msg"><span class="tok-em">Flash complete</span> in 00:47.31</span></div></div>
<div class="console-foot" style="font-size:10.5px">
<div class="stats">
<span><strong id="statLines">24</strong> lines</span>
<span><strong id="statBytes">1,516,528</strong> B transferred</span>
<span><strong id="statElapsed">00:47</strong> elapsed</span>
</div>
<div><span id="statState">COMPLETE</span></div>
</div>
</section>
</div>
</div>
<!-- Drawer toggle (layout-drawer only) -->
<button class="drawer-toggle" id="drawerToggle">
<svg viewBox="0 0 24 24" width="14" height="14" fill="none" stroke="currentColor" stroke-width="2.2" stroke-linecap="round" stroke-linejoin="round"><polyline points="4 17 10 11 4 5"></polyline><line x1="12" y1="19" x2="20" y2="19"></line></svg>
Flash Output
</button>
<!-- TWEAKS -->
<div class="tweaks on" id="tweaks">
<h3>Tweaks</h3>
<div class="tw-row">
<label for="tw-state">Simulated state</label>
<select id="tw-state">
<option value="idle">Idle — no port</option>
<option value="connected">Port connected</option>
<option value="flashing">Flashing…</option>
<option value="done">Flash complete</option>
<option value="error">Error</option>
</select>
</div>
<div class="tw-row">
<label for="tw-baud">Baud</label>
<select id="tw-baud">
<option>115200</option>
<option selected="">460800</option>
<option>921600</option>
<option>1500000</option>
</select>
</div>
<div class="tw-row">
<label for="tw-accent">Accent</label>
<select id="tw-accent">
<option value="#22e07a" selected="">Bell Green</option>
<option value="#5ab0ff">Azure</option>
<option value="#b48bff">Violet</option>
<option value="#ffb347">Amber</option>
</select>
</div>
<div class="tw-row">
<label for="tw-layout">Layout</label>
<select id="tw-layout">
<option value="split" selected="">Split (40/60)</option>
<option value="stacked">Stacked</option>
<option value="drawer">Log as drawer</option>
</select>
</div>
</div>
<script>
(() => {
// ---------- Edit mode defaults (persisted) ----------
const TWEAK_DEFAULTS = /*EDITMODE-BEGIN*/{
"state": "flashing",
"baud": "460800",
"accent": "#22e07a",
"layout": "split"
}/*EDITMODE-END*/;
// ---------- Host edit-mode wiring ----------
window.addEventListener('message', (e) => {
const d = e.data || {};
if (d.type === '__activate_edit_mode') document.getElementById('tweaks').classList.add('on');
if (d.type === '__deactivate_edit_mode') document.getElementById('tweaks').classList.remove('on');
});
window.parent.postMessage({type:'__edit_mode_available'}, '*');
const persist = (edits) => {
Object.assign(TWEAK_DEFAULTS, edits);
window.parent.postMessage({type:'__edit_mode_set_keys', edits}, '*');
};
// ---------- State machine ----------
const portBarTop = document.getElementById('portBarTop');
const portTxtTop = document.getElementById('portTxtTop');
const mainAction = document.getElementById('mainAction');
const mainActionTxt = document.getElementById('mainActionTxt');
const stepSubFlash = document.getElementById('step-sub-flash');
const consoleBody = document.getElementById('consoleBody');
const consoleEmpty = document.getElementById('consoleEmpty');
const statLines = document.getElementById('statLines');
const statBytes = document.getElementById('statBytes');
const statElapsed = document.getElementById('statElapsed');
const statState = document.getElementById('statState');
const baudMeta = document.getElementById('baudMeta');
const parts = {
bootloader: document.querySelector('[data-part="bootloader"]'),
partition: document.querySelector('[data-part="partition"]'),
nvs: document.querySelector('[data-part="nvs"]'),
firmware: document.querySelector('[data-part="firmware"]'),
};
const setPartPct = (k, pct) => {
const el = parts[k];
el.querySelector('.bar-fill').style.width = pct + '%';
el.querySelector('.pct').textContent = Math.round(pct) + '%';
el.classList.toggle('complete', pct >= 100);
el.classList.toggle('active', pct > 0 && pct < 100);
};
let timer = null;
let startTime = 0;
const fmtTime = (ms) => {
const s = Math.floor(ms/1000);
return String(Math.floor(s/60)).padStart(2,'0') + ':' + String(s%60).padStart(2,'0');
};
const clearConsole = () => {
consoleBody.innerHTML = '';
consoleBody.appendChild(consoleEmpty);
consoleEmpty.style.display = '';
};
const addLog = (level, msg) => {
if (consoleEmpty.parentElement) consoleEmpty.style.display = 'none';
const now = new Date();
const t = `${String(now.getHours()).padStart(2,'0')}:${String(now.getMinutes()).padStart(2,'0')}:${String(now.getSeconds()).padStart(2,'0')}.${String(now.getMilliseconds()).padStart(3,'0')}`;
const div = document.createElement('div');
div.className = 'log-line';
div.innerHTML = `<span class="log-time" style="font-size:12px">${t}</span><span class="log-level ${level}">${level.toUpperCase()}</span><span class="log-msg">${msg}</span>`;
consoleBody.appendChild(div);
consoleBody.scrollTop = consoleBody.scrollHeight;
statLines.textContent = consoleBody.querySelectorAll('.log-line').length;
};
const seedLogs = {
connected: [
['info', 'WebSerial: requesting user-gated port…'],
['ok', 'Port granted → <span class="tok-em">/dev/cu.usbserial-0142A91B</span>'],
['info', 'Opening @ <span class="tok-num">115200</span> baud (handshake)'],
['dbg', 'Detecting chip… <span class="tok-mut">(up to 3s)</span>'],
['ok', 'Chip detected: <span class="tok-em">ESP32-S3</span> · rev <span class="tok-num">0.2</span> · flash <span class="tok-num">8MB</span>'],
['info', 'MAC: <span class="tok-hex">84:F7:03:1A:2E:D0</span>'],
['info', 'Uploading stub flasher → <span class="tok-file">stub_flasher_s3.bin</span>'],
['ok', 'Stub running. Changing baud → <span class="tok-num">460800</span>'],
['info', 'Ready. Waiting for FLASH DEVICE.'],
],
done: [
['ok', 'Bootloader verified (<span class="tok-hex">md5 d4c3…a912</span>)'],
['ok', 'Partition Table verified (<span class="tok-hex">md5 01fe…88b1</span>)'],
['ok', 'NVS verified (<span class="tok-hex">md5 7a22…e0d4</span>)'],
['ok', 'Firmware verified (<span class="tok-hex">md5 5b9c…72a1</span>)'],
['info', 'Hard-resetting device…'],
['ok', '<span class="tok-em">Flash complete</span> in 00:47.31'],
],
error: [
['info', 'Opening @ 115200 baud…'],
['warn', 'Sync attempt 1/10 failed'],
['warn', 'Sync attempt 2/10 failed'],
['err', 'Timed out waiting for packet header. Check BOOT pin / cable.'],
['err', 'Flash aborted. <span class="tok-em">SerialException: Failed to connect to ESP32-S3</span>'],
],
};
const pushSeed = (key) => {
clearConsole();
(seedLogs[key] || []).forEach(([lv,m]) => addLog(lv, m));
};
let currentState = 'idle';
const applyState = (s) => {
currentState = s;
if (timer) { clearInterval(timer); timer = null; }
// Reset progress
Object.keys(parts).forEach(k => setPartPct(k, 0));
// Port bar
portBarTop.classList.remove('connected','flashing');
switch (s) {
case 'idle':
portTxtTop.innerHTML = 'No port selected';
mainActionTxt.textContent = 'Select COM Port';
mainAction.className = 'btn btn-primary btn-lg';
stepSubFlash.textContent = 'awaiting port';
statState.textContent = 'IDLE';
clearConsole();
break;
case 'connected':
portBarTop.classList.add('connected');
portTxtTop.innerHTML = '<strong>/dev/cu.usbserial-0142A91B</strong> &nbsp;·&nbsp; ESP32-S3 · 8MB';
mainActionTxt.textContent = 'Flash Device';
mainAction.className = 'btn btn-primary btn-lg';
stepSubFlash.textContent = 'port ready';
statState.textContent = 'READY';
pushSeed('connected');
break;
case 'flashing':
portBarTop.classList.add('flashing');
portTxtTop.innerHTML = '<strong>/dev/cu.usbserial-0142A91B</strong> &nbsp;·&nbsp; writing…';
mainActionTxt.textContent = 'Abort Flash';
mainAction.className = 'btn btn-danger btn-lg';
stepSubFlash.textContent = 'writing…';
statState.textContent = 'WRITING';
pushSeed('connected');
addLog('info', 'Starting flash sequence…');
startTime = Date.now();
// Simulated sequence
const plan = [
{ key:'bootloader', duration: 1800, label:'bootloader.bin @ 0x1000', bytes: 24576 },
{ key:'partition', duration: 600, label:'partitions.bin @ 0x8000', bytes: 3072 },
{ key:'nvs', duration: 1200, label:'nvs_vesper.bin @ 0x9000', bytes: 20480 },
{ key:'firmware', duration: 5200, label:'bellsys-vesper-2.7.3.bin @ 0x10000', bytes: 1468400 },
];
let i = 0, stageStart = Date.now(), totalBytes = 0;
addLog('info', `Writing <span class="tok-file">${plan[0].label}</span>`);
timer = setInterval(() => {
const p = plan[i];
const t = Date.now() - stageStart;
const pct = Math.min(100, (t / p.duration) * 100);
setPartPct(p.key, pct);
statBytes.textContent = (totalBytes + Math.round(p.bytes * pct/100)).toLocaleString();
statElapsed.textContent = fmtTime(Date.now() - startTime);
if (pct >= 100) {
totalBytes += p.bytes;
addLog('ok', `Wrote <span class="tok-file">${p.label}</span> (<span class="tok-num">${p.bytes.toLocaleString()}</span> B) in <span class="tok-num">${(p.duration/1000).toFixed(2)}s</span>`);
i += 1;
if (i >= plan.length) {
clearInterval(timer); timer = null;
setTimeout(() => applyState('done'), 400);
} else {
stageStart = Date.now();
addLog('info', `Writing <span class="tok-file">${plan[i].label}</span>`);
}
}
}, 60);
break;
case 'done':
portBarTop.classList.add('connected');
portTxtTop.innerHTML = '<strong>/dev/cu.usbserial-0142A91B</strong> &nbsp;·&nbsp; complete';
mainActionTxt.textContent = 'Continue to Verify →';
mainAction.className = 'btn btn-primary btn-lg';
stepSubFlash.textContent = 'complete';
statState.textContent = 'COMPLETE';
Object.keys(parts).forEach(k => setPartPct(k, 100));
if (!consoleBody.querySelector('.log-line')) pushSeed('connected');
seedLogs.done.forEach(([lv,m]) => addLog(lv, m));
statElapsed.textContent = '00:47';
statBytes.textContent = '1,516,528';
break;
case 'error':
portBarTop.classList.remove('connected','flashing');
portTxtTop.innerHTML = '<strong>/dev/cu.usbserial-0142A91B</strong> &nbsp;·&nbsp; <span style="color:var(--danger)">sync failed</span>';
mainActionTxt.textContent = 'Retry Flash';
mainAction.className = 'btn btn-primary btn-lg';
stepSubFlash.textContent = 'error';
statState.textContent = 'ERROR';
pushSeed('error');
break;
}
};
// ---------- Wire up ----------
document.querySelectorAll('.segmented button').forEach(b => {
b.addEventListener('click', () => {
document.querySelectorAll('.segmented button').forEach(x => x.classList.remove('active'));
b.classList.add('active');
});
});
mainAction.addEventListener('click', () => {
if (currentState === 'idle') applyState('connected');
else if (currentState === 'connected') applyState('flashing');
else if (currentState === 'flashing') applyState('connected');
else if (currentState === 'done') applyState('idle');
else if (currentState === 'error') applyState('flashing');
});
// Tweaks
const twState = document.getElementById('tw-state');
const twBaud = document.getElementById('tw-baud');
const twAccent = document.getElementById('tw-accent');
twState.addEventListener('change', () => { persist({state: twState.value}); applyState(twState.value); });
twBaud.addEventListener('change', () => { persist({baud: twBaud.value}); baudMeta.textContent = twBaud.value; });
twAccent.addEventListener('change', () => {
persist({accent: twAccent.value});
document.documentElement.style.setProperty('--accent', twAccent.value);
});
const twLayout = document.getElementById('tw-layout');
const applyLayout = (l) => {
document.body.classList.remove('layout-split','layout-stacked','layout-drawer');
document.body.classList.add('layout-' + l);
document.body.classList.remove('drawer-open');
};
twLayout.addEventListener('change', () => { persist({layout: twLayout.value}); applyLayout(twLayout.value); });
document.getElementById('drawerToggle').addEventListener('click', (e) => {
e.stopPropagation();
document.body.classList.add('drawer-open');
});
document.addEventListener('click', (e) => {
if (!document.body.classList.contains('layout-drawer')) return;
if (!document.body.classList.contains('drawer-open')) return;
const card = document.querySelector('.console-card');
if (card && !card.contains(e.target)) {
document.body.classList.remove('drawer-open');
}
});
// Apply initial tweak values
twState.value = TWEAK_DEFAULTS.state;
twBaud.value = TWEAK_DEFAULTS.baud;
twAccent.value = TWEAK_DEFAULTS.accent;
twLayout.value = TWEAK_DEFAULTS.layout;
baudMeta.textContent = TWEAK_DEFAULTS.baud;
document.documentElement.style.setProperty('--accent', TWEAK_DEFAULTS.accent);
applyLayout(TWEAK_DEFAULTS.layout);
applyState(TWEAK_DEFAULTS.state);
})();
</script>
</body></html>