<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="utf-8" />
  <meta name="viewport" content="width=device-width, initial-scale=1" />
  <title>Multimodel Generation</title>
  <style>
    :root{
      --bg:#0f172a; /* slate-900 */
      --panel:#111827; /* gray-900 */
      --card:#1f2937; /* gray-800 */
      --text:#e5e7eb; /* gray-200 */
      --muted:#9ca3af; /* gray-400 */
      --accent:#22c55e; /* green-500 */
      --accent-2:#38bdf8; /* sky-400 */
      --warn:#f59e0b; /* amber-500 */
      --bad:#ef4444; /* red-500 */
      --good:#10b981; /* emerald-500 */
      --radius:18px;
    }
    *{box-sizing:border-box}
    html,body{height:100%}
    body{
      margin:0; font-family: ui-rounded, system-ui, -apple-system, Segoe UI, Roboto, Noto Sans, Helvetica, Arial, "Apple Color Emoji","Segoe UI Emoji";
      background: radial-gradient(1200px 1200px at 10% -10%, #1e293b 0, var(--bg) 60%);
      color:var(--text);
    }
    header{
      position:sticky; top:0; z-index:30;
      backdrop-filter: blur(10px);
      background: linear-gradient(180deg, rgba(15,23,42,.9), rgba(15,23,42,.6));
      border-bottom:1px solid rgba(255,255,255,.06);
    }
    .wrap{max-width:1200px; margin-inline:auto; padding:16px}
    .toolbar{display:flex; gap:10px; flex-wrap:wrap; align-items:center; justify-content:space-between}
    .toolbar .right{display:flex; gap:10px; align-items:center; flex-wrap:wrap}
    h1{margin:0; font-size:20px; letter-spacing:.6px}
    .btn{appearance:none; border:none; padding:10px 14px; border-radius:12px; font-weight:700; cursor:pointer; color:#0b1220;
      background:linear-gradient(180deg, #fff, #dbeafe); box-shadow: inset 0 1px 0 #fff, 0 8px 20px rgba(0,0,0,.25);
      transition: transform .06s ease, box-shadow .2s ease; font-size:14px;}
    .btn:active{transform: translateY(1px)}
    .btn.ghost{background:transparent; color:var(--text); border:1px solid rgba(255,255,255,.12); box-shadow:none}
    .btn.primary{background:linear-gradient(180deg, #34d399, #10b981); color:#072b1e}
    .btn.sky{background:linear-gradient(180deg, #7dd3fc, #38bdf8)}
    .btn.warn{background:linear-gradient(180deg, #fde68a, #f59e0b)}
    main{max-width:1200px; margin:24px auto; padding:0 16px}
    .grid{display:grid; gap:16px; grid-template-columns: repeat(auto-fill, minmax(260px, 1fr));}
    .card{
      background:linear-gradient(180deg, rgba(255,255,255,.04), rgba(255,255,255,.02));
      border:1px solid rgba(255,255,255,.08); border-radius:var(--radius); overflow:hidden; position:relative;
      box-shadow: 0 20px 40px rgba(0,0,0,.25);
    }
    .thumb{aspect-ratio: 1 / 1; background:#0b1220; display:grid; place-items:center; overflow:hidden}
    .thumb img{width:100%; height:100%; object-fit:cover}
    .content{padding:12px 12px 14px; display:flex; flex-direction:column; gap:10px}
    .text{min-height:48px; line-height:1.25; font-size:18px; font-weight:700}
    .controls{display:flex; align-items:center; gap:8px;}
    .controls button{flex:1}
    .small{font-size:12px; color:var(--muted)}
    .pill{display:inline-block; padding:4px 10px; border-radius:999px; background:#0b1220; color:#c7d2fe; font-weight:700; font-size:11px; border:1px solid rgba(99,102,241,.4)}
    .player{position:fixed; right:16px; bottom:16px; z-index:50; background:var(--panel); border:1px solid rgba(255,255,255,.08); border-radius:16px; padding:12px; box-shadow:0 20px 40px rgba(0,0,0,.35); width:min(420px, calc(100% - 32px));}
    .player .row{display:flex; align-items:center; gap:10px}
    .player .row + .row{margin-top:10px}
    .progress{height:8px; background:#111827; border-radius:999px; overflow:hidden; border:1px solid rgba(255,255,255,.06)}
    .progress > span{display:block; height:100%; width:0; background:linear-gradient(90deg, var(--accent), var(--accent-2));}
    .filters{display:flex; gap:8px; flex-wrap:wrap}
    input[type="search"]{background:rgba(255,255,255,.06); color:var(--text); border:1px solid rgba(255,255,255,.12); border-radius:12px; padding:10px 12px; width:230px}
    label.switch{display:inline-flex; align-items:center; gap:8px; cursor:pointer;}
    label.switch input{appearance:none; width:40px; height:22px; background:#111827; border-radius:999px; position:relative; border:1px solid rgba(255,255,255,.12)}
    label.switch input::after{content:""; position:absolute; left:2px; top:2px; width:18px; height:18px; background:#fff; border-radius:50%; transition:.2s}
    label.switch input:checked{background:linear-gradient(90deg, var(--accent), var(--accent-2))}
    label.switch input:checked::after{transform:translateX(18px)}
    .empty{opacity:.8; text-align:center; padding:80px 12px}
    .empty h2{margin:0 0 8px}
    .kbd{font: 600 12px ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; padding:2px 6px; border-radius:6px; background:#0b1220; border:1px solid rgba(255,255,255,.12)}
    @media (prefers-reduced-motion: no-preference){
      .card{transition: transform .15s ease, box-shadow .15s ease}
      .card:hover{transform: translateY(-3px); box-shadow: 0 24px 50px rgba(0,0,0,.35)}
    }
  </style>
</head>
<body>
  <header>
    <div class="wrap toolbar">
      <div class="right">
        <div class="filters">
          <input id="q" type="search" placeholder="Search words… (e.g. zebra)" />
          <label class="switch"><input id="hideText" type="checkbox"/> Hide text</label>
        </div>
        <button class="btn sky" id="shuffleBtn">Randomize</button>
      </div>
    </div>
  </header>

  <main>
    <div id="grid" class="grid" aria-live="polite"></div>
    <div id="empty" class="empty">
      <h2>No matching items found</h2>
      <p>Try adjusting your search query.</p>
    </div>
  </main>

  <section class="player" aria-label="Player">
    <div class="row">
      <button class="btn" id="prevBtn">⟨ Prev</button>
      <button class="btn primary" id="playBtn">▶ Play</button>
      <button class="btn" id="nextBtn">Next ⟩</button>
    </div>
    <div class="row">
      <div style="flex:1">
        <div id="nowText" class="small">—</div>
        <div class="progress" aria-label="Progress"><span id="bar"></span></div>
      </div>
    </div>
  </section>

  <script>
  const state = {
    items: [],
    filtered: [],
    idx: -1,
    audioEl: new Audio()
  };

  const els = {
    grid: document.getElementById('grid'),
    empty: document.getElementById('empty'),
    q: document.getElementById('q'),
    hideText: document.getElementById('hideText'),
    playBtn: document.getElementById('playBtn'),
    prevBtn: document.getElementById('prevBtn'),
    nextBtn: document.getElementById('nextBtn'),
    shuffleBtn: document.getElementById('shuffleBtn'),
    nowText: document.getElementById('nowText'),
    bar: document.getElementById('bar'),
  };

  window.addEventListener('DOMContentLoaded', () => {
    // Hard-coded dataset
    const dataset = `
0001|a zebra grazed peacefully on the golden grassland
0002|a rubber ball bounced across the playground
0003|a broom leaned against the kitchen wall
0004|a red fire truck raced to the burning house
0005|a golden retriever chased a ball across the yard
0006|a teddy bear wore a ribbon around its neck
0007|a farm dog ran circles around the flock of sheep
0008|a panther crept quietly in the dark forest
0009|a white rabbit hopped happily across the meadow
0010|a storybook opened with pictures of castles and dragons
0011|a stag stood proudly at the forest edge
0012|a frost sparkled on the early morning grass
0013|a pair of scissors snipped through the paper
0014|a wheat field waved golden in the summer wind
0015|a white kitten chased a butterfly in the yard
0016|the horizon glowed pink with sunset light
0017|a delivery van parked outside the bakery shop
0018|a long green train rolled through the mountain valley
0019|a spinning top twirled quickly on the wooden floor
0020|a leopard stretched on a high tree branch
0021|a pencil scribbled quickly on the page
0022|a clock ticked loudly on the wall
0023|a drum thumped loudly in the parade
0024|a rocket blasted high into the night sky of stars
0025|a rabbit twitched its nose in the fresh grass
0026|a bicycle rang its bell on the garden path
0027|a proud rooster crowed loudly at sunrise
0028|a bulldozer cleared rubble from the old site
0029|a piano played soft notes on the stage
0030|a rainbow reappeared after the heavy rain
0031|a guinea pig squeaked inside its little cage
0032|a white dove fluttered above the town square
0033|a train whistled loudly as it entered the station
0034|a black goat munched grass near the stone wall
0035|a duckling paddled quickly behind its mother duck
0036|an ambulance rushed through traffic with sirens loud
0037|spring rain tapped gently on the fresh leaves
0038|a jaguar prowled near the forest river
0039|a walrus rested with its tusks on the icy beach
0040|a group of sheep huddled together in the shade
0041|a bus picked up children in the pouring rain
0042|a calm blue lake reflected the snowy peaks
0043|a pine forest whispered in the winter wind
0044|a crocodile basked lazily on the muddy bank
0045|a silver bell rang sweetly at the cottage door
0046|tulips colored the garden in springtime
0047|a goose honked loudly as it crossed the path
0048|a cherry tree bloomed pink in the spring garden
0049|a waterfall crashed loudly into the pool below
0050|a police car flashed blue lights in the dark street
0051|a wooden boat rocked gently on the calm lake
0052|a doll wore a red dress with white shoes
0053|a storm cloud gathered dark in the sky
0054|a horse neighed loudly in the village stable
0055|a sheep grazed quietly under the tall oak tree
0056|a horse cart rattled along the village road
0057|a chimpanzee clapped loudly at sunset
0058|a falcon dived swiftly toward its prey
0059|the golden sun rose slowly above the quiet hills
0060|a ginger cat purred loudly on the soft cushion
0061|a foal trotted beside its mother horse
0062|a puzzle lay unfinished on the floor
0063|a vulture circled slowly above the dry canyon
0064|a bright balloon floated above the crowded fairground
0065|a tennis racket leaned against the wall
0066|the night sky filled with shining stars
0067|a jeep bounced across the rocky desert road
0068|a camel rested under the hot desert sun
0069|a pink piglet rolled playfully in the muddy puddle
0070|a chair stood quietly beside the desk
0071|a hot air balloon drifted slowly into the sunrise sky
0072|a spade dug into the soft sand
0073|a cushion rested on the sofa seat
0074|a canyon opened wide between the cliffs
0075|an orange rolled across the kitchen counter
0076|a penguin waddled slowly across the snowy ice
0077|a green turtle crawled across the sandy beach
0078|a bison grazed heavily on the wide prairie
0079|a lighthouse shone brightly across the stormy sea
0080|a ball of yarn rolled across the floor
0081|a skateboard rolled down the hill with speed
0082|a blue sailboat floated gently on the summer sea
0083|a rooster crowed again as the sun climbed higher
0084|a trumpet shone brightly under the stage lights
0085|a basketball bounced on the court floor
0086|a rabbit nibbled fresh carrots in the garden
0087|a breeze shook the tall reeds by the river
0088|a glider floated silently in the summer sky
0089|a black car cruised along the highway at night
0090|a warm breeze rustled through the green grass
0091|a lavender field spread purple across the hill
0092|a bamboo grove swayed gently in the morning air
0093|a tram rolled slowly through the city center
0094|a brown bear caught a fish in the rushing river
0095|a brown horse galloped quickly across the open field
0096|a black cow chewed grass slowly in the green pasture
0097|a violin sang softly in the orchestra
0098|a white goose spread its wings beside the pond
0099|a rainbow stretched across the blue sky after rain
0100|a parakeet chirped brightly inside the living room
`.trim();
    loadFromContent(dataset);
  });

  function loadFromContent(txt){
    const lines = txt.split(/\r?\n/).map(l => l.trim()).filter(Boolean);
    const items = [];
    for (const line of lines){
      const [name, ...rest] = line.split('|');
      if (!name || rest.length === 0) continue;
      const text = rest.join('|').trim();
      const audio = guessAudioPath(name);
      const image = guessImagePathFromName(name);
      items.push({ name, text, audio, image });
    }
    state.items = items;
    state.filtered = items.slice();
    render();
    if (items.length) select(0);
  }

  function guessAudioPath(name){
    return `sources/${name}.wav`;
  }

  function guessImagePathFromName(name){
    return `sources/${name}.png`;
  }

  function loadAndPlay(it){
    const a = state.audioEl;
    a.pause();
    try {
      a.src = it.audio;
      a.currentTime = 0;
      a.play().catch(() => toast('Error playing audio. Serve files via a local server (e.g., python -m http.server).'));
      els.playBtn.textContent = '⏸ Pause';
      els.nowText.textContent = it.text;
    } catch (e) {
      toast('Failed to load audio file: ' + it.audio);
    }
  }

  function render(){
    const q = (els.q.value || '').toLowerCase();
    const hideText = els.hideText.checked;
    const pool = state.items.filter(it => it.text.toLowerCase().includes(q));
    state.filtered = pool;
    els.grid.innerHTML = '';
    if (!pool.length){ els.empty.hidden = false; return; }
    els.empty.hidden = true;

    for (const [i, it] of pool.entries()){
      const card = document.createElement('article');
      card.className = 'card';
      card.innerHTML = `
        <div class="thumb"><img src="${escapeHtml(it.image)}" alt="Picture for ${escapeHtml(it.text)}" onerror="this.src=''; this.parentElement.style.background='#0b1220'"/></div>
        <div class="content">
          <div class="text" ${hideText? 'style="filter: blur(10px)"':''}>${escapeHtml(it.text)}</div>
          <div class="controls">
            <button class="btn primary" data-i="${i}">▶ Play</button>
          </div>
          <div class="small"><span class="pill">${escapeHtml(it.name)}</span></div>
        </div>`;

      card.querySelector('button.btn.primary').addEventListener('click', () => select(i));
      els.grid.appendChild(card);
    }
  }

  function select(i){
    if (!state.filtered.length) return;
    state.idx = Math.max(0, Math.min(i, state.filtered.length - 1));
    const it = state.filtered[state.idx];
    loadAndPlay(it);
  }

  els.playBtn.addEventListener('click', () => {
    if (!state.filtered.length) return;
    const a = state.audioEl;
    if (a.paused){ a.play().catch(() => toast('Error playing audio.')); els.playBtn.textContent = '⏸ Pause'; }
    else { a.pause(); els.playBtn.textContent = '▶ Play'; }
  });

  els.prevBtn.addEventListener('click', () => { select((state.idx - 1 + state.filtered.length) % state.filtered.length); });
  els.nextBtn.addEventListener('click', () => { select((state.idx + 1) % state.filtered.length); });
  els.shuffleBtn.addEventListener('click', () => {
    shuffleInPlace(state.items);
    render();
    if (state.items.length) select(0);
  });
  els.q.addEventListener('input', render);
  els.hideText.addEventListener('change', render);

  state.audioEl.addEventListener('timeupdate', () => {
    const a = state.audioEl;
    const pct = a.duration ? (a.currentTime / a.duration) * 100 : 0;
    els.bar.style.width = pct + '%';
  });

  state.audioEl.addEventListener('ended', () => {
    els.playBtn.textContent = '▶ Play';
  });

  document.addEventListener('keydown', e => {
    if (e.target === els.q) return;
    if (e.code === 'Space'){ e.preventDefault(); els.playBtn.click(); }
    else if (e.key === 'ArrowRight'){ els.nextBtn.click(); }
    else if (e.key === 'ArrowLeft'){ els.prevBtn.click(); }
  });

  function toast(msg){
    const d = document.createElement('div');
    d.textContent = msg;
    d.style.cssText = 'position:fixed; left:50%; bottom:24px; transform:translateX(-50%); background:linear-gradient(90deg, var(--accent), var(--accent-2)); color:#08251b; font-weight:800; padding:10px 14px; border-radius:999px; box-shadow:0 10px 30px rgba(0,0,0,.35);';
    document.body.appendChild(d);
    setTimeout(() => d.remove(), 1800);
  }

  function shuffleInPlace(arr){
    for (let i=arr.length-1; i>0; i--){ const j = Math.floor(Math.random()*(i+1)); [arr[i], arr[j]] = [arr[j], arr[i]]; }
  }

  function escapeHtml(s){
    return s.replace(/[&<>"']/g, c => ({'&':'&amp;','<':'&lt;','>':'&gt;','"':'&quot;','\'':'&#39;'}[c]));
  }
  </script>
</body>
</html>