mirror of
https://codeberg.org/PostERG/xamxam.git
synced 2026-05-06 19:19:19 +02:00
SQLite performance (Database::__construct): - PRAGMA journal_mode = WAL: eliminates full-DB read locks on write, safe for concurrent PHP-FPM workers - PRAGMA synchronous = NORMAL: durable on commit without full fsync per write - PRAGMA cache_size = -8000: ~8 MB page cache per connection Accessibility foundation (WCAG 2.1 AA): - common.css: add .sr-only utility, .skip-link (hidden until focused), global :focus-visible (2px purple outline, 2px offset), prefers-reduced-motion guard; remove bare outline:none from .site-search__input - admin.css: same :focus-visible, skip-link, and motion guard scoped to admin purple; remove outline:none from .admin-input/.admin-select/ .admin-textarea and .admin-filters select (both had :focus border rules already, so focus is still visually communicated) - search.css: remove outline:none from .search-filter-select (already has :focus border-color rule) - All 5 public pages (index, search, tfe, apropos, licence): add <a href="#main-content" class="skip-link"> as first child of <body>; add id="main-content" to <main> - templates/admin/head.php: same skip link; aria-label="Navigation admin" on <nav>; id="main-content" on all 10 admin <main> elements All 4 test suites pass (unit, integration, security, rate-limit).
171 lines
7.0 KiB
PHP
171 lines
7.0 KiB
PHP
<?php
|
||
// Load configuration
|
||
require_once __DIR__ . '/../config/bootstrap.php';
|
||
require_once APP_ROOT . '/src/Database.php';
|
||
|
||
$page = isset($_GET["page"]) ? max(1, intval($_GET["page"])) : 1;
|
||
$year = isset($_GET["year"]) ? intval($_GET["year"]) : null;
|
||
$itemsPerPage = 24;
|
||
|
||
// Default home view: random theses from latest year (no year filter, no explicit page)
|
||
$isDefaultView = (!$year && $page === 1);
|
||
|
||
try {
|
||
$db = Database::getInstance();
|
||
$offset = ($page - 1) * $itemsPerPage;
|
||
$availableYears = $db->getAvailableYears();
|
||
|
||
if ($year) {
|
||
$itemsToLoad = $db->searchTheses(['year' => $year], $itemsPerPage, $offset);
|
||
$totalItems = $db->countSearchResults(['year' => $year]);
|
||
} elseif ($isDefaultView) {
|
||
$latestYear = $db->getLatestPublishedYear();
|
||
$itemsToLoad = $db->getLatestYearTheses($itemsPerPage);
|
||
$totalItems = count($itemsToLoad); // no pagination on default view
|
||
} else {
|
||
$itemsToLoad = $db->getPublishedTheses($itemsPerPage, $offset);
|
||
$totalItems = $db->countPublishedTheses();
|
||
}
|
||
|
||
$totalPages = $isDefaultView ? 1 : (int)ceil($totalItems / $itemsPerPage);
|
||
|
||
// Batch-load cover images for theses that have no banner_path
|
||
$coverMap = [];
|
||
if (!empty($itemsToLoad)) {
|
||
$needCover = array_column(
|
||
array_filter($itemsToLoad, fn($t) => empty($t['banner_path'])),
|
||
'id'
|
||
);
|
||
if (!empty($needCover)) {
|
||
$ph = implode(',', array_fill(0, count($needCover), '?'));
|
||
$cStmt = $db->getConnection()->prepare("
|
||
SELECT thesis_id, file_path FROM thesis_files
|
||
WHERE file_type = 'cover' AND thesis_id IN ($ph)
|
||
");
|
||
$cStmt->execute($needCover);
|
||
foreach ($cStmt->fetchAll() as $row) {
|
||
$coverMap[$row['thesis_id']] = $row['file_path'];
|
||
}
|
||
}
|
||
}
|
||
} catch (Exception $e) {
|
||
error_log("Error loading theses: " . $e->getMessage());
|
||
$itemsToLoad = [];
|
||
$totalPages = 0;
|
||
$availableYears = [];
|
||
$totalItems = 0;
|
||
$latestYear = null;
|
||
$isDefaultView = false;
|
||
$coverMap = [];
|
||
}
|
||
|
||
$currentNav = '';
|
||
?>
|
||
<!DOCTYPE html>
|
||
<html lang="fr">
|
||
<head>
|
||
<meta charset="UTF-8">
|
||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||
<title>Posterg</title>
|
||
<link rel="icon" type="image/svg+xml" href="/assets/admin_favicon.svg">
|
||
<link rel="stylesheet" href="assets/modern-normalize.min.css">
|
||
<link rel="stylesheet" href="assets/common.css">
|
||
<link rel="stylesheet" href="assets/main.css">
|
||
<?php if (php_sapi_name() === 'cli-server'): ?>
|
||
<script>
|
||
(function poll() {
|
||
fetch('/live-reload.php').then(r=>r.json()).then(d=>{
|
||
if(d.changed) location.reload(); else setTimeout(poll,1000);
|
||
}).catch(()=>setTimeout(poll,2000));
|
||
})();
|
||
</script>
|
||
<?php endif; ?>
|
||
</head>
|
||
<body class="home-body">
|
||
<a href="#main-content" class="skip-link">Aller au contenu principal</a>
|
||
|
||
<?php include APP_ROOT . '/templates/nav.php'; ?>
|
||
<?php include APP_ROOT . '/templates/search-bar.php'; ?>
|
||
|
||
<?php if ($year): ?>
|
||
<div class="filter-info">
|
||
Année : <?= htmlspecialchars($year) ?>
|
||
<a href="index.php" class="clear-filter">✕ Réinitialiser</a>
|
||
</div>
|
||
<?php elseif ($isDefaultView && !empty($latestYear)): ?>
|
||
<div class="filter-info home-latest-label">
|
||
Découvrez les TFE de <?= (int)$latestYear ?> — sélection aléatoire
|
||
</div>
|
||
<?php endif; ?>
|
||
|
||
<main class="home-main" id="main-content">
|
||
<div class="cards-container">
|
||
<?php foreach ($itemsToLoad as $item): ?>
|
||
<a href="tfe.php?id=<?= (int)$item["id"] ?>" class="card-link">
|
||
<div class="card">
|
||
<div class="card__media">
|
||
<?php
|
||
// Resolve thumbnail: banner_path → cover file → gradient placeholder
|
||
$thumb = null;
|
||
|
||
// 1. Banner path (dedicated home thumbnail)
|
||
if (!empty($item['banner_path'])) {
|
||
$thumb = $item['banner_path'];
|
||
}
|
||
|
||
// 2. Cover image from thesis_files (batch-loaded above)
|
||
if (!$thumb && isset($coverMap[$item['id']])) {
|
||
$thumb = $coverMap[$item['id']];
|
||
}
|
||
|
||
// 3. Fall through to gradient
|
||
?>
|
||
<?php if ($thumb): ?>
|
||
<img src="/media.php?path=<?= urlencode($thumb) ?>"
|
||
alt="<?= htmlspecialchars($item['title']) ?>"
|
||
loading="lazy">
|
||
<?php else: ?>
|
||
<?php
|
||
$hue = ($item['id'] * 47 + 160) % 360;
|
||
$hue2 = ($hue + 40) % 360;
|
||
?>
|
||
<div class="card__media--gradient"
|
||
style="background:linear-gradient(135deg,hsl(<?= $hue ?>,60%,65%),hsl(<?= $hue2 ?>,55%,45%));">
|
||
<span class="card__gradient-author"><?= htmlspecialchars($item['authors'] ?? '') ?></span>
|
||
<span class="card__gradient-title"><?= htmlspecialchars($item['title']) ?></span>
|
||
</div>
|
||
<?php endif; ?>
|
||
</div>
|
||
<div class="card__info">
|
||
<p class="authors"><?= htmlspecialchars($item["authors"] ?? '') ?><?php if (!empty($item['authors']) && !empty($item['title'])): ?> – <?php endif; ?><?= htmlspecialchars($item["title"]) ?></p>
|
||
</div>
|
||
</div>
|
||
</a>
|
||
<?php endforeach; ?>
|
||
|
||
<?php if (empty($itemsToLoad)): ?>
|
||
<p style="padding:2rem;color:#666;">Aucun mémoire trouvé.</p>
|
||
<?php endif; ?>
|
||
</div>
|
||
|
||
<?php if ($totalPages > 1): ?>
|
||
<div class="pagination-wrap">
|
||
<?php $yearParam = $year ? '&year=' . (int)$year : ''; ?>
|
||
<a href="?page=1<?= $yearParam ?>"
|
||
class="pagination-btn <?= $page <= 1 ? 'disabled' : '' ?>">«</a>
|
||
<a href="?page=<?= max(1, $page - 1) . $yearParam ?>"
|
||
class="pagination-btn <?= $page <= 1 ? 'disabled' : '' ?>">‹</a>
|
||
<span class="pagination-info">
|
||
<span class="page-current"><?= $page ?></span> / <?= $totalPages ?>
|
||
</span>
|
||
<a href="?page=<?= min($totalPages, $page + 1) . $yearParam ?>"
|
||
class="pagination-btn <?= $page >= $totalPages ? 'disabled' : '' ?>">›</a>
|
||
<a href="?page=<?= $totalPages . $yearParam ?>"
|
||
class="pagination-btn <?= $page >= $totalPages ? 'disabled' : '' ?>">»</a>
|
||
</div>
|
||
<?php endif; ?>
|
||
</main>
|
||
|
||
</body>
|
||
</html>
|