<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Answer Flow: ZS → CoT</title>
<style>
  @import url('https://fonts.googleapis.com/css2?family=IBM+Plex+Mono:wght@600;700&family=Inter:wght@400;600;700&display=swap');
  html, body { margin: 0; padding: 0; width: 100%; height: 100%; font-family: 'Inter', sans-serif; background: #0f1117; overflow: hidden; display: flex; }
  .left { flex: 0 0 62%; height: 100%; position: relative; }
  svg { display: block; width: 100%; height: 100%; }
  .right { flex: 1; height: 100%; display: flex; flex-direction: column; justify-content: center; align-items: flex-start; padding: 20px 24px; gap: 0; border-left: 1px solid #1e2030; }
  .flip-pct { font-size: 44px; font-weight: 700; font-family: 'IBM Plex Mono', monospace; color: #f87171; line-height: 1; }
  .flip-label { font-size: 13px; font-weight: 600; color: #f87171; margin-top: 2px; }
  .model-info { font-size: 12px; color: #6b7080; margin-top: 6px; }
  .divider { width: 40px; height: 1px; background: #2a2d3a; margin: 16px 0; }
  .legend { display: flex; flex-direction: column; gap: 8px; }
  .leg-item { display: flex; align-items: center; gap: 8px; font-size: 12px; color: #8b8fa3; }
  .leg-dot { width: 10px; height: 10px; border-radius: 50%; flex-shrink: 0; }
  .leg-count { font-weight: 600; font-family: 'IBM Plex Mono', monospace; color: #c8cad4; min-width: 36px; text-align: right; }
  .controls { display: flex; flex-wrap: wrap; gap: 5px; margin-top: 16px; }
  .btn { padding: 5px 12px; border: 1px solid #2a2d3a; border-radius: 14px; background: #1e2030; color: #8b8fa3; font-size: 11px; font-weight: 600; cursor: pointer; transition: all 0.2s; font-family: 'Inter', sans-serif; }
  .btn.active { background: #3b3f54; color: #fff; border-color: #4e79a7; }
  .btn:hover { background: #2a2d3a; color: #fff; }
  .tooltip { position: absolute; background: rgba(20,22,35,0.95); border: 1px solid #3b3f54; border-radius: 8px; padding: 10px 14px; font-size: 13px; line-height: 1.5; pointer-events: none; color: #c8cad4; box-shadow: 0 4px 12px rgba(0,0,0,0.4); z-index: 20; }
</style>
</head>
<body>
<div class="left">
  <div class="tooltip" id="tip" style="display:none;"></div>
  <svg id="chart"></svg>
</div>
<div class="right" id="rightPanel">
  <div class="flip-pct" id="flipPct">16.2%</div>
  <div class="flip-label">of correct answers flipped</div>
  <div class="model-info" id="modelInfo">Qwen2.5-7B · GSM8K · n=1319</div>
  <div class="divider"></div>
  <div class="legend" id="legend"></div>
  <div class="controls" id="controls">
    <button class="btn active" data-model="qwen">Qwen (75%)</button>
    <button class="btn" data-model="llama31">Llama-3.1 (23%)</button>
    <button class="btn" data-model="llama2">Llama-2 (16%)</button>
    <button class="btn" data-model="mistral">Mistral (7%)</button>
  </div>
</div>
<script src="https://d3js.org/d3.v7.min.js"></script>
<script>
const models = {
  qwen: { name:"Qwen2.5-7B", n:1319, zs_c:241, zs_i:1078, cot_c:995, ci:39, ic:793, cc:202, ii:285 },
  llama31: { name:"Llama-3.1-8B", n:1319, zs_c:192, zs_i:1127, cot_c:309, ci:131, ic:248, cc:61, ii:879 },
  llama2: { name:"Llama-2-7B", n:1319, zs_c:55, zs_i:1264, cot_c:204, ci:46, ic:195, cc:9, ii:1069 },
  mistral: { name:"Mistral-7B", n:1319, zs_c:59, zs_i:1260, cot_c:95, ci:52, ic:88, cc:7, ii:1172 }
};

const svg = d3.select("#chart");
const tip = d3.select("#tip");
let active = "qwen";

document.querySelectorAll(".btn").forEach(b => {
  b.addEventListener("click", () => {
    document.querySelectorAll(".btn").forEach(x=>x.classList.remove("active"));
    b.classList.add("active");
    active = b.dataset.model;
    draw();
  });
});

function draw() {
  const svgEl = document.querySelector(".left");
  const W = svgEl.clientWidth, H = svgEl.clientHeight;
  svg.selectAll("*").remove();
  const m = models[active];
  const g = svg.append("g");

  const topPad = 28, botPad = 16;
  const drawH = H - topPad - botPad;
  const midY = topPad + drawH / 2;

  const colX = [W * 0.22, W * 0.78];
  const maxH = drawH * 0.85;
  const barW = 40;

  const zsCorrectH = (m.zs_c / m.n) * maxH;
  const zsIncorrectH = (m.zs_i / m.n) * maxH;
  const zsTop = midY - (zsCorrectH + zsIncorrectH) / 2;

  const cotCorrectH = (m.cot_c / m.n) * maxH;
  const cotIncorrectH = ((m.n - m.cot_c) / m.n) * maxH;
  const cotTop = midY - (cotCorrectH + cotIncorrectH) / 2;

  // ZS bars
  g.append("rect").attr("x",colX[0]-barW/2).attr("y",zsTop).attr("width",barW).attr("height",zsCorrectH).attr("fill","#4ade80").attr("rx",4).attr("opacity",0.85);
  g.append("rect").attr("x",colX[0]-barW/2).attr("y",zsTop+zsCorrectH+2).attr("width",barW).attr("height",zsIncorrectH).attr("fill","#6b7080").attr("rx",4).attr("opacity",0.5);

  // CoT bars
  g.append("rect").attr("x",colX[1]-barW/2).attr("y",cotTop).attr("width",barW).attr("height",cotCorrectH).attr("fill","#4ade80").attr("rx",4).attr("opacity",0.85);
  g.append("rect").attr("x",colX[1]-barW/2).attr("y",cotTop+cotCorrectH+2).attr("width",barW).attr("height",cotIncorrectH).attr("fill","#6b7080").attr("rx",4).attr("opacity",0.5);

  // Flows
  const ccH = (m.cc / m.n) * maxH;
  drawFlow(g, colX[0]+barW/2, zsTop, ccH, colX[1]-barW/2, cotTop, ccH, "#4ade80", 0.25, `Stayed correct: ${m.cc}`);

  const ciH = Math.max((m.ci / m.n) * maxH, 3);
  drawFlow(g, colX[0]+barW/2, zsTop+ccH, ciH, colX[1]-barW/2, cotTop+cotCorrectH+2, ciH, "#f87171", 0.6, `⚡ FLIPPED: ${m.ci} correct → incorrect`);

  const icH = (m.ic / m.n) * maxH;
  drawFlow(g, colX[0]+barW/2, zsTop+zsCorrectH+2, icH, colX[1]-barW/2, cotTop+ccH, icH, "#60a5fa", 0.25, `Rescued: ${m.ic} incorrect → correct`);

  const iiH = (m.ii / m.n) * maxH;
  drawFlow(g, colX[0]+barW/2, zsTop+zsCorrectH+2+icH, iiH, colX[1]-barW/2, cotTop+cotCorrectH+2+ciH, iiH, "#6b7080", 0.12, `Stayed wrong: ${m.ii}`);

  // Column headers
  g.append("text").attr("x",colX[0]).attr("y",zsTop-10).attr("text-anchor","middle").attr("fill","#8b8fa3").attr("font-size","12px").attr("font-weight",700).text("Zero-Shot");
  g.append("text").attr("x",colX[1]).attr("y",cotTop-10).attr("text-anchor","middle").attr("fill","#8b8fa3").attr("font-size","12px").attr("font-weight",700).text("Chain-of-Thought");

  // Bar count labels (beside bars)
  g.append("text").attr("x",colX[0]-barW/2-5).attr("y",zsTop+zsCorrectH/2+4).attr("text-anchor","end").attr("fill","#4ade80").attr("font-size","11px").attr("font-weight",600).text(`${m.zs_c} ✓`);
  g.append("text").attr("x",colX[0]-barW/2-5).attr("y",zsTop+zsCorrectH+2+Math.min(zsIncorrectH/2,40)+4).attr("text-anchor","end").attr("fill","#6b7080").attr("font-size","10px").text(`${m.zs_i} ✗`);

  g.append("text").attr("x",colX[1]+barW/2+5).attr("y",cotTop+Math.min(cotCorrectH/2,40)+4).attr("text-anchor","start").attr("fill","#4ade80").attr("font-size","11px").attr("font-weight",600).text(`${m.cot_c} ✓`);
  g.append("text").attr("x",colX[1]+barW/2+5).attr("y",cotTop+cotCorrectH+2+Math.min(cotIncorrectH/2,40)+4).attr("text-anchor","start").attr("fill","#6b7080").attr("font-size","10px").text(`${m.n-m.cot_c} ✗`);

  // Update right panel
  const flipPct = ((m.ci / m.zs_c) * 100).toFixed(1);
  document.getElementById("flipPct").textContent = flipPct + "%";
  document.getElementById("modelInfo").textContent = `${m.name} · GSM8K · n=${m.n}`;

  const legData = [
    { c:"#4ade80", t:"Stayed correct", n:m.cc },
    { c:"#f87171", t:"Flipped (C→I)", n:m.ci },
    { c:"#60a5fa", t:"Rescued (I→C)", n:m.ic },
    { c:"#6b7080", t:"Stayed wrong", n:m.ii }
  ];
  const legEl = document.getElementById("legend");
  legEl.innerHTML = "";
  legData.forEach(it => {
    const row = document.createElement("div");
    row.className = "leg-item";
    row.innerHTML = `<span class="leg-dot" style="background:${it.c}"></span><span class="leg-count">${it.n}</span><span>${it.t}</span>`;
    legEl.appendChild(row);
  });
}

function drawFlow(g, x1, y1s, h1, x2, y2s, h2, color, opacity, label) {
  const path = d3.path();
  const cx = (x1+x2)/2;
  path.moveTo(x1, y1s);
  path.bezierCurveTo(cx, y1s, cx, y2s, x2, y2s);
  path.lineTo(x2, y2s + h2);
  path.bezierCurveTo(cx, y2s+h2, cx, y1s+h1, x1, y1s+h1);
  path.closePath();

  g.append("path").attr("d", path.toString()).attr("fill", color).attr("opacity", opacity)
    .style("cursor","pointer")
    .on("mouseover", e => { tip.style("display","block").html(`<b style="color:${color}">${label}</b>`); })
    .on("mousemove", e => { tip.style("left",(e.pageX+12)+"px").style("top",(e.pageY-10)+"px"); })
    .on("mouseout", () => tip.style("display","none"));
}

draw();
window.addEventListener("resize", draw);
</script>
</body>
</html>
