<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>H₂⁺ Molecular Orbital Visualization</title>
    <script src="https://cdn.plot.ly/plotly-latest.min.js"></script>
    <style>
        body {
            font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
            margin: 0;
            background-color: #f8f9fa;
            color: #212529;
        }
        #app-container {
            display: flex;
            flex-direction: column;
            align-items: center;
            width: 100%;
            padding: 20px;
            box-sizing: border-box;
        }
        #control-panel {
            width: 100%;
            max-width: 800px;
            padding: 20px;
            margin-bottom: 20px;
            background-color: #ffffff;
            border-radius: 8px;
            box-shadow: 0 2px 4px rgba(0,0,0,0.1);
            text-align: center;
        }
        #control-panel label {
            font-weight: bold;
            margin-right: 10px;
        }
        #slider-r {
            width: 300px;
            vertical-align: middle;
        }
        #slider-r-value {
            display: inline-block;
            min-width: 45px;
            font-family: monospace;
            font-weight: bold;
            color: #007bff;
        }
        .plot-container {
            width: 100%;
            margin-bottom: 20px;
            background-color: #ffffff;
            border-radius: 8px;
            box-shadow: 0 2px 4px rgba(0,0,0,0.1);
        }
        #potential-energy-plot-container {
            height: 450px;
            max-width: 1000px;
        }
        #density-plots-container {
            display: flex;
            flex-wrap: wrap;
            justify-content: space-around;
            width: 100%;
            max-width: 1200px;
        }
        .density-plot {
            width: 100%;
            max-width: 550px;
            height: 500px;
            margin: 10px;
            background-color: #ffffff;
            border-radius: 8px;
            box-shadow: 0 2px 4px rgba(0,0,0,0.1);
        }
        h2 {
            text-align: center;
            margin-top: 0;
            font-weight: 300;
        }
    </style>
</head>
<body>

    <div id="app-container">
        <h1>H₂⁺ Molecule: LCAO Approximation</h1>

        <div id="control-panel">
            <label for="slider-r">R (Å):</label>
            <input type="range" id="slider-r" min="0.001" max="4.0" step="0.001" value="0.74">
            <output id="slider-r-value">0.740</output>
        </div>

        <div id="potential-energy-plot-container" class="plot-container"></div>

        <div id="density-plots-container">
            <div id="bonding-plot-container" class="density-plot">
                 <h2>Bonding Orbital (σ)</h2>
            </div>
            <div id="antibonding-plot-container" class="density-plot">
                <h2>Antibonding Orbital (σ*)</h2>
            </div>
        </div>
    </div>

    <script>
        window.onload = function() {
            // --- DOM Element References ---
            const sliderR = document.getElementById('slider-r');
            const sliderRValue = document.getElementById('slider-r-value');

            // --- Constants ---
            const ANGSTROM_TO_BOHR = 1 / 0.529177;

            // --- Plotting Setup ---
            const GRID_SIZE = 50;
            const GRID_MIN = -3.5;
            const GRID_MAX = 3.5;
            
            const common3DLayout = {
                title: '',
                showlegend: false,
                autosize: true,
                margin: { l: 0, r: 0, b: 0, t: 40 },
                scene: {
                    xaxis: { range: [GRID_MIN, GRID_MAX], visible: false },
                    yaxis: { range: [GRID_MIN, GRID_MAX], visible: false },
                    zaxis: { range: [0, 0.45], visible: false },
                    camera: {
                        eye: { x: 1.25, y: -1.25, z: 1.25 }
                    },
                    aspectmode: 'cube'
                }
            };

            const bondingLayout = JSON.parse(JSON.stringify(common3DLayout));
            bondingLayout.title = { text: 'Bonding Orbital (σ) Electron Density', y: 0.95 };

            const antibondingLayout = JSON.parse(JSON.stringify(common3DLayout));
            antibondingLayout.title = { text: 'Antibonding Orbital (σ*) Electron Density', y: 0.95 };


            // --- Core Calculation Functions ---

            function calculatePotentialEnergyCurves() {
                const r_au_values = [];
                const r_angstrom_values = [];
                const e_bonding = [];
                const e_antibonding = [];

                for (let r_au = 0.2; r_au <= 15; r_au += 0.05) {
                    r_au_values.push(r_au);
                    r_angstrom_values.push(r_au / ANGSTROM_TO_BOHR);

                    const S = (1 + r_au + Math.pow(r_au, 2) / 3) * Math.exp(-r_au);
                    const J = (1 / r_au) - (1 + 1 / r_au) * Math.exp(-2 * r_au);
                    const K = (1 + r_au) * Math.exp(-r_au);

                    e_bonding.push((J + K) / (1 + S));
                    e_antibonding.push((J - K) / (1 - S));
                }
                return { r_angstrom_values, e_bonding, e_antibonding };
            }

            function generate3DGridData(R_angstroms) {
                const R_au = R_angstroms * ANGSTROM_TO_BOHR;
                const x_grid = [];
                const y_grid = [];
                const z_bonding_data = [];
                const z_antibonding_data = [];
                const step = (GRID_MAX - GRID_MIN) / (GRID_SIZE - 1);

                // Pre-calculate constants for the given R
                const S_ab = (1 + R_au + Math.pow(R_au, 2) / 3) * Math.exp(-R_au);
                const N_b = 1 / Math.sqrt(2 * (1 + S_ab));
                const N_a = 1 / Math.sqrt(2 * (1 - S_ab));
                
                const nucleusA_x = -R_au / 2;
                const nucleusB_x = R_au / 2;

                for (let i = 0; i < GRID_SIZE; i++) {
                    const y = GRID_MIN + i * step;
                    y_grid.push(y);
                    const z_bonding_row = [];
                    const z_antibonding_row = [];

                    for (let j = 0; j < GRID_SIZE; j++) {
                        const x = GRID_MIN + j * step;
                        if (i === 0) x_grid.push(x);

                        // Calculate distances to nuclei in the x-y plane (z=0)
                        const dist_A = Math.sqrt(Math.pow(x - nucleusA_x, 2) + Math.pow(y, 2));
                        const dist_B = Math.sqrt(Math.pow(x - nucleusB_x, 2) + Math.pow(y, 2));

                        const phi_A = Math.exp(-dist_A);
                        const phi_B = Math.exp(-dist_B);

                        // Calculate molecular orbital wavefunctions
                        const psi_b = N_b * (phi_A + phi_B);
                        const psi_a = N_a * (phi_A - phi_B);

                        // Calculate probability densities
                        z_bonding_row.push(psi_b * psi_b);
                        z_antibonding_row.push(psi_a * psi_a);
                    }
                    z_bonding_data.push(z_bonding_row);
                    z_antibonding_data.push(z_antibonding_row);
                }

                return { x_grid, y_grid, z_bonding_data, z_antibonding_data };
            }

            // --- Plotting Functions ---

            function drawPotentialEnergyPlot() {
                const { r_angstrom_values, e_bonding, e_antibonding } = calculatePotentialEnergyCurves();

                const bondingTrace = {
                    x: r_angstrom_values,
                    y: e_bonding,
                    type: 'scatter',
                    mode: 'lines',
                    name: 'Bonding PE (σ)',
                    line: { color: 'blue' }
                };

                const antibondingTrace = {
                    x: r_angstrom_values,
                    y: e_antibonding,
                    type: 'scatter',
                    mode: 'lines',
                    name: 'Antibonding PE (σ*)',
                    line: { color: 'red' }
                };

                const layout = {
                    title: 'Potential Energy of H₂⁺',
                    xaxis: {
                        title: 'Internuclear Distance R (Å)',
                        range: [0, 8]
                    },
                    yaxis: {
                        title: 'Energy (Hartrees)',
                        range: [-0.2, 0.4],
                        zerolinecolor: '#cccccc',
                        zerolinewidth: 1
                    },
                    legend: {
                        x: 0.6,
                        y: 0.95
                    },
                    margin: { t: 50, b: 50, l: 60, r: 30 }
                };

                Plotly.newPlot('potential-energy-plot-container', [bondingTrace, antibondingTrace], layout);
            }

            function initialize3DPlots() {
                const initialR = parseFloat(sliderR.value);
                sliderRValue.textContent = initialR.toFixed(3);
                const { x_grid, y_grid, z_bonding_data, z_antibonding_data } = generate3DGridData(initialR);
                
                const bondingTrace = {
                    x: x_grid,
                    y: y_grid,
                    z: z_bonding_data,
                    type: 'surface',
                    colorscale: 'YlOrBr',
                    showscale: false,
                    cmin: 0,
                    cmax: 0.45
                };

                const antibondingTrace = {
                    x: x_grid,
                    y: y_grid,
                    z: z_antibonding_data,
                    type: 'surface',
                    colorscale: 'YlOrBr',
                    showscale: false,
                    cmin: 0,
                    cmax: 0.45
                };
                
                // Remove the h2 titles before plotting
                document.querySelector('#bonding-plot-container h2').style.display = 'none';
                document.querySelector('#antibonding-plot-container h2').style.display = 'none';


                Plotly.newPlot('bonding-plot-container', [bondingTrace], bondingLayout);
                Plotly.newPlot('antibonding-plot-container', [antibondingTrace], antibondingLayout);
            }
            
            function update3DPlots() {
                const currentR = parseFloat(sliderR.value);
                sliderRValue.textContent = currentR.toFixed(3);
                
                const { z_bonding_data, z_antibonding_data } = generate3DGridData(currentR);
                
                // Use restyle for efficient updates
                Plotly.restyle('bonding-plot-container', 'z', [z_bonding_data]);
                Plotly.restyle('antibonding-plot-container', 'z', [z_antibonding_data]);
            }

            // --- Event Listeners ---
            sliderR.addEventListener('input', update3DPlots);

            // --- Initialization ---
            drawPotentialEnergyPlot();
            initialize3DPlots();
        };
    </script>

</body>
</html>