<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Best Effort Global Warming Trajectories</title>
    <script src="https://cdn.plot.ly/plotly-latest.min.js"></script>
    <style>
        body {
            font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
            background-color: #e9e9e9;
            display: flex;
            justify-content: center;
            align-items: flex-start;
            padding-top: 20px;
            margin: 0;
            min-height: 100vh;
        }
        .main-container {
            background-color: #f9f9f9;
            border: 1px solid #ccc;
            box-shadow: 0 4px 8px rgba(0,0,0,0.1);
            padding: 20px;
            width: 90%;
            max-width: 900px;
            box-sizing: border-box;
        }
        h1 {
            font-size: 1.5em;
            text-align: center;
            margin-top: 0;
            margin-bottom: 10px;
            font-weight: 500;
        }
        p {
            text-align: center;
            color: #555;
            font-size: 0.9em;
            margin-top: 0;
            margin-bottom: 20px;
        }
        #controls-container {
            display: flex;
            flex-direction: column;
            gap: 8px;
            padding: 15px;
            margin-bottom: 20px;
            border: 1px solid #ddd;
            background-color: #f0f0f0;
            border-radius: 4px;
        }
        .control-row {
            display: flex;
            align-items: center;
            gap: 15px;
        }
        .control-row label {
            width: 420px;
            text-align: right;
            font-size: 0.9em;
            color: #333;
        }
        .control-row input[type="range"] {
            flex-grow: 1;
            margin: 0;
        }
        .control-row span {
            width: 50px;
            text-align: left;
            font-family: 'Courier New', Courier, monospace;
            font-size: 0.9em;
            font-weight: bold;
        }
        #plot-div {
            width: 100%;
            height: 500px;
        }
    </style>
</head>
<body>
    <div class="main-container">
        <h1>Best Effort Global Warming Trajectories</h1>
        <p>Adjust the sliders to see how different parameters affect the projected atmospheric carbon levels and emission rates over time.</p>

        <div id="controls-container">
            <div class="control-row">
                <label for="slider-residence-time">Atmospheric CO2 residence time (years)</label>
                <input type="range" id="slider-residence-time" min="50" max="500" step="1" value="200">
                <span id="value-residence-time">200</span>
            </div>
            <div class="control-row">
                <label for="slider-transition-pace">Transition to constant reduction pace (years)</label>
                <input type="range" id="slider-transition-pace" min="0" max="100" step="1" value="25">
                <span id="value-transition-pace">25</span>
            </div>
            <div class="control-row">
                <label for="slider-wedges">Number of constant pace wedges (above emission stabilization at t=0)</label>
                <input type="range" id="slider-wedges" min="0" max="10" step="0.01" value="4">
                <span id="value-wedges">4.00</span>
            </div>
        </div>

        <div id="plot-div"></div>
    </div>

    <script>
        document.addEventListener('DOMContentLoaded', function () {
            // --- DOM Element References ---
            const residenceTimeSlider = document.getElementById('slider-residence-time');
            const transitionPaceSlider = document.getElementById('slider-transition-pace');
            const wedgesSlider = document.getElementById('slider-wedges');

            const residenceTimeValue = document.getElementById('value-residence-time');
            const transitionPaceValue = document.getElementById('value-transition-pace');
            const wedgesValue = document.getElementById('value-wedges');

            // --- Model Constants ---
            const C0 = 850;      // Initial atmospheric carbon (GtC)
            const E0 = 8.5;      // Initial annual emission rate (GtC/yr)
            const C_eq = 586;    // Pre-industrial equilibrium atmospheric carbon (GtC)
            const E_floor = 1.5;   // Floor for emission rate (GtC/yr)
            const rise_rate = 0.02; // Rate of emission increase (GtC/yr^2)
            const time_step = 0.5; // Years for numerical integration
            const max_time = 300;   // Years

            // --- Plotly Layout ---
            const plotLayout = {
                title: 'Atmospheric carbon (GtC; blue) and annual emissions (GtC/yr × 100; purple) versus time',
                xaxis: { title: 'years', range: [0, 300], zeroline: true },
                yaxis: { title: 'GtC or GtC/yr × 100', range: [0, 1600], zeroline: true },
                showlegend: false,
                margin: { l: 70, r: 20, b: 50, t: 80 },
                paper_bgcolor: '#f9f9f9',
                plot_bgcolor: 'white',
                grid: { color: '#ddd' }
            };

            // --- Core Calculation and Plotting Function ---
            function updatePlot() {
                // 1. Get current slider values
                const t_res = parseFloat(residenceTimeSlider.value);
                const t_trans = parseFloat(transitionPaceSlider.value);
                const N_wedges = parseFloat(wedgesSlider.value);

                // 2. Initialize data arrays
                const time_values = [];
                const emission_values_plot = [];
                const carbon_values = [];

                // 3. Perform numerical simulation
                const E_peak = E0 + rise_rate * t_trans;
                const reduction_rate = N_wedges / 50;
                
                let current_C = C0;

                for (let t = 0; t <= max_time; t += time_step) {
                    // a. Calculate Emission Rate E(t)
                    let current_E;
                    if (t < t_trans) {
                        current_E = E0 + rise_rate * t;
                    } else {
                        const E_linear = E_peak - reduction_rate * (t - t_trans);
                        current_E = Math.max(E_floor, E_linear);
                    }

                    // b. Calculate Atmospheric Carbon C(t) using Euler's method
                    if (t > 0) {
                        const C_prev = carbon_values[carbon_values.length - 1];
                        const dC = (current_E - (C_prev - C_eq) / t_res) * time_step;
                        current_C = C_prev + dC;
                    }

                    // 4. Store values for plotting
                    time_values.push(t);
                    emission_values_plot.push(100 * current_E);
                    carbon_values.push(current_C);
                }

                // 5. Define Plotly Traces
                const carbon_trace = {
                    x: time_values,
                    y: carbon_values,
                    type: 'scatter',
                    mode: 'lines',
                    name: 'Atmospheric Carbon',
                    line: { color: 'blue', width: 2 }
                };

                const emission_trace = {
                    x: time_values,
                    y: emission_values_plot,
                    type: 'scatter',
                    mode: 'lines',
                    name: 'Annual Emissions',
                    line: { color: 'purple', width: 2 }
                };

                const ceiling_trace = {
                    x: [0, max_time],
                    y: [1200, 1200],
                    type: 'scatter',
                    mode: 'lines',
                    name: 'Target Ceiling',
                    line: { color: 'black', width: 2, dash: 'dash' }
                };

                // 6. Update the plot
                Plotly.react('plot-div', [carbon_trace, emission_trace, ceiling_trace], plotLayout);
            }

            // --- Event Listeners ---
            function setupEventListeners() {
                residenceTimeSlider.addEventListener('input', () => {
                    residenceTimeValue.textContent = residenceTimeSlider.value;
                    updatePlot();
                });

                transitionPaceSlider.addEventListener('input', () => {
                    transitionPaceValue.textContent = transitionPaceSlider.value;
                    updatePlot();
                });

                wedgesSlider.addEventListener('input', () => {
                    wedgesValue.textContent = parseFloat(wedgesSlider.value).toFixed(2);
                    updatePlot();
                });
            }

            // --- Initial Setup ---
            setupEventListeners();
            updatePlot(); // Initial plot on page load
        });
    </script>
</body>
</html>