| <!DOCTYPE html> |
| <html lang="en"> |
| <head> |
| <meta charset="UTF-8"> |
| <meta name="viewport" content="width=device-width, initial-scale=1.0"> |
| <title>Webcam Eye Tracker</title> |
| <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css"> |
| <style> |
| :root { |
| --primary-color: #5d69b2; |
| --secondary-color: #3a416f; |
| --accent-color: #ff7e5f; |
| --bg-color: #f5f7fa; |
| --text-color: #333; |
| } |
| |
| * { |
| margin: 0; |
| padding: 0; |
| box-sizing: border-box; |
| } |
| |
| body { |
| font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif; |
| background-color: var(--bg-color); |
| color: var(--text-color); |
| display: flex; |
| flex-direction: column; |
| align-items: center; |
| min-height: 100vh; |
| padding: 2rem; |
| position: relative; |
| overflow-x: hidden; |
| } |
| |
| header { |
| text-align: center; |
| margin-bottom: 2rem; |
| width: 100%; |
| } |
| |
| h1 { |
| color: var(--primary-color); |
| margin-bottom: 0.5rem; |
| font-size: 2.2rem; |
| } |
| |
| .subtitle { |
| color: var(--secondary-color); |
| opacity: 0.8; |
| margin-bottom: 1.5rem; |
| } |
| |
| .tracker-container { |
| display: flex; |
| flex-direction: column; |
| align-items: center; |
| width: 100%; |
| max-width: 800px; |
| gap: 2rem; |
| } |
| |
| .camera-container { |
| position: relative; |
| width: 100%; |
| max-width: 640px; |
| border-radius: 12px; |
| overflow: hidden; |
| box-shadow: 0 10px 30px rgba(0, 0, 0, 0.1); |
| } |
| |
| #video { |
| width: 100%; |
| display: block; |
| background-color: #000; |
| } |
| |
| #canvas { |
| position: absolute; |
| top: 0; |
| left: 0; |
| width: 100%; |
| } |
| |
| .metrics-container { |
| display: flex; |
| flex-wrap: wrap; |
| gap: 1rem; |
| justify-content: center; |
| width: 100%; |
| } |
| |
| .metric-card { |
| background-color: white; |
| border-radius: 12px; |
| padding: 1.5rem; |
| min-width: 200px; |
| box-shadow: 0 5px 15px rgba(0, 0, 0, 0.05); |
| flex-grow: 1; |
| transition: transform 0.3s ease, box-shadow 0.3s ease; |
| } |
| |
| .metric-card:hover { |
| transform: translateY(-5px); |
| box-shadow: 0 10px 25px rgba(0, 0, 0, 0.1); |
| } |
| |
| .metric-title { |
| color: var(--primary-color); |
| font-size: 0.9rem; |
| text-transform: uppercase; |
| letter-spacing: 1px; |
| margin-bottom: 0.5rem; |
| display: flex; |
| align-items: center; |
| gap: 0.5rem; |
| } |
| |
| .metric-value { |
| font-size: 1.8rem; |
| font-weight: bold; |
| color: var(--secondary-color); |
| } |
| |
| .metric-unit { |
| font-size: 0.9rem; |
| color: #888; |
| margin-left: 0.3rem; |
| } |
| |
| .chart-container { |
| width: 100%; |
| height: 200px; |
| background-color: white; |
| border-radius: 12px; |
| padding: 1rem; |
| box-shadow: 0 5px 15px rgba(0, 0, 0, 0.05); |
| position: relative; |
| overflow: hidden; |
| } |
| |
| canvas#gaze-chart { |
| width: 100%; |
| height: 100%; |
| } |
| |
| .controls { |
| display: flex; |
| gap: 1rem; |
| margin-top: 1rem; |
| flex-wrap: wrap; |
| justify-content: center; |
| } |
| |
| button { |
| padding: 0.8rem 1.5rem; |
| background-color: var(--primary-color); |
| color: white; |
| border: none; |
| border-radius: 8px; |
| cursor: pointer; |
| font-weight: 600; |
| transition: all 0.3s ease; |
| display: flex; |
| align-items: center; |
| gap: 0.5rem; |
| } |
| |
| button:hover { |
| background-color: var(--secondary-color); |
| transform: translateY(-2px); |
| } |
| |
| button.secondary { |
| background-color: white; |
| color: var(--primary-color); |
| border: 1px solid #ddd; |
| } |
| |
| button.secondary:hover { |
| background-color: #f0f0f0; |
| } |
| |
| .loading { |
| display: flex; |
| flex-direction: column; |
| align-items: center; |
| justify-content: center; |
| gap: 1rem; |
| padding: 2rem; |
| background-color: white; |
| border-radius: 12px; |
| box-shadow: 0 5px 15px rgba(0, 0, 0, 0.05); |
| width: 100%; |
| } |
| |
| .spinner { |
| border: 4px solid rgba(0, 0, 0, 0.1); |
| border-radius: 50%; |
| border-top: 4px solid var(--primary-color); |
| width: 40px; |
| height: 40px; |
| animation: spin 1s linear infinite; |
| } |
| |
| @keyframes spin { |
| 0% { transform: rotate(0deg); } |
| 100% { transform: rotate(360deg); } |
| } |
| |
| .error-message { |
| color: #e74c3c; |
| background-color: #fceae9; |
| padding: 1rem; |
| border-radius: 8px; |
| max-width: 100%; |
| text-align: center; |
| } |
| |
| footer { |
| margin-top: 3rem; |
| text-align: center; |
| color: #888; |
| font-size: 0.9rem; |
| } |
| |
| @media (max-width: 600px) { |
| .tracker-container { |
| gap: 1rem; |
| } |
| |
| .metric-card { |
| min-width: 150px; |
| padding: 1rem; |
| } |
| |
| .metric-value { |
| font-size: 1.5rem; |
| } |
| } |
| </style> |
| </head> |
| <body> |
| <header> |
| <h1><i class="fas fa-eye"></i> Webcam Eye Tracker</h1> |
| <p class="subtitle">Real-time eye tracking using your webcam and face detection</p> |
| </header> |
|
|
| <div class="tracker-container"> |
| <div class="loading" id="loading"> |
| <div class="spinner"></div> |
| <p>Loading face detection models...</p> |
| </div> |
|
|
| <div class="camera-container" id="camera-container" style="display: none;"> |
| <video id="video" width="640" height="480" autoplay muted playsinline></video> |
| <canvas id="canvas" width="640" height="480"></canvas> |
| </div> |
|
|
| <div class="metrics-container"> |
| <div class="metric-card"> |
| <div class="metric-title"> |
| <i class="fas fa-crosshairs"></i> Eye Position |
| </div> |
| <div class="metric-value" id="eye-position"> |
| <span id="eye-x">0</span>, <span id="eye-y">0</span> |
| </div> |
| </div> |
|
|
| <div class="metric-card"> |
| <div class="metric-title"> |
| <i class="fas fa-running"></i> Movement Speed |
| </div> |
| <div class="metric-value" id="movement-speed">0<span class="metric-unit">px/s</span></div> |
| </div> |
|
|
| <div class="metric-card"> |
| <div class="metric-title"> |
| <i class="fas fa-history"></i> Time Tracked |
| </div> |
| <div class="metric-value" id="time-tracked">0<span class="metric-unit">s</span></div> |
| </div> |
|
|
| <div class="metric-card"> |
| <div class="metric-title"> |
| <i class="fas fa-bullseye"></i> Fixations |
| </div> |
| <div class="metric-value" id="fixation-count">0</div> |
| </div> |
| </div> |
|
|
| <div class="chart-container"> |
| <canvas id="gaze-chart"></canvas> |
| </div> |
|
|
| <div class="controls"> |
| <button id="start-btn"><i class="fas fa-play"></i> Start Tracking</button> |
| <button id="reset-btn" class="secondary"><i class="fas fa-redo"></i> Reset</button> |
| <button id="debug-btn" class="secondary"><i class="fas fa-bug"></i> Toggle Debug</button> |
| </div> |
|
|
| <div class="error-message" id="error-message" style="display: none;"></div> |
| </div> |
|
|
| <footer> |
| <p>Webcam Eye Tracker © 2024 | Uses face-api.js for face and eye detection</p> |
| </footer> |
|
|
| |
| <script src="https://cdn.jsdelivr.net/npm/face-api.js@0.22.2/dist/face-api.min.js"></script> |
| |
| <script src="https://cdn.jsdelivr.net/npm/chart.js"></script> |
|
|
| <script> |
| document.addEventListener('DOMContentLoaded', async function() { |
| |
| const video = document.getElementById('video'); |
| const canvas = document.getElementById('canvas'); |
| const ctx = canvas.getContext('2d'); |
| const startBtn = document.getElementById('start-btn'); |
| const resetBtn = document.getElementById('reset-btn'); |
| const debugBtn = document.getElementById('debug-btn'); |
| const eyeX = document.getElementById('eye-x'); |
| const eyeY = document.getElementById('eye-y'); |
| const movementSpeed = document.getElementById('movement-speed'); |
| const timeTracked = document.getElementById('time-tracked'); |
| const fixationCount = document.getElementById('fixation-count'); |
| const loadingElement = document.getElementById('loading'); |
| const cameraContainer = document.getElementById('camera-container'); |
| const errorMessage = document.getElementById('error-message'); |
| |
| |
| let trackingActive = false; |
| let startTime = 0; |
| let lastPosition = { x: 0, y: 0 }; |
| let lastTime = 0; |
| let currentSpeed = 0; |
| let fixations = 0; |
| let fixationStartTime = 0; |
| let isFixated = false; |
| let gazeHistory = []; |
| let showDebug = false; |
| let modelsLoaded = false; |
| let stream = null; |
| |
| |
| const chartCtx = document.getElementById('gaze-chart').getContext('2d'); |
| const gazeChart = new Chart(chartCtx, { |
| type: 'line', |
| data: { |
| labels: [], |
| datasets: [{ |
| label: 'Eye Movement Speed (px/s)', |
| data: [], |
| borderColor: '#5d69b2', |
| backgroundColor: 'rgba(93, 105, 178, 0.1)', |
| tension: 0.4, |
| fill: true |
| }] |
| }, |
| options: { |
| responsive: true, |
| maintainAspectRatio: false, |
| scales: { |
| y: { |
| beginAtZero: true |
| } |
| }, |
| animation: { |
| duration: 0 |
| } |
| } |
| }); |
| |
| |
| async function loadModels() { |
| try { |
| await faceapi.nets.tinyFaceDetector.loadFromUri('https://justadudewhohacks.github.io/face-api.js/models'); |
| await faceapi.nets.faceLandmark68Net.loadFromUri('https://justadudewhohacks.github.io/face-api.js/models'); |
| modelsLoaded = true; |
| |
| |
| loadingElement.style.display = 'none'; |
| cameraContainer.style.display = 'block'; |
| |
| |
| initCamera(); |
| } catch (error) { |
| console.error('Error loading models:', error); |
| showError("Failed to load face detection models. Please check your internet connection and try again."); |
| } |
| } |
| |
| |
| async function initCamera() { |
| try { |
| stream = await navigator.mediaDevices.getUserMedia({ |
| video: { |
| width: { ideal: 640 }, |
| height: { ideal: 480 }, |
| facingMode: 'user' |
| }, |
| audio: false |
| }); |
| video.srcObject = stream; |
| video.play(); |
| } catch (error) { |
| console.error('Camera error:', error); |
| if (error.name === 'NotAllowedError') { |
| showError("Camera access was denied. Please allow camera access to use this feature."); |
| } else if (error.name === 'NotFoundError') { |
| showError("No camera found. Please connect a webcam to use this feature."); |
| } else { |
| showError("Failed to access camera. Please try again."); |
| } |
| } |
| } |
| |
| |
| function showError(message) { |
| errorMessage.textContent = message; |
| errorMessage.style.display = 'block'; |
| loadingElement.style.display = 'none'; |
| } |
| |
| |
| function getEyePosition(landmarks) { |
| if (!landmarks || !landmarks.getLeftEye || !landmarks.getRightEye) { |
| return { x: 0, y: 0 }; |
| } |
| |
| const leftEye = landmarks.getLeftEye(); |
| const rightEye = landmarks.getRightEye(); |
| |
| |
| const leftEyeCenter = leftEye.reduce((sum, point) => { |
| return { x: sum.x + point.x, y: sum.y + point.y }; |
| }, { x: 0, y: 0 }); |
| |
| const rightEyeCenter = rightEye.reduce((sum, point) => { |
| return { x: sum.x + point.x, y: sum.y + point.y }; |
| }, { x: 0, y: 0 }); |
| |
| leftEyeCenter.x /= leftEye.length; |
| leftEyeCenter.y /= leftEye.length; |
| rightEyeCenter.x /= rightEye.length; |
| rightEyeCenter.y /= rightEye.length; |
| |
| |
| return { |
| x: (leftEyeCenter.x + rightEyeCenter.x) / 2, |
| y: (leftEyeCenter.y + rightEyeCenter.y) / 2 |
| }; |
| } |
| |
| |
| function updateMetrics(position) { |
| const now = Date.now(); |
| const timeElapsed = (now - lastTime) / 1000; |
| |
| if (timeElapsed > 0) { |
| const dx = position.x - lastPosition.x; |
| const dy = position.y - lastPosition.y; |
| const distance = Math.sqrt(dx * dx + dy * dy); |
| currentSpeed = distance / timeElapsed; |
| |
| |
| if (distance < 15) { |
| if (!isFixated) { |
| isFixated = true; |
| fixationStartTime = now; |
| } |
| |
| |
| if (isFixated && now - fixationStartTime > 200) { |
| fixations++; |
| fixationCount.textContent = fixations; |
| isFixated = false; |
| } |
| } else { |
| isFixated = false; |
| } |
| } |
| |
| |
| movementSpeed.textContent = Math.round(currentSpeed); |
| timeTracked.textContent = Math.round((now - startTime) / 1000); |
| eyeX.textContent = Math.round(position.x); |
| eyeY.textContent = Math.round(position.y); |
| |
| |
| if (gazeHistory.length > 50) { |
| gazeHistory.shift(); |
| gazeChart.data.labels.shift(); |
| gazeChart.data.datasets[0].data.shift(); |
| } |
| |
| gazeHistory.push(currentSpeed); |
| gazeChart.data.labels.push(''); |
| gazeChart.data.datasets[0].data.push(currentSpeed); |
| gazeChart.update(); |
| |
| |
| lastPosition = position; |
| lastTime = now; |
| } |
| |
| |
| async function processVideo() { |
| if (!trackingActive || !modelsLoaded) { |
| requestAnimationFrame(processVideo); |
| return; |
| } |
| |
| try { |
| |
| const options = new faceapi.TinyFaceDetectorOptions({ |
| inputSize: 128, |
| scoreThreshold: 0.5 |
| }); |
| |
| const result = await faceapi.detectSingleFace(video, options) |
| .withFaceLandmarks(); |
| |
| |
| ctx.clearRect(0, 0, canvas.width, canvas.height); |
| |
| if (result) { |
| const { landmarks, detection } = result; |
| |
| |
| const eyePosition = getEyePosition(landmarks); |
| |
| |
| updateMetrics(eyePosition); |
| |
| |
| if (showDebug) { |
| |
| const box = detection.box; |
| ctx.strokeStyle = '#00FF00'; |
| ctx.lineWidth = 2; |
| ctx.strokeRect(box.x, box.y, box.width, box.height); |
| |
| |
| faceapi.draw.drawFaceLandmarks(canvas, landmarks); |
| |
| |
| ctx.fillStyle = '#FF0000'; |
| ctx.beginPath(); |
| ctx.arc(eyePosition.x, eyePosition.y, 5, 0, 2 * Math.PI); |
| ctx.fill(); |
| } |
| } |
| } catch (error) { |
| console.error('Detection error:', error); |
| } |
| |
| requestAnimationFrame(processVideo); |
| } |
| |
| |
| function startTracking() { |
| if (!modelsLoaded) { |
| showError("Face detection models not loaded yet. Please wait."); |
| return; |
| } |
| |
| trackingActive = true; |
| startTime = Date.now(); |
| lastTime = Date.now(); |
| startBtn.innerHTML = '<i class="fas fa-pause"></i> Pause Tracking'; |
| startBtn.style.backgroundColor = '#ff7e5f'; |
| |
| |
| processVideo(); |
| } |
| |
| |
| function pauseTracking() { |
| trackingActive = false; |
| startBtn.innerHTML = '<i class="fas fa-play"></i> Resume Tracking'; |
| startBtn.style.backgroundColor = '#5d69b2'; |
| } |
| |
| |
| function resetTracking() { |
| pauseTracking(); |
| startTime = 0; |
| currentSpeed = 0; |
| fixations = 0; |
| gazeHistory = []; |
| |
| |
| eyeX.textContent = '0'; |
| eyeY.textContent = '0'; |
| movementSpeed.textContent = '0'; |
| timeTracked.textContent = '0'; |
| fixationCount.textContent = '0'; |
| |
| |
| gazeChart.data.labels = []; |
| gazeChart.data.datasets[0].data = []; |
| gazeChart.update(); |
| |
| |
| ctx.clearRect(0, 0, canvas.width, canvas.height); |
| } |
| |
| |
| startBtn.addEventListener('click', function() { |
| if (trackingActive) { |
| pauseTracking(); |
| } else { |
| startTracking(); |
| } |
| }); |
| |
| resetBtn.addEventListener('click', resetTracking); |
| |
| debugBtn.addEventListener('click', function() { |
| showDebug = !showDebug; |
| debugBtn.innerHTML = showDebug ? |
| '<i class="fas fa-eye-slash"></i> Hide Debug' : |
| '<i class="fas fa-eye"></i> Show Debug'; |
| }); |
| |
| |
| loadModels(); |
| }); |
| </script> |
| </body> |
| </html> |