Compare commits
No commits in common. "main" and "3.0.1" have entirely different histories.
@ -1,4 +1,8 @@
|
||||
# Cyla
|
||||
Site d'hébergement de fichiers simple, réduit au strict nécessaire.
|
||||
|
||||
|
||||
Le projet se base sur la première version de [QuadFile](https://github.com/QuadPiece/QuadFile).
|
||||
|
||||
|
||||
![image](https://concepts.esenjin.xyz/cyla/fichiers/6787d3fe886e4_1736954878.png)
|
87
admin.php
87
admin.php
@ -289,57 +289,56 @@ $files = $fileData['files'];
|
||||
<div class="file-list">
|
||||
<?php foreach ($files as $file): ?>
|
||||
<div class="file-item">
|
||||
<div class="file-preview">
|
||||
<?php if ($file['preview_type'] === 'image'): ?>
|
||||
<img src="<?php echo $file['path'] . Cyla::escape($file['name']); ?>"
|
||||
alt="<?php echo Cyla::escape($file['name']); ?>">
|
||||
<?php else: ?>
|
||||
<div class="preview-placeholder">
|
||||
<?php echo strtoupper($file['extension']); ?>
|
||||
<div class="file-preview">
|
||||
<?php if ($file['preview_type'] === 'image'): ?>
|
||||
<img src="fichiers/<?php echo Cyla::escape($file['name']); ?>"
|
||||
alt="<?php echo Cyla::escape($file['name']); ?>">
|
||||
<?php else: ?>
|
||||
<div class="preview-placeholder">
|
||||
<?php echo strtoupper($file['extension']); ?>
|
||||
</div>
|
||||
<?php endif; ?>
|
||||
</div>
|
||||
|
||||
<div class="file-info">
|
||||
<p class="file-name"><?php echo Cyla::escape($file['name']); ?></p>
|
||||
<p class="file-meta">
|
||||
<?php echo Cyla::escape(round($file['size'] / 1024, 2)); ?> Ko
|
||||
· <?php echo date('d/m/Y H:i', $file['uploaded']); ?>
|
||||
</p>
|
||||
<div class="file-actions">
|
||||
<a href="share.php?file=<?php echo urlencode($file['name']); ?>"
|
||||
class="btn btn-secondary"
|
||||
target="_blank">Voir</a>
|
||||
<button class="btn"
|
||||
onclick="copyShareLink('<?php echo SITE_URL; ?>share.php?file=<?php echo urlencode($file['name']); ?>')">
|
||||
Copier le lien
|
||||
</button>
|
||||
<button class="btn btn-danger"
|
||||
onclick="confirmDelete('<?php echo Cyla::escape($file['name']); ?>')">
|
||||
×
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<?php endif; ?>
|
||||
</div>
|
||||
|
||||
<div class="file-info">
|
||||
<p class="file-name"><?php echo Cyla::escape($file['name']); ?></p>
|
||||
<p class="file-meta">
|
||||
<?php echo Cyla::escape(round($file['size'] / 1024, 2)); ?> Ko
|
||||
· <?php echo date('d/m/Y H:i', $file['uploaded']); ?>
|
||||
· <?php echo $file['path']; ?>
|
||||
</p>
|
||||
<div class="file-actions">
|
||||
<a href="share.php?file=<?php echo urlencode($file['name']); ?>&path=<?php echo urlencode($file['path']); ?>"
|
||||
class="btn btn-secondary"
|
||||
target="_blank">Voir</a>
|
||||
<button class="btn"
|
||||
onclick="copyShareLink('<?php echo SITE_URL; ?>share.php?file=<?php echo urlencode($file['name']); ?>&path=<?php echo urlencode($file['path']); ?>')">
|
||||
Copier le lien
|
||||
</button>
|
||||
<button class="btn btn-danger"
|
||||
onclick="confirmDelete('<?php echo Cyla::escape($file['name']); ?>', '<?php echo Cyla::escape($file['path']); ?>')">
|
||||
×
|
||||
</button>
|
||||
</div>
|
||||
<!-- Formulaire caché pour la suppression -->
|
||||
<form id="deleteForm-<?php echo Cyla::escape($file['name']); ?>"
|
||||
action="admin.php"
|
||||
method="POST"
|
||||
style="display: none;">
|
||||
<input type="hidden" name="action" value="delete_file">
|
||||
<input type="hidden" name="filename" value="<?php echo Cyla::escape($file['name']); ?>">
|
||||
<input type="hidden" name="csrf_token" value="<?php echo Cyla::generateCSRFToken(); ?>">
|
||||
</form>
|
||||
<?php endforeach; ?>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Formulaire caché pour la suppression -->
|
||||
<form id="deleteForm-<?php echo Cyla::escape($file['name']); ?>"
|
||||
action="admin.php"
|
||||
method="POST"
|
||||
style="display: none;">
|
||||
<input type="hidden" name="action" value="delete_file">
|
||||
<input type="hidden" name="filename" value="<?php echo Cyla::escape($file['name']); ?>">
|
||||
<input type="hidden" name="path" value="<?php echo Cyla::escape($file['path']); ?>">
|
||||
<input type="hidden" name="csrf_token" value="<?php echo Cyla::generateCSRFToken(); ?>">
|
||||
</form>
|
||||
<?php endforeach; ?>
|
||||
|
||||
<?php if ($fileData['totalPages'] > 1): ?>
|
||||
<div class="pagination">
|
||||
<?php if ($fileData['currentPage'] > 1): ?>
|
||||
<a href="?page=<?php echo $fileData['currentPage'] - 1; ?>"
|
||||
class="pagination-link">«</a>
|
||||
class="pagination-link">« Précédent</a>
|
||||
<?php endif; ?>
|
||||
|
||||
<?php
|
||||
@ -366,7 +365,7 @@ $files = $fileData['files'];
|
||||
|
||||
if ($fileData['currentPage'] < $fileData['totalPages']): ?>
|
||||
<a href="?page=<?php echo $fileData['currentPage'] + 1; ?>"
|
||||
class="pagination-link">»</a>
|
||||
class="pagination-link">Suivant »</a>
|
||||
<?php endif; ?>
|
||||
</div>
|
||||
<?php endif; ?>
|
||||
|
@ -7,14 +7,10 @@ if (!defined('CYLA_CORE')) {
|
||||
|
||||
// Site configuration
|
||||
define('SITE_NAME', 'Cyla');
|
||||
define('SITE_VERSION', '3.0.2');
|
||||
define('SITE_VERSION', '3.0.1');
|
||||
define('SITE_URL', 'https://concepts.esenjin.xyz/cyla/');
|
||||
|
||||
// Files configuration
|
||||
define('LEGACY_UPLOAD_DIRS', [
|
||||
__DIR__ . '/v1/img/fichiers/',
|
||||
__DIR__ . '/v2/file/'
|
||||
]);
|
||||
define('UPLOAD_DIR', __DIR__ . '/fichiers/');
|
||||
define('MAX_FILE_SIZE', 100 * 1024 * 1024); // 100 Mo en octets
|
||||
define('ALLOWED_EXTENSIONS', [
|
||||
|
69
core.php
69
core.php
@ -138,17 +138,7 @@ class Cyla {
|
||||
*/
|
||||
public static function listFiles($page = 1, $perPage = 20) {
|
||||
$files = [];
|
||||
$allFiles = [];
|
||||
|
||||
// Ajouter les fichiers du dossier principal
|
||||
$allFiles = array_merge($allFiles, glob(UPLOAD_DIR . '*'));
|
||||
|
||||
// Ajouter les fichiers des dossiers hérités
|
||||
foreach (LEGACY_UPLOAD_DIRS as $dir) {
|
||||
if (is_dir($dir)) {
|
||||
$allFiles = array_merge($allFiles, glob($dir . '*'));
|
||||
}
|
||||
}
|
||||
$allFiles = glob(UPLOAD_DIR . '*');
|
||||
|
||||
// Trier les fichiers par date de modification (plus récent en premier)
|
||||
usort($allFiles, function($a, $b) {
|
||||
@ -158,7 +148,7 @@ class Cyla {
|
||||
// Calculer la pagination
|
||||
$total = count($allFiles);
|
||||
$totalPages = ceil($total / $perPage);
|
||||
$page = max(1, min($page, $totalPages));
|
||||
$page = max(1, min($page, $totalPages)); // Garantir que la page est dans les limites
|
||||
$offset = ($page - 1) * $perPage;
|
||||
|
||||
// Récupérer uniquement les fichiers de la page courante
|
||||
@ -166,24 +156,12 @@ class Cyla {
|
||||
|
||||
foreach ($pageFiles as $file) {
|
||||
$info = pathinfo($file);
|
||||
$relativePath = '';
|
||||
|
||||
// Déterminer le chemin relatif selon le dossier
|
||||
if (strpos($file, UPLOAD_DIR) === 0) {
|
||||
$relativePath = 'fichiers/';
|
||||
} elseif (strpos($file, __DIR__ . '/v1/img/fichiers/') === 0) {
|
||||
$relativePath = 'v1/img/fichiers/';
|
||||
} elseif (strpos($file, __DIR__ . '/v2/file/') === 0) {
|
||||
$relativePath = 'v2/file/';
|
||||
}
|
||||
|
||||
$files[] = [
|
||||
'name' => basename($file),
|
||||
'size' => filesize($file),
|
||||
'extension' => strtolower($info['extension'] ?? ''),
|
||||
'extension' => strtolower($info['extension']),
|
||||
'uploaded' => filemtime($file),
|
||||
'preview_type' => getPreviewType($info['extension'] ?? ''),
|
||||
'path' => $relativePath // Ajout du chemin relatif
|
||||
'preview_type' => getPreviewType($info['extension'])
|
||||
];
|
||||
}
|
||||
|
||||
@ -195,50 +173,23 @@ class Cyla {
|
||||
'perPage' => $perPage
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* Supprime un fichier
|
||||
* @param string $filename Nom du fichier à supprimer
|
||||
* @param string $path Chemin relatif du fichier
|
||||
* @return array ['success' => bool, 'error' => string|null]
|
||||
*/
|
||||
public static function deleteFile($filename, $path = 'fichiers/') {
|
||||
// Déterminer le chemin complet selon le dossier
|
||||
$basePath = '';
|
||||
switch ($path) {
|
||||
case 'v1/img/fichiers/':
|
||||
$basePath = __DIR__ . '/v1/img/fichiers/';
|
||||
break;
|
||||
case 'v2/file/':
|
||||
$basePath = __DIR__ . '/v2/file/';
|
||||
break;
|
||||
default:
|
||||
$basePath = UPLOAD_DIR;
|
||||
}
|
||||
public static function deleteFile($filename) {
|
||||
$filepath = UPLOAD_DIR . $filename;
|
||||
|
||||
$filepath = $basePath . $filename;
|
||||
|
||||
// Vérifier que le fichier existe et est dans le bon dossier
|
||||
// Vérifier que le fichier existe et est dans le dossier d'upload
|
||||
if (!file_exists($filepath) || !is_file($filepath)) {
|
||||
return ['success' => false, 'error' => 'Fichier introuvable'];
|
||||
}
|
||||
|
||||
// Vérifier que le fichier est bien dans un des dossiers autorisés
|
||||
// Vérifier que le fichier est bien dans le dossier d'upload
|
||||
$realpath = realpath($filepath);
|
||||
$allowed = false;
|
||||
|
||||
if (strpos($realpath, realpath(UPLOAD_DIR)) === 0) {
|
||||
$allowed = true;
|
||||
} else {
|
||||
foreach (LEGACY_UPLOAD_DIRS as $dir) {
|
||||
if (strpos($realpath, realpath($dir)) === 0) {
|
||||
$allowed = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (!$allowed) {
|
||||
$uploadDir = realpath(UPLOAD_DIR);
|
||||
if (strpos($realpath, $uploadDir) !== 0) {
|
||||
return ['success' => false, 'error' => 'Chemin de fichier non autorisé'];
|
||||
}
|
||||
|
||||
|
135
css/style.css
135
css/style.css
@ -450,11 +450,10 @@ footer {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
gap: var(--spacing-xs);
|
||||
margin: var(--spacing-lg) auto;
|
||||
gap: var(--spacing-sm);
|
||||
margin-top: var(--spacing-lg);
|
||||
padding-top: var(--spacing-md);
|
||||
border-top: 1px solid var(--color-border);
|
||||
max-width: 100%;
|
||||
}
|
||||
|
||||
.pagination-link {
|
||||
@ -463,15 +462,13 @@ footer {
|
||||
justify-content: center;
|
||||
min-width: 2rem;
|
||||
height: 2rem;
|
||||
padding: 0 0.5rem;
|
||||
padding: 0 var(--spacing-sm);
|
||||
color: var(--color-text);
|
||||
text-decoration: none;
|
||||
background-color: var(--color-bg);
|
||||
border: 1px solid var(--color-border);
|
||||
border-radius: var(--border-radius);
|
||||
transition: all 0.2s ease;
|
||||
white-space: nowrap;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.pagination-link:hover {
|
||||
@ -486,8 +483,7 @@ footer {
|
||||
|
||||
.pagination-ellipsis {
|
||||
color: var(--color-text-muted);
|
||||
padding: 0 0.25rem;
|
||||
user-select: none;
|
||||
padding: 0 var(--spacing-xs);
|
||||
}
|
||||
|
||||
/* Bouton de suppression */
|
||||
@ -557,14 +553,6 @@ footer {
|
||||
@media (max-width: 768px) {
|
||||
.pagination {
|
||||
flex-wrap: wrap;
|
||||
gap: var(--spacing-xs);
|
||||
}
|
||||
|
||||
.pagination-link {
|
||||
min-width: 2.25rem;
|
||||
height: 2.25rem;
|
||||
padding: 0 var(--spacing-sm);
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.file-actions {
|
||||
@ -603,118 +591,3 @@ footer {
|
||||
from { opacity: 1; }
|
||||
to { opacity: 0; }
|
||||
}
|
||||
|
||||
.share-container {
|
||||
max-width: 800px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.share-container h1 {
|
||||
color: var(--color-primary);
|
||||
margin-bottom: var(--spacing-sm);
|
||||
word-break: break-all;
|
||||
}
|
||||
|
||||
.file-meta {
|
||||
color: var(--color-text-muted);
|
||||
margin-bottom: var(--spacing-lg);
|
||||
}
|
||||
|
||||
.preview-container {
|
||||
background-color: var(--color-bg);
|
||||
border: 1px solid var(--color-border);
|
||||
border-radius: var(--border-radius);
|
||||
margin-bottom: var(--spacing-lg);
|
||||
min-height: 200px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.preview-content {
|
||||
max-width: 100%;
|
||||
max-height: 600px;
|
||||
}
|
||||
|
||||
.text-preview {
|
||||
width: 100%;
|
||||
max-height: 600px;
|
||||
overflow: auto;
|
||||
padding: var(--spacing-md);
|
||||
background-color: var(--color-bg);
|
||||
color: var(--color-text);
|
||||
font-family: monospace;
|
||||
white-space: pre-wrap;
|
||||
}
|
||||
|
||||
.no-preview {
|
||||
text-align: center;
|
||||
padding: var(--spacing-lg);
|
||||
color: var(--color-text-muted);
|
||||
}
|
||||
|
||||
.extension-badge {
|
||||
background-color: var(--color-bg-alt);
|
||||
color: var(--color-primary);
|
||||
padding: var(--spacing-md) var(--spacing-lg);
|
||||
border-radius: var(--border-radius);
|
||||
font-size: 2rem;
|
||||
font-weight: bold;
|
||||
margin-bottom: var(--spacing-md);
|
||||
}
|
||||
|
||||
.share-actions {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: var(--spacing-md);
|
||||
}
|
||||
|
||||
.share-link {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: var(--spacing-xs);
|
||||
}
|
||||
|
||||
.input-group {
|
||||
display: flex;
|
||||
gap: var(--spacing-sm);
|
||||
}
|
||||
|
||||
.input-group input {
|
||||
flex: 1;
|
||||
}
|
||||
|
||||
/* Styles responsifs */
|
||||
@media (max-width: 768px) {
|
||||
.input-group {
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.input-group .btn {
|
||||
width: 100%;
|
||||
}
|
||||
}
|
||||
|
||||
/* Animation de notification */
|
||||
.notification {
|
||||
position: fixed;
|
||||
bottom: var(--spacing-lg);
|
||||
right: var(--spacing-lg);
|
||||
background-color: var(--color-success);
|
||||
color: white;
|
||||
padding: var(--spacing-md) var(--spacing-lg);
|
||||
border-radius: var(--border-radius);
|
||||
animation: slideIn 0.3s ease-out, fadeOut 0.3s ease-in 2.7s forwards;
|
||||
z-index: 1000;
|
||||
}
|
||||
|
||||
@keyframes slideIn {
|
||||
from { transform: translateY(100%); opacity: 0; }
|
||||
to { transform: translateY(0); opacity: 1; }
|
||||
}
|
||||
|
||||
@keyframes fadeOut {
|
||||
from { opacity: 1; }
|
||||
to { opacity: 0; }
|
||||
}
|
186
share.php
186
share.php
@ -10,60 +10,25 @@ if (!isset($_GET['file'])) {
|
||||
$error = 'Aucun fichier spécifié';
|
||||
} else {
|
||||
$filename = $_GET['file'];
|
||||
$path = $_GET['path'] ?? 'fichiers/';
|
||||
$filepath = UPLOAD_DIR . $filename;
|
||||
|
||||
// Déterminer le chemin complet selon le dossier
|
||||
$basePath = '';
|
||||
switch ($path) {
|
||||
case 'v1/img/fichiers/':
|
||||
$basePath = __DIR__ . '/v1/img/fichiers/';
|
||||
break;
|
||||
case 'v2/file/':
|
||||
$basePath = __DIR__ . '/v2/file/';
|
||||
break;
|
||||
default:
|
||||
$basePath = UPLOAD_DIR;
|
||||
$path = 'fichiers/'; // Assurer que le path par défaut est correct
|
||||
}
|
||||
|
||||
$filepath = $basePath . $filename;
|
||||
|
||||
// Vérifier que le fichier existe et est un fichier régulier
|
||||
if (!file_exists($filepath) || !is_file($filepath)) {
|
||||
// Vérifier si le fichier existe
|
||||
if (!file_exists($filepath)) {
|
||||
$error = 'Fichier introuvable';
|
||||
} else {
|
||||
// Vérifier que le chemin est sécurisé
|
||||
$realpath = realpath($filepath);
|
||||
$allowed = false;
|
||||
|
||||
if (strpos($realpath, realpath(UPLOAD_DIR)) === 0) {
|
||||
$allowed = true;
|
||||
} else {
|
||||
foreach (LEGACY_UPLOAD_DIRS as $dir) {
|
||||
if (strpos($realpath, realpath($dir)) === 0) {
|
||||
$allowed = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (!$allowed) {
|
||||
$error = 'Accès non autorisé';
|
||||
} else {
|
||||
// Récupérer les informations du fichier
|
||||
$file_info = [
|
||||
'name' => $filename,
|
||||
'size' => filesize($filepath),
|
||||
'extension' => strtolower(pathinfo($filename, PATHINFO_EXTENSION)),
|
||||
'preview_type' => getPreviewType(pathinfo($filename, PATHINFO_EXTENSION))
|
||||
];
|
||||
}
|
||||
// Récupérer les informations du fichier
|
||||
$file_info = [
|
||||
'name' => $filename,
|
||||
'size' => filesize($filepath),
|
||||
'extension' => strtolower(pathinfo($filename, PATHINFO_EXTENSION)),
|
||||
'preview_type' => getPreviewType(pathinfo($filename, PATHINFO_EXTENSION))
|
||||
];
|
||||
}
|
||||
}
|
||||
|
||||
// Construction des URLs
|
||||
$file_url = SITE_URL . $path . ($file_info ? rawurlencode($file_info['name']) : '');
|
||||
$share_url = SITE_URL . 'share.php?file=' . ($file_info ? rawurlencode($file_info['name']) : '') . '&path=' . urlencode($path);
|
||||
// Construction de l'URL absolue du fichier
|
||||
$file_url = SITE_URL . 'fichiers/' . ($file_info ? rawurlencode($file_info['name']) : '');
|
||||
$share_url = SITE_URL . 'share.php?file=' . ($file_info ? rawurlencode($file_info['name']) : '');
|
||||
|
||||
// Contenu de la page
|
||||
$pageTitle = $file_info ? $file_info['name'] : 'Fichier introuvable';
|
||||
@ -106,13 +71,8 @@ ob_start(); ?>
|
||||
</audio>
|
||||
<?php elseif ($file_info['preview_type'] === 'text'): ?>
|
||||
<pre class="text-preview"><?php
|
||||
// Lire et afficher le contenu du fichier texte de manière sécurisée
|
||||
$content = file_get_contents($filepath);
|
||||
if ($content !== false) {
|
||||
echo Cyla::escape($content);
|
||||
} else {
|
||||
echo "Erreur lors de la lecture du fichier";
|
||||
}
|
||||
echo Cyla::escape($content);
|
||||
?></pre>
|
||||
<?php else: ?>
|
||||
<div class="no-preview">
|
||||
@ -182,6 +142,124 @@ ob_start(); ?>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<style>
|
||||
/* Styles spécifiques à la page de partage */
|
||||
.share-container {
|
||||
max-width: 800px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.share-container h1 {
|
||||
color: var(--color-primary);
|
||||
margin-bottom: var(--spacing-sm);
|
||||
word-break: break-all;
|
||||
}
|
||||
|
||||
.file-meta {
|
||||
color: var(--color-text-muted);
|
||||
margin-bottom: var(--spacing-lg);
|
||||
}
|
||||
|
||||
.preview-container {
|
||||
background-color: var(--color-bg);
|
||||
border: 1px solid var(--color-border);
|
||||
border-radius: var(--border-radius);
|
||||
margin-bottom: var(--spacing-lg);
|
||||
min-height: 200px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.preview-content {
|
||||
max-width: 100%;
|
||||
max-height: 600px;
|
||||
}
|
||||
|
||||
.text-preview {
|
||||
width: 100%;
|
||||
max-height: 600px;
|
||||
overflow: auto;
|
||||
padding: var(--spacing-md);
|
||||
background-color: var(--color-bg);
|
||||
color: var(--color-text);
|
||||
font-family: monospace;
|
||||
white-space: pre-wrap;
|
||||
}
|
||||
|
||||
.no-preview {
|
||||
text-align: center;
|
||||
padding: var(--spacing-lg);
|
||||
color: var(--color-text-muted);
|
||||
}
|
||||
|
||||
.extension-badge {
|
||||
background-color: var(--color-bg-alt);
|
||||
color: var(--color-primary);
|
||||
padding: var(--spacing-md) var(--spacing-lg);
|
||||
border-radius: var(--border-radius);
|
||||
font-size: 2rem;
|
||||
font-weight: bold;
|
||||
margin-bottom: var(--spacing-md);
|
||||
}
|
||||
|
||||
.share-actions {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: var(--spacing-md);
|
||||
}
|
||||
|
||||
.share-link {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: var(--spacing-xs);
|
||||
}
|
||||
|
||||
.input-group {
|
||||
display: flex;
|
||||
gap: var(--spacing-sm);
|
||||
}
|
||||
|
||||
.input-group input {
|
||||
flex: 1;
|
||||
}
|
||||
|
||||
/* Styles responsifs */
|
||||
@media (max-width: 768px) {
|
||||
.input-group {
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.input-group .btn {
|
||||
width: 100%;
|
||||
}
|
||||
}
|
||||
|
||||
/* Animation de notification */
|
||||
.notification {
|
||||
position: fixed;
|
||||
bottom: var(--spacing-lg);
|
||||
right: var(--spacing-lg);
|
||||
background-color: var(--color-success);
|
||||
color: white;
|
||||
padding: var(--spacing-md) var(--spacing-lg);
|
||||
border-radius: var(--border-radius);
|
||||
animation: slideIn 0.3s ease-out, fadeOut 0.3s ease-in 2.7s forwards;
|
||||
z-index: 1000;
|
||||
}
|
||||
|
||||
@keyframes slideIn {
|
||||
from { transform: translateY(100%); opacity: 0; }
|
||||
to { transform: translateY(0); opacity: 1; }
|
||||
}
|
||||
|
||||
@keyframes fadeOut {
|
||||
from { opacity: 1; }
|
||||
to { opacity: 0; }
|
||||
}
|
||||
</style>
|
||||
|
||||
<script>
|
||||
// Fonction pour gérer la copie
|
||||
async function handleCopy(elementId, message) {
|
||||
|
Loading…
x
Reference in New Issue
Block a user