<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<style>
  *{margin:0;padding:0;box-sizing:border-box}
  html,body{width:100%;height:100%;overflow:hidden;font-family:'Helvetica Neue',Arial,sans-serif;background:#fafafa}
  .wrap{display:flex;flex-direction:column;height:100%;padding:12px 16px}
  .top{display:flex;gap:20px;align-items:flex-start;padding-bottom:10px;border-bottom:1px solid #e0e0e0;margin-bottom:6px;flex-wrap:wrap}
  .slider-group{display:flex;flex-direction:column;gap:2px}
  .slider-group label{font-size:11px;color:#555;font-weight:500}
  .slider-group input[type=range]{width:140px;cursor:pointer}
  .slider-val{font-size:11px;color:#2c3e50;font-weight:600;font-variant-numeric:tabular-nums}
  .chart-area{flex:1;position:relative;min-height:0}
  canvas{width:100%;height:100%;display:block}
  .note{font-size:10px;color:#999;padding-top:4px;text-align:center}
</style>
</head>
<body>
<div class="wrap">
  <div class="top">
    <div class="slider-group">
      <label>Power-law exponent γ: <span class="slider-val" id="gammaVal">2.7</span></label>
      <input type="range" id="gamma" min="1.5" max="3.5" step="0.1" value="2.7">
    </div>
    <div class="slider-group">
      <label>Initial ρ<sub>T</sub>: <span class="slider-val" id="rhoTVal">0.35</span></label>
      <input type="range" id="rhoT0" min="0.05" max="0.80" step="0.05" value="0.35">
    </div>
    <div class="slider-group">
      <label>Incentive u (test-time compute): <span class="slider-val" id="uVal">1.0</span></label>
      <input type="range" id="uCtrl" min="0.2" max="3.0" step="0.1" value="1.0">
    </div>
    <div class="slider-group">
      <label>Social influence strength: <span class="slider-val" id="sVal">1.0</span></label>
      <input type="range" id="sCtrl" min="0.2" max="2.5" step="0.1" value="1.0">
    </div>
  </div>
  <div class="chart-area"><canvas id="cv"></canvas></div>
  <div class="note">Population state ρ = (ρ<sub>T</sub>, ρ<sub>H</sub>, ρ<sub>D</sub>) evolution over interaction rounds, computed via the mean-field ODE (Eq. 1). Adjust parameters to explore Theorem 1 predictions.</div>
</div>

<script>
const cv=document.getElementById('cv'),ctx=cv.getContext('2d');
const COLORS={T:'#27ae60',H:'#e74c3c',D:'#95a5a6'};

function resize(){
  const r=cv.parentElement.getBoundingClientRect();
  cv.width=r.width*devicePixelRatio;cv.height=r.height*devicePixelRatio;
  cv.style.width=r.width+'px';cv.style.height=r.height+'px';
  ctx.setTransform(devicePixelRatio,0,0,devicePixelRatio,0,0);
}

// Simplified MFD ODE solver using RUM-inspired transition kernel
function solveMFD(gamma, rhoT0, u, socialStrength) {
  const T_MAX = 30; // interaction rounds
  const dt = 0.05;
  const steps = Math.floor(T_MAX / dt);
  
  // Initial conditions
  let rhoT = rhoT0;
  let rhoH = (1 - rhoT0) * 0.65; // most non-truthful start hallucinating
  let rhoD = 1 - rhoT - rhoH;
  
  let trajectory = [{t:0, T:rhoT, H:rhoH, D:rhoD}];
  
  // Power-law edge weighting: lower gamma = more hub influence
  const hubFactor = Math.pow(3.5 - gamma + 0.5, 1.2);
  
  for (let s = 0; s < steps; s++) {
    // Edge-weighted truth probability (θ in the paper)
    // Hubs with high out-degree contribute more to θ
    let theta = rhoT * (1 + (hubFactor - 1) * 0.3);
    theta = Math.max(0, Math.min(1, theta));
    
    // RUM-based transition rates (logit model)
    // κ_{H→T}: hallucinating to truthful
    let deltaHT = (-0.5 + 2.0 * u * socialStrength * theta);
    let kHT = 1 / (1 + Math.exp(-deltaHT));
    
    // κ_{T→H}: truthful to hallucinating  
    let deltaTH = (-1.0 * u + 1.5 * socialStrength * (1 - theta));
    let kTH = 1 / (1 + Math.exp(-deltaTH));
    
    // κ_{D→T}: don't-know to truthful
    let deltaDT = (-0.3 + 1.8 * u * socialStrength * theta);
    let kDT = 1 / (1 + Math.exp(-deltaDT));
    
    // κ_{D→H}: don't-know to hallucinating
    let deltaDH = (-0.5 + 1.3 * socialStrength * (1 - theta));
    let kDH = 1 / (1 + Math.exp(-deltaDH));
    
    // κ_{T→D} and κ_{H→D} (small)
    let kTD = 0.02 / u;
    let kHD = 0.05;
    
    // MFD ODE: dρ/dt = F(κ, u, Q)ρ
    let dT = kHT * rhoH + kDT * rhoD - (kTH + kTD) * rhoT;
    let dH = kTH * rhoT + kDH * rhoD - (kHT + kHD) * rhoH;
    let dD = kTD * rhoT + kHD * rhoH - (kDT + kDH) * rhoD;
    
    rhoT += dT * dt;
    rhoH += dH * dt;
    rhoD += dD * dt;
    
    // Project to simplex
    rhoT = Math.max(0, rhoT);
    rhoH = Math.max(0, rhoH);
    rhoD = Math.max(0, rhoD);
    let sum = rhoT + rhoH + rhoD;
    rhoT /= sum; rhoH /= sum; rhoD /= sum;
    
    let time = (s + 1) * dt;
    if (Math.abs(time - Math.round(time)) < dt / 2) {
      trajectory.push({t: Math.round(time), T: rhoT, H: rhoH, D: rhoD});
    }
  }
  return trajectory;
}

function drawChart() {
  const gamma = +document.getElementById('gamma').value;
  const rhoT0 = +document.getElementById('rhoT0').value;
  const u = +document.getElementById('uCtrl').value;
  const s = +document.getElementById('sCtrl').value;
  
  document.getElementById('gammaVal').textContent = gamma.toFixed(1);
  document.getElementById('rhoTVal').textContent = rhoT0.toFixed(2);
  document.getElementById('uVal').textContent = u.toFixed(1);
  document.getElementById('sVal').textContent = s.toFixed(1);
  
  const traj = solveMFD(gamma, rhoT0, u, s);
  
  const W = cv.width / devicePixelRatio;
  const H = cv.height / devicePixelRatio;
  const pad = {l:52, r:16, t:16, b:36};
  const cw = W - pad.l - pad.r;
  const ch = H - pad.t - pad.b;
  
  ctx.clearRect(0, 0, W, H);
  
  // Grid
  ctx.strokeStyle = '#eee'; ctx.lineWidth = 0.5;
  for (let v = 0; v <= 1; v += 0.2) {
    let y = pad.t + ch * (1 - v);
    ctx.beginPath(); ctx.moveTo(pad.l, y); ctx.lineTo(pad.l + cw, y); ctx.stroke();
    ctx.fillStyle = '#999'; ctx.font = '10px Helvetica Neue, sans-serif'; ctx.textAlign = 'right';
    ctx.fillText(v.toFixed(1), pad.l - 6, y + 3);
  }
  for (let t = 0; t <= 30; t += 5) {
    let x = pad.l + (t / 30) * cw;
    ctx.beginPath(); ctx.moveTo(x, pad.t); ctx.lineTo(x, pad.t + ch); ctx.stroke();
    ctx.fillStyle = '#999'; ctx.font = '10px Helvetica Neue, sans-serif'; ctx.textAlign = 'center';
    ctx.fillText(t, x, pad.t + ch + 14);
  }
  
  // Axis labels
  ctx.fillStyle = '#666'; ctx.font = '11px Helvetica Neue, sans-serif';
  ctx.textAlign = 'center';
  ctx.fillText('Interaction Rounds', pad.l + cw / 2, H - 4);
  ctx.save(); ctx.translate(12, pad.t + ch / 2); ctx.rotate(-Math.PI / 2);
  ctx.fillText('Proportion ρ', 0, 0); ctx.restore();
  
  // Axis frame
  ctx.strokeStyle = '#ccc'; ctx.lineWidth = 1;
  ctx.strokeRect(pad.l, pad.t, cw, ch);
  
  // Draw trajectories
  function drawLine(key, color, dashed) {
    ctx.beginPath();
    ctx.strokeStyle = color; ctx.lineWidth = 2.5;
    if (dashed) ctx.setLineDash([6, 3]); else ctx.setLineDash([]);
    traj.forEach((p, i) => {
      let x = pad.l + (p.t / 30) * cw;
      let y = pad.t + ch * (1 - p[key]);
      if (i === 0) ctx.moveTo(x, y); else ctx.lineTo(x, y);
    });
    ctx.stroke();
    ctx.setLineDash([]);
    
    // Endpoint dot
    let last = traj[traj.length - 1];
    let ex = pad.l + (last.t / 30) * cw;
    let ey = pad.t + ch * (1 - last[key]);
    ctx.beginPath(); ctx.arc(ex, ey, 4, 0, Math.PI * 2);
    ctx.fillStyle = color; ctx.fill();
    ctx.strokeStyle = '#fff'; ctx.lineWidth = 1.5; ctx.stroke();
    
    // Fixed point label
    ctx.fillStyle = color; ctx.font = 'bold 11px Helvetica Neue, sans-serif';
    ctx.textAlign = 'left';
    let label = key === 'T' ? 'ρ*_T=' : key === 'H' ? 'ρ*_H=' : 'ρ*_D=';
    ctx.fillText(label + last[key].toFixed(2), ex + 8, ey + (key === 'T' ? -4 : key === 'H' ? 12 : 4));
  }
  
  // Fill area under truthful curve
  ctx.beginPath();
  ctx.moveTo(pad.l, pad.t + ch);
  traj.forEach(p => {
    let x = pad.l + (p.t / 30) * cw;
    let y = pad.t + ch * (1 - p.T);
    ctx.lineTo(x, y);
  });
  ctx.lineTo(pad.l + (traj[traj.length-1].t / 30) * cw, pad.t + ch);
  ctx.closePath();
  ctx.fillStyle = '#27ae6015'; ctx.fill();
  
  drawLine('T', COLORS.T, false);
  drawLine('H', COLORS.H, false);
  drawLine('D', COLORS.D, true);
}

resize();
drawChart();
window.addEventListener('resize', () => { resize(); drawChart(); });
['gamma', 'rhoT0', 'uCtrl', 'sCtrl'].forEach(id => {
  document.getElementById(id).addEventListener('input', drawChart);
});
</script>
</body>
</html>
