Spaces:
Running
Running
<html lang="en"> | |
<head> | |
<meta charset="UTF-8"> | |
<meta name="viewport" content="width=device-width, initial-scale=1.0"> | |
<title>Smart Home Security - Face Recognition</title> | |
<script src="https://cdn.jsdelivr.net/npm/face-api.js"></script> | |
<style> | |
:root { | |
--primary: #2962ff; | |
--dark: #1a237e; | |
--light: #f5f5f5; | |
--success: #00c853; | |
--danger: #d50000; | |
} | |
* { | |
margin: 0; | |
padding: 0; | |
box-sizing: border-box; | |
font-family: 'Segoe UI', sans-serif; | |
} | |
body { | |
background: var(--light); | |
min-height: 100vh; | |
} | |
.container { | |
max-width: 1200px; | |
margin: 0 auto; | |
padding: 20px; | |
} | |
.header { | |
background: var(--dark); | |
color: white; | |
padding: 1rem; | |
border-radius: 10px; | |
margin-bottom: 2rem; | |
display: flex; | |
justify-content: space-between; | |
align-items: center; | |
} | |
.main-content { | |
display: grid; | |
grid-template-columns: 2fr 1fr; | |
gap: 2rem; | |
} | |
.video-container { | |
background: white; | |
padding: 20px; | |
border-radius: 10px; | |
box-shadow: 0 4px 6px rgba(0,0,0,0.1); | |
} | |
#video { | |
width: 100%; | |
border-radius: 5px; | |
} | |
.controls { | |
background: white; | |
padding: 20px; | |
border-radius: 10px; | |
box-shadow: 0 4px 6px rgba(0,0,0,0.1); | |
} | |
.status { | |
text-align: center; | |
padding: 1rem; | |
margin: 1rem 0; | |
border-radius: 5px; | |
font-weight: bold; | |
} | |
.status.authorized { | |
background: var(--success); | |
color: white; | |
} | |
.status.unauthorized { | |
background: var(--danger); | |
color: white; | |
} | |
.btn { | |
background: var(--primary); | |
color: white; | |
border: none; | |
padding: 10px 20px; | |
border-radius: 5px; | |
cursor: pointer; | |
transition: 0.3s; | |
width: 100%; | |
margin: 5px 0; | |
} | |
.btn:hover { | |
background: var(--dark); | |
} | |
.registered-faces { | |
margin-top: 2rem; | |
} | |
.face-item { | |
display: flex; | |
align-items: center; | |
padding: 10px; | |
background: #f8f9fa; | |
margin: 5px 0; | |
border-radius: 5px; | |
} | |
.face-item img { | |
width: 50px; | |
height: 50px; | |
border-radius: 50%; | |
margin-right: 10px; | |
} | |
.loading { | |
position: fixed; | |
top: 0; | |
left: 0; | |
right: 0; | |
bottom: 0; | |
background: rgba(0,0,0,0.8); | |
display: flex; | |
justify-content: center; | |
align-items: center; | |
color: white; | |
font-size: 1.2rem; | |
} | |
@keyframes spin { | |
to { transform: rotate(360deg); } | |
} | |
.spinner { | |
width: 40px; | |
height: 40px; | |
border: 4px solid #f3f3f3; | |
border-top: 4px solid var(--primary); | |
border-radius: 50%; | |
animation: spin 1s linear infinite; | |
margin-right: 10px; | |
} | |
</style> | |
</head> | |
<body> | |
<div class="container"> | |
<div class="header"> | |
<h1>🏠 Smart Home Security</h1> | |
<div>System Status: <span id="system-status">Active</span></div> | |
</div> | |
<div class="main-content"> | |
<div class="video-container"> | |
<video id="video" autoplay muted></video> | |
<div class="status" id="access-status">Waiting for face detection...</div> | |
</div> | |
<div class="controls"> | |
<h2>Control Panel</h2> | |
<button class="btn" id="add-face">Add New Face</button> | |
<button class="btn" id="test-detection">Test Detection</button> | |
<div class="registered-faces"> | |
<h3>Registered Users</h3> | |
<div id="faces-list"> | |
<!-- Registered faces will be listed here --> | |
</div> | |
</div> | |
</div> | |
</div> | |
</div> | |
<div class="loading" id="loading" style="display: none;"> | |
<div class="spinner"></div> | |
Loading face detection models... | |
</div> | |
<script> | |
// Mock database of registered faces | |
const registeredFaces = [ | |
{ id: 1, name: 'John Doe', image: '' }, | |
{ id: 2, name: 'Jane Smith', image: '' } | |
]; | |
// Initialize face detection | |
async function initFaceDetection() { | |
try { | |
document.getElementById('loading').style.display = 'flex'; | |
// Load face-api models (in real implementation) | |
// await faceapi.nets.tinyFaceDetector.loadFromUri('/models'); | |
// await faceapi.nets.faceLandmark68Net.loadFromUri('/models'); | |
// await faceapi.nets.faceRecognitionNet.loadFromUri('/models'); | |
// Access webcam | |
const video = document.getElementById('video'); | |
const stream = await navigator.mediaDevices.getUserMedia({ video: true }); | |
video.srcObject = stream; | |
document.getElementById('loading').style.display = 'none'; | |
} catch (error) { | |
console.error('Error initializing face detection:', error); | |
document.getElementById('loading').style.display = 'none'; | |
} | |
} | |
// Simulate face detection | |
function detectFace() { | |
const accessStatus = document.getElementById('access-status'); | |
const isAuthorized = Math.random() > 0.5; // Simulate detection | |
if (isAuthorized) { | |
accessStatus.textContent = 'Access Granted ✅'; | |
accessStatus.className = 'status authorized'; | |
} else { | |
accessStatus.textContent = 'Access Denied ❌'; | |
accessStatus.className = 'status unauthorized'; | |
} | |
} | |
// Display registered faces | |
function displayRegisteredFaces() { | |
const facesList = document.getElementById('faces-list'); | |
facesList.innerHTML = registeredFaces.map(face => ` | |
<div class="face-item"> | |
<img src="${face.image}" alt="${face.name}"> | |
<div> | |
<h4>${face.name}</h4> | |
<small>ID: ${face.id}</small> | |
</div> | |
</div> | |
`).join(''); | |
} | |
// Event listeners | |
document.getElementById('add-face').addEventListener('click', () => { | |
const name = prompt('Enter name for new face:'); | |
if (name) { | |
registeredFaces.push({ | |
id: registeredFaces.length + 1, | |
name, | |
image: '' | |
}); | |
displayRegisteredFaces(); | |
} | |
}); | |
document.getElementById('test-detection').addEventListener('click', detectFace); | |
// Initialize application | |
window.addEventListener('load', () => { | |
initFaceDetection(); | |
displayRegisteredFaces(); | |
}); | |
</script> | |
</body> | |
</html> |