Mise à jour du projet

This commit is contained in:
2026-04-02 22:30:40 +02:00
parent 98825db072
commit 085cf33114
6 changed files with 448 additions and 588 deletions

22
.env
View File

@@ -1,15 +1,25 @@
# Configuration de la base de données # ============================================================
# Smart Parking v2.0 — Variables d'environnement
# Copiez ce fichier en .env à la racine du projet
# ============================================================
# ── Base de données ──────────────────────────────────────────
DB_HOST=db DB_HOST=db
DB_PORT=3306 DB_PORT=3306
DB_USER=smartparking_user DB_USER=smartparking_user
DB_PASSWORD=smartparking_pass DB_PASSWORD=smartparking_pass
DB_NAME=smartparking DB_NAME=smartparking
# Configuration JWT # ── JWT ──────────────────────────────────────────────────────
JWT_SECRET=une_chaine_tres_longue_et_secrete_à_changer # ⚠️ Changez cette valeur ! Mettez une chaîne longue et aléatoire.
JWT_SECRET=une_chaine_tres_longue_et_secrete_changez_moi_absolument
# Port du serveur # ── Serveur ──────────────────────────────────────────────────
PORT=3000 PORT=3000
# Mode environnement
NODE_ENV=production NODE_ENV=production
# ── MQTT (Mosquitto) ─────────────────────────────────────────
# Si Mosquitto tourne dans Docker (service "mqtt") → mqtt
# Si Mosquitto est installé directement sur le Pi → localhost
MQTT_HOST=mqtt
MQTT_PORT=1883

View File

@@ -1,56 +1,50 @@
/** /**
* ============================================ * ============================================
* ADMIN.JS - Panel d'administration * ADMIN.JS - Panel d'administration
* Smart Parking - BTS CIEL IR * Smart Parking v3.0
* CORRIGÉ :
* - Suppression du graphique d'occupation
* - Historique affiche la date complète
* (jour + mois + année + heure + minute)
* ============================================ * ============================================
*/ */
// Initialisation
document.addEventListener('DOMContentLoaded', () => { document.addEventListener('DOMContentLoaded', () => {
console.log('⚙️ Initialisation du panel admin...'); console.log('⚙️ Initialisation du panel admin...');
// Vérifier si l'utilisateur est admin
if (!isAdmin()) return; if (!isAdmin()) return;
initAdminPanel(); initAdminPanel();
}); });
/**
* Vérifie si l'utilisateur est admin
*/
function isAdmin() { function isAdmin() {
const user = JSON.parse(localStorage.getItem('smart_parking_user') || 'null'); const user = JSON.parse(localStorage.getItem('smart_parking_user') || 'null');
return user && user.role === 'admin'; return user && user.role === 'admin';
} }
/**
* Initialise le panel admin
*/
function initAdminPanel() { function initAdminPanel() {
loadAdminStats(); loadAdminStats();
initPlacesControl(); initPlacesControl();
loadUsersTable(); loadUsersTable();
loadReservationsTable(); loadReservationsTable();
initOccupancyChart();
loadHistoryLog(); loadHistoryLog();
// Rafraîchissement périodique // Rafraîchissement périodique toutes les 10 secondes
setInterval(() => { setInterval(() => {
loadAdminStats(); loadAdminStats();
loadReservationsTable(); loadReservationsTable();
loadHistoryLog();
}, 10000); }, 10000);
} }
/** // ============================================
* Charge les statistiques admin // STATISTIQUES ADMIN
*/ // ============================================
function loadAdminStats() { function loadAdminStats() {
const users = JSON.parse(localStorage.getItem('smart_parking_users') || '[]'); const users = JSON.parse(localStorage.getItem('smart_parking_users') || '[]');
const reservations = JSON.parse(localStorage.getItem('smart_parking_reservations') || '[]'); const reservations = JSON.parse(localStorage.getItem('smart_parking_reservations') || '[]');
const spots = JSON.parse(localStorage.getItem('smart_parking_spots') || '[]'); const spots = JSON.parse(localStorage.getItem('smart_parking_spots') || '[]');
// Calculer les stats const totalUsers = users.length + 1;
const totalUsers = users.length + 1; // +1 pour l'admin par défaut
const totalReservations = reservations.length; const totalReservations = reservations.length;
const totalRevenue = reservations const totalRevenue = reservations
.filter(r => r.status === 'active' || r.status === 'completed') .filter(r => r.status === 'active' || r.status === 'completed')
@@ -62,33 +56,27 @@ function loadAdminStats() {
? Math.round(((occupied + reserved) / spots.length) * 100) ? Math.round(((occupied + reserved) / spots.length) * 100)
: 0; : 0;
// Mettre à jour l'affichage
document.getElementById('adminTotalUsers').textContent = totalUsers; document.getElementById('adminTotalUsers').textContent = totalUsers;
document.getElementById('adminTotalReservations').textContent = totalReservations; document.getElementById('adminTotalReservations').textContent = totalReservations;
document.getElementById('adminTotalRevenue').textContent = totalRevenue + '€'; document.getElementById('adminTotalRevenue').textContent = totalRevenue + '€';
document.getElementById('adminOccupancyRate').textContent = occupancyRate + '%'; document.getElementById('adminOccupancyRate').textContent = occupancyRate + '%';
} }
/** // ============================================
* Initialise le contrôle des places // GESTION DES PLACES
*/ // ============================================
function initPlacesControl() { function initPlacesControl() {
const spots = JSON.parse(localStorage.getItem('smart_parking_spots') || '[]'); const spots = JSON.parse(localStorage.getItem('smart_parking_spots') || '[]');
// Mettre à jour le champ du nombre de places
const spotsInput = document.getElementById('adminTotalSpots'); const spotsInput = document.getElementById('adminTotalSpots');
if (spotsInput) { if (spotsInput) spotsInput.value = spots.length || 10;
spotsInput.value = spots.length || 10;
}
// Bouton de mise à jour
document.getElementById('updateSpotsBtn')?.addEventListener('click', () => { document.getElementById('updateSpotsBtn')?.addEventListener('click', () => {
const newCount = parseInt(document.getElementById('adminTotalSpots').value); const newCount = parseInt(document.getElementById('adminTotalSpots').value);
if (newCount < 5 || newCount > 50) { if (newCount < 5 || newCount > 50) {
Dashboard.showToast('Le nombre de places doit être entre 5 et 50', 'error'); Dashboard.showToast('Le nombre de places doit être entre 5 et 50', 'error');
return; return;
} }
if (window.ParkingMap) { if (window.ParkingMap) {
window.ParkingMap.setTotalSpots(newCount); window.ParkingMap.setTotalSpots(newCount);
renderAdminPlacesList(); renderAdminPlacesList();
@@ -96,13 +84,9 @@ function initPlacesControl() {
} }
}); });
// Rendre la liste des places
renderAdminPlacesList(); renderAdminPlacesList();
} }
/**
* Rend la liste des places dans l'admin
*/
function renderAdminPlacesList() { function renderAdminPlacesList() {
const container = document.getElementById('adminPlacesList'); const container = document.getElementById('adminPlacesList');
if (!container) return; if (!container) return;
@@ -113,60 +97,42 @@ function renderAdminPlacesList() {
<div <div
class="admin-place-item ${spot.status}" class="admin-place-item ${spot.status}"
onclick="toggleSpotStatus(${spot.id})" onclick="toggleSpotStatus(${spot.id})"
title="Place ${spot.number} - Cliquez pour changer" title="Place ${spot.number} Cliquez pour changer"
> >
${spot.number} ${spot.number}
</div> </div>
`).join(''); `).join('');
} }
/**
* Change le statut d'une place (admin)
*/
function toggleSpotStatus(spotId) { function toggleSpotStatus(spotId) {
const spots = JSON.parse(localStorage.getItem('smart_parking_spots') || '[]'); const spots = JSON.parse(localStorage.getItem('smart_parking_spots') || '[]');
const spot = spots.find(s => s.id === spotId); const spot = spots.find(s => s.id === spotId);
if (!spot) return; if (!spot) return;
// Cycle: free -> occupied -> reserved -> free
const cycle = ['free', 'occupied', 'reserved']; const cycle = ['free', 'occupied', 'reserved'];
const currentIndex = cycle.indexOf(spot.status); const nextStatus = cycle[(cycle.indexOf(spot.status) + 1) % cycle.length];
const nextStatus = cycle[(currentIndex + 1) % cycle.length];
spot.status = nextStatus; spot.status = nextStatus;
spot.lastUpdate = new Date().toISOString(); spot.lastUpdate = new Date().toISOString();
localStorage.setItem('smart_parking_spots', JSON.stringify(spots)); localStorage.setItem('smart_parking_spots', JSON.stringify(spots));
// Rafraîchir
renderAdminPlacesList(); renderAdminPlacesList();
if (window.ParkingMap) { if (window.ParkingMap) window.ParkingMap.refresh();
window.ParkingMap.refresh();
}
loadAdminStats(); loadAdminStats();
Dashboard.showToast(`Place ${spot.number} - ${getStatusLabel(nextStatus)}`, 'success'); Dashboard.showToast(`Place ${spot.number} ${getStatusLabel(nextStatus)}`, 'success');
} }
/** // ============================================
* Charge le tableau des utilisateurs // TABLEAU UTILISATEURS
*/ // ============================================
function loadUsersTable() { function loadUsersTable() {
const tbody = document.getElementById('adminUsersTable'); const tbody = document.getElementById('adminUsersTable');
if (!tbody) return; if (!tbody) return;
const users = JSON.parse(localStorage.getItem('smart_parking_users') || '[]'); const users = JSON.parse(localStorage.getItem('smart_parking_users') || '[]');
// Ajouter l'admin par défaut
const allUsers = [ const allUsers = [
{ { id: 0, name: 'Administrateur', email: 'admin@smartparking.fr', phone: '01 23 45 67 89', role: 'admin' },
id: 0,
name: 'Administrateur',
email: 'admin@smartparking.fr',
phone: '01 23 45 67 89',
role: 'admin'
},
...users ...users
]; ];
@@ -184,9 +150,8 @@ function loadUsersTable() {
<td> <td>
<div class="table-actions"> <div class="table-actions">
${user.role !== 'admin' ? ` ${user.role !== 'admin' ? `
<button class="btn btn-danger btn-small btn-icon-only" onclick="deleteUser(${user.id})" title="Supprimer"> <button class="btn btn-danger btn-small btn-icon-only"
🗑️ onclick="deleteUser(${user.id})" title="Supprimer">🗑️</button>
</button>
` : '-'} ` : '-'}
</div> </div>
</td> </td>
@@ -194,9 +159,6 @@ function loadUsersTable() {
`).join(''); `).join('');
} }
/**
* Supprime un utilisateur
*/
function deleteUser(userId) { function deleteUser(userId) {
if (!confirm('Êtes-vous sûr de vouloir supprimer cet utilisateur ?')) return; if (!confirm('Êtes-vous sûr de vouloir supprimer cet utilisateur ?')) return;
@@ -204,7 +166,6 @@ function deleteUser(userId) {
users = users.filter(u => u.id !== userId); users = users.filter(u => u.id !== userId);
localStorage.setItem('smart_parking_users', JSON.stringify(users)); localStorage.setItem('smart_parking_users', JSON.stringify(users));
// Supprimer aussi ses réservations
let reservations = JSON.parse(localStorage.getItem('smart_parking_reservations') || '[]'); let reservations = JSON.parse(localStorage.getItem('smart_parking_reservations') || '[]');
reservations = reservations.filter(r => r.userId !== userId); reservations = reservations.filter(r => r.userId !== userId);
localStorage.setItem('smart_parking_reservations', JSON.stringify(reservations)); localStorage.setItem('smart_parking_reservations', JSON.stringify(reservations));
@@ -212,13 +173,13 @@ function deleteUser(userId) {
loadUsersTable(); loadUsersTable();
loadReservationsTable(); loadReservationsTable();
loadAdminStats(); loadAdminStats();
Dashboard.showToast('Utilisateur supprimé', 'success'); Dashboard.showToast('Utilisateur supprimé', 'success');
} }
/** // ============================================
* Charge le tableau des réservations // TABLEAU RÉSERVATIONS
*/ // ============================================
function loadReservationsTable() { function loadReservationsTable() {
const tbody = document.getElementById('adminReservationsTable'); const tbody = document.getElementById('adminReservationsTable');
if (!tbody) return; if (!tbody) return;
@@ -226,7 +187,7 @@ function loadReservationsTable() {
const reservations = JSON.parse(localStorage.getItem('smart_parking_reservations') || '[]'); const reservations = JSON.parse(localStorage.getItem('smart_parking_reservations') || '[]');
if (reservations.length === 0) { if (reservations.length === 0) {
tbody.innerHTML = '<tr><td colspan="8" style="text-align: center; color: var(--text-muted);">Aucune réservation</td></tr>'; tbody.innerHTML = '<tr><td colspan="8" style="text-align:center;color:var(--text-muted)">Aucune réservation</td></tr>';
return; return;
} }
@@ -235,7 +196,7 @@ function loadReservationsTable() {
<td>#${res.id}</td> <td>#${res.id}</td>
<td>${res.userName}</td> <td>${res.userName}</td>
<td>Place ${res.spotNumber}</td> <td>Place ${res.spotNumber}</td>
<td>${formatDate(res.date)}</td> <td>${formatDateShort(res.date)}</td>
<td>${res.startTime} - ${res.endTime}</td> <td>${res.startTime} - ${res.endTime}</td>
<td>${res.price}€</td> <td>${res.price}€</td>
<td> <td>
@@ -246,12 +207,10 @@ function loadReservationsTable() {
<td> <td>
<div class="table-actions"> <div class="table-actions">
${res.status === 'active' ? ` ${res.status === 'active' ? `
<button class="btn btn-success btn-small btn-icon-only" onclick="completeReservation(${res.id})" title="Terminer"> <button class="btn btn-success btn-small btn-icon-only"
onclick="completeReservation(${res.id})" title="Terminer">✓</button>
</button> <button class="btn btn-danger btn-small btn-icon-only"
<button class="btn btn-danger btn-small btn-icon-only" onclick="adminCancelReservation(${res.id})" title="Annuler"> onclick="adminCancelReservation(${res.id})" title="Annuler">✕</button>
</button>
` : '-'} ` : '-'}
</div> </div>
</td> </td>
@@ -259,113 +218,42 @@ function loadReservationsTable() {
`).join(''); `).join('');
} }
/**
* Termine une réservation
*/
function completeReservation(reservationId) { function completeReservation(reservationId) {
let reservations = JSON.parse(localStorage.getItem('smart_parking_reservations') || '[]'); let reservations = JSON.parse(localStorage.getItem('smart_parking_reservations') || '[]');
const reservation = reservations.find(r => r.id === reservationId); const reservation = reservations.find(r => r.id === reservationId);
if (!reservation) return;
if (reservation) {
reservation.status = 'completed'; reservation.status = 'completed';
localStorage.setItem('smart_parking_reservations', JSON.stringify(reservations)); localStorage.setItem('smart_parking_reservations', JSON.stringify(reservations));
// Libérer la place if (window.ParkingMap) window.ParkingMap.setSpotStatus(reservation.spotId, 'free');
if (window.ParkingMap) {
window.ParkingMap.setSpotStatus(reservation.spotId, 'free');
}
loadReservationsTable(); loadReservationsTable();
loadAdminStats(); loadAdminStats();
Dashboard.showToast('Réservation terminée', 'success'); Dashboard.showToast('Réservation terminée', 'success');
}
} }
/**
* Annule une réservation (admin)
*/
function adminCancelReservation(reservationId) { function adminCancelReservation(reservationId) {
if (!confirm('Êtes-vous sûr de vouloir annuler cette réservation ?')) return; if (!confirm('Êtes-vous sûr de vouloir annuler cette réservation ?')) return;
let reservations = JSON.parse(localStorage.getItem('smart_parking_reservations') || '[]'); let reservations = JSON.parse(localStorage.getItem('smart_parking_reservations') || '[]');
const reservation = reservations.find(r => r.id === reservationId); const reservation = reservations.find(r => r.id === reservationId);
if (!reservation) return;
if (reservation) {
reservation.status = 'cancelled'; reservation.status = 'cancelled';
localStorage.setItem('smart_parking_reservations', JSON.stringify(reservations)); localStorage.setItem('smart_parking_reservations', JSON.stringify(reservations));
// Libérer la place if (window.ParkingMap) window.ParkingMap.setSpotStatus(reservation.spotId, 'free');
if (window.ParkingMap) {
window.ParkingMap.setSpotStatus(reservation.spotId, 'free');
}
loadReservationsTable(); loadReservationsTable();
loadAdminStats(); loadAdminStats();
Dashboard.showToast('Réservation annulée', 'success'); Dashboard.showToast('Réservation annulée', 'success');
}
} }
/** // ============================================
* Initialise le graphique d'occupation // HISTORIQUE — CORRIGÉ : date complète
*/ // ============================================
function initOccupancyChart() {
const ctx = document.getElementById('adminOccupancyChart');
if (!ctx) return;
// Générer des données d'exemple
const labels = [];
const data = [];
for (let i = 6; i >= 0; i--) {
const date = new Date();
date.setDate(date.getDate() - i);
labels.push(date.toLocaleDateString('fr-FR', { weekday: 'short' }));
data.push(Math.floor(Math.random() * 40) + 30); // 30-70% d'occupation
}
new Chart(ctx, {
type: 'bar',
data: {
labels: labels,
datasets: [{
label: 'Taux d\'occupation (%)',
data: data,
backgroundColor: 'rgba(99, 102, 241, 0.5)',
borderColor: 'rgba(99, 102, 241, 1)',
borderWidth: 1,
borderRadius: 4
}]
},
options: {
responsive: true,
maintainAspectRatio: false,
plugins: {
legend: {
labels: { color: '#f1f5f9' }
}
},
scales: {
x: {
grid: { color: '#334155' },
ticks: { color: '#94a3b8' }
},
y: {
min: 0,
max: 100,
grid: { color: '#334155' },
ticks: {
color: '#94a3b8',
callback: value => value + '%'
}
}
}
}
});
}
/**
* Charge l'historique
*/
function loadHistoryLog() { function loadHistoryLog() {
const container = document.getElementById('adminLogContainer'); const container = document.getElementById('adminLogContainer');
if (!container) return; if (!container) return;
@@ -373,54 +261,61 @@ function loadHistoryLog() {
const history = JSON.parse(localStorage.getItem('smart_parking_history') || '[]'); const history = JSON.parse(localStorage.getItem('smart_parking_history') || '[]');
if (history.length === 0) { if (history.length === 0) {
container.innerHTML = '<p style="color: var(--text-muted); text-align: center;">Aucun historique</p>'; container.innerHTML = '<p style="color:var(--text-muted);text-align:center">Aucun historique</p>';
return; return;
} }
container.innerHTML = history.slice(0, 20).map(item => ` container.innerHTML = history.slice(0, 50).map(item => `
<div class="log-item"> <div class="log-item">
<span class="log-time">${formatTime(item.timestamp)}</span> <span class="log-time">${formatDateComplete(item.timestamp)}</span>
<span><strong>${item.action}:</strong> ${item.details}</span> <span><strong>${item.action} :</strong> ${item.details}</span>
</div> </div>
`).join(''); `).join('');
} }
// ============================================
// FONCTIONS DE FORMAT DATE
// ============================================
/** /**
* Formate une date * CORRIGÉ — Affiche la date complète dans l'historique
* Avant : seulement "14:32"
* Après : "12/06/2025 à 14:32"
*/ */
function formatDate(dateString) { function formatDateComplete(dateString) {
if (!dateString) return 'N/A';
const date = new Date(dateString); const date = new Date(dateString);
return date.toLocaleDateString('fr-FR', { const jour = String(date.getDate()).padStart(2, '0');
day: '2-digit', const mois = String(date.getMonth() + 1).padStart(2, '0');
month: '2-digit' const annee = date.getFullYear();
const heure = String(date.getHours()).padStart(2, '0');
const min = String(date.getMinutes()).padStart(2, '0');
return `${jour}/${mois}/${annee} à ${heure}:${min}`;
}
function formatDateShort(dateString) {
if (!dateString) return 'N/A';
return new Date(dateString).toLocaleDateString('fr-FR', {
day: '2-digit', month: '2-digit'
}); });
} }
/**
* Formate une heure
*/
function formatTime(dateString) {
const date = new Date(dateString);
return date.toLocaleTimeString('fr-FR', {
hour: '2-digit',
minute: '2-digit'
});
}
/**
* Retourne le label du statut
*/
function getStatusLabel(status) { function getStatusLabel(status) {
const labels = { return {
'pending': 'En attente', pending: 'En attente',
'active': 'Active', active: 'Active',
'completed': 'Terminée', completed: 'Terminée',
'cancelled': 'Annulée' cancelled: 'Annulée',
}; free: 'Libre',
return labels[status] || status; occupied: 'Occupée',
reserved: 'Réservée'
}[status] || status;
} }
// Exporter les fonctions // ============================================
// EXPORT
// ============================================
window.AdminModule = { window.AdminModule = {
refresh: () => { refresh: () => {
loadAdminStats(); loadAdminStats();

View File

@@ -1,22 +1,21 @@
/** /**
* ============================================ * ============================================
* RESERVATION.JS - Système de réservation * RESERVATION.JS - Système de réservation
* Smart Parking - BTS CIEL IR * Smart Parking v3.0
* MODIFIÉ : Suppression du QR code, remplacement * CORRIGÉ : ne bloque plus une place pour
* par une confirmation simple avec badge * toujours — vérifie uniquement si
* une voiture est physiquement là
* ============================================ * ============================================
*/ */
// Tarifs
const PRICING = { const PRICING = {
30: 2, // 30 min = 2€ 30: 2,
60: 3, // 1h = 3€ 60: 3,
120: 5, // 2h = 5€ 120: 5,
240: 8, // 4h = 8€ 240: 8,
480: 15 // 8h (journée) = 15€ 480: 15
}; };
// Horaires disponibles
const TIME_SLOTS = [ const TIME_SLOTS = [
'06:00', '06:30', '07:00', '07:30', '08:00', '08:30', '09:00', '09:30', '06:00', '06:30', '07:00', '07:30', '08:00', '08:30', '09:00', '09:30',
'10:00', '10:30', '11:00', '11:30', '12:00', '12:30', '13:00', '13:30', '10:00', '10:30', '11:00', '11:30', '12:00', '12:30', '13:00', '13:30',
@@ -25,31 +24,23 @@ const TIME_SLOTS = [
'22:00' '22:00'
]; ];
// État de la réservation en cours
let currentReservation = null; let currentReservation = null;
// Initialisation
document.addEventListener('DOMContentLoaded', () => { document.addEventListener('DOMContentLoaded', () => {
console.log('📅 Initialisation du système de réservation...'); console.log('📅 Initialisation du système de réservation...');
initReservationForm(); initReservationForm();
initDatePicker(); initDatePicker();
initTimeSlots(); initTimeSlots();
initPricePreview(); initPricePreview();
initConfirmationModal(); // MODIFIÉ : était initPaymentModal() initConfirmationModal();
}); });
/**
* Initialise le formulaire de réservation
*/
function initReservationForm() { function initReservationForm() {
const form = document.getElementById('reservationForm'); const form = document.getElementById('reservationForm');
if (!form) return; if (!form) return;
form.addEventListener('submit', handleReservationSubmit); form.addEventListener('submit', handleReservationSubmit);
} }
/**
* Initialise le sélecteur de date
*/
function initDatePicker() { function initDatePicker() {
const dateInput = document.getElementById('resDate'); const dateInput = document.getElementById('resDate');
if (!dateInput) return; if (!dateInput) return;
@@ -58,9 +49,6 @@ function initDatePicker() {
dateInput.value = today; dateInput.value = today;
} }
/**
* Initialise les créneaux horaires
*/
function initTimeSlots() { function initTimeSlots() {
const select = document.getElementById('resStartTime'); const select = document.getElementById('resStartTime');
if (!select) return; if (!select) return;
@@ -72,20 +60,16 @@ function initTimeSlots() {
select.appendChild(option); select.appendChild(option);
}); });
// Sélectionner le prochain créneau disponible
const now = new Date(); const now = new Date();
const currentHour = now.getHours(); const currentHour = now.getHours();
const currentMinutes = now.getMinutes(); const currentMins = now.getMinutes();
const nextSlot = TIME_SLOTS.find(t => { const nextSlot = TIME_SLOTS.find(t => {
const [h, m] = t.split(':').map(Number); const [h, m] = t.split(':').map(Number);
return h > currentHour || (h === currentHour && m > currentMinutes); return h > currentHour || (h === currentHour && m > currentMins);
}); });
if (nextSlot) select.value = nextSlot; if (nextSlot) select.value = nextSlot;
} }
/**
* Initialise la prévisualisation du prix
*/
function initPricePreview() { function initPricePreview() {
const durationSelect = document.getElementById('resDuration'); const durationSelect = document.getElementById('resDuration');
if (!durationSelect) return; if (!durationSelect) return;
@@ -93,9 +77,6 @@ function initPricePreview() {
updatePricePreview(); updatePricePreview();
} }
/**
* Met à jour la prévisualisation du prix
*/
function updatePricePreview() { function updatePricePreview() {
const duration = parseInt(document.getElementById('resDuration').value); const duration = parseInt(document.getElementById('resDuration').value);
const price = PRICING[duration] || 0; const price = PRICING[duration] || 0;
@@ -104,10 +85,13 @@ function updatePricePreview() {
/** /**
* Gère la soumission du formulaire * Gère la soumission du formulaire
* MODIFIÉ : la réservation est enregistrée ici directement, *
* puis le modal de confirmation s'affiche. * CORRIGÉ : on n'empêche plus la réservation si la place
* est "reserved" dans le localStorage. On laisse le serveur
* vérifier les conflits d'horaire. On bloque uniquement si
* une voiture est physiquement détectée (status "occupied").
*/ */
function handleReservationSubmit(e) { async function handleReservationSubmit(e) {
e.preventDefault(); e.preventDefault();
const user = JSON.parse(localStorage.getItem('smart_parking_user') || 'null'); const user = JSON.parse(localStorage.getItem('smart_parking_user') || 'null');
@@ -127,12 +111,13 @@ function handleReservationSubmit(e) {
return; return;
} }
// Vérifier que la place est toujours libre // CORRIGÉ : on bloque uniquement si une voiture est physiquement là
// Une place "reserved" peut quand même être réservée à un autre horaire
const spots = JSON.parse(localStorage.getItem('smart_parking_spots') || '[]'); const spots = JSON.parse(localStorage.getItem('smart_parking_spots') || '[]');
const spot = spots.find(s => s.id === spotId); const spot = spots.find(s => s.id === spotId);
if (!spot || spot.status !== 'free') { if (spot && spot.status === 'occupied') {
Dashboard.showToast('Cette place n\'est plus disponible', 'error'); Dashboard.showToast('Une voiture est déjà sur cette place', 'error');
if (window.ParkingMap) window.ParkingMap.refresh(); if (window.ParkingMap) window.ParkingMap.refresh();
return; return;
} }
@@ -142,39 +127,78 @@ function handleReservationSubmit(e) {
endDate.setMinutes(endDate.getMinutes() + duration); endDate.setMinutes(endDate.getMinutes() + duration);
const endTime = endDate.toTimeString().slice(0, 5); const endTime = endDate.toTimeString().slice(0, 5);
// Construire l'objet réservation // Essayer de créer la réservation via l'API
// Le serveur vérifiera les conflits d'horaire
const token = localStorage.getItem('smart_parking_token');
if (token) {
try {
const response = await fetch('/api/reservations', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
'Authorization': 'Bearer ' + token
},
body: JSON.stringify({
spotId, date, startTime, endTime,
duration, vehicle: vehicle.toUpperCase(),
price: PRICING[duration]
})
});
const data = await response.json();
if (!data.success) {
// Le serveur a détecté un conflit ou une erreur
Dashboard.showToast(data.message, 'error');
return;
}
// Succès via API
currentReservation = { currentReservation = {
id: Date.now(), id: data.data.id,
userId: user.id, userId: user.id,
userName: user.name, userName: user.name,
spotId: spotId, spotId: spotId,
spotNumber: spot.number, spotNumber: spot ? spot.number : spotId,
date: date, date, startTime, endTime, duration,
startTime: startTime,
endTime: endTime,
duration: duration,
vehicle: vehicle.toUpperCase(), vehicle: vehicle.toUpperCase(),
price: PRICING[duration], price: PRICING[duration],
status: 'active', status: 'active',
createdAt: new Date().toISOString() createdAt: new Date().toISOString()
}; };
// ---- Enregistrement immédiat (plus de bouton "J'ai payé") ---- } catch (_err) {
// Mode offline : enregistrement local
currentReservation = creerReservationLocale(
user, spotId, spot, date, startTime, endTime, duration, vehicle
);
}
} else {
// Pas de token : mode offline
currentReservation = creerReservationLocale(
user, spotId, spot, date, startTime, endTime, duration, vehicle
);
}
// Sauvegarder la réservation // Sauvegarder dans le localStorage
let reservations = JSON.parse(localStorage.getItem('smart_parking_reservations') || '[]'); let reservations = JSON.parse(localStorage.getItem('smart_parking_reservations') || '[]');
reservations.push(currentReservation); reservations.push(currentReservation);
localStorage.setItem('smart_parking_reservations', JSON.stringify(reservations)); localStorage.setItem('smart_parking_reservations', JSON.stringify(reservations));
// Mettre la place en "réservée" sur la carte // Mettre à jour la carte seulement si la réservation est pour aujourd'hui
if (window.ParkingMap) { const today = new Date().toISOString().split('T')[0];
window.ParkingMap.setSpotStatus(currentReservation.spotId, 'reserved'); const now = new Date();
const resStart = new Date(date + 'T' + startTime);
const diffMin = (resStart - now) / 60000;
if (date === today && diffMin <= 30 && window.ParkingMap) {
window.ParkingMap.setSpotStatus(spotId, 'reserved');
} }
// Ajouter à l'historique admin
addToHistory( addToHistory(
'Réservation', 'Réservation',
`Place ${currentReservation.spotNumber} réservée par ${currentReservation.userName} - ${currentReservation.price}` `Place ${currentReservation.spotNumber} réservée le ${date} de ${startTime} à ${endTime}${PRICING[duration]}`
); );
// Réinitialiser le formulaire // Réinitialiser le formulaire
@@ -182,95 +206,84 @@ function handleReservationSubmit(e) {
initDatePicker(); initDatePicker();
updatePricePreview(); updatePricePreview();
// Afficher le modal de confirmation
showConfirmationModal(); showConfirmationModal();
} }
/**
* Crée une réservation en mode offline (localStorage)
*/
function creerReservationLocale(user, spotId, spot, date, startTime, endTime, duration, vehicle) {
return {
id: Date.now(),
userId: user.id,
userName: user.name,
spotId: spotId,
spotNumber: spot ? spot.number : spotId,
date, startTime, endTime, duration,
vehicle: vehicle.toUpperCase(),
price: PRICING[duration],
status: 'active',
createdAt: new Date().toISOString()
};
}
// ============================================ // ============================================
// MODAL DE CONFIRMATION (remplace le QR code) // MODAL DE CONFIRMATION
// ============================================ // ============================================
/**
* Initialise les événements du modal de confirmation
* REMPLACE : initPaymentModal()
*/
function initConfirmationModal() { function initConfirmationModal() {
document.getElementById('closeConfirmationModal')?.addEventListener('click', hideConfirmationModal); document.getElementById('closeConfirmationModal')?.addEventListener('click', hideConfirmationModal);
document.getElementById('closeConfirmationBtn')?.addEventListener('click', hideConfirmationModal); document.getElementById('closeConfirmationBtn')?.addEventListener('click', hideConfirmationModal);
} }
/**
* Affiche le modal de confirmation
* REMPLACE : showPaymentModal() + generateQRCode()
*/
function showConfirmationModal() { function showConfirmationModal() {
if (!currentReservation) return; if (!currentReservation) return;
const modal = document.getElementById('confirmationModal'); const modal = document.getElementById('confirmationModal');
// Remplir le récapitulatif
document.getElementById('paySpot').textContent = 'Place ' + currentReservation.spotNumber; document.getElementById('paySpot').textContent = 'Place ' + currentReservation.spotNumber;
document.getElementById('payDate').textContent = formatDate(currentReservation.date); document.getElementById('payDate').textContent = formatDate(currentReservation.date);
document.getElementById('payTime').textContent = currentReservation.startTime + ' - ' + currentReservation.endTime; document.getElementById('payTime').textContent = currentReservation.startTime + ' - ' + currentReservation.endTime;
document.getElementById('payDuration').textContent = formatDuration(currentReservation.duration); document.getElementById('payDuration').textContent = formatDuration(currentReservation.duration);
document.getElementById('payTotal').textContent = currentReservation.price + '€'; document.getElementById('payTotal').textContent = currentReservation.price + '€';
// Afficher le modal
modal.classList.remove('hidden'); modal.classList.remove('hidden');
} }
/**
* Cache le modal de confirmation
* REMPLACE : hidePaymentModal()
*/
function hideConfirmationModal() { function hideConfirmationModal() {
document.getElementById('confirmationModal').classList.add('hidden'); document.getElementById('confirmationModal').classList.add('hidden');
// Rediriger vers "Mes réservations" après fermeture
if (window.Dashboard) { if (window.Dashboard) {
Dashboard.navigateToPage('my-reservations'); Dashboard.navigateToPage('my-reservations');
document.querySelector('[data-page="my-reservations"]')?.classList.add('active'); document.querySelector('[data-page="my-reservations"]')?.classList.add('active');
document.querySelector('[data-page="reservation"]')?.classList.remove('active'); document.querySelector('[data-page="reservation"]')?.classList.remove('active');
} }
// Rafraîchir les stats du profil
if (window.Dashboard) Dashboard.refreshStats(); if (window.Dashboard) Dashboard.refreshStats();
} }
// ============================================ // ============================================
// FONCTIONS UTILITAIRES // UTILITAIRES
// ============================================ // ============================================
/**
* Ajoute une entrée à l'historique
*/
function addToHistory(action, details) { function addToHistory(action, details) {
let history = JSON.parse(localStorage.getItem('smart_parking_history') || '[]'); let history = JSON.parse(localStorage.getItem('smart_parking_history') || '[]');
history.unshift({ history.unshift({
id: Date.now(), id: Date.now(),
action: action, action,
details: details, details,
timestamp: new Date().toISOString() timestamp: new Date().toISOString()
}); });
if (history.length > 100) history = history.slice(0, 100); if (history.length > 100) history = history.slice(0, 100);
localStorage.setItem('smart_parking_history', JSON.stringify(history)); localStorage.setItem('smart_parking_history', JSON.stringify(history));
} }
/**
* Formate une date DD/MM/YYYY
*/
function formatDate(dateString) { function formatDate(dateString) {
const date = new Date(dateString); return new Date(dateString).toLocaleDateString('fr-FR', {
return date.toLocaleDateString('fr-FR', { day: '2-digit', month: '2-digit', year: 'numeric'
day: '2-digit',
month: '2-digit',
year: 'numeric'
}); });
} }
/**
* Formate une durée en minutes vers texte lisible
*/
function formatDuration(minutes) { function formatDuration(minutes) {
if (minutes >= 480) return 'Journée (8h)'; if (minutes >= 480) return 'Journée (8h)';
if (minutes >= 60) { if (minutes >= 60) {
@@ -281,10 +294,4 @@ function formatDuration(minutes) {
return `${minutes} min`; return `${minutes} min`;
} }
// Exporter les fonctions publiques window.Reservation = { PRICING, TIME_SLOTS, formatDuration, addToHistory };
window.Reservation = {
PRICING,
TIME_SLOTS,
formatDuration,
addToHistory
};

View File

@@ -6,8 +6,7 @@
<title>Smart Parking - Dashboard</title> <title>Smart Parking - Dashboard</title>
<link rel="stylesheet" href="../css/style.css"> <link rel="stylesheet" href="../css/style.css">
<link rel="stylesheet" href="../css/dashboard.css"> <link rel="stylesheet" href="../css/dashboard.css">
<!-- SUPPRIMÉ : qrcode.min.js (plus nécessaire) --> <!-- Chart.js retiré car le graphique d'occupation a été supprimé -->
<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
</head> </head>
<body> <body>
<!-- Header --> <!-- Header -->
@@ -21,25 +20,19 @@
</div> </div>
<nav class="nav"> <nav class="nav">
<a href="#map" class="nav-link active" data-page="map"> <a href="#map" class="nav-link active" data-page="map">
<span class="nav-icon">🗺️</span> <span class="nav-icon">🗺️</span>Carte
Carte
</a> </a>
<a href="#reservation" class="nav-link" data-page="reservation"> <a href="#reservation" class="nav-link" data-page="reservation">
<span class="nav-icon">📅</span> <span class="nav-icon">📅</span>Réservation
Réservation
</a> </a>
<a href="#my-reservations" class="nav-link" data-page="my-reservations"> <a href="#my-reservations" class="nav-link" data-page="my-reservations">
<span class="nav-icon">🎫</span> <span class="nav-icon">🎫</span>Mes réservations
Mes réservations
</a> </a>
<a href="#profile" class="nav-link" data-page="profile"> <a href="#profile" class="nav-link" data-page="profile">
<span class="nav-icon">👤</span> <span class="nav-icon">👤</span>Profil
Profil
</a> </a>
<!-- Lien admin (visible uniquement pour admin) -->
<a href="#admin" class="nav-link admin-only hidden" data-page="admin"> <a href="#admin" class="nav-link admin-only hidden" data-page="admin">
<span class="nav-icon">⚙️</span> <span class="nav-icon">⚙️</span>Admin
Admin
</a> </a>
</nav> </nav>
<div class="header-right"> <div class="header-right">
@@ -48,24 +41,19 @@
<span id="userRole" class="user-role">Client</span> <span id="userRole" class="user-role">Client</span>
</div> </div>
<button id="logoutBtn" class="btn btn-secondary btn-small"> <button id="logoutBtn" class="btn btn-secondary btn-small">
<span class="btn-icon">🚪</span> <span class="btn-icon">🚪</span>Déconnexion
Déconnexion
</button> </button>
</div> </div>
</div> </div>
</header> </header>
<!-- Main Content -->
<main class="main"> <main class="main">
<!-- PAGE: CARTE DES PLACES -->
<!-- ═══ PAGE : CARTE DES PLACES ═══ -->
<section id="map" class="page active"> <section id="map" class="page active">
<div class="container"> <div class="container">
<h2 class="page-title"> <h2 class="page-title"><span class="icon">🗺️</span>Carte du Parking</h2>
<span class="icon">🗺️</span>
Carte du Parking
</h2>
<!-- Statistiques des places -->
<div class="stats-grid"> <div class="stats-grid">
<div class="stat-card free"> <div class="stat-card free">
<div class="stat-icon"></div> <div class="stat-icon"></div>
@@ -97,30 +85,16 @@
</div> </div>
</div> </div>
<!-- Carte des places -->
<div class="parking-section"> <div class="parking-section">
<div class="parking-map-container"> <div class="parking-map-container">
<h3>Vue du parking</h3> <h3>Vue du parking</h3>
<div class="parking-map" id="parkingMap"> <div class="parking-map" id="parkingMap"></div>
<!-- Les places seront générées par JS -->
</div>
<div class="legend"> <div class="legend">
<div class="legend-item"> <div class="legend-item"><span class="legend-color free"></span><span>Libre</span></div>
<span class="legend-color free"></span> <div class="legend-item"><span class="legend-color occupied"></span><span>Occupée</span></div>
<span>Libre</span> <div class="legend-item"><span class="legend-color reserved"></span><span>Réservée</span></div>
</div>
<div class="legend-item">
<span class="legend-color occupied"></span>
<span>Occupée</span>
</div>
<div class="legend-item">
<span class="legend-color reserved"></span>
<span>Réservée</span>
</div> </div>
</div> </div>
</div>
<!-- Détails de la place sélectionnée -->
<div class="spot-details-container"> <div class="spot-details-container">
<h3>Détails de la place</h3> <h3>Détails de la place</h3>
<div id="spotDetails" class="spot-details"> <div id="spotDetails" class="spot-details">
@@ -129,42 +103,23 @@
</div> </div>
</div> </div>
<!-- Tarifs -->
<div class="pricing-section"> <div class="pricing-section">
<h3>💰 Nos tarifs</h3> <h3>💰 Nos tarifs</h3>
<div class="pricing-cards"> <div class="pricing-cards">
<div class="pricing-card"> <div class="pricing-card"><h4>30 minutes</h4><span class="price">2€</span></div>
<h4>30 minutes</h4> <div class="pricing-card"><h4>1 heure</h4><span class="price">3€</span></div>
<span class="price">2</span> <div class="pricing-card"><h4>2 heures</h4><span class="price">5</span></div>
</div> <div class="pricing-card"><h4>4 heures</h4><span class="price">8€</span></div>
<div class="pricing-card"> <div class="pricing-card"><h4>Journée</h4><span class="price">15€</span></div>
<h4>1 heure</h4>
<span class="price">3€</span>
</div>
<div class="pricing-card">
<h4>2 heures</h4>
<span class="price">5€</span>
</div>
<div class="pricing-card">
<h4>4 heures</h4>
<span class="price">8€</span>
</div>
<div class="pricing-card">
<h4>Journée</h4>
<span class="price">15€</span>
</div>
</div> </div>
</div> </div>
</div> </div>
</section> </section>
<!-- PAGE: RÉSERVATION --> <!-- ═══ PAGE : RÉSERVATION ═══ -->
<section id="reservation" class="page hidden"> <section id="reservation" class="page hidden">
<div class="container"> <div class="container">
<h2 class="page-title"> <h2 class="page-title"><span class="icon">📅</span>Réserver une place</h2>
<span class="icon">📅</span>
Réserver une place
</h2>
<div class="reservation-form-container"> <div class="reservation-form-container">
<form id="reservationForm" class="reservation-form"> <form id="reservationForm" class="reservation-form">
@@ -180,7 +135,6 @@
<input type="date" id="resDate" class="form-control" required> <input type="date" id="resDate" class="form-control" required>
</div> </div>
</div> </div>
<div class="form-row"> <div class="form-row">
<div class="form-group"> <div class="form-group">
<label for="resStartTime">Heure d'arrivée</label> <label for="resStartTime">Heure d'arrivée</label>
@@ -191,34 +145,29 @@
<div class="form-group"> <div class="form-group">
<label for="resDuration">Durée</label> <label for="resDuration">Durée</label>
<select id="resDuration" class="form-control" required> <select id="resDuration" class="form-control" required>
<option value="30">30 min - 2€</option> <option value="30">30 min 2€</option>
<option value="60">1h - 3€</option> <option value="60">1h 3€</option>
<option value="120" selected>2h - 5€</option> <option value="120" selected>2h 5€</option>
<option value="240">4h - 8€</option> <option value="240">4h 8€</option>
<option value="480">Journée (8h) - 15€</option> <option value="480">Journée (8h) 15€</option>
</select> </select>
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="resVehicle">Plaque d'immatriculation</label> <label for="resVehicle">Plaque d'immatriculation</label>
<input type="text" id="resVehicle" class="form-control" placeholder="AB-123-CD" required> <input type="text" id="resVehicle" class="form-control" placeholder="AB-123-CD" required>
</div> </div>
<div class="price-preview"> <div class="price-preview">
<span>Prix total :</span> <span>Prix total :</span>
<span id="previewPrice" class="price-amount">5€</span> <span id="previewPrice" class="price-amount">5€</span>
</div> </div>
<!-- MODIFIÉ : bouton "Valider la réservation" au lieu de "Procéder au paiement" -->
<button type="submit" class="btn btn-primary btn-block"> <button type="submit" class="btn btn-primary btn-block">
<span class="btn-icon"></span> <span class="btn-icon"></span>Valider la réservation
Valider la réservation
</button> </button>
</form> </form>
</div> </div>
<!-- REMPLACÉ : Modal de confirmation (sans QR code) --> <!-- Modal de confirmation -->
<div id="confirmationModal" class="modal hidden"> <div id="confirmationModal" class="modal hidden">
<div class="modal-content"> <div class="modal-content">
<div class="modal-header"> <div class="modal-header">
@@ -226,43 +175,22 @@
<button class="modal-close" id="closeConfirmationModal">&times;</button> <button class="modal-close" id="closeConfirmationModal">&times;</button>
</div> </div>
<div class="modal-body"> <div class="modal-body">
<!-- Récapitulatif -->
<div class="payment-summary"> <div class="payment-summary">
<h4>Récapitulatif</h4> <h4>Récapitulatif</h4>
<div class="summary-row"> <div class="summary-row"><span>Place :</span><span id="paySpot">-</span></div>
<span>Place :</span> <div class="summary-row"><span>Date :</span><span id="payDate">-</span></div>
<span id="paySpot">-</span> <div class="summary-row"><span>Heure :</span><span id="payTime">-</span></div>
<div class="summary-row"><span>Durée :</span><span id="payDuration">-</span></div>
<div class="summary-row total"><span>Total :</span><span id="payTotal">-</span></div>
</div> </div>
<div class="summary-row">
<span>Date :</span>
<span id="payDate">-</span>
</div>
<div class="summary-row">
<span>Heure :</span>
<span id="payTime">-</span>
</div>
<div class="summary-row">
<span>Durée :</span>
<span id="payDuration">-</span>
</div>
<div class="summary-row total">
<span>Total :</span>
<span id="payTotal">-</span>
</div>
</div>
<!-- Message de confirmation -->
<div class="confirmation-message"> <div class="confirmation-message">
<div class="confirmation-icon">🏷️</div> <div class="confirmation-icon">🏷️</div>
<h4>Merci pour votre réservation !</h4> <h4>Merci pour votre réservation !</h4>
<p>Votre place est bien réservée.<br> <p>Votre place est bien réservée.<br>
Vous recevrez votre <strong>badge d'accès dans les 24 heures</strong>.</p> Vous recevrez votre <strong>badge d'accès dans les 24 heures</strong>.</p>
</div> </div>
<!-- Bouton fermer -->
<button id="closeConfirmationBtn" class="btn btn-primary btn-block"> <button id="closeConfirmationBtn" class="btn btn-primary btn-block">
<span class="btn-icon"></span> <span class="btn-icon"></span>Fermer
Fermer
</button> </button>
</div> </div>
</div> </div>
@@ -270,44 +198,30 @@
</div> </div>
</section> </section>
<!-- PAGE: MES RÉSERVATIONS --> <!-- ═══ PAGE : MES RÉSERVATIONS ═══ -->
<section id="my-reservations" class="page hidden"> <section id="my-reservations" class="page hidden">
<div class="container"> <div class="container">
<h2 class="page-title"> <h2 class="page-title"><span class="icon">🎫</span>Mes réservations</h2>
<span class="icon">🎫</span> <div id="myReservationsList" class="reservations-list"></div>
Mes réservations
</h2>
<div id="myReservationsList" class="reservations-list">
<!-- Généré par JS -->
</div>
<div id="noReservations" class="empty-state"> <div id="noReservations" class="empty-state">
<span class="empty-icon">📭</span> <span class="empty-icon">📭</span>
<p>Vous n'avez aucune réservation active</p> <p>Vous n'avez aucune réservation</p>
<a href="#reservation" class="btn btn-primary">Faire une réservation</a> <a href="#reservation" class="btn btn-primary">Faire une réservation</a>
</div> </div>
</div> </div>
</section> </section>
<!-- PAGE: PROFIL --> <!-- ═══ PAGE : PROFIL ═══ -->
<section id="profile" class="page hidden"> <section id="profile" class="page hidden">
<div class="container"> <div class="container">
<h2 class="page-title"> <h2 class="page-title"><span class="icon">👤</span>Mon profil</h2>
<span class="icon">👤</span>
Mon profil
</h2>
<div class="profile-container"> <div class="profile-container">
<div class="profile-card"> <div class="profile-card">
<div class="profile-header"> <div class="profile-header">
<div class="profile-avatar"> <div class="profile-avatar"><span id="profileAvatar">👤</span></div>
<span id="profileAvatar">👤</span>
</div>
<h3 id="profileName">-</h3> <h3 id="profileName">-</h3>
<span id="profileRole" class="role-badge">Client</span> <span id="profileRole" class="role-badge">Client</span>
</div> </div>
<form id="profileForm" class="profile-form"> <form id="profileForm" class="profile-form">
<div class="form-group"> <div class="form-group">
<label>Nom complet</label> <label>Nom complet</label>
@@ -323,15 +237,14 @@
</div> </div>
<div class="form-group"> <div class="form-group">
<label>Nouveau mot de passe</label> <label>Nouveau mot de passe</label>
<input type="password" id="profileNewPassword" class="form-control" placeholder="Laisser vide pour ne pas changer"> <input type="password" id="profileNewPassword" class="form-control"
placeholder="Laisser vide pour ne pas changer">
</div> </div>
<button type="submit" class="btn btn-primary"> <button type="submit" class="btn btn-primary">
<span class="btn-icon">💾</span> <span class="btn-icon">💾</span>Mettre à jour
Mettre à jour
</button> </button>
</form> </form>
</div> </div>
<div class="profile-stats"> <div class="profile-stats">
<h3>Statistiques</h3> <h3>Statistiques</h3>
<div class="stats-cards"> <div class="stats-cards">
@@ -353,13 +266,10 @@
</div> </div>
</section> </section>
<!-- PAGE: ADMIN (visible uniquement pour admin) --> <!-- ═══ PAGE : ADMIN ═══ -->
<section id="admin" class="page hidden admin-page"> <section id="admin" class="page hidden admin-page">
<div class="container"> <div class="container">
<h2 class="page-title"> <h2 class="page-title"><span class="icon">⚙️</span>Administration</h2>
<span class="icon">⚙️</span>
Administration
</h2>
<!-- Stats admin --> <!-- Stats admin -->
<div class="admin-stats-grid"> <div class="admin-stats-grid">
@@ -388,87 +298,66 @@
<div class="form-group"> <div class="form-group">
<label>Nombre total de places</label> <label>Nombre total de places</label>
<div class="input-group"> <div class="input-group">
<input type="number" id="adminTotalSpots" class="form-control" min="5" max="50" value="10"> <input type="number" id="adminTotalSpots" class="form-control"
min="5" max="50" value="10">
<button id="updateSpotsBtn" class="btn btn-primary">Mettre à jour</button> <button id="updateSpotsBtn" class="btn btn-primary">Mettre à jour</button>
</div> </div>
</div> </div>
</div> </div>
<div class="admin-places-list" id="adminPlacesList"> <div class="admin-places-list" id="adminPlacesList"></div>
<!-- Généré par JS -->
</div>
</div> </div>
<!-- Gestion des utilisateurs --> <!-- Utilisateurs -->
<div class="admin-section"> <div class="admin-section">
<h3>👥 Utilisateurs</h3> <h3>👥 Utilisateurs</h3>
<div class="table-container"> <div class="table-container">
<table class="data-table"> <table class="data-table">
<thead> <thead>
<tr> <tr>
<th>ID</th> <th>ID</th><th>Nom</th><th>Email</th>
<th>Nom</th> <th>Téléphone</th><th>Rôle</th><th>Actions</th>
<th>Email</th>
<th>Téléphone</th>
<th>Rôle</th>
<th>Actions</th>
</tr> </tr>
</thead> </thead>
<tbody id="adminUsersTable"> <tbody id="adminUsersTable"></tbody>
<!-- Généré par JS -->
</tbody>
</table> </table>
</div> </div>
</div> </div>
<!-- Toutes les réservations --> <!-- Réservations -->
<div class="admin-section"> <div class="admin-section">
<h3>📅 Toutes les réservations</h3> <h3>📅 Toutes les réservations</h3>
<div class="table-container"> <div class="table-container">
<table class="data-table"> <table class="data-table">
<thead> <thead>
<tr> <tr>
<th>ID</th> <th>ID</th><th>Client</th><th>Place</th><th>Date</th>
<th>Client</th> <th>Horaire</th><th>Prix</th><th>Statut</th><th>Actions</th>
<th>Place</th>
<th>Date</th>
<th>Horaire</th>
<th>Prix</th>
<th>Statut</th>
<th>Actions</th>
</tr> </tr>
</thead> </thead>
<tbody id="adminReservationsTable"> <tbody id="adminReservationsTable"></tbody>
<!-- Généré par JS -->
</tbody>
</table> </table>
</div> </div>
</div> </div>
<!-- Graphique d'occupation --> <!-- ═══ GRAPHIQUE D'OCCUPATION SUPPRIMÉ ici ═══ -->
<div class="admin-section">
<h3>📊 Statistiques d'occupation</h3>
<div class="chart-container">
<canvas id="adminOccupancyChart"></canvas>
</div>
</div>
<!-- Historique --> <!-- Historique -->
<div class="admin-section"> <div class="admin-section">
<h3>📜 Historique</h3> <h3>📜 Historique</h3>
<div class="log-container" id="adminLogContainer"> <div class="log-container" id="adminLogContainer"></div>
<!-- Généré par JS -->
</div>
</div> </div>
</div> </div>
</section> </section>
</main> </main>
<!-- Toast notifications -->
<div id="toastContainer" class="toast-container"></div> <div id="toastContainer" class="toast-container"></div>
<script src="../js/dashboard.js"></script> <script src="../js/dashboard.js"></script>
<script src="../js/map.js"></script> <script src="../js/map.js"></script>
<script src="../js/reservation.js"></script> <script src="../js/reservation.js"></script>
<script src="../js/admin.js"></script> <script src="../js/admin.js"></script>
<!-- Chart.js et admin.js n'utilisent plus le graphique -->
</body> </body>
</html> </html>

View File

@@ -1,8 +1,10 @@
/** /**
* ============================================ * ============================================
* DATABASE.JS - Gestion MariaDB * DATABASE.JS - Gestion MariaDB
* Smart Parking v2.0 * Smart Parking v3.0
* AJOUTÉ : expireReservations() — libère auto les places * AJOUTÉ : checkReservationConflict()
* → vérifie les conflits d'horaire
* au lieu de bloquer toute la place
* ============================================ * ============================================
*/ */
@@ -123,7 +125,7 @@ async function initDatabase() {
[i, `SENSOR_${String(i).padStart(3, '0')}`, 'free'] [i, `SENSOR_${String(i).padStart(3, '0')}`, 'free']
); );
} }
console.log('✅ 10 places créées (toutes libres)'); console.log('✅ 10 places créées');
} }
} catch (err) { } catch (err) {
@@ -227,6 +229,34 @@ async function createReservation(userId, spotId, date, startTime, endTime, durat
return { id: result.insertId }; return { id: result.insertId };
} }
/**
* ⭐ NOUVELLE FONCTION — Vérification des conflits d'horaire
*
* Problème corrigé : avant, quand une place était réservée,
* elle restait bloquée pour TOUS les jours et TOUTES les heures.
*
* Maintenant on vérifie uniquement s'il y a une réservation
* qui se chevauche sur la MÊME date et le MÊME créneau horaire.
*
* Exemple :
* Place 2 réservée aujourd'hui 10h-11h ✅
* Place 2 réservée aujourd'hui 14h-15h ✅ (pas de conflit)
* Place 2 réservée demain 10h-11h ✅ (pas de conflit)
* Place 2 réservée aujourd'hui 10h30-11h30 ❌ (conflit !)
*/
async function checkReservationConflict(spotId, date, startTime, endTime) {
const [rows] = await pool.query(`
SELECT id FROM reservations
WHERE spot_id = ?
AND date = ?
AND status IN ('active', 'pending')
AND start_time < ?
AND end_time > ?
`, [spotId, date, endTime, startTime]);
return rows.length > 0; // true = conflit, false = créneau libre
}
async function getReservationById(id) { async function getReservationById(id) {
const [rows] = await pool.query( const [rows] = await pool.query(
`SELECT r.*, s.number AS spot_number `SELECT r.*, s.number AS spot_number
@@ -270,16 +300,10 @@ async function updateReservationStatus(id, status) {
} }
/** /**
* ⭐ NOUVELLE FONCTION — Expiration automatique des réservations * Expiration automatique des réservations
* * Appelée toutes les 60 secondes par server.js
* Cherche toutes les réservations actives dont la date+heure de fin
* est déjà dépassée, les passe en "completed" et libère les places.
*
* Appelée toutes les 60 secondes par server.js.
* Cela résout le problème des places qui restent "réservées" indéfiniment.
*/ */
async function expireReservations() { async function expireReservations() {
// Trouver les réservations actives dont l'heure de fin est passée
const [expiredRows] = await pool.query(` const [expiredRows] = await pool.query(`
SELECT r.id, r.spot_id, r.user_id, s.number AS spot_number SELECT r.id, r.spot_id, r.user_id, s.number AS spot_number
FROM reservations r FROM reservations r
@@ -289,20 +313,14 @@ async function expireReservations() {
`); `);
for (const res of expiredRows) { for (const res of expiredRows) {
// Passer la réservation en "completed"
await pool.query( await pool.query(
"UPDATE reservations SET status = 'completed' WHERE id = ?", "UPDATE reservations SET status = 'completed' WHERE id = ?",
[res.id] [res.id]
); );
// Libérer la place (la passer en "free")
// (le capteur Arduino prendra le relais ensuite si une voiture est encore là)
await pool.query( await pool.query(
"UPDATE spots SET status = 'free', last_update = NOW() WHERE id = ?", "UPDATE spots SET status = 'free', last_update = NOW() WHERE id = ?",
[res.spot_id] [res.spot_id]
); );
// Ajouter à l'historique
await pool.query( await pool.query(
'INSERT INTO history (action, details, user_id) VALUES (?, ?, ?)', 'INSERT INTO history (action, details, user_id) VALUES (?, ?, ?)',
[ [
@@ -364,7 +382,7 @@ async function getStats(days = 7) {
} }
// ============================================ // ============================================
// MQTT EVENTS // MQTT
// ============================================ // ============================================
async function recordMqttEvent(topic, message) { async function recordMqttEvent(topic, message) {
@@ -388,18 +406,13 @@ module.exports = {
pool, pool,
initDatabase, initDatabase,
closeDatabase, closeDatabase,
// Utilisateurs
createUser, getUserByEmail, getUserById, getAllUsers, updateUser, deleteUser, createUser, getUserByEmail, getUserById, getAllUsers, updateUser, deleteUser,
// Places
createSpot, getAllSpots, getSpotById, updateSpotStatus, deleteAllSpots, createSpot, getAllSpots, getSpotById, updateSpotStatus, deleteAllSpots,
// Réservations createReservation, checkReservationConflict,
createReservation, getReservationById, getReservationsByUser, getReservationById, getReservationsByUser,
getAllReservations, updateReservationStatus, getAllReservations, updateReservationStatus,
expireReservations, // ← NOUVEAU expireReservations,
// Historique
addHistory, getHistory, addHistory, getHistory,
// Stats
recordStats, getStats, recordStats, getStats,
// MQTT
recordMqttEvent recordMqttEvent
}; };

View File

@@ -1,9 +1,10 @@
/** /**
* ============================================ * ============================================
* API ROUTES - Routes de l'API REST * API ROUTES - Routes de l'API REST
* Smart Parking - BTS CIEL IR * Smart Parking v3.0
* CORRIGÉ : annulation libère bien la place * CORRIGÉ : réservation vérifie les conflits
* ajout route /complete pour l'admin * d'horaire au lieu de bloquer la place
* définitivement
* ============================================ * ============================================
*/ */
@@ -110,11 +111,7 @@ router.post('/spots/init', authenticateToken, requireAdmin, async (req, res) =>
const spotCount = Math.min(Math.max(parseInt(req.body.count) || 10, 5), 50); const spotCount = Math.min(Math.max(parseInt(req.body.count) || 10, 5), 50);
await db.deleteAllSpots(); await db.deleteAllSpots();
for (let i = 1; i <= spotCount; i++) { for (let i = 1; i <= spotCount; i++) {
let status = 'free'; await db.createSpot(i, `SENSOR_${String(i).padStart(3, '0')}`, 'free');
const rand = Math.random();
if (rand > 0.85) status = 'reserved';
else if (rand > 0.60) status = 'occupied';
await db.createSpot(i, `SENSOR_${String(i).padStart(3, '0')}`, status);
} }
await db.addHistory('Réinitialisation places', `${spotCount} places créées`, req.user.id); await db.addHistory('Réinitialisation places', `${spotCount} places créées`, req.user.id);
res.json({ success: true, message: `${spotCount} places créées` }); res.json({ success: true, message: `${spotCount} places créées` });
@@ -145,21 +142,71 @@ router.get('/reservations/all', authenticateToken, requireAdmin, async (req, res
} }
}); });
/**
* POST /api/reservations
*
* CORRIGÉ : on ne bloque plus la place entière définitivement.
* On vérifie uniquement s'il y a un CONFLIT d'horaire sur
* la même date et le même créneau.
*
* Exemple de ce qui est maintenant possible :
* Place 2 — 10h-11h aujourd'hui ✅
* Place 2 — 14h-15h aujourd'hui ✅ (pas de conflit)
* Place 2 — 10h-11h demain ✅ (pas de conflit)
* Place 2 — 10h30-11h30 aujourd'hui ❌ (conflit !)
*/
router.post('/reservations', authenticateToken, async (req, res) => { router.post('/reservations', authenticateToken, async (req, res) => {
try { try {
const { spotId, date, startTime, endTime, duration, vehicle, price } = req.body; const { spotId, date, startTime, endTime, duration, vehicle, price } = req.body;
if (!spotId || !date || !startTime || !endTime || !duration || !price) if (!spotId || !date || !startTime || !endTime || !duration || !price)
return res.status(400).json({ success: false, message: 'Tous les champs sont requis' }); return res.status(400).json({ success: false, message: 'Tous les champs sont requis' });
const spot = await db.getSpotById(spotId); const spot = await db.getSpotById(spotId);
if (!spot || spot.status !== 'free') if (!spot)
return res.status(409).json({ success: false, message: "Cette place n'est plus disponible" }); return res.status(404).json({ success: false, message: 'Place introuvable' });
// CORRIGÉ : bloquer uniquement si une voiture est physiquement là
if (spot.status === 'occupied')
return res.status(409).json({ success: false, message: "Une voiture est déjà sur cette place" });
// CORRIGÉ : vérifier les conflits d'horaire au lieu du statut global
const conflict = await db.checkReservationConflict(spotId, date, startTime, endTime);
if (conflict)
return res.status(409).json({
success: false,
message: `Cette place est déjà réservée sur ce créneau. Choisissez un autre horaire ou une autre date.`
});
const paymentCode = 'PARK' + Date.now().toString().slice(-8); const paymentCode = 'PARK' + Date.now().toString().slice(-8);
const result = await db.createReservation( const result = await db.createReservation(
req.user.id, spotId, date, startTime, endTime, duration, vehicle, price, paymentCode req.user.id, spotId, date, startTime, endTime, duration, vehicle, price, paymentCode
); );
// On ne change le statut de la place QUE si la réservation est pour aujourd'hui
// et que l'heure de début est maintenant ou dans moins de 30 minutes
const now = new Date();
const today = now.toISOString().split('T')[0];
const resStart = new Date(`${date}T${startTime}`);
const diffMin = (resStart - now) / 60000;
if (date === today && diffMin <= 30) {
await db.updateSpotStatus(spotId, 'reserved'); await db.updateSpotStatus(spotId, 'reserved');
await db.addHistory('Nouvelle réservation', `Place ${spot.number} - ${price}EUR`, req.user.id); }
res.status(201).json({ success: true, message: 'Réservation créée', data: { id: result.id, paymentCode } }); // Pour une réservation future, le statut de la place reste inchangé
// Le timer d'expiration (server.js) le mettra à jour au bon moment
await db.addHistory(
'Nouvelle réservation',
`Place ${spot.number} réservée le ${date} de ${startTime} à ${endTime}${price}EUR`,
req.user.id
);
res.status(201).json({
success: true,
message: 'Réservation créée',
data: { id: result.id, paymentCode }
});
} catch (err) { } catch (err) {
console.error('❌ Erreur create reservation:', err.message); console.error('❌ Erreur create reservation:', err.message);
res.status(500).json({ success: false, message: 'Erreur serveur' }); res.status(500).json({ success: false, message: 'Erreur serveur' });
@@ -168,7 +215,6 @@ router.post('/reservations', authenticateToken, async (req, res) => {
/** /**
* PUT /api/reservations/:id/cancel * PUT /api/reservations/:id/cancel
* CORRIGÉ : libère désormais la place associée
*/ */
router.put('/reservations/:id/cancel', authenticateToken, async (req, res) => { router.put('/reservations/:id/cancel', authenticateToken, async (req, res) => {
try { try {
@@ -179,10 +225,10 @@ router.put('/reservations/:id/cancel', authenticateToken, async (req, res) => {
return res.status(403).json({ success: false, message: 'Accès refusé' }); return res.status(403).json({ success: false, message: 'Accès refusé' });
await db.updateReservationStatus(req.params.id, 'cancelled'); await db.updateReservationStatus(req.params.id, 'cancelled');
await db.updateSpotStatus(reservation.spot_id, 'free'); // ← BUG CORRIGÉ ICI await db.updateSpotStatus(reservation.spot_id, 'free');
await db.addHistory( await db.addHistory(
'Annulation réservation', 'Annulation réservation',
`Reservation #${req.params.id} annulee - place ${reservation.spot_id} liberee`, `Réservation #${req.params.id} annulée place ${reservation.spot_number} libérée`,
req.user.id req.user.id
); );
res.json({ success: true, message: 'Réservation annulée' }); res.json({ success: true, message: 'Réservation annulée' });
@@ -193,7 +239,7 @@ router.put('/reservations/:id/cancel', authenticateToken, async (req, res) => {
}); });
/** /**
* PUT /api/reservations/:id/complete (admin uniquement) * PUT /api/reservations/:id/complete (admin)
*/ */
router.put('/reservations/:id/complete', authenticateToken, requireAdmin, async (req, res) => { router.put('/reservations/:id/complete', authenticateToken, requireAdmin, async (req, res) => {
try { try {
@@ -205,7 +251,7 @@ router.put('/reservations/:id/complete', authenticateToken, requireAdmin, async
await db.updateSpotStatus(reservation.spot_id, 'free'); await db.updateSpotStatus(reservation.spot_id, 'free');
await db.addHistory( await db.addHistory(
'Réservation terminée', 'Réservation terminée',
`Reservation #${req.params.id} terminee - place ${reservation.spot_id} liberee`, `Réservation #${req.params.id} terminée place ${reservation.spot_number} libérée`,
req.user.id req.user.id
); );
res.json({ success: true, message: 'Réservation terminée' }); res.json({ success: true, message: 'Réservation terminée' });
@@ -244,7 +290,7 @@ router.get('/history', authenticateToken, requireAdmin, async (req, res) => {
}); });
router.get('/status', (_req, res) => { router.get('/status', (_req, res) => {
res.json({ success: true, message: 'Smart Parking API operationnelle', version: '1.0.0', timestamp: new Date().toISOString() }); res.json({ success: true, message: 'Smart Parking API opérationnelle', version: '3.0.0', timestamp: new Date().toISOString() });
}); });
module.exports = router; module.exports = router;