<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width,initial-scale=1" />
<title>Spring-Mass Oscillator: Plot and Animation (p5.js)</title>
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.9.0/p5.min.js"></script>
<style>
  :root {
    --bg: #f2f3f7;
    --panel: #ffffff;
    --text: #222;
    --muted: #6b7280;
    --accent: #6363B2;
    --grid: #e5e7eb;
    --axis: #111827;
  }
  html, body {
    margin: 0;
    padding: 0;
    background: var(--bg);
    color: var(--text);
    font-family: system-ui, -apple-system, Segoe UI, Roboto, Ubuntu, Cantarell, "Helvetica Neue", Arial, "Noto Sans", "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
  }
  #app {
    max-width: 1200px;
    margin: 20px auto 40px;
    padding: 0 16px;
    display: flex;
    flex-direction: column;
    gap: 16px;
  }

  /* Control Panel */
  .control-panel {
    position: relative;
    background: var(--panel);
    border-radius: 10px;
    box-shadow: 0 1px 3px rgba(0,0,0,0.07), 0 4px 24px rgba(0,0,0,0.06);
    padding: 16px 16px 10px;
  }
  .cp-row {
    display: flex;
    flex-wrap: wrap;
    align-items: center;
    gap: 18px 24px;
  }
  .control-group {
    display: grid;
    grid-template-columns: auto 240px auto;
    align-items: center;
    gap: 10px;
    padding: 6px 8px;
    background: #fafafa;
    border-radius: 8px;
  }
  .control-group label {
    font-size: 14px;
    color: var(--muted);
    text-transform: lowercase;
  }
  .control-group input[type="range"] {
    width: 240px;
  }
  .value {
    min-width: 56px;
    text-align: right;
    font-variant-numeric: tabular-nums;
    font-weight: 600;
  }

  .time-controls {
    display: flex;
    align-items: center;
    gap: 8px;
    padding: 6px 8px;
    background: #fafafa;
    border-radius: 8px;
  }
  .time-controls label {
    margin-right: 6px;
    font-size: 14px;
    color: var(--muted);
    text-transform: lowercase;
  }
  button {
    appearance: none;
    border: 1px solid #d1d5db;
    background: white;
    color: #111827;
    padding: 6px 10px;
    border-radius: 8px;
    cursor: pointer;
    font-size: 14px;
    line-height: 1;
    transition: background 0.15s ease, transform 0.03s ease, border-color 0.15s ease;
  }
  button:hover {
    background: #f2f4f8;
  }
  button:active {
    transform: translateY(1px);
  }

  #btn-play-pause {
    font-weight: 700;
    border-color: #c7cce3;
    background: #eef0ff;
    color: #2b2b7f;
  }

  #btn-reset {
    position: absolute;
    right: 10px;
    top: 10px;
    border-color: #fca5a5;
    background: #fff1f1;
    color: #991b1b;
    font-weight: 700;
    width: 28px;
    height: 28px;
    padding: 0;
  }

  /* Visualization area */
  .viz {
    display: flex;
    gap: 12px;
    align-items: stretch;
    min-height: 300px;
  }
  #plot-canvas-container, #animation-canvas-container {
    background: var(--panel);
    border-radius: 10px;
    box-shadow: 0 1px 3px rgba(0,0,0,0.07), 0 4px 24px rgba(0,0,0,0.06);
    display: flex;
    justify-content: center;
    align-items: center;
    overflow: hidden;
    position: relative;
  }
  #plot-canvas-container {
    flex: 1 1 70%;
    min-width: 400px;
    min-height: 320px;
  }
  #animation-canvas-container {
    flex: 1 1 30%;
    min-width: 220px;
    min-height: 320px;
  }

  /* A11y minor */
  .sr-only {
    position: absolute;
    left: -10000px;
    width: 1px;
    height: 1px;
    overflow: hidden;
  }

  @media (max-width: 900px) {
    .control-group {
      grid-template-columns: auto 1fr auto;
    }
    .viz {
      flex-direction: column;
    }
    #plot-canvas-container, #animation-canvas-container {
      min-height: 280px;
    }
  }
</style>
</head>
<body>
  <div id="app">
    <!-- Control Panel Section -->
    <div class="control-panel">
      <button id="btn-reset" title="Reset">x</button>
      <div class="cp-row">
        <!-- Amplitude Control -->
        <div class="control-group" id="amplitude-control">
          <label for="slider-amplitude">amplitude</label>
          <input type="range" id="slider-amplitude" min="0.1" max="1.0" step="0.001" value="0.3" />
          <span class="value" id="amplitude-value">0.300</span>
        </div>
        <!-- Stiffness Control -->
        <div class="control-group" id="stiffness-control">
          <label for="slider-stiffness">stiffness</label>
          <input type="range" id="slider-stiffness" min="0.1" max="10.0" step="0.01" value="1.0" />
          <span class="value" id="stiffness-value">1.00</span>
        </div>
        <!-- Mass Control -->
        <div class="control-group" id="mass-control">
          <label for="slider-mass">mass</label>
          <input type="range" id="slider-mass" min="0.1" max="10.0" step="0.01" value="1.0" />
          <span class="value" id="mass-value">1.00</span>
        </div>
        <!-- Time Controls -->
        <div class="time-controls">
          <label for="btn-play-pause">time</label>
          <button id="btn-step-back" title="Step back">◀</button>
          <button id="btn-play-pause" title="Play/Pause">▶</button>
          <button id="btn-step-forward" title="Step forward">▶|</button>
        </div>
      </div>
    </div>

    <!-- Visualization Container -->
    <div class="viz">
      <div id="plot-canvas-container" aria-label="Plot Canvas"></div>
      <div id="animation-canvas-container" aria-label="Animation Canvas"></div>
    </div>
  </div>

<script>
  // Shared simulation state
  let amplitude = 0.3;
  let stiffness = 1.0;
  let mass = 1.0;
  let angularFrequency = Math.sqrt(stiffness / mass);
  let time = 0.0;
  let isPlaying = false;
  let positionHistory = [];
  let xMax = 25;       // initial x-axis max for plot
  const xMin = 0;
  const dtPlay = 0.05; // time increment per frame during play
  const dtStep = 0.1;  // time increment for step buttons
  const seriesColor = "#6363B2";

  // DOM elements
  const elAmp = document.getElementById('slider-amplitude');
  const elStiff = document.getElementById('slider-stiffness');
  const elMass = document.getElementById('slider-mass');
  const elAmpVal = document.getElementById('amplitude-value');
  const elStiffVal = document.getElementById('stiffness-value');
  const elMassVal = document.getElementById('mass-value');
  const btnPlayPause = document.getElementById('btn-play-pause');
  const btnStepBack = document.getElementById('btn-step-back');
  const btnStepForward = document.getElementById('btn-step-forward');
  const btnReset = document.getElementById('btn-reset');

  function fmt3(v) { return (+v).toFixed(3); }
  function fmt2(v) { return (+v).toFixed(2); }

  function recalcOmega() {
    angularFrequency = Math.sqrt(stiffness / mass);
  }

  function addCurrentPoint() {
    const y = amplitude * Math.sin(angularFrequency * time);
    positionHistory.push({ t: time, y: y });
    if (time > xMax) {
      // Extend xMax in chunks to keep the grid tidy
      const chunk = 25;
      while (time > xMax) xMax += chunk;
    }
  }

  function setPlayState(playing) {
    isPlaying = playing;
    if (isPlaying) {
      btnPlayPause.textContent = '||';
      if (plotP5) plotP5.loop();
      if (animP5) animP5.loop();
    } else {
      btnPlayPause.textContent = '▶';
      if (plotP5) plotP5.noLoop();
      if (animP5) animP5.noLoop();
      // ensure final static redraw reflects last state
      redrawBoth();
    }
  }

  function resetSimulation() {
    amplitude = 0.3;
    stiffness = 1.0;
    mass = 1.0;
    elAmp.value = amplitude;
    elStiff.value = stiffness;
    elMass.value = mass;
    elAmpVal.textContent = fmt3(amplitude);
    elStiffVal.textContent = fmt2(stiffness);
    elMassVal.textContent = fmt2(mass);
    recalcOmega();
    time = 0;
    positionHistory = [];
    xMax = 25;
    setPlayState(false);
    redrawBoth();
  }

  function onSliderChange() {
    amplitude = parseFloat(elAmp.value);
    stiffness = parseFloat(elStiff.value);
    mass = parseFloat(elMass.value);
    elAmpVal.textContent = fmt3(amplitude);
    elStiffVal.textContent = fmt2(stiffness);
    elMassVal.textContent = fmt2(mass);
    recalcOmega();
    time = 0;
    positionHistory = [];
    xMax = 25;
    setPlayState(false);
    redrawBoth();
  }

  function stepTime(dir) {
    if (isPlaying) return; // step only when paused
    time += (dir > 0 ? dtStep : -dtStep);
    if (time < 0) time = 0;
    addCurrentPoint();
    redrawBoth();
  }

  function redrawBoth() {
    if (plotP5) plotP5.redraw();
    if (animP5) animP5.redraw();
  }

  // Initialize UI events
  elAmp.addEventListener('input', onSliderChange);
  elStiff.addEventListener('input', onSliderChange);
  elMass.addEventListener('input', onSliderChange);

  btnPlayPause.addEventListener('click', () => setPlayState(!isPlaying));
  btnStepBack.addEventListener('click', () => stepTime(-1));
  btnStepForward.addEventListener('click', () => stepTime(+1));
  btnReset.addEventListener('click', resetSimulation);

  // p5 instances
  let plotP5, animP5;

  // Plot Sketch (left)
  const plotSketch = (p) => {
    let margin = { left: 54, right: 16, top: 18, bottom: 36 };
    let w = 800, h = 420;

    function containerSize() {
      const el = document.getElementById('plot-canvas-container');
      const cw = Math.max(400, el.clientWidth);
      const ch = Math.max(320, el.clientHeight);
      return { w: cw, h: ch };
    }

    function xToPx(x) {
      const xmin = xMin, xmax = xMax;
      const innerW = w - margin.left - margin.right;
      return margin.left + (x - xmin) / (xmax - xmin) * innerW;
    }

    function yToPx(y) {
      const ymin = -1, ymax = 1;
      const innerH = h - margin.top - margin.bottom;
      // y positive up -> screen y decreases
      return margin.top + (ymax - y) / (ymax - ymin) * innerH;
    }

    function drawGridAndAxes() {
      p.push();
      p.noFill();

      // Background
      p.background(255);

      // Grid
      p.strokeWeight(1);
      p.stroke('#eaecef');

      // Vertical grid lines (time)
      const xStep = 5;
      for (let xg = 0; xg <= xMax + 1e-9; xg += xStep) {
        const px = xToPx(xg);
        p.stroke('#e5e7eb');
        p.line(px, margin.top, px, h - margin.bottom);
      }

      // Horizontal grid lines (y)
      const yTicks = [-1, -0.5, 0, 0.5, 1];
      yTicks.forEach(yt => {
        const py = yToPx(yt);
        p.stroke('#e5e7eb');
        p.line(margin.left, py, w - margin.right, py);
      });

      // Axes
      p.stroke('#111827');
      p.strokeWeight(1.5);
      // y=0 axis
      const y0 = yToPx(0);
      p.line(margin.left, y0, w - margin.right, y0);
      // x=0 axis
      const x0 = xToPx(0);
      p.line(x0, margin.top, x0, h - margin.bottom);

      // Axis labels
      p.fill('#374151');
      p.noStroke();
      p.textSize(12);
      p.textAlign(p.CENTER, p.TOP);
      for (let xg = 0; xg <= xMax + 1e-9; xg += xStep) {
        const px = xToPx(xg);
        p.text(xg.toFixed(0), px, h - margin.bottom + 6);
      }
      p.textAlign(p.RIGHT, p.CENTER);
      [-1, 0, 1].forEach(yt => {
        const py = yToPx(yt);
        p.text(yt.toString(), margin.left - 8, py);
      });

      // Titles (subtle)
      p.textAlign(p.LEFT, p.TOP);
      p.fill('#6b7280');
      p.textSize(13);
      p.text('position y(t)', margin.left + 4, margin.top + 2);
      p.pop();
    }

    function drawWaveform() {
      if (positionHistory.length < 2) return;
      p.push();
      p.noFill();
      p.stroke(seriesColor);
      p.strokeWeight(2);

      p.beginShape();
      for (let i = 0; i < positionHistory.length; i++) {
        const pt = positionHistory[i];
        const px = xToPx(pt.t);
        const py = yToPx(pt.y);
        p.vertex(px, py);
      }
      p.endShape();

      p.pop();
    }

    p.setup = function() {
      const size = containerSize();
      w = size.w; h = size.h;
      const cnv = p.createCanvas(w, h);
      cnv.parent('plot-canvas-container');
      p.frameRate(60);
      p.noLoop(); // start paused
    };

    p.windowResized = function() {
      const size = containerSize();
      w = size.w; h = size.h;
      p.resizeCanvas(w, h);
      p.redraw();
    };

    p.draw = function() {
      // Update time if playing (only in this sketch to avoid double increment)
      if (isPlaying) {
        time += dtPlay;
        addCurrentPoint();
      }

      drawGridAndAxes();
      drawWaveform();
    };
  };

  // Animation Sketch (right)
  const animSketch = (p) => {
    let w = 320, h = 480;
    let margin = { left: 20, right: 20, top: 18, bottom: 24 };
    const yMin = -1.2, yMax = 1.2;

    function containerSize() {
      const el = document.getElementById('animation-canvas-container');
      const cw = Math.max(220, el.clientWidth);
      const ch = Math.max(320, el.clientHeight);
      return { w: cw, h: ch };
    }

    function yToPx(y) {
      const innerH = h - margin.top - margin.bottom;
      return margin.top + (yMax - y) / (yMax - yMin) * innerH;
    }
    function pxToY(y_px) {
      const innerH = h - margin.top - margin.bottom;
      const frac = (y_px - margin.top) / innerH;
      return yMax - frac * (yMax - yMin);
    }

    p.setup = function() {
      const size = containerSize();
      w = size.w; h = size.h;
      const cnv = p.createCanvas(w, h);
      cnv.parent('animation-canvas-container');
      p.frameRate(60);
      p.noLoop(); // controlled by play state
    };

    p.windowResized = function() {
      const size = containerSize();
      w = size.w; h = size.h;
      p.resizeCanvas(w, h);
      p.redraw();
    };

    function drawAxes() {
      p.push();
      p.background(255);

      // Vertical axis at center
      const cx = Math.round(w / 2);
      p.stroke('#111827');
      p.strokeWeight(1.5);
      p.line(cx, margin.top, cx, h - margin.bottom);

      // Grid-like y labels at -1, 0, 1
      p.noStroke();
      p.fill('#374151');
      p.textSize(12);
      p.textAlign(p.RIGHT, p.CENTER);
      [-1, 0, 1].forEach(yt => {
        const py = yToPx(yt);
        p.fill('#e5e7eb');
        p.rect(margin.left, py - 0.5, w - margin.left - margin.right, 1);
        p.fill('#374151');
        p.text(yt.toString(), cx - 10, py);
      });

      // Title
      p.fill('#6b7280');
      p.textAlign(p.LEFT, p.TOP);
      p.textSize(13);
      p.text('spring-mass animation', margin.left + 2, margin.top + 2);
      p.pop();
    }

    function drawSpringAndMass() {
      const cx = Math.round(w / 2);

      // Anchor at top
      const anchorY = margin.top + 22;
      p.stroke(0);
      p.strokeWeight(3);
      p.line(cx - 40, anchorY - 8, cx + 40, anchorY - 8); // ceiling bar

      // equilibrium pixel y (world y=0)
      const y0px = yToPx(0);

      // Current world y (positive up), compute pixel
      const yWorld = amplitude * Math.sin(angularFrequency * time);
      const yPx = yToPx(yWorld); // pixel coordinate of the mass center in world coordinates (vertical reference)

      // Define mass rectangle size
      const massW = Math.min(80, Math.max(42, w * 0.18));
      const massH = Math.min(60, Math.max(34, h * 0.10));

      // Mass center and top in pixels:
      // Place mass center around world y mapped, but ensure spring base length from anchor
      const massCenterY = yPx;
      const massTopY = massCenterY - massH / 2;

      // Spring geometry from anchor to massTopY
      const springStartY = anchorY;
      const springEndY = massTopY;
      const springLen = Math.max(30, springEndY - springStartY);

      // Draw spring as zigzag
      const coils = 12;
      const halfAmp = Math.min(16, w * 0.06);
      p.stroke(0);
      p.strokeWeight(2);
      p.noFill();

      let xLeft = cx - halfAmp;
      let xRight = cx + halfAmp;

      p.beginShape();
      // Start vertical segment
      p.vertex(cx, springStartY);
      const segment = springLen / (coils * 2);
      let y = springStartY;
      for (let i = 0; i < coils; i++) {
        y += segment;
        p.vertex(xLeft, y);
        y += segment;
        p.vertex(xRight, y);
      }
      // Last vertical down to end
      p.vertex(cx, springEndY);
      p.endShape();

      // Mass block
      p.fill(seriesColor);
      p.stroke('#2f2f7a');
      p.strokeWeight(2);
      p.rectMode(p.CORNER);
      p.rect(cx - massW / 2, massTopY, massW, massH, 8);
    }

    p.draw = function() {
      drawAxes();
      drawSpringAndMass();
    };
  };

  // Instantiate p5 sketches
  plotP5 = new p5(plotSketch);
  animP5 = new p5(animSketch);

  // Initial state on load
  window.addEventListener('load', () => {
    // Ensure UI reflects defaults
    elAmpVal.textContent = fmt3(amplitude);
    elStiffVal.textContent = fmt2(stiffness);
    elMassVal.textContent = fmt2(mass);
    recalcOmega();
    time = 0;
    positionHistory = [];
    xMax = 25;
    setPlayState(false);
    redrawBoth();
  });
</script>
</body>
</html>