<!DOCTYPE html>
<html>
<head>
    <title>Ground Displacement Visualization</title>
    <script src="https://cdn.plot.ly/plotly-latest.min.js"></script>
    <style>
        body {
            font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Helvetica, Arial, sans-serif;
            background-color: #f0f0f0;
            color: #333;
            margin: 0;
            padding: 0;
            display: flex;
            justify-content: center;
            align-items: center;
            height: 100vh;
        }
        #app-container {
            width: 90vw;
            height: 90vh;
            max-width: 1000px;
            max-height: 800px;
            background-color: #e8e8e8;
            border: 1px solid #ccc;
            border-radius: 8px;
            box-shadow: 0 4px 12px rgba(0,0,0,0.1);
            display: flex;
            flex-direction: column;
            overflow: hidden;
        }
        #control-panel {
            padding: 20px;
            border-bottom: 1px solid #ccc;
            background-color: #f7f7f7;
            display: flex;
            flex-wrap: wrap;
            gap: 15px 20px;
            align-items: center;
        }
        .control-group {
            display: flex;
            align-items: center;
            gap: 10px;
        }
        .control-group label {
            white-space: nowrap;
            font-size: 14px;
            min-width: 120px;
        }
        input[type="range"] {
            flex-grow: 1;
            min-width: 150px;
            -webkit-appearance: none;
            height: 5px;
            background: #d3d3d3;
            outline: none;
            opacity: 0.7;
            transition: opacity .2s;
            border-radius: 5px;
        }
        input[type="range"]:hover {
            opacity: 1;
        }
        input[type="range"]::-webkit-slider-thumb {
            -webkit-appearance: none;
            appearance: none;
            width: 18px;
            height: 18px;
            background: white;
            border: 1px solid #999;
            cursor: pointer;
            border-radius: 50%;
        }
        input[type="range"]::-moz-range-thumb {
            width: 18px;
            height: 18px;
            background: white;
            border: 1px solid #999;
            cursor: pointer;
            border-radius: 50%;
        }
        .value-display {
            font-family: monospace;
            font-size: 14px;
            min-width: 40px;
            text-align: right;
        }
        .btn-group {
            display: inline-flex;
            border: 1px solid #b0b0b0;
            border-radius: 5px;
            overflow: hidden;
        }
        .btn-group button {
            background-color: #f0f0f0;
            border: none;
            border-left: 1px solid #b0b0b0;
            padding: 5px 12px;
            cursor: pointer;
            font-size: 13px;
            transition: background-color 0.2s;
        }
        .btn-group button:first-child {
            border-left: none;
        }
        .btn-group button:hover {
            background-color: #e0e0e0;
        }
        .btn-group button.active {
            background-color: #c0c0c0;
            color: black;
            font-weight: 500;
        }
        #plot-container {
            flex-grow: 1;
            position: relative;
        }
        #plot {
            width: 100%;
            height: 100%;
        }
    </style>
</head>
<body>

<div id="app-container">
    <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 id="value-dip" class="value-display">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 id="value-depth" class="value-display">500</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 id="value-limit" class="value-display">1000</span>
        </div>
        <div class="control-group">
            <label>displacement component</label>
            <div id="btn-group-component" class="btn-group">
                <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" class="btn-group">
                <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>
    <div id="plot-container">
        <div id="plot"></div>
    </div>
</div>

<script>
    document.addEventListener('DOMContentLoaded', function() {
        // --- DOM Element References ---
        const sliderDip = document.getElementById('slider-dip');
        const valueDip = document.getElementById('value-dip');
        const sliderDepth = document.getElementById('slider-depth');
        const valueDepth = document.getElementById('value-depth');
        const sliderLimit = document.getElementById('slider-limit');
        const valueLimit = document.getElementById('value-limit');
        
        const btnGroupComponent = document.getElementById('btn-group-component');
        const btnGroupFaultType = document.getElementById('btn-group-fault-type');
        
        const plotDiv = document.getElementById('plot');

        // --- State Management ---
        const state = {
            dip: parseInt(sliderDip.value),
            depth: parseInt(sliderDepth.value),
            limit: parseInt(sliderLimit.value),
            component: 'Z',
            faultType: 'tensile'
        };

        // --- Core Calculation Logic ---
        function calculateDisplacements(x, y, dipRad, depth, faultType) {
            // This is a simplified, phenomenological model designed to qualitatively match the snapshot shapes.
            // It is not a precise implementation of the Okada (1992) model.
            const nu = 0.25;
            const scale = 5e7 / (depth + 1);
            
            let ux = 0, uy = 0, uz = 0;
            const R_sq = x * x + y * y + depth * depth;
            if (R_sq === 0) return { ux, uy, uz };

            const R = Math.sqrt(R_sq);
            const R3 = R * R_sq;
            const R5 = R3 * R_sq;
            
            if (faultType === 'tensile') {
                const dip_effect_y = -depth * Math.tan(dipRad);
                const y_shifted = y - dip_effect_y;
                const x_shifted = x;
                const R_eff_sq = x_shifted * x_shifted + y_shifted * y_shifted + depth * depth;
                const R_eff_3 = Math.pow(R_eff_sq, 1.5);

                if (R_eff_3 > 0) {
                    let base_uz = scale * depth / R_eff_3;
                    uz = base_uz * (1 - 0.7 * Math.sin(dipRad) * y_shifted / Math.sqrt(y_shifted*y_shifted + depth*depth));
                    ux = -base_uz * x_shifted / R_eff_sq * Math.sin(dipRad) * 10;
                    uy = -base_uz * y_shifted / R_eff_sq;
                }
            } else if (faultType === 'normal') { // This is a dip-slip fault
                uz = scale * (-y * depth * Math.cos(dipRad) / R5) * 50;
                ux = scale * (-x * y / R5) * 150;
                uy = scale * ((depth * depth - y * y) / R5) * 50;
            } else if (faultType === 'strike-slip') {
                uz = scale * (x * Math.sin(dipRad) * depth / R5) * 500;
                uy = scale * (-2 * x * y / R5) * Math.cos(dipRad) * 5000;
                ux = scale * ((y * y - x * x) / R5) * 5000;
            }

            return { 
                ux: isNaN(ux) ? 0 : ux, 
                uy: isNaN(uy) ? 0 : uy, 
                uz: isNaN(uz) ? 0 : uz 
            };
        }


        // --- Plotting Logic ---
        function updatePlot() {
            // a. Read current values
            const dipRad = state.dip * Math.PI / 180;
            const gridSize = 50;
            
            // c. Create grid coordinates
            const x_coords = Array.from({ length: gridSize }, (_, i) => -state.limit + (i * 2 * state.limit) / (gridSize - 1));
            const y_coords = Array.from({ length: gridSize }, (_, i) => -state.limit + (i * 2 * state.limit) / (gridSize - 1));
            
            // d. Create empty z_values array
            const z_values = [];
            
            // e. Iterate and calculate
            for (let j = 0; j < gridSize; j++) {
                const y = y_coords[j];
                const row = [];
                for (let i = 0; i < gridSize; i++) {
                    const x = x_coords[i];
                    
                    const { ux, uy, uz } = calculateDisplacements(x, y, dipRad, state.depth, state.faultType);
                    
                    let displacement;
                    switch (state.component) {
                        case 'X': displacement = ux; break;
                        case 'Y': displacement = uy; break;
                        case 'Z': displacement = uz; break;
                    }
                    
                    // iii. Convert from meters to millimeters
                    row.push(displacement * 1000);
                }
                z_values.push(row);
            }
            
            // f. Call Plotly update
            const data = [{
                type: 'surface',
                x: x_coords,
                y: y_coords,
                z: z_values,
                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: [-state.limit, state.limit], gridcolor: '#777' },
                    yaxis: { title: 'y (m)', range: [-state.limit, state.limit], gridcolor: '#777' },
                    zaxis: { title: 'z (mm)', autorange: true, gridcolor: '#777' },
                    camera: {
                        eye: { x: 1.8, y: 1.8, z: 0.8 }
                    }
                },
                margin: { l: 20, r: 20, b: 20, t: 20 }
            };

            Plotly.react(plotDiv, data, layout, {responsive: true});
        }

        // --- Event Listeners ---
        sliderDip.addEventListener('input', (e) => {
            state.dip = parseInt(e.target.value);
            valueDip.textContent = state.dip;
            updatePlot();
        });

        sliderDepth.addEventListener('input', (e) => {
            state.depth = parseInt(e.target.value);
            valueDepth.textContent = state.depth;
            updatePlot();
        });

        sliderLimit.addEventListener('input', (e) => {
            state.limit = parseInt(e.target.value);
            valueLimit.textContent = state.limit;
            updatePlot();
        });
        
        function handleButtonGroupClick(e, group, stateKey) {
            if (e.target.tagName === 'BUTTON') {
                const selectedButton = e.target;
                
                if (selectedButton.classList.contains('active')) return;
                
                const currentActive = group.querySelector('.active');
                if (currentActive) {
                    currentActive.classList.remove('active');
                }
                selectedButton.classList.add('active');
                
                if (stateKey === 'component') {
                    state.component = selectedButton.textContent;
                } else if (stateKey === 'faultType') {
                    state.faultType = selectedButton.id.substring('btn-fault-'.length);
                }
                
                updatePlot();
            }
        }
        
        btnGroupComponent.addEventListener('click', (e) => handleButtonGroupClick(e, btnGroupComponent, 'component'));
        btnGroupFaultType.addEventListener('click', (e) => handleButtonGroupClick(e, btnGroupFaultType, 'faultType'));

        // --- Initialization ---
        function init() {
            valueDip.textContent = state.dip;
            valueDepth.textContent = state.depth;
            valueLimit.textContent = state.limit;
            updatePlot();
        }

        init();
        
        window.addEventListener('resize', () => {
             Plotly.Plots.resize(plotDiv);
        });
    });
</script>

</body>
</html>