<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Reaction‑Reactor Demo</title>
<style>
  body {font-family: Arial, Helvetica, sans-serif; margin:0; padding:0; background:#fff;}
  #demo-container {max-width:900px; margin:auto; padding:20px;}
  #control-panel {
    display:grid;
    grid-template-columns:repeat(2,1fr);
    gap:15px;
    background:#f0f0f0;
    border:1px solid #ccc;
    padding:15px;
  }
  #control-panel div {
    display:flex;
    align-items:center;
    gap:10px;
  }
  #control-panel label {flex:1; min-width:180px;}
  #control-panel input[type=range] {flex:2;}
  #control-panel span {width:50px; text-align:right;}
  #plot-container {
    width:100%; height:500px;
    background:#fff;
    border:1px solid #ccc;
    margin-top:20px;
  }
</style>
</head>
<body>

<div id="demo-container">

  <!-- Control Panel -->
  <div id="control-panel">
    <div>
      <label for="slider-ua">heat transfer coefficient (cal/(dm²·K·s))</label>
      <input type="range" id="slider-ua" min="0" max="20" step="0.1" value="0">
      <span id="ua-value">0</span>
    </div>

    <div>
      <label for="slider-kr0">reverse reaction pre‑exponential factor (1/s)</label>
      <input type="range" id="slider-kr0" min="0" max="10" step="0.1" value="0">
      <span id="kr0-value">0</span>
    </div>

    <div>
      <label for="slider-tf">feed temperature (K)</label>
      <input type="range" id="slider-tf" min="250" max="350" step="1" value="265">
      <span id="tf-value">265</span>
    </div>

    <div>
      <label for="slider-tau">residence time (s)</label>
      <input type="range" id="slider-tau" min="100" max="1000" step="10" value="400">
      <span id="tau-value">400</span>
    </div>
  </div>

  <!-- Plot area -->
  <div id="plot-container"></div>
</div>

<!-- Plotly CDN -->
<script src="https://cdn.plot.ly/plotly-latest.min.js"></script>

<script>
/* ---------- constants ---------- */
const C_A0 = 10;               // mmol/dm³
const R = 1.987;               // cal/(mol·K)
const E_f = 25000;             // cal/mol
const dH = -200000;            // cal/mol
const E_r = E_f - dH;          // cal/mol
const k_f0 = 1e15;             // 1/s
const rhoCp = 20000;           // cal/(dm³·K)

/* ---------- UI elements ---------- */
const uaSlider   = document.getElementById('slider-ua');
const kr0Slider  = document.getElementById('slider-kr0');
const tfSlider   = document.getElementById('slider-tf');
const tauSlider  = document.getElementById('slider-tau');

const uaValSpan  = document.getElementById('ua-value');
const kr0ValSpan = document.getElementById('kr0-value');
const tfValSpan  = document.getElementById('tf-value');
const tauValSpan = document.getElementById('tau-value');

/* ---------- helper functions ---------- */
function fmtSci(num) {
  // format number like 9.0 × 10¹²
  const exp = Math.floor(Math.log10(num));
  const mant = (num / Math.pow(10, exp)).toFixed(1);
  return `${mant} × 10${exp}`;
}

/* ---------- main calculation & plotting ---------- */
function updateAll() {
  // read slider values
  const UA   = parseFloat(uaSlider.value);
  const kr0  = parseFloat(kr0Slider.value) * 1e12; // convert to real units
  const T_f  = parseFloat(tfSlider.value);
  const tau  = parseFloat(tauSlider.value);

  // update displayed numbers
  uaValSpan.textContent   = UA.toFixed(1);
  kr0ValSpan.textContent  = fmtSci(kr0);
  tfValSpan.textContent   = T_f.toFixed(0);
  tauValSpan.textContent  = tau.toFixed(0);

  // temperature range for the plot
  const T_min = 250, T_max = 400;
  const N = 200;
  const Ts = Array.from({length:N}, (_,i)=> T_min + (T_max-T_min)*i/(N-1));

  // compute mass‑balance curve (green)
  const C_B_mb = Ts.map(T=> {
    const k_f = k_f0 * Math.exp(-E_f/(R*T));
    const k_r = kr0 * Math.exp(-E_r/(R*T));
    const C_B = (tau * C_A0 * k_f) / (1 + tau*(k_f + k_r));
    return C_B;
  });

  // compute energy‑balance curve (blue)
  const C_B_eb = Ts.map(T=> {
    // assume coolant temperature = feed temperature
    const deltaT = T - T_f;
    const C_B = (rhoCp*deltaT + UA*tau*deltaT) / (-dH);
    return C_B;
  });

  // build Plotly traces
  const traceMB = {
    x: Ts,
    y: C_B_mb,
    mode: 'lines',
    name: 'mass balance',
    line: {color:'#008000', width:2}
  };
  const traceEB = {
    x: Ts,
    y: C_B_eb,
    mode: 'lines',
    name: 'energy balance',
    line: {color:'#0000FF', width:2}
  };

  const layout = {
    title: '',
    xaxis: {title:'temperature (K)', range:[T_min,T_max], gridcolor:'#e0e0e0'},
    yaxis: {title:'product concentration (mmol/dm³)', range:[0,12], gridcolor:'#e0e0e0'},
    legend: {orientation:'h', x:0.5, xanchor:'center', y:-0.2},
    margin:{l:60,r:20,b:60,t:30},
    annotations:[
      {x:360, y:C_B_mb[findIdx(Ts,360)], xref:'x', yref:'y',
       text:'mass balance', showarrow:true,
       arrowhead:2, ax:0, ay:-30, font:{color:'#008000'}},
      {x:280, y:C_B_eb[findIdx(Ts,280)], xref:'x', yref:'y',
       text:'energy balance', showarrow:true,
       arrowhead:2, ax:0, ay:-30, font:{color:'#0000FF'}}
    ]
  };

  Plotly.react('plot-container', [traceMB, traceEB], layout, {responsive:true});
}

/* helper to find index of a value in the array (linear search is fine for 200 pts) */
function findIdx(arr,val){
  for(let i=0;i<arr.length;i++){
    if(Math.abs(arr[i]-val)<0.5) return i;
  }
  return -1;
}

/* ---------- event listeners ---------- */
[uaSlider, kr0Slider, tfSlider, tauSlider].forEach(sl=> {
  sl.addEventListener('input', updateAll);
});

/* initial draw */
updateAll();
</script>

</body>
</html>