<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>CSTR Steady State Analysis</title>
    <script src="https://cdn.plot.ly/plotly-latest.min.js"></script>
    <style>
        body {
            font-family: sans-serif;
            background-color: #e9e9e9;
            display: flex;
            justify-content: center;
            align-items: flex-start;
            padding: 20px;
            margin: 0;
        }

        #demo-container {
            width: 750px;
            max-width: 100%;
            padding: 20px;
            background-color: white;
            border: 1px solid #aaa;
            box-shadow: 2px 2px 5px rgba(0, 0, 0, 0.1);
        }

        #control-panel {
            display: grid;
            grid-template-columns: 1fr 1fr;
            gap: 15px 30px;
            padding: 20px;
            background-color: #f0f0f0;
            border: 1px solid #ccc;
            border-radius: 5px;
            margin-bottom: 20px;
        }

        #control-panel>div {
            display: grid;
            grid-template-areas:
                "label label"
                "slider value";
            grid-template-columns: 1fr auto;
            align-items: center;
            gap: 5px;
        }

        #control-panel label {
            grid-area: label;
            font-size: 13px;
            color: #333;
        }

        #control-panel input[type="range"] {
            grid-area: slider;
            width: 100%;
            margin: 0;
        }

        #control-panel span {
            grid-area: value;
            font-size: 13px;
            min-width: 75px;
            text-align: right;
            padding-left: 5px;
            white-space: nowrap;
        }

        #plot-container {
            border: 1px solid #ccc;
        }
    </style>
</head>
<body>

    <div id="demo-container">

        <!-- Control Panel Section -->
        <div id="control-panel">

            <!-- Heat Transfer Coefficient Slider -->
            <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"></span>
            </div>

            <!-- Reverse Reaction Pre-exponential Factor Slider -->
            <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"></span>
            </div>

            <!-- Feed Temperature Slider -->
            <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"></span>
            </div>

            <!-- Residence Time Slider -->
            <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"></span>
            </div>

        </div>

        <!-- Visualization Section -->
        <div id="plot-container"></div>
    </div>

    <script>
        document.addEventListener('DOMContentLoaded', function () {
            // --- DOM Elements ---
            const sliderUA = document.getElementById('slider-ua');
            const sliderKr0 = document.getElementById('slider-kr0');
            const sliderTf = document.getElementById('slider-tf');
            const sliderTau = document.getElementById('slider-tau');

            const uaValueSpan = document.getElementById('ua-value');
            const kr0ValueSpan = document.getElementById('kr0-value');
            const tfValueSpan = document.getElementById('tf-value');
            const tauValueSpan = document.getElementById('tau-value');

            const plotContainer = document.getElementById('plot-container');

            // --- Mathematical Constants ---
            const C_A0 = 10;
            const R = 1.987;
            const E_f = 25000;
            const deltaH_rxn = -200000;
            const E_r = E_f - deltaH_rxn;
            const k_f0 = 1e15;
            const rhoCp = 20000;

            // --- Helper Function for Formatting kr0 Value ---
            function formatKr0(valueStr) {
                const val = parseFloat(valueStr);
                if (val === 0) {
                    return "0.";
                }
                let formatted = val.toFixed(1);
                // For integer values (e.g., 9.0), format as "9." to match snapshot
                if (val % 1 === 0) {
                    formatted = val.toFixed(0) + ".";
                }
                return `${formatted} x 10\u00B9\u00B2`; // Using unicode superscripts ¹²
            }

            // --- Main Plotting Function ---
            function updatePlot() {
                // 1. Read Slider Values
                const UA = parseFloat(sliderUA.value);
                const kr0_slider = parseFloat(sliderKr0.value);
                const Tf = parseFloat(sliderTf.value);
                const tau = parseFloat(sliderTau.value);

                const kr0_actual = kr0_slider * 1e12;

                // 2. Update Value Displays
                uaValueSpan.textContent = UA.toFixed(1);
                kr0ValueSpan.textContent = formatKr0(kr0_slider);
                tfValueSpan.textContent = Tf;
                tauValueSpan.textContent = tau;

                // 3. Recalculate Curves
                const T_values = [];
                const Cb_mass = [];
                const Cb_energy = [];

                for (let T = 250; T <= 400; T += 1) {
                    T_values.push(T);

                    // Mass Balance Curve
                    const k_f = k_f0 * Math.exp(-E_f / (R * T));
                    const k_r = kr0_actual * Math.exp(-E_r / (R * T));
                    const cb_m = (tau * k_f * C_A0) / (1 + tau * (k_f + k_r));
                    Cb_mass.push(cb_m);

                    // Energy Balance Line
                    const cb_e = ((T - Tf) * (rhoCp + UA * tau)) / (-deltaH_rxn);
                    Cb_energy.push(cb_e);
                }

                // 4. Update Plot
                const massBalanceTrace = {
                    x: T_values,
                    y: Cb_mass,
                    mode: 'lines',
                    line: {
                        color: '#008000',
                        width: 2.5
                    }
                };

                const energyBalanceTrace = {
                    x: T_values,
                    y: Cb_energy,
                    mode: 'lines',
                    line: {
                        color: '#0000FF',
                        width: 2.5
                    }
                };
                
                // Position annotations relative to the calculated curves
                const massAnnotationY = Cb_mass[T_values.indexOf(360)];
                const energyAnnotationY = Cb_energy[T_values.indexOf(290)];

                const layout = {
                    xaxis: {
                        title: 'temperature (K)',
                        range: [250, 400],
                        gridcolor: '#e0e0e0',
                        zeroline: false
                    },
                    yaxis: {
                        title: 'product concentration (mmol/dm³)',
                        range: [0, 12],
                        gridcolor: '#e0e0e0',
                        zeroline: false
                    },
                    annotations: [{
                        x: 360,
                        y: massAnnotationY,
                        text: '<b>mass balance</b>',
                        font: { color: '#008000', size: 16 },
                        showarrow: false,
                        xanchor: 'left',
                        yanchor: 'bottom',
                        xshift: 5,
                        yshift: 5
                    }, {
                        x: 290,
                        y: energyAnnotationY,
                        text: '<b>energy balance</b>',
                        font: { color: '#0000FF', size: 16 },
                        showarrow: false,
                        xanchor: 'left',
                        yanchor: 'bottom',
                        xshift: 5,
                        yshift: 5
                    }],
                    showlegend: false,
                    font: {
                        family: 'sans-serif',
                        size: 14
                    },
                    margin: { l: 80, r: 30, b: 60, t: 20 },
                    paper_bgcolor: 'white',
                    plot_bgcolor: 'white'
                };

                Plotly.react(plotContainer, [massBalanceTrace, energyBalanceTrace], layout, {responsive: true});
            }

            // --- Event Listeners ---
            [sliderUA, sliderKr0, sliderTf, sliderTau].forEach(slider => {
                slider.addEventListener('input', updatePlot);
            });

            // --- Initial Plot ---
            updatePlot();
        });
    </script>
</body>
</html>