<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Fault Displacement Visualizer</title>
<script src="https://cdn.plot.ly/plotly-latest.min.js"></script>
<style>
  body {
    font-family: Arial, Helvetica, sans-serif;
    margin: 0;
    padding: 0;
    background:#f5f5f5;
  }
  #app-container {
    max-width: 1200px;
    margin: 0 auto;
    padding: 20px;
    display: flex;
    flex-direction: column;
    gap: 20px;
  }
  #control-panel {
    background:#fff;
    padding:15px;
    border-radius:8px;
    box-shadow:0 2px 6px rgba(0,0,0,.1);
    display:flex;
    flex-wrap:wrap;
    gap:20px;
    align-items:center;
  }
  .control-group {
    display:flex;
    flex-direction:column;
    min-width:150px;
  }
  .control-group label {
    margin-bottom:4px;
    font-weight:bold;
  }
  .control-group input[type=range] {
    width:180px;
  }
  .value-display {
    margin-top:2px;
    font-size:0.9em;
    color:#555;
  }
  #btn-group-component, #btn-group-fault-type {
    display:flex;
    gap:5px;
  }
  button {
    padding:6px 12px;
    border:none;
    border-radius:4px;
    background:#e0e0e0;
    cursor:pointer;
    font-size:0.9rem;
  }
  button.active {
    background:#ff9800;
    color:#fff;
    border:2px solid #ff9800;
  }
  #plot-container {
    background:#fff;
    padding:15px;
    border-radius:8px;
    box-shadow:0 2px 6px rgba(0,0,0,.1);
  }
</style>
</head>
<body>
<div id="app-container">
  <!-- Control Panel -->
  <div id="control-panel">
    <div class="control-group">
      <label for="slider-dip">fault dip</label>
      <input type="range" id="slider-dip" min="0" max="90" step="1" value="0">
      <span class="value-display" id="value-dip">0°</span>
    </div>

    <div class="control-group">
      <label for="slider-depth">fault depth</label>
      <input type="range" id="slider-depth" min="100" max="1000" step="10" value="500">
      <span class="value-display" id="value-depth">500 m</span>
    </div>

    <div class="control-group">
      <label for="slider-limit">x and y plot limit</label>
      <input type="range" id="slider-limit" min="500" max="2000" step="50" value="1000">
      <span class="value-display" id="value-limit">1000 m</span>
    </div>

    <div class="control-group">
      <label>displacement component</label>
      <div id="btn-group-component">
        <button id="btn-component-x">X</button>
        <button id="btn-component-y">Y</button>
        <button id="btn-component-z" class="active">Z</button>
      </div>
    </div>

    <div class="control-group">
      <label>fault type</label>
      <div id="btn-group-fault-type">
        <button id="btn-fault-tensile" class="active">tensile</button>
        <button id="btn-fault-strike-slip">strike‑slip</button>
        <button id="btn-fault-normal">normal</button>
      </div>
    </div>
  </div>

  <!-- Visualization Area -->
  <div id="plot-container">
    <div id="plot"></div>
  </div>
</div>

<script>
/* ---------- Global State ---------- */
let dipDeg = 0;          // degrees
let dipRad = 0;          // radians
let depthM = 500;        // metres
let limitM = 1000;       // metres
let comp = 'Z';          // X, Y or Z
let faultType = 'tensile';

const nu = 0.25;         // Poisson's ratio
const b = 1.0;           // dislocation magnitude (m)

/* ---------- Helper Functions ---------- */
function degToRad(d) { return d * Math.PI / 180; }

function calculateDisplacements(x, y, dip, depth, faultType) {
  // Simple analytical model for demonstration
  const R = Math.sqrt(x*x + y*y + depth*depth); // distance from source
  const theta = Math.atan2(Math.sin(dipRad), Math.cos(dipRad)); // slip direction

  // Displacement components (in metres)
  let ux = 0, uy = 0, uz = 0;
  switch (faultType) {
    case 'tensile':
      ux = b/(4*Math.PI*R) * Math.cos(theta);
      uy = b/(4*Math.PI*R) * Math.sin(theta);
      uz = b/(4*Math.PI*R) * Math.sin(dipRad);
      break;
    case 'strike-slip':
      ux = b/(4*Math.PI*R) * Math.cos(theta);
      uy = b/(4*Math.PI*R) * Math.sin(theta);
      uz = 0;
      break;
    case 'normal':
      ux = 0;
      uy = 0;
      uz = b/(4*Math.PI*R);
      break;
  }
  return {ux, uy, uz};
}

/* ---------- UI Update Functions ---------- */
function updateSliderDisplays() {
  document.getElementById('value-dip').textContent = `${dipDeg}°`;
  document.getElementById('value-depth').textContent = `${depthM} m`;
  document.getElementById('value-limit').textContent = `${limitM} m`;
}

function setActiveButtons() {
  document.querySelectorAll('#btn-group-component button')
          .forEach(b=>b.classList.toggle('active', b.id===`btn-component-${comp}`));
  document.querySelectorAll('#btn-group-fault-type button')
          .forEach(b=>b.classList.toggle('active', b.id===`btn-fault-${faultType}`));
}

/* ---------- Plotting ---------- */
function generateGrid(limit) {
  const step = limit / 50; // 50 points per axis
  const xs = [];
  const ys = [];
  for (let i = -limit; i <= limit; i += step) xs.push(i);
  for (let j = -limit; j <= limit; j += step) ys.push(j);
  return {xs, ys};
}

function computeZValues() {
  const {xs, ys} = generateGrid(limitM);
  const zVals = [];

  for (let i = 0; i < xs.length; i++) {
    const row = [];
    const x = xs[i];
    for (let j = 0; j < ys.length; j++) {
      const y = ys[j];
      const disp = calculateDisplacements(x, y, dipRad, depthM, faultType);
      const val = comp === 'X' ? disp.ux :
                  comp === 'Y' ? disp.uy : disp.uz;
      row.push(val * 1000); // convert to mm
    }
    zVals.push(row);
  }
  return {xs, ys, zVals};
}

function drawPlot() {
  const {xs, ys, zVals} = computeZValues();

  const trace = {
    type: 'surface',
    x: xs,
    y: ys,
    z: zVals,
    colorscale: [
      [0, '#FFA500'],
      [1, '#FFD700']
    ],
    showscale: false,
    contours: {
      x: {show: true, color: 'black', width: 0.5},
      y: {show: true, color: 'black', width: 0.5},
      z: {show: false}
    },
    lighting: {
      ambient: 0.8,
      diffuse: 0.8,
      specular: 0.2
    }
  };

  const layout = {
    title: '',
    scene: {
      xaxis: {title: 'x (m)', range: [-limitM, limitM]},
      yaxis: {title: 'y (m)', range: [-limitM, limitM]},
      zaxis: {title: 'z (mm)', autorange: true},
      camera: {eye: {x: 1.8, y: 1.8, z: 0.8}}
    },
    margin: {l: 20, r: 20, b: 20, t: 20}
  };

  Plotly.react('plot', [trace], layout, {responsive: true});
}

/* ---------- Event Listeners ---------- */
document.getElementById('slider-dip').addEventListener('input', e => {
  dipDeg = Number(e.target.value);
  dipRad = degToRad(dipDeg);
  updateSliderDisplays();
  drawPlot();
});

document.getElementById('slider-depth').addEventListener('input', e => {
  depthM = Number(e.target.value);
  updateSliderDisplays();
  drawPlot();
});

document.getElementById('slider-limit').addEventListener('input', e => {
  limitM = Number(e.target.value);
  updateSliderDisplays();
  drawPlot();
});

document.getElementById('btn-group-component').addEventListener('click', e => {
  if (e.target.tagName !== 'BUTTON') return;
  comp = e.target.id.split('-')[1]; // X, Y or Z
  setActiveButtons();
  drawPlot();
});

document.getElementById('btn-group-fault-type').addEventListener('click', e => {
  if (e.target.tagName !== 'BUTTON') return;
  faultType = e.target.id.split('-')[1]; // tensile, strike-slip, normal
  setActiveButtons();
  drawPlot();
});

/* ---------- Initialization ---------- */
function init() {
  // set initial displays
  updateSliderDisplays();
  setActiveButtons();
  drawPlot();
}
init();
</script>
</body>
</html>