mirror of
https://codeberg.org/PostERG/xamxam.git
synced 2026-05-06 19:19:19 +02:00
Move CSV import to inline dialog on list page
This commit is contained in:
@@ -10,6 +10,211 @@ if (empty($_SESSION['csrf_token'])) {
|
||||
$pageTitle = "Liste des TFE";
|
||||
require_once __DIR__ . '/../../src/Database.php';
|
||||
|
||||
// ── CSV Import (inline, submitted to this same page) ─────────────────────────
|
||||
$importMessage = '';
|
||||
$importErrors = [];
|
||||
$importResults = [];
|
||||
$importDone = false;
|
||||
|
||||
if ($_SERVER['REQUEST_METHOD'] === 'POST' && isset($_FILES['csv_file'])) {
|
||||
if (!isset($_POST['csrf_token']) || !hash_equals($_SESSION['csrf_token'], $_POST['csrf_token'])) {
|
||||
$importErrors[] = "Erreur de sécurité : token invalide.";
|
||||
} else {
|
||||
$importedCount = 0;
|
||||
$skippedCount = 0;
|
||||
try {
|
||||
$importDb = new Database();
|
||||
$importPdo = $importDb->getPDO();
|
||||
|
||||
if ($_FILES['csv_file']['error'] !== UPLOAD_ERR_OK) {
|
||||
throw new Exception("Erreur lors du téléversement du fichier.");
|
||||
}
|
||||
|
||||
$handle = fopen($_FILES['csv_file']['tmp_name'], 'r');
|
||||
if (!$handle) throw new Exception("Impossible d'ouvrir le fichier CSV.");
|
||||
|
||||
fgetcsv($handle, 0, ',', '"', '');
|
||||
fgetcsv($handle, 0, ',', '"', '');
|
||||
fgetcsv($handle, 0, ',', '"', '');
|
||||
fgetcsv($handle, 0, ',', '"', ''); // skip 4 header rows
|
||||
|
||||
$orientationMap = [
|
||||
'SC'=>'Sculpture','VI'=>'Vidéographie','CA'=>"Cinéma d'animation",
|
||||
'IP'=>'Installation-Performance','PE'=>'Peinture','PH'=>'Photographie',
|
||||
'DE'=>'Dessin','AN'=>'Arts Numériques','GR'=>'Graphisme',
|
||||
'TY'=>'Typographie','DN'=>'Design Numérique','IL'=>'Illustration',
|
||||
'BD'=>'Bande-Dessinée','SE'=>'Sérigraphie','GV'=>'Gravure',
|
||||
];
|
||||
|
||||
$lineNumber = 5;
|
||||
while (($row = fgetcsv($handle, 0, ',', '"', '')) !== false) {
|
||||
$lineNumber++;
|
||||
if (empty($row[0]) && empty($row[1])) continue;
|
||||
try {
|
||||
$importDb->beginTransaction();
|
||||
|
||||
$identifier = trim($row[0] ?? '');
|
||||
$title = trim($row[1] ?? '');
|
||||
$subtitle = trim($row[2] ?? '');
|
||||
$authorsRaw = trim($row[3] ?? '');
|
||||
$contact = trim($row[4] ?? '');
|
||||
$supervisorsRaw = trim($row[5] ?? '');
|
||||
$formatsRaw = trim($row[6] ?? '');
|
||||
$year = intval($row[7] ?? 0);
|
||||
$apCode = trim($row[8] ?? '');
|
||||
$orientationCode = trim($row[9] ?? '');
|
||||
$finalityName = trim($row[10] ?? '');
|
||||
$keywordsRaw = trim($row[11] ?? '');
|
||||
$synopsis = trim($row[12] ?? '');
|
||||
$context = trim($row[13] ?? '');
|
||||
$remarks = trim($row[14] ?? '');
|
||||
$languageRaw = trim($row[15] ?? '');
|
||||
$access = trim($row[16] ?? '');
|
||||
$license = trim($row[17] ?? '');
|
||||
$sizeInfo = trim($row[18] ?? '');
|
||||
$juryPoints = !empty($row[19]) ? floatval($row[19]) : null;
|
||||
$baiuLink = trim($row[20] ?? '');
|
||||
|
||||
if (empty($title) || empty($year)) throw new Exception("Titre et année requis.");
|
||||
|
||||
$orientationName = $orientationMap[$orientationCode] ?? null;
|
||||
$orientationId = null;
|
||||
if ($orientationName) {
|
||||
$s = $importPdo->prepare("SELECT id FROM orientations WHERE name = ?");
|
||||
$s->execute([$orientationName]);
|
||||
$r = $s->fetch(); $orientationId = $r ? $r['id'] : null;
|
||||
}
|
||||
|
||||
$apProgramId = null;
|
||||
if (!empty($apCode)) {
|
||||
$s = $importPdo->prepare("SELECT id FROM ap_programs WHERE code = ?");
|
||||
$s->execute([$apCode]);
|
||||
$r = $s->fetch(); $apProgramId = $r ? $r['id'] : null;
|
||||
}
|
||||
|
||||
$finalityId = null;
|
||||
if (!empty($finalityName)) {
|
||||
$s = $importPdo->prepare("SELECT id FROM finality_types WHERE name = ?");
|
||||
$s->execute([$finalityName]);
|
||||
$r = $s->fetch(); $finalityId = $r ? $r['id'] : null;
|
||||
}
|
||||
|
||||
$accessTypeId = null;
|
||||
if (!empty($access)) {
|
||||
$s = $importPdo->prepare("SELECT id FROM access_types WHERE name = ?");
|
||||
$s->execute([ucfirst(strtolower($access))]);
|
||||
$r = $s->fetch(); $accessTypeId = $r ? $r['id'] : null;
|
||||
}
|
||||
if ($accessTypeId === null) $accessTypeId = 1;
|
||||
|
||||
if (!empty($identifier)) {
|
||||
$s = $importPdo->prepare("SELECT id FROM theses WHERE identifier = ?");
|
||||
$s->execute([$identifier]);
|
||||
if ($s->fetch()) {
|
||||
$importDb->rollback();
|
||||
$skippedCount++;
|
||||
$importResults[] = ['type'=>'skip', 'msg'=>"Ligne $lineNumber: identifiant \"$identifier\" déjà présent, ignoré."];
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
$s = $importPdo->prepare("
|
||||
INSERT INTO theses (
|
||||
identifier, title, subtitle, year,
|
||||
orientation_id, ap_program_id, finality_id,
|
||||
synopsis, context_note, remarks,
|
||||
file_size_info, jury_points, baiu_link,
|
||||
access_type_id, is_published, submitted_at
|
||||
) VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?,1,CURRENT_TIMESTAMP)
|
||||
");
|
||||
$s->execute([
|
||||
!empty($identifier) ? $identifier : null, $title,
|
||||
!empty($subtitle) ? $subtitle : null, $year,
|
||||
$orientationId, $apProgramId, $finalityId,
|
||||
!empty($synopsis) ? $synopsis : null,
|
||||
!empty($context) ? $context : null,
|
||||
!empty($remarks) ? $remarks : null,
|
||||
!empty($sizeInfo) ? $sizeInfo : null,
|
||||
$juryPoints,
|
||||
!empty($baiuLink) ? $baiuLink : null,
|
||||
$accessTypeId,
|
||||
]);
|
||||
$thesisId = $importPdo->lastInsertId();
|
||||
|
||||
if (!empty($authorsRaw)) {
|
||||
foreach (array_map('trim', explode(',', $authorsRaw)) as $idx => $name) {
|
||||
if ($name) {
|
||||
$aId = $importDb->findOrCreateAuthor($name, $idx === 0 ? $contact : null);
|
||||
$s = $importPdo->prepare("INSERT INTO thesis_authors (thesis_id, author_id, author_order) VALUES (?,?,?)");
|
||||
$s->execute([$thesisId, $aId, $idx + 1]);
|
||||
}
|
||||
}
|
||||
}
|
||||
if (!empty($supervisorsRaw)) {
|
||||
foreach (array_map('trim', explode(',', $supervisorsRaw)) as $idx => $name) {
|
||||
if ($name) {
|
||||
$sId = $importDb->findOrCreateSupervisor($name);
|
||||
$s = $importPdo->prepare("INSERT INTO thesis_supervisors (thesis_id, supervisor_id, supervisor_order) VALUES (?,?,?)");
|
||||
$s->execute([$thesisId, $sId, $idx + 1]);
|
||||
}
|
||||
}
|
||||
}
|
||||
if (!empty($keywordsRaw)) {
|
||||
foreach (array_slice(array_map('trim', explode(',', $keywordsRaw)), 0, 10) as $kw) {
|
||||
if ($kw) {
|
||||
$tId = $importDb->findOrCreateTag($kw);
|
||||
if ($tId) {
|
||||
$s = $importPdo->prepare("INSERT INTO thesis_tags (thesis_id, tag_id) VALUES (?,?)");
|
||||
$s->execute([$thesisId, $tId]);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (!empty($languageRaw)) {
|
||||
$s = $importPdo->prepare("SELECT id FROM languages WHERE name = ?");
|
||||
$s->execute([ucfirst(strtolower($languageRaw))]);
|
||||
$r = $s->fetch();
|
||||
if ($r) {
|
||||
$s2 = $importPdo->prepare("INSERT INTO thesis_languages (thesis_id, language_id) VALUES (?,?)");
|
||||
$s2->execute([$thesisId, $r['id']]);
|
||||
}
|
||||
}
|
||||
if (!empty($formatsRaw)) {
|
||||
foreach (array_map('trim', explode(',', $formatsRaw)) as $fmt) {
|
||||
if ($fmt) {
|
||||
$s = $importPdo->prepare("SELECT id FROM format_types WHERE name = ?");
|
||||
$s->execute([ucfirst(strtolower($fmt))]);
|
||||
$r = $s->fetch();
|
||||
if ($r) {
|
||||
$s2 = $importPdo->prepare("INSERT INTO thesis_formats (thesis_id, format_id) VALUES (?,?)");
|
||||
$s2->execute([$thesisId, $r['id']]);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
$importDb->commit();
|
||||
$importedCount++;
|
||||
$importResults[] = ['type'=>'ok', 'msg'=>"\"$title\" (ID: $thesisId)"];
|
||||
|
||||
} catch (Exception $e) {
|
||||
$importDb->rollback();
|
||||
$skippedCount++;
|
||||
$importResults[] = ['type'=>'error', 'msg'=>"Ligne $lineNumber: " . $e->getMessage()];
|
||||
error_log("Import error on line $lineNumber: " . $e->getMessage());
|
||||
}
|
||||
}
|
||||
fclose($handle);
|
||||
$importMessage = "Import terminé : $importedCount TFE importés, $skippedCount ignorés.";
|
||||
$importDone = true;
|
||||
} catch (Exception $e) {
|
||||
$importErrors[] = $e->getMessage();
|
||||
error_log("CSV import error: " . $e->getMessage());
|
||||
}
|
||||
}
|
||||
$_SESSION['csrf_token'] = bin2hex(random_bytes(32));
|
||||
}
|
||||
|
||||
try {
|
||||
$db = new Database();
|
||||
$searchQuery = isset($_GET['search']) ? trim($_GET['search']) : '';
|
||||
@@ -72,26 +277,32 @@ document.addEventListener('DOMContentLoaded', () => {
|
||||
</script>
|
||||
|
||||
<main id="main-content">
|
||||
<h1>Liste des TFE</h1>
|
||||
<div class="admin-list-header">
|
||||
<div class="admin-list-header__top">
|
||||
<h1>Liste des TFE</h1>
|
||||
<dl class="admin-stats">
|
||||
<div class="admin-stat">
|
||||
<dt class="admin-stat__label">TFE total</dt>
|
||||
<dd class="admin-stat__number"><?= $stats['total'] ?></dd>
|
||||
</div>
|
||||
<div class="admin-stat">
|
||||
<dt class="admin-stat__label">Publiés</dt>
|
||||
<dd class="admin-stat__number"><?= $stats['published'] ?></dd>
|
||||
</div>
|
||||
<div class="admin-stat">
|
||||
<dt class="admin-stat__label">En attente</dt>
|
||||
<dd class="admin-stat__number"><?= $stats['pending'] ?></dd>
|
||||
</div>
|
||||
</dl>
|
||||
</div>
|
||||
<button type="button" class="admin-btn" id="import-dialog-btn"
|
||||
onclick="document.getElementById('import-dialog').showModal()">
|
||||
Importer un CSV
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<?php include APP_ROOT . '/templates/partials/flash-messages.php'; ?>
|
||||
|
||||
<!-- Stats (always reflects full DB, independent of active filters) -->
|
||||
<dl class="admin-stats">
|
||||
<div class="admin-stat">
|
||||
<dt class="admin-stat__label">TFE total</dt>
|
||||
<dd class="admin-stat__number"><?= $stats['total'] ?></dd>
|
||||
</div>
|
||||
<div class="admin-stat">
|
||||
<dt class="admin-stat__label">Publiés</dt>
|
||||
<dd class="admin-stat__number"><?= $stats['published'] ?></dd>
|
||||
</div>
|
||||
<div class="admin-stat">
|
||||
<dt class="admin-stat__label">En attente</dt>
|
||||
<dd class="admin-stat__number"><?= $stats['pending'] ?></dd>
|
||||
</div>
|
||||
</dl>
|
||||
|
||||
<!-- Filters -->
|
||||
<form class="admin-filters" method="get" action="/admin/">
|
||||
<input type="text" name="search" placeholder="Titre, auteur..."
|
||||
@@ -216,4 +427,66 @@ document.addEventListener('DOMContentLoaded', () => {
|
||||
?>
|
||||
</main>
|
||||
|
||||
<!-- ══════════════════════════════════════════════════════════════
|
||||
IMPORT DIALOG
|
||||
══════════════════════════════════════════════════════════════ -->
|
||||
<dialog id="import-dialog" class="admin-dialog" aria-labelledby="import-dialog-title">
|
||||
<div class="admin-dialog__header">
|
||||
<h2 id="import-dialog-title">Importer une liste de TFE</h2>
|
||||
<button type="button" class="admin-dialog__close" aria-label="Fermer"
|
||||
onclick="document.getElementById('import-dialog').close()">✕</button>
|
||||
</div>
|
||||
|
||||
<?php if (!empty($importErrors)): ?>
|
||||
<div role="alert" data-type="error" class="admin-dialog__alert">
|
||||
<strong>⚠ Erreurs :</strong>
|
||||
<ul class="admin-error-list">
|
||||
<?php foreach ($importErrors as $err): ?>
|
||||
<li><?= htmlspecialchars($err) ?></li>
|
||||
<?php endforeach; ?>
|
||||
</ul>
|
||||
</div>
|
||||
<?php endif; ?>
|
||||
|
||||
<?php if ($importMessage): ?>
|
||||
<p role="status" data-type="success">✓ <?= htmlspecialchars($importMessage) ?></p>
|
||||
<?php endif; ?>
|
||||
|
||||
<form method="post" enctype="multipart/form-data" class="admin-form">
|
||||
<input type="hidden" name="csrf_token" value="<?= htmlspecialchars($_SESSION['csrf_token']) ?>">
|
||||
|
||||
<div>
|
||||
<label for="csv_file">Fichier CSV</label>
|
||||
<div class="admin-file-input">
|
||||
<input type="file" id="csv_file" name="csv_file" accept=".csv" required>
|
||||
<small class="admin-file-hint">
|
||||
Colonnes : Identifiant, Titre, Sous-titre, Auteur·ice(s), Contact, Promoteur·ice(s), Format, Année, AP, Orientation, Finalité, Mots-clés, Synopsis, Contexte, Remarques, Langue, Autorisation, License, taille, Points sur 20, lien BAIU<br>
|
||||
Quatre premières lignes ignorées — Séparateur : virgule — UTF-8
|
||||
</small>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="admin-form-footer">
|
||||
<button type="submit" class="admin-btn">Importer</button>
|
||||
<button type="button" class="admin-btn-secondary"
|
||||
onclick="document.getElementById('import-dialog').close()">Annuler</button>
|
||||
</div>
|
||||
</form>
|
||||
|
||||
<?php if (!empty($importResults)): ?>
|
||||
<div class="admin-import-results">
|
||||
<h3 class="admin-import-results__title">Résultats</h3>
|
||||
<ul class="admin-import-log">
|
||||
<?php foreach ($importResults as $r): ?>
|
||||
<li class="admin-import-log__item admin-import-log__item--<?= $r['type'] ?>"><?= htmlspecialchars($r['msg']) ?></li>
|
||||
<?php endforeach; ?>
|
||||
</ul>
|
||||
</div>
|
||||
<?php endif; ?>
|
||||
</dialog>
|
||||
|
||||
<?php if ($importDone || !empty($importErrors)): ?>
|
||||
<script>document.getElementById('import-dialog').showModal();</script>
|
||||
<?php endif; ?>
|
||||
|
||||
<?php require_once APP_ROOT . '/templates/admin/footer.php'; ?>
|
||||
|
||||
Reference in New Issue
Block a user