<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Progressive CoT Trajectory</title>
<style>
  html, body { margin: 0; padding: 0; overflow: hidden; width: 100%; height: 100%; font-family: 'Georgia', serif; background: #fafafa; }
  svg { display: block; width: 100%; height: 100%; }
  .controls { position: absolute; top: 12px; left: 50%; transform: translateX(-50%); display: flex; gap: 8px; z-index: 10; }
  .btn { padding: 6px 14px; border: 2px solid #ddd; border-radius: 20px; background: #fff; font-size: 12px; font-weight: 600; cursor: pointer; transition: all 0.2s; font-family: 'Georgia', serif; }
  .btn.active { border-color: #333; background: #333; color: #fff; }
  .btn:hover { border-color: #666; }
  .tooltip { position: absolute; background: rgba(255,255,255,0.96); border: 1px solid #ccc; border-radius: 6px; padding: 8px 12px; font-size: 12px; pointer-events: none; box-shadow: 0 2px 8px rgba(0,0,0,0.12); }
  .slider-container { position: absolute; bottom: 20px; left: 50%; transform: translateX(-50%); display: flex; align-items: center; gap: 12px; background: rgba(255,255,255,0.9); padding: 8px 16px; border-radius: 20px; box-shadow: 0 2px 6px rgba(0,0,0,0.1); }
  .slider-label { font-size: 13px; font-weight: 600; color: #333; white-space: nowrap; }
  input[type=range] { width: 200px; }
</style>
</head>
<body>
<div class="controls">
  <button class="btn active" data-model="all">All Models</button>
  <button class="btn" data-model="qwen">Qwen</button>
  <button class="btn" data-model="llama31">Llama-3.1</button>
  <button class="btn" data-model="llama2">Llama-2</button>
  <button class="btn" data-model="mistral">Mistral</button>
</div>
<div class="slider-container">
  <span class="slider-label">Token checkpoint:</span>
  <input type="range" id="tokenSlider" min="0" max="9" value="9" step="1">
  <span class="slider-label" id="tokenLabel">512 tokens</span>
</div>
<div class="tooltip" id="tip" style="display:none;"></div>
<svg id="chart"></svg>
<script src="https://d3js.org/d3.v7.min.js"></script>
<script>
const checkpoints = [16,32,48,64,96,128,192,256,384,512];
const models = {
  qwen: { name: "Qwen2.5-7B", color: "#4e79a7", zs: 18.3, cot: 75.4,
    acc: [37.8, 51.2, 58.6, 63.4, 68.1, 71.2, 73.8, 74.6, 75.2, 75.4] },
  llama31: { name: "Llama-3.1-8B", color: "#e15759", zs: 14.6, cot: 23.4,
    acc: [8.2, 11.5, 14.3, 16.8, 19.1, 20.7, 21.9, 22.6, 23.1, 23.4] },
  llama2: { name: "Llama-2-7B", color: "#f28e2b", zs: 4.2, cot: 15.5,
    acc: [3.1, 5.8, 8.2, 9.7, 11.4, 12.8, 13.9, 14.6, 15.2, 15.5] },
  mistral: { name: "Mistral-7B", color: "#76b7b2", zs: 4.5, cot: 7.2,
    acc: [2.1, 3.4, 4.2, 4.8, 5.5, 6.0, 6.4, 6.7, 7.0, 7.2] }
};

const svg = d3.select("#chart");
const tip = d3.select("#tip");
let activeModel = "all";
let maxCheckpoint = 9;

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

document.getElementById("tokenSlider").addEventListener("input", e => {
  maxCheckpoint = +e.target.value;
  document.getElementById("tokenLabel").textContent = checkpoints[maxCheckpoint] + " tokens";
  draw();
});

function draw() {
  const W = window.innerWidth, H = window.innerHeight;
  const m = { top: 55, right: 120, bottom: 80, left: 60 };
  const w = W - m.left - m.right, h = H - m.top - m.bottom;
  svg.selectAll("*").remove();
  const g = svg.append("g").attr("transform", `translate(${m.left},${m.top})`);

  const visibleCPs = checkpoints.slice(0, maxCheckpoint + 1);
  const x = d3.scalePoint().domain(visibleCPs.map(String)).range([0, w]).padding(0.1);
  const y = d3.scaleLinear().domain([0, 82]).range([h, 0]);

  g.append("g").attr("class","axis").attr("transform",`translate(0,${h})`).call(d3.axisBottom(x).tickFormat(d=>d+"t"));
  g.append("g").attr("class","axis").call(d3.axisLeft(y).ticks(6).tickFormat(d=>d+"%"));
  g.append("text").attr("x",w/2).attr("y",h+45).attr("text-anchor","middle").attr("font-size","13px").attr("fill","#333").attr("font-weight",600).text("CoT Token Checkpoint");
  g.append("text").attr("transform","rotate(-90)").attr("x",-h/2).attr("y",-45).attr("text-anchor","middle").attr("font-size","13px").attr("fill","#333").attr("font-weight",600).text("Accuracy (%)");

  const keys = activeModel === "all" ? Object.keys(models) : [activeModel];

  keys.forEach(key => {
    const m2 = models[key];
    const pts = visibleCPs.map((cp, i) => ({ cp: String(cp), acc: m2.acc[i] }));
    const line = d3.line().x(d => x(d.cp)).y(d => y(d.acc)).curve(d3.curveMonotoneX);

    // ZS baseline
    g.append("line").attr("x1",0).attr("x2",w).attr("y1",y(m2.zs)).attr("y2",y(m2.zs))
      .attr("stroke",m2.color).attr("stroke-width",1).attr("stroke-dasharray","4,3").attr("opacity",0.5);

    // Line
    g.append("path").datum(pts).attr("d",line).attr("fill","none")
      .attr("stroke",m2.color).attr("stroke-width",2.5).attr("opacity", keys.length>1?0.8:1);

    // Dots
    g.selectAll(`.dot-${key}`).data(pts).enter().append("circle")
      .attr("cx", d=>x(d.cp)).attr("cy", d=>y(d.acc)).attr("r",5)
      .attr("fill",m2.color).attr("stroke","#fff").attr("stroke-width",1.5)
      .style("cursor","pointer")
      .on("mouseover", (e,d) => {
        tip.style("display","block").html(
          `<b style="color:${m2.color}">${m2.name}</b><br>
           Checkpoint: ${d.cp} tokens<br>
           Accuracy: <b>${d.acc.toFixed(1)}%</b><br>
           ZS baseline: ${m2.zs}%`
        );
      })
      .on("mousemove", e => { tip.style("left",(e.pageX+12)+"px").style("top",(e.pageY-10)+"px"); })
      .on("mouseout", () => tip.style("display","none"));

    // Label
    const lastPt = pts[pts.length-1];
    g.append("text").attr("x",x(lastPt.cp)+10).attr("y",y(lastPt.acc)+4)
      .attr("font-size","12px").attr("fill",m2.color).attr("font-weight",600)
      .text(m2.name.split("-")[0]);
  });
}

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