<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Interactive Tangent</title>
    <style>
        body {
            margin: 0;
            padding: 0;
            overflow: hidden; 
            width: 100vw;
            height: 100vh;
            background-color: white; /* Clean white background */
            font-family: sans-serif;
            position: relative;
        }

        /* The container now just acts as a full-screen wrapper */
        .canvas-container {
            background-color: rgba(255, 100, 0, 0.1);
            width: 100%;
            height: 100%;
            cursor: crosshair;
            position: relative;
        }

        canvas {
            display: block;
            width: 100%;
            height: 100%;
        }

        /* Slope Readout Styling */
        .slope-readout {
            position: absolute;
            top: 15px;
            right: 15px;
            background-color: rgba(255, 255, 255, 0.9);
            border: 1px solid #ccc;
            padding: 6px 10px;
            border-radius: 4px;
            font-family: monospace;
            font-size: 0.9rem;
            pointer-events: none; /* Allows clicks to pass through */
            user-select: none;
        }

        .slope-label { color: #666; font-weight: bold; }
        .slope-value { color: #ff3333; font-weight: bold; }
    </style>
</head>
<body>
    
    <div class="canvas-container" id="container">
        <canvas id="graphCanvas"></canvas>
        
        <div class="slope-readout">
            <span class="slope-label">Slope:</span> 
            <span class="slope-value" id="slopeValue">0.00</span>
        </div>
    </div>

<script>
    const canvas = document.getElementById('graphCanvas');
    const container = document.getElementById('container');
    const ctx = canvas.getContext('2d');
    const slopeOutput = document.getElementById('slopeValue'); 

    // Global variables for dimensions
    let width, height, originX, originY;
    
    // Scale 120 gives a good view of the sine wave
    const scaleX = 120; 
    const scaleY = 120; 

    // --- Math Functions ---
    function f(x) { return Math.sin(x); }
    function slope(x) { return Math.cos(x); }

    // --- Coordinate Helpers ---
    function toMathX(canvasX) { return (canvasX - originX) / scaleX; }
    function toCanvasX(mathX) { return (mathX * scaleX) + originX; }
    function toCanvasY(mathY) { return originY - (mathY * scaleY); }

    // --- Drawing Logic ---
    function drawCurve() {
        ctx.beginPath();
        ctx.strokeStyle = "black";
        ctx.lineWidth = 4; // Slightly thinner for a cleaner look
        
        for (let pixelX = 0; pixelX <= width; pixelX++) {
            let mathX = toMathX(pixelX);
            let mathY = f(mathX);
            let pixelY = toCanvasY(mathY);
            
            if (pixelX === 0) ctx.moveTo(pixelX, pixelY);
            else ctx.lineTo(pixelX, pixelY);
        }
        ctx.stroke();
    }

    function drawTangent(mouseX) {
        ctx.clearRect(0, 0, width, height);
        
        // Optional: Very faint axis lines for reference
        ctx.beginPath();
        ctx.strokeStyle = "#333333";
        ctx.lineWidth = 1;
        ctx.moveTo(0, originY); ctx.lineTo(width, originY); 
        ctx.moveTo(originX, 0); ctx.lineTo(originX, height); 
        ctx.stroke();

        drawCurve();

        // Default to slight offset if no mouse input (initial load)
        if (mouseX === null) mouseX = originX + 60;

        let currentMathX = toMathX(mouseX);
        let currentMathY = f(currentMathX);
        let currentSlope = slope(currentMathX);

        slopeOutput.innerText = currentSlope.toFixed(2);

        let canvasX = mouseX; 
        let canvasY = toCanvasY(currentMathY);

        // Draw Dot
        ctx.beginPath();
        ctx.fillStyle = "#ff3333";
        ctx.arc(canvasX, canvasY, 6, 0, Math.PI * 2);
        ctx.fill();

        // Draw Tangent Line
        let xLeft = -10; 
        let yLeft = currentSlope * (xLeft - currentMathX) + currentMathY;
        let xRight = 10;
        let yRight = currentSlope * (xRight - currentMathX) + currentMathY;

        ctx.beginPath();
        ctx.strokeStyle = "#ff3333";
        ctx.lineWidth = 2.5; 
        ctx.moveTo(toCanvasX(xLeft), toCanvasY(yLeft));
        ctx.lineTo(toCanvasX(xRight), toCanvasY(yRight));
        ctx.stroke();
    }

    // --- Resizing Logic ---
    function resizeCanvas() {
        const box = container.getBoundingClientRect();
        canvas.width = box.width;
        canvas.height = box.height;

        width = canvas.width;
        height = canvas.height;
        originX = width / 2;
        originY = height / 2;

        drawTangent(null);
    }

    window.addEventListener('resize', resizeCanvas);
    resizeCanvas(); 

    // --- Interaction ---
    container.addEventListener('mousemove', (e) => {
        const rect = container.getBoundingClientRect();
        const mouseX = e.clientX - rect.left;
        window.requestAnimationFrame(() => {
            drawTangent(mouseX);
        });
    });

</script>
</body>
</html>