<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Best Effort Global Warming Trajectories</title>

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

<style>
    body {
        font-family: Arial, Helvetica, sans-serif;
        margin: 20px;
        background-color: #fafafa;
    }
    h1 {
        text-align: center;
        margin-bottom: 0.5em;
    }
    p {
        text-align: center;
        max-width: 800px;
        margin: 0 auto 1.5em;
        line-height: 1.4;
    }
    #controls-container {
        display: flex;
        flex-wrap: wrap;
        justify-content: center;
        gap: 1.5rem;
        margin-bottom: 1.5rem;
    }
    .control-group {
        display: flex;
        flex-direction: column;
        align-items: stretch;
        width: 260px;
    }
    .control-group label {
        margin-bottom: 0.3rem;
        font-weight: bold;
    }
    .control-group input[type=range] {
        width: 100%;
    }
    .control-group span.value {
        margin-top: 0.2rem;
        text-align: right;
        font-family: monospace;
        font-size: 0.9rem;
        color: #555;
    }
    #plot-div {
        max-width: 900px;
        margin: 0 auto;
    }
</style>
</head>
<body>

<h1>Best Effort Global Warming Trajectories</h1>
<p>
    This demo visualises how different climate‑policy parameters affect the evolution of atmospheric carbon
    dioxide and annual emissions. Move the sliders to see the curves update in real time.
</p>

<div id="controls-container">
    <div class="control-group">
        <label for="slider-residence-time">Atmospheric CO₂ residence time (years)</label>
        <input type="range" id="slider-residence-time"
               min="50" max="500" step="1" value="200">
        <span class="value" id="value-residence-time">200</span>
    </div>

    <div class="control-group">
        <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 class="value" id="value-transition-pace">25</span>
    </div>

    <div class="control-group">
        <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 class="value" id="value-wedges">4.00</span>
    </div>
</div>

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

<script>
/* ---------- Constants of the model ---------- */
const C0      = 850;   // initial atmospheric carbon (GtC)
const E0      = 8.5;   // initial emission rate (GtC/yr)
const C_eq    = 586;   // pre‑industrial equilibrium carbon (GtC)
const E_floor = 1.5;   // emission floor (GtC/yr)
const rise_rate = 0.02; // GtC/yr², rate of emission increase during transition
const dt      = 0.5;   // integration step (years)
const maxTime = 300;   // years

/* ---------- Grab DOM elements ---------- */
const sliderResidence = document.getElementById('slider-residence-time');
const sliderTransition = document.getElementById('slider-transition-pace');
const sliderWedges     = document.getElementById('slider-wedges');

const outResidence = document.getElementById('value-residence-time');
const outTransition = document.getElementById('value-transition-pace');
const outWedges     = document.getElementById('value-wedges');

/* ---------- Helper: format wedges to 2 decimals ---------- */
function fmtWedges(v){ return Number(v).toFixed(2); }

/* ---------- Update displayed numbers ---------- */
function updateOutputs(){
    outResidence.textContent = sliderResidence.value;
    outTransition.textContent = sliderTransition.value;
    outWedges.textContent = fmtWedges(sliderWedges.value);
}

/* ---------- Core simulation ---------- */
function simulate(){
    const t_res   = Number(sliderResidence.value);   // years
    const t_trans = Number(sliderTransition.value); // years
    const N_wedges= Number(sliderWedges.value);     // wedges

    const times = [];
    const emissions = []; // E(t) in GtC/yr
    const carbon   = []; // C(t) in GtC

    let Cprev = C0;

    // initialise emission at t=0
    let Ecur = E0 + rise_rate * 0; // already at E0

    for(let t=0; t<=maxTime+dt; t+=dt){
        times.push(t);

        // ----- emission rate -----
        if(t < t_trans){
            Ecur = E0 + rise_rate * t;
        }else{
            const reductionRate = N_wedges / 50; // wedges per year, scaled
            const Elinear = E0 + rise_rate * t_trans - reductionRate * (t - t_trans);
            Ecur = Math.max(E_floor, Elinear);
        }

        // store for plotting (purple curve)
        emissions.push(Ecur);

        // ----- atmospheric carbon -----
        const dC = (Ecur - (Cprev - C_eq) / t_res) * dt;
        Cprev = Cprev + dC;
        carbon.push(Cprev);
    }

    return {times, emissions, carbon};
}

/* ---------- Plotting with Plotly ---------- */
let plotInitialized = false;

function drawPlot(){
    const {times, emissions, carbon} = simulate();

    const traceEmission = {
        x: times,
        y: emissions.map(v=>v*100), // scale for visual convenience
        mode: 'lines',
        name: 'Annual Emissions',
        line: {color: 'purple', width: 2}
    };

    const traceCarbon = {
        x: times,
        y: carbon,
        mode: 'lines',
        name: 'Atmospheric Carbon',
        line: {color: 'blue', width: 2}
    };

    const traceCeiling = {
        x: [0, maxTime],
        y: [1200, 1200], // constant ~2× pre‑industrial
        mode: 'lines',
        name: 'Target Ceiling',
        line: {color: 'black', width: 2, dash: 'dash'}
    };

    const layout = {
        title: 'Atmospheric carbon (GtC; blue) and annual emissions (GtC/yr × 100; purple) versus time',
        xaxis: {title: 'years', range: [0, maxTime]},
        yaxis: {title: 'GtC or GtC/yr × 100', range: [0, 1600]},
        showlegend: false,
        margin: {l:60, r:20, b:40, t:80},
        paper_bgcolor: '#f9f9f9',
        plot_bgcolor: 'white',
        grid: {color:'#ddd'}
    };

    if(!plotInitialized){
        Plotly.newPlot('plot-div', [traceCarbon, traceEmission, traceCeiling], layout);
        plotInitialized = true;
    }else{
        Plotly.react('plot-div', [traceCarbon, traceEmission, traceCeiling], layout);
    }
}

/* ---------- Event listeners ---------- */
[sliderResidence, sliderTransition, sliderWedges].forEach(sl => {
    sl.addEventListener('input', () => {
        updateOutputs();
        drawPlot();
    });
});

/* ---------- Initialisation ---------- */
updateOutputs();   // set initial numbers
drawPlot();        // draw the first chart
</script>

</body>
</html>