osm-commerces/templates/public/dashboard.html.twig

243 lines
9.2 KiB
Twig
Raw Normal View History

2025-05-26 11:32:53 +02:00
{% extends 'base.html.twig' %}
2025-06-03 10:52:15 +02:00
{% block title %}Tableau de bord{% endblock %}
2025-05-26 11:32:53 +02:00
{% block stylesheets %}
{{ parent() }}
2025-06-03 14:58:23 +02:00
<link href='{{ asset('js/maplibre/maplibre-gl.css') }}' rel='stylesheet' />
2025-05-26 11:32:53 +02:00
<style>
.hidden {
display: none;
}
2025-06-01 19:52:56 +02:00
#mapDashboard {
height: 400px;
width: 100%;
margin-bottom: 1rem;
}
2025-05-26 11:32:53 +02:00
</style>
{% endblock %}
2025-05-28 17:05:34 +02:00
{% block javascripts %}
{{ parent() }}
2025-06-03 14:58:23 +02:00
<script src='{{ asset('js/maplibre/maplibre-gl.js') }}'></script>
2025-06-01 23:35:15 +02:00
<script >
2025-06-03 12:51:20 +02:00
// Fonction pour rechercher avec Addok
async function searchPostalCode(query) {
try {
const response = await fetch(`https://api-adresse.data.gouv.fr/search/?q=${query}&type=municipality&autocomplete=1`);
const data = await response.json();
return data.features.map(feature => ({
label: `${feature.properties.city} (${feature.properties.postcode})`,
postcode: feature.properties.postcode,
city: feature.properties.city
}));
} catch (error) {
console.error('Erreur lors de la recherche:', error);
return [];
}
}
// Créer et configurer la liste de suggestions
const suggestionList = document.createElement('ul');
suggestionList.style.cssText = `
list-style: none;
padding: 0;
margin: 0;
border: 1px solid #ccc;
border-top: none;
max-height: 200px;
overflow-y: auto;
position: absolute;
background: white;
width: 100%;
z-index: 1000;
`;
// Configurer l'input de recherche
document.addEventListener('DOMContentLoaded', () => {
const searchInput = document.getElementById('app_admin_labourer');
const inputContainer = searchInput.parentElement;
// Ajouter un conteneur relatif pour le positionnement
const searchWrapper = document.createElement('div');
searchWrapper.style.position = 'relative';
inputContainer.appendChild(searchWrapper);
searchWrapper.appendChild(searchInput);
searchWrapper.appendChild(suggestionList);
let debounceTimer;
searchInput.addEventListener('input', async (e) => {
clearTimeout(debounceTimer);
debounceTimer = setTimeout(async () => {
const query = e.target.value;
if (query.length < 2) {
suggestionList.innerHTML = '';
return;
}
const suggestions = await searchPostalCode(query);
suggestionList.innerHTML = '';
if (suggestions.length === 0) {
const li = document.createElement('li');
li.style.cssText = `
padding: 8px 12px;
color: #666;
font-style: italic;
`;
li.textContent = 'Aucun résultat trouvé';
suggestionList.appendChild(li);
return;
}
suggestions.forEach(suggestion => {
const li = document.createElement('li');
li.style.cssText = `
padding: 8px 12px;
cursor: pointer;
border-bottom: 1px solid #eee;
`;
li.textContent = suggestion.label;
li.addEventListener('mouseenter', () => {
li.style.backgroundColor = '#f0f0f0';
});
li.addEventListener('mouseleave', () => {
li.style.backgroundColor = 'white';
});
li.addEventListener('click', () => {
searchInput.value = suggestion.postcode;
suggestionList.innerHTML = '';
labourer();
});
suggestionList.appendChild(li);
});
}, 300);
});
// Cacher la liste quand on clique ailleurs
document.addEventListener('click', (e) => {
if (!inputContainer.contains(e.target)) {
suggestionList.innerHTML = '';
}
});
});
2025-06-01 19:52:56 +02:00
// Définir la fonction labourer dans le scope global
2025-05-28 17:05:34 +02:00
function labourer() {
window.location.href = '/admin/labourer/' + document.getElementById('app_admin_labourer').value;
}
// Créer une carte des villes avec les codes postaux
2025-06-01 19:52:56 +02:00
let map = new maplibregl.Map({
container: 'mapDashboard',
style: 'https://api.maptiler.com/maps/basic-v2/style.json?key={{ maptiler_token }}',
center: [2.3488, 48.8534], // Paris
zoom: 10
});
2025-06-01 20:08:57 +02:00
// Fonction pour obtenir la couleur selon le pourcentage
function getColorFromPercent(percent) {
const red = Math.round(255 * (1 - percent/100));
const green = Math.round(255 * (percent/100));
return `rgb(${red}, ${green}, 0)`;
}
2025-06-01 20:08:57 +02:00
// Attendre le chargement du DOM
2025-06-01 23:35:15 +02:00
document.addEventListener('DOMContentLoaded', async function() {
2025-06-01 20:08:57 +02:00
// Récupérer le bouton labourer
const btnLabourer = document.querySelector('#labourer');
if (btnLabourer) {
// Ajouter l'écouteur d'événement click
btnLabourer.addEventListener('click', function() {
// Récupérer la valeur du code postal
const codePostal = document.querySelector('#app_admin_labourer').value;
// Rediriger vers la route de labourage avec le code postal
window.location.href = `/admin/labourer/${codePostal}`;
});
}
2025-06-01 23:35:15 +02:00
const postalCodes = [{% for stat in stats %}'{{ stat.zone }}'{% if not loop.last %}, {% endif %}{% endfor %}];
2025-06-03 12:51:20 +02:00
{% verbatim %}
2025-06-01 20:08:57 +02:00
console.log(postalCodes);
2025-06-01 19:52:56 +02:00
let postalLines = ``;
postalCodes.forEach(code => {
if (/^\d+$/.test(code)) {
2025-06-03 12:51:20 +02:00
postalLines += `
area["postal_code"="${code}"]->.searchArea_${code};
nwr["admin_level"="8"]["name"](area.searchArea_${code});`;
2025-06-01 19:52:56 +02:00
}
});
2025-06-01 23:35:15 +02:00
const query = `[out:json][timeout:25];
2025-06-03 12:51:20 +02:00
(
${postalLines}
);
out body;
>;
out skel qt;`;
{% endverbatim %}
2025-06-04 01:15:46 +02:00
2025-06-01 19:52:56 +02:00
});
2025-05-28 17:05:34 +02:00
</script>
{% endblock %}
2025-05-26 11:32:53 +02:00
{% block body %}
<div class="container mt-4">
<div class="row">
<div class="col-12">
<h1>Dashboard</h1>
</div>
<div class="col-12">
2025-06-01 19:52:56 +02:00
<h2>Statistiques : {{ stats|length }} codes postaux</h2>
2025-06-04 01:15:46 +02:00
{# <div id="mapDashboard"></div> #}
2025-06-03 12:51:20 +02:00
2025-05-28 17:05:34 +02:00
2025-05-27 12:17:46 +02:00
<table class="table table-hover table-striped table-responsive">
2025-05-26 11:32:53 +02:00
<thead>
<tr>
<th>Zone</th>
2025-06-03 10:52:15 +02:00
<th>Nombre de lieux</th>
2025-05-26 11:32:53 +02:00
<th>Complétude %</th>
2025-05-28 16:24:34 +02:00
<th>Actions</th>
2025-05-26 11:32:53 +02:00
</tr>
</thead>
<tbody>
{% for stat in stats %}
<tr>
2025-05-28 16:24:34 +02:00
<td>
2025-06-03 12:51:20 +02:00
<a href="{{ path('app_admin_stats', {'zip_code': stat.zone}) }}">{{ stat.zone }} {{ stat.name }}</a>
2025-05-28 16:24:34 +02:00
</td>
2025-06-04 01:15:46 +02:00
<td>{{ stat.placesCount }}</td>
2025-05-26 11:32:53 +02:00
<td style="background : rgba(0 , 255, 0, {{stat.completionPercent / 100 }} )">{{ stat.completionPercent }}</td>
2025-05-28 16:24:34 +02:00
<td>
<a class="btn btn-sm btn-primary" href="{{ path('app_admin_stats', {'zip_code': stat.zone}) }}"><i class="bi bi-eye"></i></a>
<a class="btn btn-sm btn-warning" href="{{ path('app_admin_labourer', {'zip_code': stat.zone}) }}"><i class="bi bi-arrow-repeat"></i></a>
<a class="btn btn-sm btn-danger" href="{{ path('app_admin_delete_by_zone', {'zip_code': stat.zone}) }}"><i class="bi bi-trash"></i></a>
</td>
2025-05-26 11:32:53 +02:00
</tr>
{% endfor %}
</tbody>
</table>
2025-06-04 01:03:07 +02:00
<h2>{{ places_count }} Lieux</h2>
2025-06-03 12:51:20 +02:00
<h2><button class="btn btn-primary" id="labourer">Labourer les mises à jour</button></h2>
<label for="app_admin_labourer">Rechercher une ville </label>
<input class="form-control" type="text" id="app_admin_labourer" value="75013">
2025-06-03 12:51:20 +02:00
2025-05-26 11:32:53 +02:00
</div>
</div>
</div>
{% endblock %}