<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Rayleigh Wave Demo</title>
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.9.0/p5.min.js"></script>
<style>
  body {font-family: Arial, sans-serif; margin:0; padding:0; background:#f0f0f0;}
  #main {display:flex; flex-direction:column; align-items:center; padding:20px;}
  #controls-container {display:flex; flex-wrap:wrap; gap:20px; justify-content:center; margin-bottom:15px;}
  .control-group {display:flex; flex-direction:column; align-items:center;}
  .control-group label {margin-bottom:4px; font-weight:bold;}
  .control-group input[type=range] {width:200px;}
  .control-group span {margin-top:4px; font-size:0.9em; min-width:45px; text-align:center;}
  #canvas-container {border:1px solid #999; background:#fff;}
</style>
</head>
<body>
<div id="main">
  <div id="controls-container">
    <div class="control-group">
      <label for="slider-time">time</label>
      <input type="range" id="slider-time" min="0" max="25" step="0.01" value="7">
      <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" step="0.01" value="4">
      <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" step="0.01" value="2">
      <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" step="0.01" value="1">
      <span id="value-latitudinal-amplitude">1.00</span>
    </div>
  </div>

  <div id="canvas-container"></div>
</div>

<script>
/* ---------- GLOBAL STATE ---------- */
let time = 7;               // current animated time (overrides slider-time while dragging)
let wavelength = 4;
let longitudinalAmp = 2;
let latitudinalAmp = 1;

/* ---------- UI ELEMENTS ---------- */
const sliderTime = document.getElementById('slider-time');
const valueTime = document.getElementById('value-time');

const sliderWavelength = document.getElementById('slider-wavelength');
const valueWavelength = document.getElementById('value-wavelength');

const sliderLongitudinal = document.getElementById('slider-longitudinal-amplitude');
const valueLongitudinal = document.getElementById('value-longitudinal-amplitude');

const sliderLatitudinal = document.getElementById('slider-latitudinal-amplitude');
const valueLatitudinal = document.getElementById('value-latitudinal-amplitude');

/* ---------- UPDATE DISPLAY ---------- */
function updateDisplays() {
  valueTime.textContent = time.toFixed(2);
  valueWavelength.textContent = wavelength.toFixed(2);
  valueLongitudinal.textContent = longitudinalAmp.toFixed(2);
  valueLatitudinal.textContent = latitudinalAmp.toFixed(2);
}

/* ---------- SLIDER INTERACTION ---------- */
// pause animation while dragging the time slider
let paused = false;
sliderTime.addEventListener('input', () => {
  paused = true;
  time = parseFloat(sliderTime.value);
  updateDisplays();
});
sliderTime.addEventListener('change', () => {
  // when release, resume animation from the new time value
  paused = false;
});

sliderWavelength.addEventListener('input', e => {
  wavelength = parseFloat(e.target.value);
  updateDisplays();
});
sliderLongitudinal.addEventListener('input', e => {
  longitudinalAmp = parseFloat(e.target.value);
  updateDisplays();
});
sliderLatitudinal.addEventListener('input', e => {
  latitudinalAmp = parseFloat(e.target.value);
  updateDisplays();
});

/* ---------- P5 SKETCH ---------- */
new p5(p => {
  const cols = 10;
  const rows = 6;
  const cellW = 60;
  const cellH = 40;
  const offsetX = 80;
  const offsetY = 80;
  const particleRadius = 8;

  let eqPoints = []; // [{x0,y0}] for each particle

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

    // calculate equilibrium points
    for (let r = 0; r < rows; r++) {
      for (let c = 0; c < cols; c++) {
        const x0 = offsetX + c * cellW;
        const y0 = offsetY + r * cellH;
        eqPoints.push({x0, y0});
      }
    }

    updateDisplays();
  };

  p.draw = function () {
    // ---- 1. Update time (unless paused) ----
    if (!paused) {
      time += 0.03; // speed of animation
      sliderTime.value = time.toFixed(2);
      valueTime.textContent = time.toFixed(2);
    }

    // ---- 2. Clear background ----
    p.background('#E0F0FF');

    // ---- 3. Draw ellipses (paths) ----
    const k = p.TWO_PI / wavelength; // wave number
    eqPoints.forEach((pt, i) => {
      const rowIdx = Math.floor(i / cols);
      const ampDecay = Math.exp(- (rowIdx / (rows - 1)) * 1.5);

      const ampX = longitudinalAmp * ampDecay * 20; // scale factor 20
      const ampY = latitudinalAmp * ampDecay * 20;

      const phase = k * pt.x0 - time;

      // ellipse path
      p.stroke('#8F998F');
      p.strokeWeight(2);
      p.noFill();
      p.ellipse(pt.x0, pt.y0, ampX * 2, ampY * 2);

      // particle position
      const dx = ampX * Math.cos(phase);
      const dy = ampY * Math.sin(phase);
      const curX = pt.x0 + dx;
      const curY = pt.y0 + dy;

      // draw particle
      p.fill('#FF0000');
      p.noStroke();
      p.ellipse(curX, curY, particleRadius * 2);
    });

    // ---- 4. Draw connecting dashed lines per row ----
    p.drawingContext.setLineDash([5, 5]);
    p.stroke('#0000FF');
    p.strokeWeight(2);
    for (let r = 0; r < rows; r++) {
      const ptsRow = eqPoints.filter((_, idx) => Math.floor(idx / cols) === r);
      // connect current positions
      p.beginShape();
      ptsRow.forEach(pt => {
        const phase = k * pt.x0 - time;
        const ampX = longitudinalAmp * Math.exp(- (r / (rows - 1)) * 1.5) * 20;
        const ampY = latitudinalAmp * Math.exp(- (r / (rows - 1)) * 1.5) * 20;
        const dx = ampX * Math.cos(phase);
        const dy = ampY * Math.sin(phase);
        p.vertex(pt.x0 + dx, pt.y0 + dy);
      });
      p.endShape();
    }
    p.drawingContext.setLineDash([]);
  };
});
</script>
</body>
</html>