<!doctype html>
<html lang="en">
<head>
  <meta charset="utf-8" />
  <meta name="viewport" content="width=device-width, initial-scale=1" />
  <title>Agent Showcase — Pro</title>
  <script src="https://cdn.tailwindcss.com"></script>
  <style>
    :root { --card: rgba(18,18,22,0.75); --ring: rgba(255,255,255,0.08); }
    .glass { background: var(--card); backdrop-filter: blur(14px); }
    .ring  { box-shadow: 0 0 0 1px var(--ring) inset; }
    .bubble { max-width: 80%; padding: .625rem .75rem; border-radius: 14px; border: 1px solid rgba(255,255,255,.08); background: rgba(255,255,255,.04) }
    .bubble.me { background: rgba(120,120,255,.14); border-color: rgba(160,160,255,.24) }
    .typing { display:inline-flex; gap:4px; align-items:center }
    .dot { width:6px; height:6px; border-radius:9999px; background:#a1a1aa; opacity:.5; animation: blink 1.2s infinite }
    .dot:nth-child(2){ animation-delay:.2s } .dot:nth-child(3){ animation-delay:.4s }
    @keyframes blink { 0%, 80%, 100%{ opacity:.3 } 40%{ opacity:1 } }
    @keyframes caret { from { border-right-color: rgba(255,255,255,.45)} to { border-right-color: transparent } }
    .typewriter { border-right:1px solid rgba(255,255,255,.45); animation: caret 1s steps(1) infinite }
    .spinner { width:14px; height:14px; border-radius:9999px; border:2px solid rgba(255,255,255,.25); border-top-color:rgba(255,255,255,.9); animation: spin .7s linear infinite }
    @keyframes spin { to { transform: rotate(360deg) } }
    .err { background: rgba(244, 63, 94, .12); border: 1px solid rgba(244, 63, 94, .35); color:#fecaca }
  </style>
</head>
<body class="min-h-screen bg-gradient-to-b from-black via-zinc-950 to-black text-zinc-100 selection:bg-indigo-500/30">
  <main class="mx-auto max-w-3xl px-5 py-10">
    <header class="mb-8">
      <h1 class="text-3xl font-semibold tracking-tight">Agent Showcase</h1>
      <p class="text-sm text-zinc-400">Clean UI. Sequential reveal. No backend changes.</p>
    </header>

    <!-- Page selector -->
    <div class="mb-6 inline-grid grid-cols-2 rounded-xl ring overflow-hidden">
      <button data-page="comedy" class="px-4 py-2 text-sm bg-zinc-900">Comedy</button>
      <button data-page="numbers" class="px-4 py-2 text-sm hover:bg-zinc-900/60">Numbers</button>
    </div>

    <!-- COMEDY -->
    <section id="page-comedy" class="space-y-4">
      <div class="glass ring rounded-2xl p-6 space-y-4">
        <div class="flex items-center justify-between">
          <h2 class="text-xl font-semibold">Comedy Duo</h2>
        </div>
        <div class="text-xs text-zinc-400">
          <div>• <strong>Start Duo</strong>: Joe and Cathy alternate for N rounds (theme optional).</div>
          <div>• <strong>Ask for one-liner</strong>: The selected comedian (picker) replies to your prompt with one short line.</div>
        </div>
        <div class="grid grid-cols-1 gap-3 sm:grid-cols-3">
          <div class="sm:col-span-2">
            <label class="text-sm text-zinc-400">Theme (optional)</label>
            <input id="comedyTheme" class="mt-1 w-full rounded-xl bg-zinc-900/70 border border-zinc-700 px-3 py-2" placeholder="airports, finals week, coffee..." />
          </div>
          <div>
            <label class="text-sm text-zinc-400">Rounds</label>
            <input id="comedyRounds" type="number" min="2" max="12" value="6" class="mt-1 w-full rounded-xl bg-zinc-900/70 border border-zinc-700 px-3 py-2" />
          </div>
        </div>
        <div class="flex flex-col gap-3">
          <div class="flex items-center gap-2">
            <button id="btnStartComedy" class="px-4 py-2 rounded-xl bg-indigo-600 hover:bg-indigo-500 flex items-center gap-2">
              <span>Start Duo</span><span class="hidden spinner" id="spinComedy"></span>
            </button>
          </div>
          <div class="flex items-center gap-2">
            <select id="oneWho" class="px-3 py-2 rounded-xl bg-zinc-900/70 border border-zinc-700">
              <option>Joe</option>
              <option>Cathy</option>
            </select>
            <input id="onePrompt" class="flex-1 px-3 py-2 rounded-xl bg-zinc-900/70 border border-zinc-700" placeholder="Write a one-liner about office hours…" />
            <button id="btnOneLiner" class="px-3 py-2 rounded-xl bg-zinc-700 hover:bg-zinc-600 flex items-center gap-2">
              <span>Ask for one-liner</span><span class="hidden spinner" id="spinOne"></span>
            </button>
          </div>
        </div>
      </div>

      <div class="glass ring rounded-2xl p-6">
        <div id="comedyErr" class="hidden err rounded-xl px-3 py-2 text-sm mb-3"></div>
        <div id="comedyOut" class="space-y-4 max-h-[28rem] overflow-y-auto pr-1"></div>
      </div>
    </section>

    <!-- NUMBERS -->
    <section id="page-numbers" class="hidden space-y-4">
      <div class="glass ring rounded-2xl p-6 space-y-4">
        <div class="flex items-center justify-between">
          <h2 class="text-xl font-semibold">Number Guessing</h2>
        </div>
        <div class="text-xs text-zinc-400">
          <div>• <strong>AI vs AI (Run)</strong>: narrator reveals each guess sequentially.</div>
          <div>• <strong>You guess</strong>: you guess a hidden number, receive feedback.</div>
          <div>• <strong>AI guesses</strong>: AI proposes guesses; you click feedback.</div>
        </div>
        <div class="grid grid-cols-1 gap-3 sm:grid-cols-4">
          <div>
            <label class="text-sm text-zinc-400">Low</label>
            <input id="low" type="number" value="1" class="mt-1 w-full rounded-xl bg-zinc-900/70 border border-zinc-700 px-3 py-2" />
          </div>
          <div>
            <label class="text-sm text-zinc-400">High</label>
            <input id="high" type="number" value="100" class="mt-1 w-full rounded-xl bg-zinc-900/70 border border-zinc-700 px-3 py-2" />
          </div>
          <div class="flex items-end">
            <label class="inline-flex items-center gap-2 text-sm text-zinc-300"><input id="banter" type="checkbox" class="accent-indigo-500" checked /> Banter</label>
          </div>
          <div class="flex items-end justify-end">
            <button id="btnRunAiAi" class="px-4 py-2 rounded-xl bg-indigo-600 hover:bg-indigo-500 flex items-center gap-2"><span>Run</span><span class="hidden spinner" id="spinAiAi"></span></button>
          </div>
        </div>
      </div>

      <div class="glass ring rounded-2xl p-6">
        <div id="numErr" class="hidden err rounded-xl px-3 py-2 text-sm mb-3"></div>

        <div class="mb-6">
          <h3 class="text-sm font-medium text-zinc-300 mb-2">AI vs AI</h3>
          <div id="aiAiOut" class="space-y-4 max-h-[22rem] overflow-y-auto pr-1"></div>
        </div>

        <div class="flex items-center gap-3 mb-3 text-sm text-zinc-400">
          <button data-subtab="you-guess" class="subtab px-3 py-1.5 rounded-lg bg-zinc-800 border border-zinc-700">You guess</button>
          <button data-subtab="ai-guesses" class="subtab px-3 py-1.5 rounded-lg hover:bg-zinc-800 border border-zinc-700/40">AI guesses</button>
        </div>

        <div id="pane-you-guess" class="space-y-3">
          <div class="flex items-center gap-2">
            <button id="btnStartYou" class="px-4 py-2 rounded-xl bg-indigo-600 hover:bg-indigo-500 flex items-center gap-2"><span>Start Game</span><span class="hidden spinner" id="spinYou"></span></button>
            <span id="youRange" class="text-sm text-zinc-400"></span>
          </div>
          <div class="flex gap-2">
            <input id="youGuess" type="number" class="flex-1 px-3 py-2 rounded-xl bg-zinc-900/70 border border-zinc-700" placeholder="Your guess" />
            <button id="btnSubmitYou" class="px-4 py-2 rounded-xl bg-zinc-700 hover:bg-zinc-600">Submit</button>
          </div>
          <div id="youOut" class="space-y-3 max-h-[20rem] overflow-y-auto pr-1"></div>
        </div>

        <div id="pane-ai-guesses" class="hidden space-y-3">
          <div class="flex items-center gap-2">
            <button id="btnStartAiG" class="px-4 py-2 rounded-xl bg-indigo-600 hover:bg-indigo-500 flex items-center gap-2"><span>Start</span><span class="hidden spinner" id="spinAiG"></span></button>
            <span class="text-sm text-zinc-400">Give feedback each turn.</span>
          </div>
          <div id="aiGControls" class="hidden flex items-center gap-2">
            <button id="btnLow" class="px-3 py-1.5 rounded-lg bg-zinc-700 hover:bg-zinc-600">Too low</button>
            <button id="btnHigh" class="px-3 py-1.5 rounded-lg bg-zinc-700 hover:bg-zinc-600">Too high</button>
            <button id="btnCorrect" class="px-3 py-1.5 rounded-lg bg-emerald-600 hover:bg-emerald-500">Correct</button>
          </div>
          <div id="aiGOut" class="space-y-3 max-h-[20rem] overflow-y-auto pr-1"></div>
        </div>
      </div>
    </section>
  </main>

  <script>
  document.addEventListener('DOMContentLoaded', () => {
    const $ = (sel) => document.querySelector(sel);

    // --- helpers
    const AVATAR = { Joe: '🧢', Cathy: '🎤', Narrator: '🎬', You: '🫵' };
    const sleep = (ms)=>new Promise(r=>setTimeout(r,ms));
    function row(speaker) {
      const r = document.createElement('div'); r.className = 'flex items-start gap-3';
      const a = document.createElement('div'); a.className = 'w-8 h-8 rounded-full bg-zinc-900 grid place-items-center ring'; a.textContent = AVATAR[speaker] || '🤖';
      const side = document.createElement('div'); side.className = 'flex-1 min-w-0 space-y-1';
      r.append(a, side); return { r, side };
    }
    function typingDots() { const t = document.createElement('div'); t.className = 'bubble'; t.innerHTML = '<span class="typing"><span class="dot"></span><span class="dot"></span><span class="dot"></span></span>'; return t; }
    function typewriterAsync(el, text, cps=36) {
      return new Promise(resolve => {
        const ms = Math.max(10, 1000/cps); let i=0;
        const tick = () => { if (i<=text.length){ el.textContent=text.slice(0,i); el.classList.toggle('typewriter', i<text.length); i++; setTimeout(tick, ms);} else {el.classList.remove('typewriter'); resolve();}};
        tick();
      });
    }
    async function revealSequential(container, items) {
      for (const it of items) {
        const { r, side } = row(it.speaker || 'Narrator');
        const dots = typingDots(); side.append(dots); container.append(r); container.scrollTop = container.scrollHeight;
        await sleep(180);
        dots.remove();
        const b = document.createElement('div'); b.className = 'bubble' + (it.mine ? ' me' : ''); side.append(b);
        await typewriterAsync(b, it.text);
        container.scrollTop = container.scrollHeight;
        await sleep(120);
      }
    }
    function setBusy(btn, spinner, busy) { if (!btn) return; btn.disabled = !!busy; spinner?.classList.toggle('hidden', !busy); }
    function showError(el, msg) { el.textContent = msg; el.classList.remove('hidden'); }

    // --- page switch
    const navBtns = document.querySelectorAll('[data-page]');
    navBtns.forEach(b => b.addEventListener('click', () => {
      navBtns.forEach(x => x.classList.remove('bg-zinc-900'));
      b.classList.add('bg-zinc-900');
      const id = b.dataset.page;
      $('#page-comedy').classList.toggle('hidden', id !== 'comedy');
      $('#page-numbers').classList.toggle('hidden', id !== 'numbers');
    }));

    // ===== COMEDY =====
    const outC = $('#comedyOut'); const errC = $('#comedyErr');

    async function startComedy() {
      outC.innerHTML = ''; errC.classList.add('hidden'); errC.textContent = '';
      const theme = ($('#comedyTheme')?.value || '').trim();
      const rounds = parseInt($('#comedyRounds')?.value || '6', 10);
      const btn = $('#btnStartComedy'); const spin = $('#spinComedy');
      try {
        setBusy(btn, spin, true);
        const res = await fetch('/api/comedy/duo', { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ theme, rounds }) });
        const data = await res.json();
        if (!res.ok || !data.ok || !Array.isArray(data.transcript)) throw new Error(data.error || `HTTP ${res.status}`);
        const seq = data.transcript.map(x => ({ speaker: x.speaker, mine: x.speaker === 'Joe', text: x.text }));
        await revealSequential(outC, seq);
      } catch (e) {
        showError(errC, `Comedy failed: ${String(e).slice(0,240)}`);
      } finally { setBusy(btn, spin, false); }
    }

    async function oneLiner() {
      errC.classList.add('hidden'); errC.textContent = '';
      const persona = $('#oneWho')?.value || 'Joe';
      const prompt = ($('#onePrompt')?.value || '').trim(); if (!prompt) return;
      const btn = $('#btnOneLiner'); const spin = $('#spinOne');
      try {
        setBusy(btn, spin, true);
        const res = await fetch('/api/comedy/one', { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ prompt, persona }) });
        const data = await res.json();
        if (!res.ok || !data.ok) throw new Error(data.error || `HTTP ${res.status}`);
        await revealSequential(outC, [{ speaker: persona, mine: persona === 'Joe', text: data.line }]);
      } catch (e) {
        showError(errC, `One-liner failed: ${String(e).slice(0,240)}`);
      } finally { setBusy(btn, spin, false); }
    }

    $('#btnStartComedy')?.addEventListener('click', startComedy);
    $('#btnOneLiner')?.addEventListener('click', oneLiner);

    // ===== NUMBERS =====
    const numErr = $('#numErr');

    // sub-tabs
    const subs = document.querySelectorAll('.subtab');
    subs.forEach(btn => btn.addEventListener('click', () => {
      subs.forEach(x => x.classList.remove('bg-zinc-800'));
      btn.classList.add('bg-zinc-800');
      const id = btn.dataset.subtab;
      $('#pane-you-guess').classList.toggle('hidden', id !== 'you-guess');
      $('#pane-ai-guesses').classList.toggle('hidden', id !== 'ai-guesses');
    }));

    // --- AI vs AI ---
    const outAiAi = $('#aiAiOut');
    async function runAiVsAi() {
      outAiAi.innerHTML = ''; numErr.classList.add('hidden'); numErr.textContent = '';
      const low = parseInt($('#low')?.value || '1', 10);
      const high = parseInt($('#high')?.value || '100', 10);
      const banter = !!($('#banter')?.checked);
      const btn = $('#btnRunAiAi'); const spin = $('#spinAiAi');
      try {
        setBusy(btn, spin, true);
        const res = await fetch('/api/number/ai-vs-ai', {
          method: 'POST',
          headers: { 'Content-Type': 'application/json' },
          body: JSON.stringify({ low, high, banter })
        });
        const text = await res.text(); // read raw once
        let data;
        try { data = JSON.parse(text); } catch { throw new Error(`Non-JSON response: ${text.slice(0,200)}`); }
        if (!res.ok || !data?.ok) throw new Error(data?.error || `HTTP ${res.status}`);
        if (!Array.isArray(data.steps)) throw new Error('Bad response: steps missing');

        // intro
        await revealSequential(outAiAi, [{ speaker:'Narrator', mine:true, text:`Secret picked in ${low}–${high}.` }]);
        // per step, sequential
        for (const s of data.steps) {
          await revealSequential(outAiAi, [{ speaker:'Narrator', text:`Guess ${s.guess} → ${s.feedback}${s.quip?'.':''}` }]);
          if (s.quip) await revealSequential(outAiAi, [{ speaker:'Narrator', text:s.quip }]);
        }
        await revealSequential(outAiAi, [{ speaker:'Narrator', mine:true, text:`Done. Secret was ${data.secret}.` }]);
      } catch (e) {
        showError(numErr, `AI vs AI failed: ${String(e).slice(0,240)}`);
      } finally { setBusy(btn, spin, false); }
    }
    $('#btnRunAiAi')?.addEventListener('click', runAiVsAi);

    // --- You guess ---
    const youOut = $('#youOut'); let sessionId = null;
    async function startYouGuess() {
      youOut.innerHTML=''; numErr.classList.add('hidden'); numErr.textContent='';
      const low = parseInt($('#low')?.value || '1', 10); const high = parseInt($('#high')?.value || '100', 10);
      const btn = $('#btnStartYou'); const spin = $('#spinYou');
      try {
        setBusy(btn, spin, true);
        const res = await fetch('/api/number/human-start', { method:'POST', headers:{'Content-Type':'application/json'}, body: JSON.stringify({ low, high }) });
        const data = await res.json(); if (!res.ok || !data.ok) throw new Error(data.error || `HTTP ${res.status}`);
        sessionId = data.sessionId; $('#youRange').textContent = `Range: ${data.range.low}–${data.range.high}`;
        await revealSequential(youOut, [{ speaker:'Narrator', text:'I picked a number. Try to guess!' }]);
      } catch(e){ showError(numErr, `Start game failed: ${String(e).slice(0,240)}`); }
      finally { setBusy(btn, spin, false); }
    }
    async function submitYourGuess() {
      if (!sessionId) return;
      const g = parseInt($('#youGuess')?.value || '0', 10);
      await revealSequential(youOut, [{ speaker:'You', mine:true, text:`My guess: ${g}` }]);
      const res = await fetch('/api/number/human-guess', { method:'POST', headers:{'Content-Type':'application/json'}, body: JSON.stringify({ sessionId, guess: g }) });
      const data = await res.json();
      if (!res.ok || !data.ok) return showError(numErr, `Submit failed: ${String(data.error||'HTTP '+res.status).slice(0,240)}`);
      await revealSequential(youOut, [{ speaker:'Narrator', text:`Feedback: ${data.feedback}` }]);
      if (data.feedback === 'correct') { sessionId = null; await revealSequential(youOut, [{ speaker:'Narrator', mine:true, text:'Nice! Game over.' }]); }
    }
    $('#btnStartYou')?.addEventListener('click', startYouGuess);
    $('#btnSubmitYou')?.addEventListener('click', submitYourGuess);

    // --- AI guesses ---
    const aiGOut = $('#aiGOut'); const aiGControls = $('#aiGControls'); const aiHist = [];
    async function nextAiGuess() {
      const low = parseInt($('#low')?.value || '1', 10); const high = parseInt($('#high')?.value || '100', 10);
      try {
        const res = await fetch('/api/number/ai-next-guess', { method:'POST', headers:{'Content-Type':'application/json'}, body: JSON.stringify({ low, high, history: aiHist }) });
        const data = await res.json(); if (!res.ok || !data.ok) throw new Error(data.error || `HTTP ${res.status}`);
        if (data.done && data.nextGuess == null) return revealSequential(aiGOut, [{ speaker:'Narrator', text:'Inconsistent feedback. Reset.' }]);
        if (data.done) return revealSequential(aiGOut, [{ speaker:'Narrator', mine:true, text:`${data.line || 'Got it!'} (${data.nextGuess})` }]);
        await revealSequential(aiGOut, [
          { speaker:'Narrator', text:`${data.line || 'My guess:'} (${data.nextGuess})` },
          { speaker:'Narrator', text:`Range now ${data.range.low}–${data.range.high}` }
        ]);
        $('#btnLow').dataset.guess = data.nextGuess; $('#btnHigh').dataset.guess = data.nextGuess; $('#btnCorrect').dataset.guess = data.nextGuess;
      } catch(e){ showError(numErr, `AI guesses failed: ${String(e).slice(0,240)}`); }
    }
    function startAiGuesses(){ aiHist.length=0; aiGOut.innerHTML=''; aiGControls.classList.remove('hidden'); nextAiGuess(); }
    function feedbackClick(kind){ return async (ev)=>{ const g=parseInt(ev.currentTarget.dataset.guess||'0',10); if(!g) return; aiHist.push({guess:g, feedback:kind}); await revealSequential(aiGOut, [{ speaker:'You', mine:true, text:kind }]); nextAiGuess(); }; }
    $('#btnStartAiG')?.addEventListener('click', startAiGuesses);
    $('#btnLow')?.addEventListener('click', feedbackClick('too low'));
    $('#btnHigh')?.addEventListener('click', feedbackClick('too high'));
    $('#btnCorrect')?.addEventListener('click', feedbackClick('correct'));
  });
  </script>
</body>
</html>
