jolie beau et ajax

This commit is contained in:
Yasder5 2026-03-02 21:01:46 +01:00
parent 7475c3ce49
commit 42ec68178a
11 changed files with 242 additions and 76 deletions

112
views/_partial/apigeo.tpl Normal file
View file

@ -0,0 +1,112 @@
<!-- Script d'autocomplétion ville/département — API Géo (gouvernement français) -->
<script>
(function () {
const input = document.getElementById('user_location');
const suggestions = document.getElementById('location-suggestions');
let debounceTimer = null;
// Ferme la liste si on clique ailleurs
document.addEventListener('click', function (e) {
if (!input.contains(e.target) && !suggestions.contains(e.target)) {
hideSuggestions();
}
});
input.addEventListener('input', function () {
const query = this.value.trim();
clearTimeout(debounceTimer);
if (query.length < 2) {
hideSuggestions();
return;
}
// Délai de 300 ms pour éviter trop d'appels API
debounceTimer = setTimeout(function () {
fetchCities(query);
}, 300);
});
// Navigation clavier dans la liste
input.addEventListener('keydown', function (e) {
const items = suggestions.querySelectorAll('.list-group-item');
const active = suggestions.querySelector('.list-group-item.active');
let index = Array.from(items).indexOf(active);
if (e.key === 'ArrowDown') {
e.preventDefault();
if (index < items.length - 1) setActive(items, index + 1);
} else if (e.key === 'ArrowUp') {
e.preventDefault();
if (index > 0) setActive(items, index - 1);
} else if (e.key === 'Enter' && active) {
e.preventDefault();
selectItem(active.dataset.value);
} else if (e.key === 'Escape') {
hideSuggestions();
}
});
function fetchCities(query) {
// On cherche par nom de commune, on récupère aussi le département
const url = 'https://geo.api.gouv.fr/communes?nom=' + encodeURIComponent(query)
+ '&fields=nom,departement&boost=population&limit=8';
fetch(url)
.then(function (res) { return res.json(); })
.then(function (data) { renderSuggestions(data); })
.catch(function () { hideSuggestions(); });
}
function renderSuggestions(cities) {
suggestions.innerHTML = '';
if (!cities || cities.length === 0) {
hideSuggestions();
return;
}
cities.forEach(function (city) {
const dept = city.departement
? city.departement.nom + ' (' + city.departement.code + ')'
: '';
const label = city.nom + (dept ? ' — ' + dept : '');
// Valeur stockée : "Ville (Département)"
const value = city.nom + (city.departement ? ' (' + city.departement.nom + ')' : '');
const li = document.createElement('li');
li.className = 'list-group-item list-group-item-action py-2 px-3';
li.style.cursor = 'pointer';
li.dataset.value = value;
li.textContent = label;
li.addEventListener('mousedown', function (e) {
// mousedown avant blur pour éviter que la liste disparaisse avant le clic
e.preventDefault();
selectItem(value);
});
suggestions.appendChild(li);
});
suggestions.style.display = 'block';
}
function selectItem(value) {
input.value = value;
hideSuggestions();
}
function hideSuggestions() {
suggestions.style.display = 'none';
suggestions.innerHTML = '';
}
function setActive(items, index) {
items.forEach(function (item) { item.classList.remove('active'); });
items[index].classList.add('active');
input.value = items[index].dataset.value;
}
})();
</script>

View file

@ -5,29 +5,34 @@
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<link rel="stylesheet" href="assests/css/style.css">
<link rel="shortcut icon" href="assests/img/Group-49.ico" type="image/x-icon">
<!-- <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css"> -->
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.8/dist/css/bootstrap.min.css" rel="stylesheet" integrity="sha384-sRIl4kxILFvY47J16cr9ZwB07vP4J8+LH7qKQnuqkuIAvNWLzeN8tE5YBujZqJLB" crossorigin="anonymous">
<script src="https://use.fontawesome.com/releases/v6.3.0/js/all.js" crossorigin="anonymous"></script>
<title>Folliow{block name="title"}{/block}</title>
</head>
<body class="d-flex flex-column min-vh-100">
<nav class="navbar navbar-expand-lg navbar-light">
<div class="container-fluid">
<!-- Logo -->
<a class="navbar-brand d-flex align-items-center" href="index.php">
<img src="assests/img/logo.png" alt="Logo" class="logo-image">
</a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav" aria-controls="navbarNav" aria-expanded="false" aria-label="Toggle navigation">
<!-- Bouton hamburger mobile -->
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav"
aria-controls="navbarNav" aria-expanded="false" aria-label="Toggle navigation">
<span class="navbar-toggler-icon"></span>
</button>
<!-- Liens de navigation -->
<div class="collapse navbar-collapse" id="navbarNav">
<ul class="navbar-nav me-auto">
<!-- Liens gauche -->
<ul class="navbar-nav me-auto mb-0">
<li class="nav-item">
<a class="nav-link" href="?ctrl=page&action=about">À propos</a>
</li>
<li class="nav-item">
<a class="nav-link" href="?ctrl=project&action=search">Rechercher</a>
</li>
@ -42,62 +47,65 @@
{/if}
{/if}
</ul>
<nav class="col-4 d-flex justify-content-end align-items-center" aria-label="Connexion utilisateur">
<!-- Liens droite (connexion / profil) -->
<nav aria-label="Connexion utilisateur">
{if !isset($smarty.session.user)}
<ul class="navbar-nav">
<ul class="navbar-nav d-flex flex-row align-items-center gap-1 mb-0">
<li class="nav-item">
<a class="nav-link" href="index.php?ctrl=user&action=signup" title="Créer un compte" aria-label="Créer un compte">
<a class="nav-link" href="index.php?ctrl=user&action=signup"
title="Créer un compte" aria-label="Créer un compte">
S'inscrire
</a>
</li>
<li class="nav-item">
<a class="nav-link" href="index.php?ctrl=user&action=login" title="Se connecter" aria-label="Se connecter">
Se connecter
<li class="nav-item">
<a class="nav-link" href="index.php?ctrl=user&action=login"
title="Se connecter" aria-label="Se connecter">
Se connecter
</a>
</li>
</ul>
</ul>
{else}
<ul class="navbar-nav">
<li class="nav-item">
<a class="nav-link" href="index.php?ctrl=user&action=user&pseudo={$smarty.session.user.user_pseudo}" title="Modifier mon compte" aria-label="Modifier mon compte">
<img src={$smarty.env.IMG_USER_PATH}{if ($smarty.env.IMG_USER_PATH|cat:($smarty.session.user.user_image))|file_exists}{$smarty.session.user.user_image}{else}images.jpg{/if}
class="rounded-circle flex-shrink-0 mt-2 ml-5"
style="width: 36px; height: 36px; object-fit: cover;"
alt="Photo de profil">
<ul class="navbar-nav d-flex flex-row align-items-center gap-2 mb-0">
<li class="nav-item d-flex align-items-center">
<a class="nav-link p-0" href="index.php?ctrl=user&action=user&pseudo={$smarty.session.user.user_pseudo}"
title="Modifier mon compte" aria-label="Modifier mon compte">
<img
src="{$smarty.env.IMG_USER_PATH}{if ($smarty.env.IMG_USER_PATH|cat:($smarty.session.user.user_image))|file_exists}{$smarty.session.user.user_image}{else}images.jpg{/if}"
class="nav-avatar"
alt="Photo de profil"
>
</a>
</li>
<li>
<a class="nav-link" href="index.php?ctrl=user&action=logout" title="Se déconnecter" aria-label="Se déconnecter">
<li class="nav-item d-flex align-items-center">
<a class="nav-link" href="index.php?ctrl=user&action=logout"
title="Se déconnecter" aria-label="Se déconnecter">
Se déconnecter
</a>
</li>
</ul>
{/if}
</nav>
</div>
</div>
</nav>
<main id="main-content" class="container my-4">
<section
class="p-4 mb-4 text-center txt_title "
aria-labelledby="page-title"
>
<div class="col-lg-8 mx-auto">
<h2 id="page-title" class="display-5 fw-semibold mb-3">
{block name="h2"}{/block}
</h2>
<p class="lead mb-2">
{block name="p"}{/block}
</p>
<p class="text-muted small mb-0">
{block name="date_maj"}{/block}
</p>
</div>
</section>
</main>
<main id="main-content" class="container my-4">
<section class="p-4 mb-4 text-center txt_title" aria-labelledby="page-title">
<div class="col-lg-8 mx-auto">
<h2 id="page-title" class="display-5 fw-semibold mb-3">
{block name="h2"}{/block}
</h2>
<p class="lead mb-2">
{block name="p"}{/block}
</p>
<p class="text-muted small mb-0">
{block name="date_maj"}{/block}
</p>
</div>
</section>
</main>
</body>
{include file="views/_partial/messages.tpl"}

View file

@ -14,7 +14,7 @@
<div class="card-body p-3 bg-light">
<div class="d-flex align-items-start gap-3">
<a href="index.php?ctrl=user&action=user&id={$objProject->getUser_id()}">
<a href="index.php?ctrl=user&action=user&pseudo={$objProject->getCreatorname()}">
<img src="{$smarty.env.IMG_USER_PATH}{if ($smarty.env.IMG_USER_PATH|cat:($objProject->getUser_image()))|file_exists}{$objProject->getUser_image()}{else}images.jpg{/if}"
class="rounded-circle flex-shrink-0 border border-2 border-white"
style="width: 64px; height: 64px; object-fit: cover; margin-top: 8px;"

View file

@ -3,7 +3,6 @@
{block name="content"}
<!-- Page : Inscription -->
<div class="py-5">
<!-- Centrage horizontal du formulaire -->
@ -150,13 +149,23 @@
<label class="form-label" for="user_location">
Localisation
</label>
<input
class="form-control"
type="text"
id="user_location"
name="user_location"
value="{$objUser->getLocation()|default:''}"
>
<div class="position-relative">
<input
class="form-control"
type="text"
id="user_location"
name="user_location"
value="{$objUser->getLocation()|default:''}"
autocomplete="off"
placeholder="Ex : Paris, Lyon..."
>
<!-- Liste déroulante des suggestions -->
<ul
id="location-suggestions"
class="list-group position-absolute w-100 shadow-sm"
style="z-index: 1000; display: none; max-height: 220px; overflow-y: auto; top: 100%; left: 0;"
></ul>
</div>
</div>
<!-- Champ optionnel : phrase d'accroche -->
@ -168,9 +177,8 @@
class="form-control"
id="user_description"
name="user_description"
value="{$objUser->getDescription()|default:''}"
rows="3"
></textarea>
>{$objUser->getDescription()|default:''}</textarea>
</div>
<!-- Bouton de soumission du formulaire -->
@ -199,4 +207,5 @@
</div>
</div>
{include file="views/_partial/apigeo.tpl"}
{/block}

View file

@ -15,18 +15,30 @@
<p class="text-muted">{$user->getMail()}</p>
{if $user->getWork()}
<p>{$user->getWork()}</p>
<div class="d-flex align-items-center gap-2 mt-3">
<i class="fa-solid fa-briefcase"></i>
<p class="mb-0">{$user->getWork()}</p>
</div>
{/if}
{if $user->getLocation()}
<p>{$user->getLocation()}</p>
<div class="d-flex align-items-center gap-2 mt-3">
<i class="fa-solid fa-location-dot"></i>
<p class="mb-0">{$user->getLocation()}</p>
</div>
{/if}
{if $user->getLocation()}
<div class="d-flex align-items-center gap-2 mt-3">
<i class="fa-regular fa-note-sticky"></i> <p class="mb-0">{$user->getDescription()}</p>
</div>
{/if}
<p class="mt-3">{$user->getDescription()}</p>
{if $smarty.session.user.user_id == $user->getId()}
<a class="btn btn-sm btn-primary flex-fill"
href="?ctrl=user&action=edit">Edit account</a>
{/if}
{/if}
</div>
</div>

View file

@ -132,18 +132,29 @@
</div>
<div class="col-12">
<label class="form-label" for="user_location">
Localisation
</label>
<input
class="form-control"
type="text"
id="user_location"
name="user_location"
value="{$objUser->getLocation()}"
>
</div>
<!-- Champ optionnel : localisation de l'utilisateur -->
<div class="col-12">
<label class="form-label" for="user_location">
Localisation
</label>
<div class="position-relative">
<input
class="form-control"
type="text"
id="user_location"
name="user_location"
value="{$objUser->getLocation()}"
autocomplete="off"
placeholder="Ex : Paris, Lyon..."
>
<!-- Liste déroulante des suggestions -->
<ul
id="location-suggestions"
class="list-group position-absolute w-100 shadow-sm"
style="z-index: 1000; display: none; max-height: 220px; overflow-y: auto; top: 100%; left: 0;"
></ul>
</div>
</div>
<div class="col-12">
@ -176,4 +187,6 @@
</div>
</div>
</main>
{include file="views/_partial/apigeo.tpl"}
{/block}