2025-01-04 16:08:38 +01:00
|
|
|
<?php
|
|
|
|
require_once 'fonctions.php';
|
|
|
|
|
|
|
|
// Vérifier l'authentification
|
|
|
|
session_start();
|
|
|
|
if (!isset($_SESSION['admin_id'])) {
|
|
|
|
header('Location: admin.php?action=login');
|
|
|
|
exit;
|
|
|
|
}
|
2025-01-06 23:53:46 +01:00
|
|
|
checkAdminSession();
|
2025-01-04 16:08:38 +01:00
|
|
|
|
|
|
|
// Récupérer le chemin courant
|
|
|
|
$currentPath = isset($_GET['path']) ? $_GET['path'] : './liste_albums_prives';
|
|
|
|
$currentPath = realpath($currentPath);
|
|
|
|
|
|
|
|
// Vérification de sécurité
|
|
|
|
if (!isSecurePrivatePath($currentPath)) {
|
|
|
|
header('Location: arbre-prive.php');
|
|
|
|
exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Gérer l'upload des images
|
|
|
|
if ($_SERVER['REQUEST_METHOD'] === 'POST') {
|
|
|
|
if (isset($_POST['action'])) {
|
|
|
|
switch ($_POST['action']) {
|
|
|
|
case 'upload':
|
|
|
|
$uploadedFiles = $_FILES['images'] ?? [];
|
2025-01-09 13:01:53 +01:00
|
|
|
$successCount = 0; // Initialiser le compteur
|
2025-01-04 16:08:38 +01:00
|
|
|
$errors = [];
|
2025-01-09 13:01:53 +01:00
|
|
|
|
2025-01-04 16:08:38 +01:00
|
|
|
// Gérer les uploads multiples
|
|
|
|
for ($i = 0; $i < count($uploadedFiles['name']); $i++) {
|
|
|
|
if ($uploadedFiles['error'][$i] === UPLOAD_ERR_OK) {
|
|
|
|
$tmpName = $uploadedFiles['tmp_name'][$i];
|
|
|
|
$fileName = sanitizeFilename($uploadedFiles['name'][$i]);
|
|
|
|
$extension = strtolower(pathinfo($fileName, PATHINFO_EXTENSION));
|
2025-01-09 13:01:53 +01:00
|
|
|
|
2025-01-04 16:08:38 +01:00
|
|
|
// Vérifier l'extension
|
|
|
|
if (in_array($extension, ALLOWED_EXTENSIONS)) {
|
|
|
|
$destination = $currentPath . '/' . $fileName;
|
|
|
|
|
|
|
|
// Vérifier si le fichier existe déjà
|
|
|
|
if (file_exists($destination)) {
|
|
|
|
$baseName = pathinfo($fileName, PATHINFO_FILENAME);
|
|
|
|
$counter = 1;
|
|
|
|
while (file_exists($destination)) {
|
|
|
|
$fileName = $baseName . '_' . $counter . '.' . $extension;
|
|
|
|
$destination = $currentPath . '/' . $fileName;
|
|
|
|
$counter++;
|
|
|
|
}
|
|
|
|
}
|
2025-01-09 13:01:53 +01:00
|
|
|
|
2025-01-04 16:08:38 +01:00
|
|
|
if (move_uploaded_file($tmpName, $destination)) {
|
2025-01-09 13:01:53 +01:00
|
|
|
$successCount++; // Incrémenter le compteur en cas de succès
|
2025-01-04 16:08:38 +01:00
|
|
|
} else {
|
|
|
|
$errors[] = "Erreur lors du déplacement de $fileName";
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$errors[] = "Extension non autorisée pour $fileName";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2025-01-09 13:01:53 +01:00
|
|
|
|
|
|
|
// Loguer l'action une fois que tous les uploads sont terminés
|
|
|
|
if ($successCount > 0) {
|
|
|
|
logAdminAction(
|
|
|
|
$_SESSION['admin_id'],
|
|
|
|
'UPLOAD_PRIVATE_IMAGES', // Notez le changement ici pour les images privées
|
|
|
|
"Téléversement de $successCount image(s) privée(s)", // Message adapté pour les images privées
|
|
|
|
$currentPath
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2025-01-04 16:08:38 +01:00
|
|
|
if ($successCount > 0) {
|
|
|
|
$_SESSION['success_message'] = "$successCount image(s) téléversée(s) avec succès.";
|
|
|
|
}
|
|
|
|
if (!empty($errors)) {
|
|
|
|
$_SESSION['error_message'] = implode("\n", $errors);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'toggle_top':
|
|
|
|
$image = $_POST['image'] ?? '';
|
|
|
|
if ($image) {
|
|
|
|
$imagePath = $currentPath . '/' . basename($image);
|
|
|
|
if (isSecurePrivatePath($imagePath) && file_exists($imagePath)) {
|
|
|
|
$info = pathinfo($imagePath);
|
|
|
|
$isTop = strpos($info['filename'], '--top--') !== false;
|
|
|
|
|
|
|
|
if ($isTop) {
|
|
|
|
// Enlever le tag top
|
|
|
|
$newName = str_replace('--top--', '', $info['filename']) . '.' . $info['extension'];
|
|
|
|
} else {
|
|
|
|
// Ajouter le tag top
|
|
|
|
$newName = $info['filename'] . '--top--.' . $info['extension'];
|
|
|
|
}
|
|
|
|
|
|
|
|
$newPath = $currentPath . '/' . $newName;
|
|
|
|
if (rename($imagePath, $newPath)) {
|
|
|
|
$_SESSION['success_message'] = $isTop ? "Image retirée des tops." : "Image mise en top.";
|
|
|
|
} else {
|
|
|
|
$_SESSION['error_message'] = "Erreur lors de la modification du statut top.";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'delete':
|
|
|
|
$images = $_POST['images'] ?? [];
|
2025-01-09 13:01:53 +01:00
|
|
|
$deleteCount = 0; // Initialiser le compteur
|
|
|
|
$errors = [];
|
2025-01-04 16:08:38 +01:00
|
|
|
|
|
|
|
foreach ($images as $image) {
|
|
|
|
$imagePath = $currentPath . '/' . basename($image);
|
2025-01-09 13:01:53 +01:00
|
|
|
if (isSecurePrivatePath($imagePath) && file_exists($imagePath)) { // Notez l'utilisation de isSecurePrivatePath
|
2025-01-04 16:08:38 +01:00
|
|
|
if (unlink($imagePath)) {
|
2025-01-09 13:01:53 +01:00
|
|
|
$deleteCount++; // Incrémenter le compteur en cas de succès
|
|
|
|
} else {
|
|
|
|
$errors[] = "Erreur lors de la suppression de " . basename($image);
|
2025-01-04 16:08:38 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-01-09 13:01:53 +01:00
|
|
|
// Loguer l'action une fois que toutes les suppressions sont terminées
|
2025-01-04 16:08:38 +01:00
|
|
|
if ($deleteCount > 0) {
|
2025-01-09 13:01:53 +01:00
|
|
|
logAdminAction(
|
|
|
|
$_SESSION['admin_id'],
|
|
|
|
'DELETE_PRIVATE_IMAGES', // Notez le changement ici pour les images privées
|
|
|
|
"Suppression de $deleteCount image(s) privée(s)", // Message adapté pour les images privées
|
|
|
|
$currentPath
|
|
|
|
);
|
2025-01-04 16:08:38 +01:00
|
|
|
$_SESSION['success_message'] = "$deleteCount image(s) supprimée(s).";
|
|
|
|
}
|
2025-01-09 13:01:53 +01:00
|
|
|
|
|
|
|
if (!empty($errors)) {
|
|
|
|
$_SESSION['error_message'] = implode("\n", $errors);
|
|
|
|
}
|
2025-01-04 16:08:38 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
header('Location: arbre-img-prive.php?path=' . urlencode($currentPath));
|
|
|
|
exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Récupérer les images du dossier courant
|
|
|
|
$images = [];
|
|
|
|
$tempImages = [];
|
|
|
|
foreach (new DirectoryIterator($currentPath) as $file) {
|
|
|
|
if ($file->isDot()) continue;
|
|
|
|
if ($file->isFile()) {
|
|
|
|
$extension = strtolower($file->getExtension());
|
|
|
|
if (in_array($extension, ALLOWED_EXTENSIONS)) {
|
|
|
|
$tempImages[] = [
|
|
|
|
'name' => $file->getFilename(),
|
|
|
|
'time' => $file->getCTime()
|
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Trier par date de création décroissante
|
|
|
|
usort($tempImages, function($a, $b) {
|
|
|
|
return $b['time'] - $a['time'];
|
|
|
|
});
|
|
|
|
|
|
|
|
// Extraire uniquement les noms de fichiers
|
|
|
|
$images = array_map(function($img) {
|
|
|
|
return $img['name'];
|
|
|
|
}, $tempImages);
|
|
|
|
|
|
|
|
$currentAlbumInfo = getAlbumInfo($currentPath);
|
2025-01-06 14:06:25 +01:00
|
|
|
$config = getSiteConfig();
|
2025-01-04 16:08:38 +01:00
|
|
|
?>
|
|
|
|
|
|
|
|
<!DOCTYPE html>
|
|
|
|
<html lang="fr">
|
|
|
|
<head>
|
|
|
|
<meta charset="UTF-8">
|
|
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
2025-01-06 14:06:25 +01:00
|
|
|
<title>Gestion des images privées - <?php echo htmlspecialchars($config['site_title']); ?></title>
|
2025-01-04 16:08:38 +01:00
|
|
|
<link rel="icon" type="image/png" href="favicon.png">
|
|
|
|
<link rel="stylesheet" href="styles.css">
|
|
|
|
<link rel="stylesheet" href="styles-admin.css">
|
|
|
|
</head>
|
|
|
|
<body class="admin-page">
|
|
|
|
<div class="admin-header">
|
|
|
|
<h1>
|
|
|
|
Images de : <?php echo htmlspecialchars($currentAlbumInfo['title']); ?>
|
|
|
|
<span class="private-badge">Privé</span>
|
|
|
|
</h1>
|
|
|
|
<div class="admin-actions">
|
|
|
|
<button onclick="document.getElementById('imageUploadForm').click()" class="action-button action-button-success">
|
|
|
|
Ajouter des images
|
|
|
|
</button>
|
|
|
|
<button onclick="deleteSelected()" id="deleteSelectedBtn" class="action-button action-button-danger">
|
|
|
|
Supprimer la sélection
|
|
|
|
</button>
|
2025-01-06 20:46:06 +01:00
|
|
|
<button onclick="toggleSelectAll()" id="selectAllBtn" class="action-button">
|
|
|
|
Tout sélectionner
|
|
|
|
</button>
|
2025-01-04 16:08:38 +01:00
|
|
|
<a href="arbre-prive.php?path=<?php echo urlencode($currentPath); ?>" class="action-button action-button-secondary">
|
|
|
|
Retour
|
|
|
|
</a>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div class="admin-content">
|
|
|
|
<?php if (isset($_SESSION['success_message'])): ?>
|
|
|
|
<div class="message success-message"><?php echo nl2br(htmlspecialchars($_SESSION['success_message'])); ?></div>
|
|
|
|
<?php unset($_SESSION['success_message']); ?>
|
|
|
|
<?php endif; ?>
|
|
|
|
|
|
|
|
<?php if (isset($_SESSION['error_message'])): ?>
|
|
|
|
<div class="message error-message"><?php echo nl2br(htmlspecialchars($_SESSION['error_message'])); ?></div>
|
|
|
|
<?php unset($_SESSION['error_message']); ?>
|
|
|
|
<?php endif; ?>
|
|
|
|
|
|
|
|
<div class="upload-zone" id="dropZone">
|
|
|
|
<p>Glissez-déposez vos images ici ou cliquez sur "Ajouter des images"</p>
|
|
|
|
<form method="post" enctype="multipart/form-data" id="uploadForm">
|
|
|
|
<input type="hidden" name="action" value="upload">
|
|
|
|
<input type="file" name="images[]" id="imageUploadForm" multiple accept=".jpg,.jpeg,.png,.gif">
|
|
|
|
</form>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<form method="post" id="imagesForm">
|
|
|
|
<input type="hidden" name="action" id="formAction" value="">
|
|
|
|
<div class="images-grid">
|
|
|
|
<?php foreach($images as $image):
|
|
|
|
$imagePath = str_replace('\\', '/', substr($currentPath, strpos($currentPath, '/liste_albums_prives/') + strlen('/liste_albums_prives/')));
|
|
|
|
$imageUrl = getBaseUrl() . '/liste_albums_prives/' . ($imagePath ? $imagePath . '/' : '') . $image;
|
|
|
|
?>
|
|
|
|
<div class="image-item">
|
|
|
|
<input type="checkbox" name="images[]" value="<?php echo htmlspecialchars($image); ?>"
|
|
|
|
class="image-checkbox" onchange="updateActionButtons()">
|
|
|
|
<div class="image-wrapper">
|
|
|
|
<img src="<?php echo htmlspecialchars($imageUrl); ?>"
|
|
|
|
alt="<?php echo htmlspecialchars($image); ?>" loading="lazy">
|
|
|
|
<div class="image-actions">
|
|
|
|
<?php
|
|
|
|
$isTop = strpos($image, '--top--') !== false;
|
|
|
|
?>
|
|
|
|
<button type="button" onclick="toggleTop('<?php echo htmlspecialchars($image); ?>')"
|
|
|
|
class="tree-button <?php echo $isTop ? 'tree-button-top' : ''; ?>"
|
|
|
|
title="<?php echo $isTop ? 'Retirer des tops' : 'Mettre en top'; ?>">
|
|
|
|
⭐
|
|
|
|
</button>
|
|
|
|
<button type="button" onclick="deleteImage('<?php echo htmlspecialchars($image); ?>')"
|
|
|
|
class="tree-button tree-button-danger">🗑️</button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<?php endforeach; ?>
|
|
|
|
</div>
|
|
|
|
</form>
|
|
|
|
</div>
|
|
|
|
|
2025-01-06 15:16:25 +01:00
|
|
|
<!-- Modale d'indication de téléversement -->
|
|
|
|
<div id="uploadModal" class="modal-upload">
|
|
|
|
<div class="modal-content">
|
|
|
|
<div class="spinner"></div>
|
|
|
|
<p>Téléversement en cours... veuillez patienter...</p>
|
|
|
|
</div>
|
|
|
|
</div>
|
2025-01-04 16:08:38 +01:00
|
|
|
|
2025-01-06 15:16:25 +01:00
|
|
|
<script>
|
|
|
|
document.addEventListener('DOMContentLoaded', function() {
|
|
|
|
const modal = document.getElementById('uploadModal');
|
2025-01-04 16:08:38 +01:00
|
|
|
const dropZone = document.getElementById('dropZone');
|
|
|
|
const uploadForm = document.getElementById('uploadForm');
|
|
|
|
const imageUploadForm = document.getElementById('imageUploadForm');
|
|
|
|
|
2025-01-06 15:16:25 +01:00
|
|
|
// Gestion du formulaire d'upload
|
|
|
|
if (uploadForm) {
|
|
|
|
uploadForm.addEventListener('submit', function(e) {
|
|
|
|
const fileInput = this.querySelector('input[type="file"]');
|
|
|
|
if (fileInput && fileInput.files && fileInput.files.length > 0) {
|
|
|
|
modal.style.display = 'block';
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2025-01-04 16:08:38 +01:00
|
|
|
|
2025-01-06 15:16:25 +01:00
|
|
|
// Gestion du drag & drop
|
|
|
|
if (dropZone) {
|
|
|
|
dropZone.addEventListener('dragover', (e) => {
|
|
|
|
e.preventDefault();
|
|
|
|
dropZone.classList.add('drag-over');
|
|
|
|
});
|
2025-01-04 16:08:38 +01:00
|
|
|
|
2025-01-06 15:16:25 +01:00
|
|
|
dropZone.addEventListener('dragleave', () => {
|
|
|
|
dropZone.classList.remove('drag-over');
|
|
|
|
});
|
|
|
|
|
|
|
|
dropZone.addEventListener('drop', (e) => {
|
|
|
|
e.preventDefault();
|
|
|
|
dropZone.classList.remove('drag-over');
|
|
|
|
|
|
|
|
const files = e.dataTransfer.files;
|
|
|
|
if (files.length > 0) {
|
|
|
|
const dataTransfer = new DataTransfer();
|
|
|
|
for (let file of files) {
|
|
|
|
dataTransfer.items.add(file);
|
|
|
|
}
|
|
|
|
imageUploadForm.files = dataTransfer.files;
|
|
|
|
modal.style.display = 'block'; // Afficher la modale avant la soumission
|
|
|
|
uploadForm.submit();
|
2025-01-04 16:08:38 +01:00
|
|
|
}
|
2025-01-06 15:16:25 +01:00
|
|
|
});
|
|
|
|
}
|
2025-01-04 16:08:38 +01:00
|
|
|
|
2025-01-06 15:16:25 +01:00
|
|
|
// Gestion du click sur "Ajouter des images"
|
|
|
|
const imageUploadInput = document.getElementById('imageUploadForm');
|
|
|
|
if (imageUploadInput) {
|
|
|
|
imageUploadInput.addEventListener('change', function() {
|
|
|
|
if (this.files && this.files.length > 0) {
|
|
|
|
modal.style.display = 'block';
|
|
|
|
uploadForm.submit();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
2025-01-06 20:46:06 +01:00
|
|
|
|
|
|
|
// Fonction pour basculer la sélection de toutes les images
|
|
|
|
function toggleSelectAll() {
|
|
|
|
const checkboxes = document.querySelectorAll('.image-checkbox');
|
|
|
|
const allChecked = document.querySelectorAll('.image-checkbox:checked').length === checkboxes.length;
|
|
|
|
|
|
|
|
checkboxes.forEach(checkbox => {
|
|
|
|
checkbox.checked = !allChecked;
|
|
|
|
});
|
|
|
|
|
|
|
|
updateActionButtons();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Modifier la fonction updateActionButtons existante
|
|
|
|
function updateActionButtons() {
|
|
|
|
const checkboxes = document.querySelectorAll('.image-checkbox');
|
|
|
|
const selectedCheckboxes = document.querySelectorAll('.image-checkbox:checked');
|
|
|
|
const count = selectedCheckboxes.length;
|
|
|
|
|
|
|
|
const deleteBtn = document.getElementById('deleteSelectedBtn');
|
|
|
|
const selectAllBtn = document.getElementById('selectAllBtn');
|
|
|
|
|
|
|
|
if (deleteBtn) {
|
|
|
|
deleteBtn.style.display = count > 0 ? 'inline-flex' : 'none';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (selectAllBtn) {
|
|
|
|
selectAllBtn.textContent = checkboxes.length === selectedCheckboxes.length ?
|
|
|
|
'Tout désélectionner' : 'Tout sélectionner';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ajouter l'initialisation au chargement de la page
|
|
|
|
document.addEventListener('DOMContentLoaded', updateActionButtons);
|
2025-01-04 16:08:38 +01:00
|
|
|
</script>
|
2025-01-06 16:54:32 +01:00
|
|
|
<button class="scroll-top" title="Retour en haut">↑</button>
|
|
|
|
<script>
|
|
|
|
const scrollBtn = document.querySelector('.scroll-top');
|
|
|
|
window.addEventListener('scroll', () => {
|
|
|
|
scrollBtn.style.display = window.scrollY > 500 ? 'flex' : 'none';
|
|
|
|
});
|
|
|
|
scrollBtn.addEventListener('click', () => {
|
|
|
|
window.scrollTo({ top: 0, behavior: 'smooth' });
|
|
|
|
});
|
2025-01-09 13:16:18 +01:00
|
|
|
|
|
|
|
// Fonction pour basculer la sélection de toutes les images
|
|
|
|
function toggleSelectAll() {
|
|
|
|
const checkboxes = document.querySelectorAll('.image-checkbox');
|
|
|
|
const allChecked = document.querySelectorAll('.image-checkbox:checked').length === checkboxes.length;
|
|
|
|
|
|
|
|
checkboxes.forEach(checkbox => {
|
|
|
|
checkbox.checked = !allChecked;
|
|
|
|
});
|
|
|
|
|
|
|
|
updateActionButtons();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fonction de suppression d'une seule image
|
|
|
|
function deleteImage(imageName) {
|
|
|
|
if (confirm('Êtes-vous sûr de vouloir supprimer cette image ?')) {
|
|
|
|
const form = document.getElementById('imagesForm');
|
|
|
|
form.innerHTML = `
|
|
|
|
<input type="hidden" name="action" value="delete">
|
|
|
|
<input type="hidden" name="images[]" value="${imageName}">
|
|
|
|
`;
|
|
|
|
form.submit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fonction de suppression multiple
|
|
|
|
function deleteSelected() {
|
|
|
|
const checkboxes = document.querySelectorAll('.image-checkbox:checked');
|
|
|
|
if (checkboxes.length > 0 && confirm('Êtes-vous sûr de vouloir supprimer les images sélectionnées ?')) {
|
|
|
|
document.getElementById('formAction').value = 'delete';
|
|
|
|
document.getElementById('imagesForm').submit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fonction de mise à jour de l'état des boutons d'action
|
|
|
|
function updateActionButtons() {
|
|
|
|
const checkboxes = document.querySelectorAll('.image-checkbox');
|
|
|
|
const selectedCheckboxes = document.querySelectorAll('.image-checkbox:checked');
|
|
|
|
const count = selectedCheckboxes.length;
|
|
|
|
|
|
|
|
const deleteBtn = document.getElementById('deleteSelectedBtn');
|
|
|
|
const selectAllBtn = document.getElementById('selectAllBtn');
|
|
|
|
|
|
|
|
if (deleteBtn) {
|
|
|
|
deleteBtn.style.display = count > 0 ? 'inline-flex' : 'none';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (selectAllBtn) {
|
|
|
|
selectAllBtn.textContent = checkboxes.length === selectedCheckboxes.length ?
|
|
|
|
'Tout désélectionner' : 'Tout sélectionner';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initialisation des boutons au chargement
|
|
|
|
document.addEventListener('DOMContentLoaded', updateActionButtons);
|
2025-01-06 16:54:32 +01:00
|
|
|
</script>
|
2025-01-05 21:55:22 +01:00
|
|
|
<?php include 'footer.php'; ?>
|
2025-01-04 16:08:38 +01:00
|
|
|
</body>
|
|
|
|
</html>
|