<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Rayleigh Wave Visualization</title>
    <script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.9.0/p5.min.js"></script>
    <style>
        body {
            font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif;
            display: flex;
            justify-content: center;
            align-items: center;
            flex-direction: column;
            background-color: #f0f0f0;
            margin: 0;
            padding: 20px;
        }
        #main-container {
            max-width: 750px;
            width: 100%;
            padding: 20px;
            background-color: #fff;
            border-radius: 8px;
            box-shadow: 0 4px 8px rgba(0,0,0,0.1);
        }
        #controls-container {
            display: flex;
            flex-direction: column;
            gap: 15px;
            margin-bottom: 20px;
        }
        .control-group {
            display: flex;
            align-items: center;
            gap: 15px;
        }
        .control-group label {
            width: 160px;
            flex-shrink: 0;
            text-align: right;
            color: #333;
        }
        .control-group input[type="range"] {
            flex-grow: 1;
            margin: 0;
        }
        .control-group span {
            width: 50px;
            flex-shrink: 0;
            text-align: left;
            font-variant-numeric: tabular-nums;
            color: #555;
        }
        #canvas-container {
            border: 1px solid #ccc;
            border-radius: 4px;
            overflow: hidden;
            display: flex;
            justify-content: center;
            align-items: center;
        }
    </style>
</head>
<body>
    <div id="main-container">
        <div id="controls-container">
            <div class="control-group">
                <label for="slider-time">time</label>
                <input type="range" id="slider-time" min="0" max="25" value="7.0" step="0.01">
                <span id="value-time">7.00</span>
            </div>
            <div class="control-group">
                <label for="slider-wavelength">wavelength</label>
                <input type="range" id="slider-wavelength" min="1" max="8" value="4.0" step="0.01">
                <span id="value-wavelength">4.00</span>
            </div>
            <div class="control-group">
                <label for="slider-longitudinal-amplitude">longitudinal amplitude</label>
                <input type="range" id="slider-longitudinal-amplitude" min="0" max="3" value="2.0" step="0.01">
                <span id="value-longitudinal-amplitude">2.00</span>
            </div>
            <div class="control-group">
                <label for="slider-latitudinal-amplitude">latitudinal amplitude</label>
                <input type="range" id="slider-latitudinal-amplitude" min="0" max="3" value="1.0" step="0.01">
                <span id="value-latitudinal-amplitude">1.00</span>
            </div>
        </div>
        <div id="canvas-container"></div>
    </div>

    <script>
        // Global p5.js sketch variables
        let sliderTime, valueTime;
        let sliderWavelength, valueWavelength;
        let sliderLongAmp, valueLongAmp;
        let sliderLatAmp, valueLatAmp;

        let particles = [];
        const gridCols = 10;
        const gridRows = 6;
        
        let time;
        let isDraggingTime = false;

        function setup() {
            let canvas = createCanvas(700, 450);
            canvas.parent('canvas-container');

            // Get references to HTML elements
            sliderTime = document.getElementById('slider-time');
            valueTime = document.getElementById('value-time');
            sliderWavelength = document.getElementById('slider-wavelength');
            valueWavelength = document.getElementById('value-wavelength');
            sliderLongAmp = document.getElementById('slider-longitudinal-amplitude');
            valueLongAmp = document.getElementById('value-longitudinal-amplitude');
            sliderLatAmp = document.getElementById('slider-latitudinal-amplitude');
            valueLatAmp = document.getElementById('value-latitudinal-amplitude');

            time = parseFloat(sliderTime.value);

            // Add event listeners to update values
            sliderTime.addEventListener('input', () => valueTime.textContent = parseFloat(sliderTime.value).toFixed(2));
            sliderWavelength.addEventListener('input', () => valueWavelength.textContent = parseFloat(sliderWavelength.value).toFixed(2));
            sliderLongAmp.addEventListener('input', () => valueLongAmp.textContent = parseFloat(sliderLongAmp.value).toFixed(2));
            sliderLatAmp.addEventListener('input', () => valueLatAmp.textContent = parseFloat(sliderLatAmp.value).toFixed(2));

            // Add event listeners for time slider interaction
            sliderTime.addEventListener('pointerdown', () => isDraggingTime = true);
            sliderTime.addEventListener('pointerup', () => {
                isDraggingTime = false;
                time = parseFloat(sliderTime.value); // Sync animation time on release
            });

            // Initialize particle equilibrium positions
            const paddingX = 60;
            const paddingY = 60;
            const drawableWidth = width - 2 * paddingX;
            const drawableHeight = height - 2 * paddingY;
            const spacingX = drawableWidth / (gridCols - 1);
            const spacingY = drawableHeight / (gridRows - 1);

            for (let r = 0; r < gridRows; r++) {
                for (let c = 0; c < gridCols; c++) {
                    particles.push({
                        x0: paddingX + c * spacingX,
                        y0: paddingY + r * spacingY,
                        row: r
                    });
                }
            }
        }

        function draw() {
            // 1. Update State
            if (!isDraggingTime) {
                time += 0.03;
                if (time > parseFloat(sliderTime.max)) {
                    time = parseFloat(sliderTime.min);
                }
                sliderTime.value = time;
                valueTime.textContent = time.toFixed(2);
            } else {
                time = parseFloat(sliderTime.value);
            }

            // Get current parameter values
            const wavelength = parseFloat(sliderWavelength.value);
            const longAmp = parseFloat(sliderLongAmp.value);
            const latAmp = parseFloat(sliderLatAmp.value);

            // 2. Clear Canvas
            background('#E0F0FF');

            // 3. Particle Grid Logic
            const k = TWO_PI / (wavelength * 100); // Scale wavelength for pixel space
            const currentPositions = [];

            // First loop: Calculate positions and draw elliptical paths
            for (let i = 0; i < particles.length; i++) {
                const p = particles[i];

                // Amplitude decay with depth
                const ampDecay = exp(-(p.row / (gridRows - 1)) * 1.5);
                
                // Amplitudes for motion (scaled)
                const ampX = longAmp * ampDecay * 20;
                const ampY = latAmp * ampDecay * 20;

                // Draw elliptical path
                noFill();
                stroke('#8F998F');
                strokeWeight(2);
                ellipse(p.x0, p.y0, 2 * ampX, 2 * ampY);
                
                // Calculate particle position (retrograde motion)
                const phase = k * p.x0 - time;
                const dx = ampX * cos(phase);
                const dy = ampY * sin(phase);
                
                currentPositions.push({ x: p.x0 + dx, y: p.y0 + dy });
            }

            // 4. Draw Connecting Lines and Particles
            
            // Draw connecting lines for each row
            for (let r = 0; r < gridRows; r++) {
                stroke('#0000FF');
                strokeWeight(2);
                noFill();
                drawingContext.setLineDash([5, 5]); // Set dashed line style

                beginShape();
                for (let c = 0; c < gridCols; c++) {
                    const index = r * gridCols + c;
                    vertex(currentPositions[index].x, currentPositions[index].y);
                }
                endShape();
                
                drawingContext.setLineDash([]); // Reset to solid lines
            }

            // Draw particles on top
            noStroke();
            fill('#FF0000');
            for (let i = 0; i < currentPositions.length; i++) {
                circle(currentPositions[i].x, currentPositions[i].y, 8);
            }
        }
    </script>
</body>
</html>