création de la partie administration des clefs de partage

This commit is contained in:
Esenjin 2025-01-05 13:10:05 +01:00
parent 0aeab751cb
commit d251f6a3dd
3 changed files with 317 additions and 1 deletions

View File

@ -129,10 +129,21 @@ function showAdminInterface() {
</svg> </svg>
</div> </div>
<div class="menu-content"> <div class="menu-content">
<h2>Albums privés</h2> <h2>Gestion des albums privés</h2>
<p>Gérez vos albums photos privés et sécurisés.</p> <p>Gérez vos albums photos privés et sécurisés.</p>
</div> </div>
</a> </a>
<a href="clefs.php" class="admin-menu-item">
<div class="menu-icon">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
<path d="M21 2l-2 2m-7.61 7.61a5.5 5.5 0 1 1-7.778 7.778 5.5 5.5 0 0 1 7.777-7.777zm0 0L15.5 7.5m0 0l3 3L22 7l-3-3m-3.5 3.5L19 4"></path>
</svg>
</div>
<div class="menu-content">
<h2>Clés de partage</h2>
<p>Gérez les accès temporaires à vos albums privés.</p>
</div>
</a>
</div> </div>
</div> </div>
</body> </body>

244
clefs.php Normal file
View File

@ -0,0 +1,244 @@
<?php
require_once 'fonctions.php';
session_start();
// Variables pour stocker les messages
$successMessage = null;
$errorMessage = null;
// Vérification de l'authentification
if (!isset($_SESSION['admin_id'])) {
header('Location: admin.php?action=login');
exit;
}
// Initialisation des variables
$db = new SQLite3('database.sqlite');
$keys = [];
$albums = [];
$filter = $_GET['filter'] ?? 'active';
$albumFilter = $_GET['album'] ?? '';
// Gérer les actions POST
if ($_SERVER['REQUEST_METHOD'] === 'POST') {
$action = $_POST['action'] ?? '';
switch ($action) {
case 'delete_key':
$keyId = $_POST['key_id'] ?? '';
if ($keyId) {
$stmt = $db->prepare('DELETE FROM share_keys WHERE id = :id');
$stmt->bindValue(':id', $keyId, SQLITE3_INTEGER);
if ($stmt->execute()) {
$successMessage = "Clé supprimée avec succès.";
} else {
$errorMessage = "Erreur lors de la suppression de la clé.";
}
}
break;
case 'clean_expired':
$deletedCount = cleanExpiredShareKeys();
if ($deletedCount > 0) {
$successMessage = "$deletedCount clé(s) expirée(s) supprimée(s).";
} else {
$successMessage = "Aucune clé expirée à supprimer.";
}
break;
}
}
// Construire la requête SQL en fonction des filtres
$query = 'SELECT s.*, a.path, a.identifier as album_identifier
FROM share_keys s
JOIN album_identifiers a ON s.album_identifier = a.identifier
WHERE 1=1';
if ($filter === 'active') {
$query .= ' AND s.expires_at > datetime("now")';
} elseif ($filter === 'expired') {
$query .= ' AND s.expires_at <= datetime("now")';
}
if (!empty($albumFilter)) {
$query .= ' AND a.identifier = :album_identifier';
}
$query .= ' ORDER BY s.created_at DESC';
$stmt = $db->prepare($query);
if (!empty($albumFilter)) {
$stmt->bindValue(':album_identifier', $albumFilter, SQLITE3_TEXT);
}
$result = $stmt->execute();
while ($row = $result->fetchArray(SQLITE3_ASSOC)) {
$keys[] = $row;
}
// Récupérer la liste des albums pour le filtre
$albumsQuery = 'SELECT DISTINCT a.identifier, a.path
FROM album_identifiers a
ORDER BY a.path';
$albumsResult = $db->query($albumsQuery);
while ($row = $albumsResult->fetchArray(SQLITE3_ASSOC)) {
$albums[] = $row;
}
?>
<!DOCTYPE html>
<html lang="fr">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Gestion des clés de partage - ICO</title>
<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>Gestion des clés de partage</h1>
<div class="admin-actions">
<button onclick="cleanExpiredKeys()" class="action-button">
Nettoyer les clés expirées
</button>
<a href="admin.php" class="action-button action-button-secondary">Retour</a>
</div>
</div>
<div class="admin-content">
<?php if ($successMessage): ?>
<div class="message success-message"><?php echo htmlspecialchars($successMessage); ?></div>
<?php endif; ?>
<?php if ($errorMessage): ?>
<div class="message error-message"><?php echo htmlspecialchars($errorMessage); ?></div>
<?php endif; ?>
<div class="filters">
<div class="filter-group">
<label for="status-filter">Statut :</label>
<select id="status-filter" class="form-select" onchange="updateFilters()">
<option value="active" <?php echo $filter === 'active' ? 'selected' : ''; ?>>Clés actives</option>
<option value="expired" <?php echo $filter === 'expired' ? 'selected' : ''; ?>>Clés expirées</option>
<option value="all" <?php echo $filter === 'all' ? 'selected' : ''; ?>>Toutes les clés</option>
</select>
</div>
<div class="filter-group">
<label for="album-filter">Album :</label>
<select id="album-filter" class="form-select" onchange="updateFilters()">
<option value="">Tous les albums</option>
<?php foreach ($albums as $album): ?>
<option value="<?php echo htmlspecialchars($album['identifier']); ?>"
<?php echo $albumFilter === $album['identifier'] ? 'selected' : ''; ?>>
<?php echo htmlspecialchars($album['path']); ?>
</option>
<?php endforeach; ?>
</select>
</div>
</div>
<div class="keys-list">
<table class="admin-table">
<thead>
<tr>
<th>Album</th>
<th>URL de partage</th>
<th>Créée le</th>
<th>Expire le</th>
<th>Commentaire</th>
<th>Actions</th>
</tr>
</thead>
<tbody>
<?php foreach ($keys as $key):
$shareUrl = getBaseUrl() . '/galeries-privees.php?key=' . urlencode($key['key_value']);
$isExpired = strtotime($key['expires_at']) <= time();
$albumInfo = getAlbumInfo($key['path']);
?>
<tr class="<?php echo $isExpired ? 'expired-key' : ''; ?>">
<td title="<?php echo htmlspecialchars($key['path']); ?>">
<?php echo htmlspecialchars($albumInfo['title']); ?>
<?php if ($albumInfo['mature_content']): ?>
<span class="mature-warning">🔞</span>
<?php endif; ?>
</td>
<td>
<?php if (!$isExpired): ?>
<div class="share-url">
<input type="text" readonly value="<?php echo $shareUrl; ?>"
class="share-url-input" onclick="this.select()">
<button onclick="copyShareUrl(this)" class="tree-button" title="Copier">📋</button>
</div>
<?php else: ?>
<span class="expired-text">Expirée</span>
<?php endif; ?>
</td>
<td><?php echo date('d/m/Y H:i', strtotime($key['created_at'])); ?></td>
<td><?php echo date('d/m/Y H:i', strtotime($key['expires_at'])); ?></td>
<td><?php echo htmlspecialchars($key['comment']); ?></td>
<td>
<form method="post" style="display: inline;">
<input type="hidden" name="action" value="delete_key">
<input type="hidden" name="key_id" value="<?php echo $key['id']; ?>">
<button type="submit" class="tree-button tree-button-danger"
onclick="return confirm('Voulez-vous vraiment supprimer cette clé ?')">
🗑️
</button>
</form>
</td>
</tr>
<?php endforeach; ?>
<?php if (empty($keys)): ?>
<tr>
<td colspan="6" class="no-data">Aucune clé trouvée</td>
</tr>
<?php endif; ?>
</tbody>
</table>
</div>
</div>
<script>
function cleanExpiredKeys() {
if (confirm('Voulez-vous supprimer toutes les clés expirées ?')) {
const form = document.createElement('form');
form.method = 'post';
form.innerHTML = '<input type="hidden" name="action" value="clean_expired">';
document.body.appendChild(form);
form.submit();
}
}
function copyShareUrl(button) {
const input = button.previousElementSibling;
input.select();
document.execCommand('copy');
const originalText = button.innerHTML;
button.innerHTML = '✓';
button.classList.add('copied');
setTimeout(() => {
button.innerHTML = originalText;
button.classList.remove('copied');
}, 2000);
}
function updateFilters() {
const statusFilter = document.getElementById('status-filter').value;
const albumFilter = document.getElementById('album-filter').value;
let url = 'clefs.php?filter=' + statusFilter;
if (albumFilter) {
url += '&album=' + albumFilter;
}
window.location.href = url;
}
</script>
</body>
</html>

View File

@ -780,6 +780,67 @@ body {
border-left: 4px solid #2196f3; border-left: 4px solid #2196f3;
} }
/* Styles pour la page de gestion des clefs de partage */
.filters {
display: flex;
gap: 1rem;
margin-bottom: 2rem;
background: #2a2a2a;
padding: 1rem;
border-radius: 0.5rem;
}
.filter-group {
display: flex;
align-items: center;
gap: 0.5rem;
}
.share-url {
display: flex;
gap: 0.5rem;
align-items: center;
}
.share-url-input {
background: #2a2a2a;
border: none;
padding: 0.5rem;
border-radius: 0.25rem;
color: white;
width: 300px;
font-size: 0.9rem;
}
.expired-text {
color: #dc3545;
font-style: italic;
}
.expired-key {
opacity: 0.7;
}
.no-data {
text-align: center;
padding: 2rem;
color: #666;
}
.tree-button.copied {
background-color: #28a745 !important;
}
@media (max-width: 768px) {
.filters {
flex-direction: column;
}
.share-url-input {
width: 200px;
}
}
/* Styles spécifiques pour la page de gestion du carrousel */ /* Styles spécifiques pour la page de gestion du carrousel */
body[data-page="carrousel"] .upload-zone { body[data-page="carrousel"] .upload-zone {
border-color: #ff8c00; border-color: #ff8c00;