5 Commits

Author SHA1 Message Date
  vincent 54ba5d1695 php-cs-fixer 4 months ago
  vincent 8cc2eea7cf composer update 4 months ago
  vincent e2ea5d81fe ajoute la dépendance à php-cs-fixer 4 months ago
  vincent 9f3cd00a24 ajoute la license 4 months ago
  vincent 6833889cf2 commente un peu plus le projet 4 months ago
38 changed files with 1654 additions and 466 deletions
Unified View
  1. +5
    -0
      .gitignore
  2. +13
    -0
      .php-cs-fixer.dist.php
  3. +13
    -0
      COPYING
  4. +58
    -8
      README.md
  5. +6
    -4
      assets/app.js
  6. +7
    -0
      assets/controllers/clipboard_controller.js
  7. +37
    -1
      assets/controllers/josm_controller.js
  8. +45
    -11
      assets/controllers/map_controller.js
  9. +1
    -0
      assets/styles/app.css
  10. +2
    -1
      composer.json
  11. +1260
    -243
      composer.lock
  12. +8
    -5
      src/Controller/HomeController.php
  13. +3
    -0
      src/Controller/MapController.php
  14. +21
    -31
      src/Controller/ProjectController.php
  15. +56
    -86
      src/Controller/TaskController.php
  16. +2
    -3
      src/DataFixtures/AppFixtures.php
  17. +3
    -0
      src/Entity/Comment.php
  18. +3
    -2
      src/Entity/Project.php
  19. +2
    -0
      src/Entity/Tag.php
  20. +9
    -8
      src/Entity/Task.php
  21. +2
    -0
      src/Entity/User.php
  22. +15
    -3
      src/EventSubscriber/TaskLifecycleSubscriber.php
  23. +0
    -2
      src/Form/CommentType.php
  24. +1
    -2
      src/Form/CsvType.php
  25. +0
    -2
      src/Form/TaskLifecycleType.php
  26. +2
    -3
      src/Form/TaskType.php
  27. +2
    -3
      src/Repository/CommentRepository.php
  28. +5
    -6
      src/Repository/TaskRepository.php
  29. +7
    -8
      src/Security/OpenStreetMapAuthenticator.php
  30. +7
    -7
      src/Service/GeoJsonManager.php
  31. +5
    -5
      src/Service/OpenStreetMapClient.php
  32. +5
    -3
      src/Service/OsmoseClient.php
  33. +6
    -6
      src/Service/OverpassClient.php
  34. +2
    -3
      src/Service/SourceGenerator.php
  35. +6
    -9
      src/Service/TaskLifecycleManager.php
  36. +12
    -0
      symfony.lock
  37. +1
    -1
      templates/_header.html.twig
  38. +22
    -0
      templates/macro.html.twig

+ 5
- 0
.gitignore View File

@ -13,3 +13,8 @@
/public/assets/ /public/assets/
/assets/vendor/ /assets/vendor/
###< symfony/asset-mapper ### ###< symfony/asset-mapper ###
###> friendsofphp/php-cs-fixer ###
/.php-cs-fixer.php
/.php-cs-fixer.cache
###< friendsofphp/php-cs-fixer ###

+ 13
- 0
.php-cs-fixer.dist.php View File

@ -0,0 +1,13 @@
<?php
$finder = (new PhpCsFixer\Finder())
->in(__DIR__)
->exclude('var')
;
return (new PhpCsFixer\Config())
->setRules([
'@Symfony' => true,
])
->setFinder($finder)
;

+ 13
- 0
COPYING View File

@ -0,0 +1,13 @@
DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
Version 2, December 2004
Copyright (C) 2004 Sam Hocevar <sam@hocevar.net>
Everyone is permitted to copy and distribute verbatim or modified
copies of this license document, and changing it is allowed as long
as the name is changed.
DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
0. You just DO WHAT THE FUCK YOU WANT TO.

+ 58
- 8
README.md View File

@ -1,4 +1,13 @@
L'idée serait de faire une application Symfony pour gérer simplement des tâches de cartographie.
# Gestionnaire de tâches simple
Voici un outil pour faciliter le travail collaboratif autour d’un projet. Le
projet regroupe des tâches que les contributeurs OSM peuvent s’approprier et
traiter dans JOSM.
# Idée de départ
L'idée serait de faire une application Symfony pour gérer simplement des tâches
de cartographie.
* authentification OAuth2 sur OSM pour ne pas avoir à créer de compte * authentification OAuth2 sur OSM pour ne pas avoir à créer de compte
* télécommande JOSM * télécommande JOSM
@ -6,19 +15,60 @@ L'idée serait de faire une application Symfony pour gérer simplement des tâch
* carte maplibre ou leaflet synchronisée avec la liste des tâches * carte maplibre ou leaflet synchronisée avec la liste des tâches
* possibilité de modifier les tâches en masse (actions groupées) * possibilité de modifier les tâches en masse (actions groupées)
* stockage des données dans sqlite * stockage des données dans sqlite
* presets dans le projet pour envoyer à JOSM (genre hashtags du message de commit, etc)
* presets dans le projet pour envoyer à JOSM (genre hashtags du message de
commit, etc)
On créé un projet (titre, description), on peut manipuler les tâches d'un On créé un projet (titre, description), on peut manipuler les tâches d'un
projet (ajouter par import geojson, supprimer, diviser en n×n ou en surface (on en déduit n)) et pour chacune
projet (ajouter par import geojson, supprimer, diviser en n×n ou en surface (on
en déduit n)) et pour chacune
d'elle on trouve un statut historicisé (à faire, en cours, fait) pour chaque d'elle on trouve un statut historicisé (à faire, en cours, fait) pour chaque
action (mapper, vérifier). On peut imagine que certains status lockent la tâche action (mapper, vérifier). On peut imagine que certains status lockent la tâche
qui se délocke au bout d'un certain temps. qui se délocke au bout d'un certain temps.
Peut-être préférer un workflow : à mapper → mappage en cours → mappage terminé à vérifier → vérification en cours → vérifié
Où les étapes *en cours* sont lockantes et reviennent au statut précédent au bout d'une journée
Peut-être préférer un workflow : à mapper → mappage en cours → mappage terminé
à vérifier → vérification en cours → vérifié
Où les étapes *en cours* sont lockantes et reviennent au statut précédent au
bout d'une journée
Un système de commentaires arborescent sur les tâches serait pas du luxe (lien
avec le statut via la date).
On peut faire des statistiques par projet sur les statuts des tâches. Et
rappeller les commentaires par ordre antéchronologique globalement sur le
projet.
En tous cas l'idéal serait de pouvoir faire tout ça via une api et de fournir
un client web en js moderne. Un client en ligne de commande serait pas du luxe
non plus.
## Mise en place technique
Il s’agit d’un petit projet Symfony (7.1.3) donc essentiellement en PHP
(développé avec la version 8.2.23) avec un peu de Javascript (utilisation du
framework Stimulus suggéré par Symfony) et de CSS (utilisation du framework
Bootstrap). Les données sont stockées dans une base SQLite localement.
Les dépendances PHP sont gérées assez classiquement par Composer. On peut donc
les récuperer avec un simple `composer install` dans la racine de
l’application.
On peut le faire tourner en local pour tester/développer grâce à l’outil en
ligne de commande [`symfony`](https://symfony.com/download) et notamment en
démarrant un serveur local : `symfony serve -d`.
L’application peut être servie par un serveur web (nginx, Apache, etc) comme
une application Symfony classique (la racine du serveur web étant dans
`/public`) pour peu qu’il interprète le PHP. Il n’y a pas de référence à des
noms de domaines donc pas de soucis pour les adresses web absolues.
La configuration de l’application se fait dans un fichier `.env.local`
(s’inspirer du `.env` fourni) dans lequel il faut essentiellement renseigner
les variables :
Un système de commentaires arborescent sur les tâches serait pas du luxe (lien avec le statut via la date).
* `APP_TIMEZONE` a priori `Europe/Paris`
* `OSM_CLIENT_ID` et `OSM_CLIENT_SECRET` à générer dans les options de son
compte OSM (onglet « application OAuth2 » avec comme URI de redirections
l’adresse web de son instance suffixée du chemin `/osm/callback` et comme
autorisation, uniquement « Lire les préférences de l’utilisateur »)
On peut faire des statistiques par projet sur les statuts des tâches. Et rappeller les commentaires par ordre antéchronologique globalement sur le projet.
En tous cas l'idéal serait de pouvoir faire tout ça via une api et de fournir un client web en js moderne. Un client en ligne de commande serait pas du luxe non plus.

+ 6
- 4
assets/app.js View File

@ -1,13 +1,15 @@
import './bootstrap.js';
import './bootstrap.js'; // Stimulus
import './vendor/bootstrap/dist/css/bootstrap.min.css';
import './vendor/leaflet/dist/leaflet.min.css';
import './styles/app.css';
import './vendor/bootstrap/dist/css/bootstrap.min.css'; // Bootstrap
import './vendor/leaflet/dist/leaflet.min.css'; // Leaflet
import './styles/app.css'; // Nos personnalisations
import { Tooltip } from './vendor/bootstrap/bootstrap.index.js'; import { Tooltip } from './vendor/bootstrap/bootstrap.index.js';
const tooltipTriggerList = document.querySelectorAll('[data-bs-toggle="tooltip"]') const tooltipTriggerList = document.querySelectorAll('[data-bs-toggle="tooltip"]')
const tooltipList = [...tooltipTriggerList].map(tooltipTriggerEl => new Tooltip(tooltipTriggerEl)) const tooltipList = [...tooltipTriggerList].map(tooltipTriggerEl => new Tooltip(tooltipTriggerEl))
// Implémente une sécurité basique pour confirmer le clic sur tous les boutons
// qui emènent sur une adresse web susceptible de supprimer quelque-chose…
document.addEventListener('click', function (event) { document.addEventListener('click', function (event) {
if (event.target.matches('a[href*="remove"]')) { if (event.target.matches('a[href*="remove"]')) {
if (!confirm(event.target.innerText + ' ?')) { if (!confirm(event.target.innerText + ' ?')) {


+ 7
- 0
assets/controllers/clipboard_controller.js View File

@ -1,3 +1,10 @@
/**
Simple contrôleur pour copier du texte dans le presse papier en cliquant
simplement sur un bouton
Cf la macro twig `clipboard`
**/
import { Controller } from "@hotwired/stimulus" import { Controller } from "@hotwired/stimulus"
export default class extends Controller { export default class extends Controller {


+ 37
- 1
assets/controllers/josm_controller.js View File

@ -1,3 +1,40 @@
/**
Implémente la télécommande JOSM
Le HTML géré pourrait ressembler à :
```html
<button
type="button"
data-controller="josm"
data-action="click->josm#remoteControl"
data-josm-commands-value="…"
>JOSM</button>
```
la valeur `commands` prend la forme dun objet JSON dont chacune des entrées
représente une commande à enchaîner successivement et la clef est le chemin
de la commande et la valeur un objet dont chacune des entrées représente les
paramètres de la commande.
Par exemple :
```json
{
"version": {},
"imagery": {
"id": "osmfr",
}
}
```
Avec cette valeur de `commands`, les appels successifs aux endpoints `/version`
puis `imagery?id=osmfr` seront effectués.
Cf <https://josm.openstreetmap.de/wiki/Help/RemoteControlCommands>
TODO Gérer le cas il ny a pas de JOSM en face pourrait être plus user-friendly.
**/
import { Controller } from '@hotwired/stimulus'; import { Controller } from '@hotwired/stimulus';
export default class extends Controller { export default class extends Controller {
@ -6,7 +43,6 @@ export default class extends Controller {
} }
remoteControl() { remoteControl() {
// cf <https://josm.openstreetmap.de/wiki/Help/RemoteControlCommands>
const baseurl = 'http://localhost:8111'; const baseurl = 'http://localhost:8111';
const _this = this, commands = JSON.parse(this.commandsValue); const _this = this, commands = JSON.parse(this.commandsValue);
for (var command in commands) { for (var command in commands) {


+ 45
- 11
assets/controllers/map_controller.js View File

@ -1,3 +1,12 @@
/**
Contrôleur qui gère la carte (charge son contenu et contrôle linteractivité)
Voir la macro Twig `map` qui produit le HTML géré par ce contrôleur.
Cf <https://leafletjs.com/reference.html>
**/
import { Controller } from '@hotwired/stimulus'; import { Controller } from '@hotwired/stimulus';
import 'leaflet'; import 'leaflet';
@ -10,13 +19,7 @@ export default class extends Controller {
} }
connect() { connect() {
const simpleIcon = L.icon({
iconUrl: this.iconValue,
iconSize: [16, 16],
iconAnchor: [8, 16],
popupAnchor: [0, 0],
});
// Constitue une collection d’icones aux couleurs Bootstrap
const iconHtml = ` const iconHtml = `
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" class="bi bi-geo-alt-fill" viewBox="0 0 16 16"> <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" class="bi bi-geo-alt-fill" viewBox="0 0 16 16">
<path fill="currentColor" d="M8 16s6-5.686 6-10A6 6 0 0 0 2 6c0 4.314 6 10 6 10m0-7a3 3 0 1 1 0-6 3 3 0 0 1 0 6"/> <path fill="currentColor" d="M8 16s6-5.686 6-10A6 6 0 0 0 2 6c0 4.314 6 10 6 10m0-7a3 3 0 1 1 0-6 3 3 0 0 1 0 6"/>
@ -30,35 +33,50 @@ export default class extends Controller {
}; };
var geojsons, _this = this, map = L.map(this.element); var geojsons, _this = this, map = L.map(this.element);
// Commence par déclarer le fond de carte classique OSM par défaut
L.tileLayer('https://tile.openstreetmap.org/{z}/{x}/{y}.png', { L.tileLayer('https://tile.openstreetmap.org/{z}/{x}/{y}.png', {
maxZoom: 19, maxZoom: 19,
attribution: '&copy; <a href="http://www.openstreetmap.org/copyright">OpenStreetMap</a>' attribution: '&copy; <a href="http://www.openstreetmap.org/copyright">OpenStreetMap</a>'
}).addTo(map); }).addTo(map);
// Crée un ensemble de couches pour mieux les manipuler
// individuellement
var layer = L.featureGroup(); var layer = L.featureGroup();
// Crée la couche dédiée à Overpass
var overpassLayer = L.featureGroup(); var overpassLayer = L.featureGroup();
if (this.overpassResultValue !== '') { if (this.overpassResultValue !== '') {
geojsons = JSON.parse(this.overpassResultValue); geojsons = JSON.parse(this.overpassResultValue);
if (geojsons.elements.length > 0) { if (geojsons.elements.length > 0) {
// Ajoute chaque forme
geojsons.elements.forEach(function (element) { geojsons.elements.forEach(function (element) {
element.members.forEach(function (member) { element.members.forEach(function (member) {
// TODO On est parti du principe que les features du
// geojson sont toutes des polylines ce qui est un peu
// réducteur, à terme il faudrait distinguer les nœuds,
// des chemins, des polygones, etc.
const polygon = L.polyline(member.geometry.map(function (coord) { return L.latLng(coord.lat, coord.lon)}), { const polygon = L.polyline(member.geometry.map(function (coord) { return L.latLng(coord.lat, coord.lon)}), {
color: '#0dcaf0',
weight: 6,
color: '#0dcaf0', // bleu info bootstrap
weight: 6, // l’idée c’est que ce soit plus gros (par défaut c’est 3) que le tracé des données des tâches parce que ce sera en dessous
opacity: 0.8, opacity: 0.8,
}).addTo(overpassLayer).bindPopup(L.popup({ }).addTo(overpassLayer).bindPopup(L.popup({
overpassElement: element,
}).setContent('…'));
overpassElement: element, // on transmet les données du geojson à la popup par là
}).setContent('…')); // le contenu définitif de la popup sera chargé plus tard en ajax
}); });
}); });
// Intervient lors de l’ouverture de la popup associée à la forme
overpassLayer.on('popupopen', function (event) { overpassLayer.on('popupopen', function (event) {
// Récupère le geojson de la forme
var element = event.popup.options.overpassElement; var element = event.popup.options.overpassElement;
// Enlève ce qui nous est inutile (les points, etc)
delete element.members; delete element.members;
// Ajoute ce qui peut-être utile (concernant la carte0
element['map'] = { element['map'] = {
'center': map.getCenter(), 'center': map.getCenter(),
'zoom': map.getZoom(), 'zoom': map.getZoom(),
}; };
// Effectue l’appel ajax pour récupérer le contenu de la popup
fetch(_this.popupUrlValue + '?' + (new URLSearchParams({ fetch(_this.popupUrlValue + '?' + (new URLSearchParams({
'element': JSON.stringify(element), 'element': JSON.stringify(element),
}))) })))
@ -73,15 +91,25 @@ export default class extends Controller {
} }
} }
// Créé la couche dédiée aux tâches
var taskLayer = L.featureGroup(); var taskLayer = L.featureGroup();
geojsons = JSON.parse(this.geojsonValue); geojsons = JSON.parse(this.geojsonValue);
if (geojsons.length > 0) { if (geojsons.length > 0) {
geojsons.forEach(function (geojson) { geojsons.forEach(function (geojson) {
// TODO Ici aussi on ne distingue pas les géométries des
// features mais ça vaudrait peut-être le coup de s’adapter un
// peu à la situation en cas de nœud, chemin ou zone…
// Par facilité on conserve les propriétés de la première feature pour plus tard
const feature0 = geojson.features[0].properties; const feature0 = geojson.features[0].properties;
// Dessine la forme de la tâche avec la proprieté `name` qui
// s’ffiche au survol et cliquable vers l’adresse web dans la
// propriété `url`
const polygon = L.geoJSON(geojson, { const polygon = L.geoJSON(geojson, {
style: function (feature) { style: function (feature) {
// Par défaut c’est un bleu par défaut de leaflet mais
// sinon on utilise les couleurs de Bootstrap
var color = 'blue'; var color = 'blue';
switch (feature.properties.color) { switch (feature.properties.color) {
case 'danger' : color = '#dc3545'; break case 'danger' : color = '#dc3545'; break
@ -94,6 +122,8 @@ export default class extends Controller {
window.location.href = event.layer.feature.properties.url; window.location.href = event.layer.feature.properties.url;
}); });
// Ajoute un marqueur au centroïde de la forme (car les
// marqueurs gardent la même taille quelque-soit le zoom
L.marker(polygon.getBounds().getCenter(), { L.marker(polygon.getBounds().getCenter(), {
icon: icons[feature0.color], icon: icons[feature0.color],
title: feature0.name, title: feature0.name,
@ -108,6 +138,8 @@ export default class extends Controller {
layer.addTo(map); layer.addTo(map);
// Si la couche Overpass n’est pas vide, ajoute le sélecteur de couches
// sur la carte
if (this.overpassResultValue !== '') { if (this.overpassResultValue !== '') {
L.control.layers({}, { L.control.layers({}, {
'Overpass': overpassLayer, 'Overpass': overpassLayer,
@ -115,6 +147,8 @@ export default class extends Controller {
}).addTo(map); }).addTo(map);
} }
// Zoome la carte pour que les données des tâches soient toutes
// visibles
map.fitBounds(taskLayer.getBounds()); map.fitBounds(taskLayer.getBounds());
} }
} }

+ 1
- 0
assets/styles/app.css View File

@ -1,3 +1,4 @@
/* Force la hauteur à un minimum de 50% de la hauteur du viewport */
.min-vh-50 { .min-vh-50 {
min-height: 50vh; min-height: 50vh;
} }

+ 2
- 1
composer.json View File

@ -1,6 +1,6 @@
{ {
"type": "project", "type": "project",
"license": "proprietary",
"license": "WTFPL",
"minimum-stability": "stable", "minimum-stability": "stable",
"prefer-stable": true, "prefer-stable": true,
"require": { "require": {
@ -42,6 +42,7 @@
}, },
"require-dev": { "require-dev": {
"doctrine/doctrine-fixtures-bundle": "^3.6", "doctrine/doctrine-fixtures-bundle": "^3.6",
"friendsofphp/php-cs-fixer": "^3.64",
"symfony/debug-bundle": "7.1.*", "symfony/debug-bundle": "7.1.*",
"symfony/maker-bundle": "^1.60", "symfony/maker-bundle": "^1.60",
"symfony/stopwatch": "7.1.*", "symfony/stopwatch": "7.1.*",


+ 1260
- 243
composer.lock
File diff suppressed because it is too large
View File


+ 8
- 5
src/Controller/HomeController.php View File

@ -3,7 +3,6 @@
namespace App\Controller; namespace App\Controller;
use KnpU\OAuth2ClientBundle\Client\ClientRegistry; use KnpU\OAuth2ClientBundle\Client\ClientRegistry;
use League\OAuth2\Client\Provider\Exception\IdentityProviderException;
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController; use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
use Symfony\Bundle\SecurityBundle\Security; use Symfony\Bundle\SecurityBundle\Security;
use Symfony\Component\HttpFoundation\Request; use Symfony\Component\HttpFoundation\Request;
@ -12,6 +11,7 @@ use Symfony\Component\Routing\Attribute\Route;
class HomeController extends AbstractController class HomeController extends AbstractController
{ {
// Page d’accueil du site
#[Route('/', name: 'app_home')] #[Route('/', name: 'app_home')]
public function index(): Response public function index(): Response
{ {
@ -19,10 +19,12 @@ class HomeController extends AbstractController
]); ]);
} }
// Page d’erreur
public function error(Request $request, $exception, $logger = null): Response public function error(Request $request, $exception, $logger = null): Response
{ {
$this->addFlash('danger', $exception->getMessage()); // . ' ' . $exception->getFile() . ':' . $exception->getLine()); $this->addFlash('danger', $exception->getMessage()); // . ' ' . $exception->getFile() . ':' . $exception->getLine());
// Si le referer est renseigné on renvoie vers celui-ci où le message d’erreur sera affiché
if ($request->headers->has('Referer')) { if ($request->headers->has('Referer')) {
return $this->redirect($request->headers->get('Referer')); return $this->redirect($request->headers->get('Referer'));
} }
@ -31,16 +33,18 @@ class HomeController extends AbstractController
]); ]);
} }
// L’oauth OSM commence ici
#[Route('/osm/request', name: 'app_osm_request')] #[Route('/osm/request', name: 'app_osm_request')]
public function osmRequest(ClientRegistry $clientRegistry): Response public function osmRequest(ClientRegistry $clientRegistry): Response
{ {
return $clientRegistry return $clientRegistry
->getClient('openstreetmap')
->getClient('openstreetmap') // cf `config/packages/knpu_oauth2_client.yaml`
->redirect([ ->redirect([
'read_prefs',
'read_prefs', // TODO est-ce qu’on a même besoin de ça ?
]); ]);
} }
// L’oauth OSM aboutit ici
#[Route('/osm/callback', name: 'app_osm_callback')] #[Route('/osm/callback', name: 'app_osm_callback')]
public function osmCallback(Request $request, ClientRegistry $clientRegistry): Response public function osmCallback(Request $request, ClientRegistry $clientRegistry): Response
{ {
@ -58,11 +62,10 @@ class HomeController extends AbstractController
return $this->redirectToRoute('app_home'); return $this->redirectToRoute('app_home');
} }
// La déconnexion passe par là mais c’est pas lié à l’oauth
#[Route('/osm/logout', name: 'app_osm_logout')] #[Route('/osm/logout', name: 'app_osm_logout')]
public function osmLogout(Security $security): Response public function osmLogout(Security $security): Response
{ {
return $security->logout(); return $security->logout();
} }
} }

+ 3
- 0
src/Controller/MapController.php View File

@ -10,9 +10,12 @@ use Symfony\Component\Routing\Attribute\Route;
#[Route('/map')] #[Route('/map')]
class MapController extends AbstractController class MapController extends AbstractController
{ {
// Produit le contenu de la popup (quand on clique sur un tracé issu d’overpass)
#[Route('/popup', name: 'app_map_popup')] #[Route('/popup', name: 'app_map_popup')]
public function popup(Request $request): Response public function popup(Request $request): Response
{ {
// Grosso modo, on récuère les données de la feature correspondante
// Cf `assets/controllers/map_controller.js`
$element = json_decode($request->query->get('element'), true); $element = json_decode($request->query->get('element'), true);
$josmCommands = [ $josmCommands = [


+ 21
- 31
src/Controller/ProjectController.php View File

@ -18,7 +18,7 @@ use Symfony\Component\Routing\Attribute\Route;
#[Route('/project')] #[Route('/project')]
class ProjectController extends AbstractController class ProjectController extends AbstractController
{ {
// Page des projets, où l'on voit tous les projets
#[Route('/', name: 'app_project')] #[Route('/', name: 'app_project')]
public function index(EntityManagerInterface $entityManager): Response public function index(EntityManagerInterface $entityManager): Response
{ {
@ -29,6 +29,7 @@ class ProjectController extends AbstractController
]); ]);
} }
// Formulaire de création d’un projet
#[Route('/create', name: 'app_project_create')] #[Route('/create', name: 'app_project_create')]
public function create(Request $request, EntityManagerInterface $entityManager): Response public function create(Request $request, EntityManagerInterface $entityManager): Response
{ {
@ -47,17 +48,11 @@ class ProjectController extends AbstractController
$entityManager->persist($project); $entityManager->persist($project);
$entityManager->flush(); $entityManager->flush();
$this->addFlash(
'success',
'Projet créé !'
);
$this->addFlash('success', 'Projet créé !');
return $this->redirectToRoute('app_project'); return $this->redirectToRoute('app_project');
} catch (\Exception $exception) { } catch (\Exception $exception) {
$this->addFlash(
'danger',
'Impossible de créer le projet !'
);
$this->addFlash('danger', 'Impossible de créer le projet !');
} }
} }
@ -66,6 +61,7 @@ class ProjectController extends AbstractController
]); ]);
} }
// Page d’un prtojet donné (où l’on voit ses tâches)
#[Route('/{slug}', name: 'app_project_show')] #[Route('/{slug}', name: 'app_project_show')]
public function show(EntityManagerInterface $entityManager, $slug): Response public function show(EntityManagerInterface $entityManager, $slug): Response
{ {
@ -77,14 +73,14 @@ class ProjectController extends AbstractController
} }
$tasks = $entityManager->getRepository(Task::class)->findByProjectPaginated($project); $tasks = $entityManager->getRepository(Task::class)->findByProjectPaginated($project);
$randomTask = $entityManager->getRepository(Task::class)->findRandomByProject($project, Task::STATUS_TODO);
$randomTask = $entityManager->getRepository(Task::class)->findRandomByProject($project, Task::STATUS_TODO); // Trouve la tâche à faire que l’on piochera
$csvForm = $this->createForm(CsvType::class, null, [ $csvForm = $this->createForm(CsvType::class, null, [
'action' => $this->generateUrl('app_project_import', ['slug' => $slug])
'action' => $this->generateUrl('app_project_import', ['slug' => $slug]),
]); ]);
$csvForm->add('submit', SubmitType::class, ['label' => 'Importer']); $csvForm->add('submit', SubmitType::class, ['label' => 'Importer']);
$comments = $entityManager->getRepository(Comment::class)->findLatestByProject($project);
$comments = $entityManager->getRepository(Comment::class)->findLatestByProject($project); // Commentaires agrégés de toutes les tâches du projet
return $this->render('project/show.html.twig', [ return $this->render('project/show.html.twig', [
'project' => $project, 'project' => $project,
@ -95,6 +91,7 @@ class ProjectController extends AbstractController
]); ]);
} }
// Import de tâches dans un projet
#[Route('/{slug}/import', name: 'app_project_import')] #[Route('/{slug}/import', name: 'app_project_import')]
public function import(Request $request, EntityManagerInterface $entityManager, $slug): Response public function import(Request $request, EntityManagerInterface $entityManager, $slug): Response
{ {
@ -112,10 +109,10 @@ class ProjectController extends AbstractController
if ($csvForm->isSubmitted() and $csvForm->isValid()) { if ($csvForm->isSubmitted() and $csvForm->isValid()) {
$csvFile = $csvForm->get('csv')->getData(); $csvFile = $csvForm->get('csv')->getData();
// TODO un peu plus de gestion d’erreur ici serait pas du luxe…
$csv = fopen($csvFile->getPathName(), 'r'); $csv = fopen($csvFile->getPathName(), 'r');
$col = array_flip(fgetcsv($csv)); $col = array_flip(fgetcsv($csv));
while ($row = fgetcsv($csv)) { while ($row = fgetcsv($csv)) {
$task = new Task(); $task = new Task();
$task->setCreatedBy($this->getUser()); $task->setCreatedBy($this->getUser());
$task->setProject($project); $task->setProject($project);
@ -140,6 +137,7 @@ class ProjectController extends AbstractController
return $this->redirectToRoute('app_project_show', ['slug' => $slug]); return $this->redirectToRoute('app_project_show', ['slug' => $slug]);
} }
// Page de modification du projet
#[Route('/{slug}/update', name: 'app_project_update')] #[Route('/{slug}/update', name: 'app_project_update')]
public function update(Request $request, EntityManagerInterface $entityManager, $slug): Response public function update(Request $request, EntityManagerInterface $entityManager, $slug): Response
{ {
@ -147,10 +145,7 @@ class ProjectController extends AbstractController
$project = $repository->findOneBySlug($slug); $project = $repository->findOneBySlug($slug);
if (!$project) { if (!$project) {
$this->addFlash(
'warning',
'Projet non trouvé !'
);
$this->addFlash('warning', 'Projet non trouvé !');
return $this->redirectToRoute('app_project'); return $this->redirectToRoute('app_project');
} }
@ -168,17 +163,11 @@ class ProjectController extends AbstractController
$entityManager->persist($project); $entityManager->persist($project);
$entityManager->flush(); $entityManager->flush();
$this->addFlash(
'success',
'Projet modifié !'
);
$this->addFlash('success', 'Projet modifié !');
return $this->redirectToRoute('app_project_show', ['slug' => $slug]); return $this->redirectToRoute('app_project_show', ['slug' => $slug]);
} catch (\Exception $exception) { } catch (\Exception $exception) {
$this->addFlash(
'danger',
'Impossible de mopdifier le projet !'
);
$this->addFlash('danger', 'Impossible de mopdifier le projet !');
} }
} }
@ -188,6 +177,7 @@ class ProjectController extends AbstractController
]); ]);
} }
// La suppression d’un projet passe par là
#[Route('/{slug}/remove', name: 'app_project_remove')] #[Route('/{slug}/remove', name: 'app_project_remove')]
public function remove(EntityManagerInterface $entityManager, $slug): Response public function remove(EntityManagerInterface $entityManager, $slug): Response
{ {
@ -206,15 +196,13 @@ class ProjectController extends AbstractController
return $this->redirectToRoute('app_project'); return $this->redirectToRoute('app_project');
} catch (\Exception $exception) { } catch (\Exception $exception) {
$this->addFlash(
'danger',
'Impossible de supprimer le projet !'
);
$this->addFlash('danger', 'Impossible de supprimer le projet !');
} }
return $this->redirectToRoute('app_project'); return $this->redirectToRoute('app_project');
} }
// Effectue la requête Overpass liée au projet
#[Route('/{slug}/overpass', name: 'app_project_overpass')] #[Route('/{slug}/overpass', name: 'app_project_overpass')]
public function overpass(OverpassClient $overpassClient, EntityManagerInterface $entityManager, $slug): Response public function overpass(OverpassClient $overpassClient, EntityManagerInterface $entityManager, $slug): Response
{ {
@ -223,20 +211,23 @@ class ProjectController extends AbstractController
if (!$project) { if (!$project) {
$this->addFlash('warning', 'Projet non trouvé !'); $this->addFlash('warning', 'Projet non trouvé !');
return $this->redirectToRoute('app_project'); return $this->redirectToRoute('app_project');
} }
if (!$project->hasOverpassQuery()) { if (!$project->hasOverpassQuery()) {
$this->addFlash('warning', 'Ce projet n’a pas de requête Overpass !'); $this->addFlash('warning', 'Ce projet n’a pas de requête Overpass !');
return $this->redirectToRoute('app_project_show', ['slug' => $project->getSlug()]); return $this->redirectToRoute('app_project_show', ['slug' => $project->getSlug()]);
} }
if ($project->hasOverpassResult() and !$project->isOverpassResultOutdated()) { if ($project->hasOverpassResult() and !$project->isOverpassResultOutdated()) {
$this->addFlash('warning', 'Merci d’attendre un peu avant de requêter de nouveau Overpass !'); $this->addFlash('warning', 'Merci d’attendre un peu avant de requêter de nouveau Overpass !');
return $this->redirectToRoute('app_project_show', ['slug' => $project->getSlug()]); return $this->redirectToRoute('app_project_show', ['slug' => $project->getSlug()]);
} }
$result = $overpassClient->query($project->getOverpassQuery());
$result = $overpassClient->query($project->getOverpassQuery());
$project->setOverpassResult($result); $project->setOverpassResult($result);
@ -245,5 +236,4 @@ class ProjectController extends AbstractController
return $this->redirectToRoute('app_project_show', ['slug' => $project->getSlug()]); return $this->redirectToRoute('app_project_show', ['slug' => $project->getSlug()]);
} }
} }

+ 56
- 86
src/Controller/TaskController.php View File

@ -8,7 +8,6 @@ use App\Entity\Task;
use App\Form\CommentType; use App\Form\CommentType;
use App\Form\TaskType; use App\Form\TaskType;
use App\Service\GeoJsonManager; use App\Service\GeoJsonManager;
use App\Service\OpenStreetMapClient;
use Doctrine\ORM\EntityManagerInterface; use Doctrine\ORM\EntityManagerInterface;
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController; use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
use Symfony\Component\Form\Extension\Core\Type\SubmitType; use Symfony\Component\Form\Extension\Core\Type\SubmitType;
@ -24,11 +23,13 @@ use Symfony\Component\Workflow\WorkflowInterface;
#[Route('/task')] #[Route('/task')]
class TaskController extends AbstractController class TaskController extends AbstractController
{ {
// Page de créatiom d’une tâche
#[Route('/create', name: 'app_task_create')] #[Route('/create', name: 'app_task_create')]
public function create(Request $request, EntityManagerInterface $entityManager): Response public function create(Request $request, EntityManagerInterface $entityManager): Response
{ {
if (!$request->query->has('slug')) { if (!$request->query->has('slug')) {
$this->addFlash( 'warning', 'Projet non spécifié !');
$this->addFlash('warning', 'Projet non spécifié !');
return $this->redirectToRoute('app_project'); return $this->redirectToRoute('app_project');
} }
@ -38,7 +39,8 @@ class TaskController extends AbstractController
$project = $repository->findOneBySlug($slug); $project = $repository->findOneBySlug($slug);
if (!$project) { if (!$project) {
$this->addFlash( 'warning', 'Projet non trouvé !');
$this->addFlash('warning', 'Projet non trouvé !');
return $this->redirectToRoute('app_project'); return $this->redirectToRoute('app_project');
} }
@ -61,10 +63,7 @@ class TaskController extends AbstractController
$entityManager->persist($task); $entityManager->persist($task);
$entityManager->flush(); $entityManager->flush();
$this->addFlash(
'success',
'Tâche créée !'
);
$this->addFlash('success', 'Tâche créée !');
return $this->redirectToRoute('app_project_show', ['slug' => $slug]); return $this->redirectToRoute('app_project_show', ['slug' => $slug]);
} catch (\Exception $exception) { } catch (\Exception $exception) {
@ -72,13 +71,13 @@ class TaskController extends AbstractController
} }
} }
return $this->render('task/create.html.twig', [ return $this->render('task/create.html.twig', [
'project' => $project, 'project' => $project,
'create_form' => $createForm, 'create_form' => $createForm,
]); ]);
} }
// Page spécifique à une tâche, où l’on trouve tout ce qui la concerne
#[Route('/{slug}', name: 'app_task_show')] #[Route('/{slug}', name: 'app_task_show')]
public function show(Request $request, EntityManagerInterface $entityManager, GeoJsonManager $geoJsonManager, $slug): Response public function show(Request $request, EntityManagerInterface $entityManager, GeoJsonManager $geoJsonManager, $slug): Response
{ {
@ -86,14 +85,12 @@ class TaskController extends AbstractController
$task = $repository->findOneBySlug($slug); $task = $repository->findOneBySlug($slug);
if (!$task) { if (!$task) {
$this->addFlash(
'warning',
'Tâche non trouvée !'
);
$this->addFlash('warning', 'Tâche non trouvée !');
if (!$request->headers->has('Referer')) { if (!$request->headers->has('Referer')) {
throw $this->createNotFoundException('Task not found'); throw $this->createNotFoundException('Task not found');
} }
return $this->redirect($request->headers->get('Referer')); return $this->redirect($request->headers->get('Referer'));
} }
@ -108,10 +105,15 @@ class TaskController extends AbstractController
'label' => 'Commenter', 'label' => 'Commenter',
]); ]);
// Programmation de la télécommande JOSM
$josmCommands = [ $josmCommands = [
// Charge l’imagerie (fond OSM par défaut)
'imagery' => [ 'imagery' => [
'id' => $project->hasImagery() ? $project->getImagery() : 'osmfr', 'id' => $project->hasImagery() ? $project->getImagery() : 'osmfr',
], ],
// Charge le XML OSM du projet danms un calque de données
// TODO C’est susceptible de planter s’il n’y a pas d’OSM dans le projet mais est-ce possible ?
'import' => [ 'import' => [
'new_layer' => true, 'new_layer' => true,
'layer_name' => $task->getName(), 'layer_name' => $task->getName(),
@ -127,14 +129,15 @@ class TaskController extends AbstractController
$geom = \geoPHP::load($task->getGeojson(), 'json'); $geom = \geoPHP::load($task->getGeojson(), 'json');
$bbox = $geom->getBBox(); $bbox = $geom->getBBox();
// Zoome sur les données et préremplit le changeset
$josmCommands['zoom'] = [ $josmCommands['zoom'] = [
'bottom' => $bbox['miny'], 'bottom' => $bbox['miny'],
'top' => $bbox['maxy'], 'top' => $bbox['maxy'],
'left' => $bbox['minx'], 'left' => $bbox['minx'],
'right' => $bbox['maxx'], 'right' => $bbox['maxx'],
'changeset_comment' => sprintf('%s %s', $project->getName(), $task->getName()), 'changeset_comment' => sprintf('%s %s', $project->getName(), $task->getName()),
'changeset_source' => $project->getSource(),
'changeset_hashtags' => $project->getHashtags(),
'changeset_source' => $project->getSource(),
'changeset_hashtags' => $project->getHashtags(),
]; ];
} }
@ -147,6 +150,7 @@ class TaskController extends AbstractController
]); ]);
} }
// Ajoute un commentaire à la tâche
#[Route('/{slug}/comment', name: 'app_task_comment')] #[Route('/{slug}/comment', name: 'app_task_comment')]
public function comment(Request $request, EntityManagerInterface $entityManager, $slug): Response public function comment(Request $request, EntityManagerInterface $entityManager, $slug): Response
{ {
@ -154,10 +158,7 @@ class TaskController extends AbstractController
$task = $repository->findOneBySlug($slug); $task = $repository->findOneBySlug($slug);
if (!$task) { if (!$task) {
$this->addFlash(
'warning',
'Tâche non trouvée !'
);
$this->addFlash('warning', 'Tâche non trouvée !');
return $this->redirect($request->headers->get('Referer')); return $this->redirect($request->headers->get('Referer'));
} }
@ -177,18 +178,15 @@ class TaskController extends AbstractController
try { try {
$entityManager->persist($comment); $entityManager->persist($comment);
$entityManager->flush(); $entityManager->flush();
} catch (\Exception $exception) { } catch (\Exception $exception) {
$this->addFlash(
'danger',
'Impossible de commenter ! ' . $exception->getMessage()
);
$this->addFlash('danger', 'Impossible de commenter ! '.$exception->getMessage());
} }
} }
return $this->redirectToRoute('app_task_show', ['slug' => $slug]); return $this->redirectToRoute('app_task_show', ['slug' => $slug]);
} }
// Modifie les informations d’une tâche
#[Route('/{slug}/update', name: 'app_task_update')] #[Route('/{slug}/update', name: 'app_task_update')]
public function update(Request $request, EntityManagerInterface $entityManager, $slug): Response public function update(Request $request, EntityManagerInterface $entityManager, $slug): Response
{ {
@ -196,10 +194,7 @@ class TaskController extends AbstractController
$task = $repository->findOneBySlug($slug); $task = $repository->findOneBySlug($slug);
if (!$task) { if (!$task) {
$this->addFlash(
'warning',
'Tâche non trouvée !'
);
$this->addFlash('warning', 'Tâche non trouvée !');
return $this->redirect($request->headers->get('Referer')); return $this->redirect($request->headers->get('Referer'));
} }
@ -219,10 +214,7 @@ class TaskController extends AbstractController
return $this->redirectToRoute('app_task_show', ['slug' => $slug]); return $this->redirectToRoute('app_task_show', ['slug' => $slug]);
} catch (\Exception $exception) { } catch (\Exception $exception) {
$this->addFlash(
'danger',
'Impossible de modifier la tâche !'
);
$this->addFlash('danger', 'Impossible de modifier la tâche !');
} }
} }
@ -233,6 +225,7 @@ class TaskController extends AbstractController
]); ]);
} }
// Supprimer une tâche
#[Route('/{slug}/remove', name: 'app_task_remove')] #[Route('/{slug}/remove', name: 'app_task_remove')]
public function remove(Request $request, EntityManagerInterface $entityManager, $slug): Response public function remove(Request $request, EntityManagerInterface $entityManager, $slug): Response
{ {
@ -240,41 +233,34 @@ class TaskController extends AbstractController
$task = $repository->findOneBySlug($slug); $task = $repository->findOneBySlug($slug);
if (!$task) { if (!$task) {
$this->addFlash(
'warning',
'Tâche non trouvée !'
);
$this->addFlash('warning', 'Tâche non trouvée !');
return $this->redirect($request->headers->get('Referer')); return $this->redirect($request->headers->get('Referer'));
} }
$project = $task->getProject(); $project = $task->getProject();
try { try {
$entityManager->remove($task); $entityManager->remove($task);
$entityManager->flush(); $entityManager->flush();
return $this->redirectToRoute('app_project_show', ['slug' => $project->getSlug()]); return $this->redirectToRoute('app_project_show', ['slug' => $project->getSlug()]);
} catch (\Exception $exception) { } catch (\Exception $exception) {
$this->addFlash(
'danger',
'Impossible de supprimer la tâche !'
);
$this->addFlash('danger', 'Impossible de supprimer la tâche !');
} }
return $this->redirectToRoute('app_project_show', ['slug' => $slug]); return $this->redirectToRoute('app_project_show', ['slug' => $slug]);
} }
// Passe une tâche d’un état à un autre
private function transition(WorkflowInterface $taskLifecycleStateMachine, EntityManagerInterface $entityManager, $slug, $transitionName): Response private function transition(WorkflowInterface $taskLifecycleStateMachine, EntityManagerInterface $entityManager, $slug, $transitionName): Response
{ {
$repository = $entityManager->getRepository(Task::class); $repository = $entityManager->getRepository(Task::class);
$task = $repository->findOneBySlug($slug); $task = $repository->findOneBySlug($slug);
if (!$task) { if (!$task) {
$this->addFlash(
'warning',
'Tâche non trouvée !'
);
$this->addFlash('warning', 'Tâche non trouvée !');
return $this->redirectToRoute('app_project'); return $this->redirectToRoute('app_project');
} }
@ -285,39 +271,41 @@ class TaskController extends AbstractController
$entityManager->flush(); $entityManager->flush();
} catch (Exception $exception) { } catch (Exception $exception) {
$this->addFlash(
'warning',
'Impossible de modifier la tâche !'
);
$this->addFlash('warning', 'Impossible de modifier la tâche !');
} }
return $this->redirectToRoute('app_task_show', ['slug' => $slug]); return $this->redirectToRoute('app_task_show', ['slug' => $slug]);
} }
// Commence une tâche
#[Route('/{slug}/start', name: 'app_task_start')] #[Route('/{slug}/start', name: 'app_task_start')]
public function start(WorkflowInterface $taskLifecycleStateMachine, EntityManagerInterface $entityManager, $slug): Response public function start(WorkflowInterface $taskLifecycleStateMachine, EntityManagerInterface $entityManager, $slug): Response
{ {
return $this->transition($taskLifecycleStateMachine, $entityManager, $slug, Task::TRANSITION_START); return $this->transition($taskLifecycleStateMachine, $entityManager, $slug, Task::TRANSITION_START);
} }
// Termine une tâche
#[Route('/{slug}/finish', name: 'app_task_finish')] #[Route('/{slug}/finish', name: 'app_task_finish')]
public function finish(WorkflowInterface $taskLifecycleStateMachine, EntityManagerInterface $entityManager, $slug): Response public function finish(WorkflowInterface $taskLifecycleStateMachine, EntityManagerInterface $entityManager, $slug): Response
{ {
return $this->transition($taskLifecycleStateMachine, $entityManager, $slug, Task::TRANSITION_FINISH); return $this->transition($taskLifecycleStateMachine, $entityManager, $slug, Task::TRANSITION_FINISH);
} }
// Abandonne une tâche
#[Route('/{slug}/cancel', name: 'app_task_cancel')] #[Route('/{slug}/cancel', name: 'app_task_cancel')]
public function cancel(WorkflowInterface $taskLifecycleStateMachine, EntityManagerInterface $entityManager, $slug): Response public function cancel(WorkflowInterface $taskLifecycleStateMachine, EntityManagerInterface $entityManager, $slug): Response
{ {
return $this->transition($taskLifecycleStateMachine, $entityManager, $slug, Task::TRANSITION_CANCEL); return $this->transition($taskLifecycleStateMachine, $entityManager, $slug, Task::TRANSITION_CANCEL);
} }
// Recommence une tâche
#[Route('/{slug}/reset', name: 'app_task_reset')] #[Route('/{slug}/reset', name: 'app_task_reset')]
public function reset(WorkflowInterface $taskLifecycleStateMachine, EntityManagerInterface $entityManager, $slug): Response public function reset(WorkflowInterface $taskLifecycleStateMachine, EntityManagerInterface $entityManager, $slug): Response
{ {
return $this->transition($taskLifecycleStateMachine, $entityManager, $slug, Task::TRANSITION_RESET); return $this->transition($taskLifecycleStateMachine, $entityManager, $slug, Task::TRANSITION_RESET);
} }
// Renvoie le geojson associé à une tâche
#[Route('/download/{slug}.geojson', name: 'app_task_geojson')] #[Route('/download/{slug}.geojson', name: 'app_task_geojson')]
public function geojson(Request $request, EntityManagerInterface $entityManager, $slug): Response public function geojson(Request $request, EntityManagerInterface $entityManager, $slug): Response
{ {
@ -325,14 +313,12 @@ class TaskController extends AbstractController
$task = $repository->findOneBySlug($slug); $task = $repository->findOneBySlug($slug);
if (!$task) { if (!$task) {
$this->addFlash(
'warning',
'Tâche non trouvée !'
);
// TODO faire pareil ailleurs où il y a des referer
$this->addFlash('warning', 'Tâche non trouvée !');
if (!$request->headers->has('Referer')) { if (!$request->headers->has('Referer')) {
throw $this->createNotFoundException('Task not found'); throw $this->createNotFoundException('Task not found');
} }
return $this->redirect($request->headers->get('Referer')); return $this->redirect($request->headers->get('Referer'));
} }
@ -350,6 +336,7 @@ class TaskController extends AbstractController
return $response; return $response;
} }
// Renvoie le gpx associé ã une tâche (concrètement il s’agit juste du geojson converti automqtiquement)
#[Route('/download/{slug}.gpx', name: 'app_task_gpx')] #[Route('/download/{slug}.gpx', name: 'app_task_gpx')]
public function gpx(Request $request, EntityManagerInterface $entityManager, $slug): Response public function gpx(Request $request, EntityManagerInterface $entityManager, $slug): Response
{ {
@ -361,6 +348,7 @@ class TaskController extends AbstractController
if (!$request->headers->has('Referer')) { if (!$request->headers->has('Referer')) {
throw $this->createNotFoundException('Task not found'); throw $this->createNotFoundException('Task not found');
} }
return $this->redirect($request->headers->get('Referer')); return $this->redirect($request->headers->get('Referer'));
} }
@ -371,6 +359,10 @@ class TaskController extends AbstractController
$geom = \geoPHP::load($task->getGeojson(), 'json'); $geom = \geoPHP::load($task->getGeojson(), 'json');
$gpx = $geom->out('gpx'); $gpx = $geom->out('gpx');
// TODO On ne vérifie rien ici en partant du principe que tout se passe
// bien. Il doit y avoir des circonstances dans lesquelles ce n’est pas
// exactement le cas.
$response = new Response($gpx); $response = new Response($gpx);
$response->headers->set('Content-Type', 'application/xml'); $response->headers->set('Content-Type', 'application/xml');
@ -382,6 +374,7 @@ class TaskController extends AbstractController
return $response; return $response;
} }
// Renvoie le XML OSM associé à la tâche
#[Route('/download/{slug}.osm', name: 'app_task_osm')] #[Route('/download/{slug}.osm', name: 'app_task_osm')]
public function osm(Request $request, EntityManagerInterface $entityManager, $slug): Response public function osm(Request $request, EntityManagerInterface $entityManager, $slug): Response
{ {
@ -393,9 +386,12 @@ class TaskController extends AbstractController
'warning', 'warning',
'Tâche non trouvée !' 'Tâche non trouvée !'
); );
return $this->redirect($request->headers->get('referer')); return $this->redirect($request->headers->get('referer'));
} }
// On est pas obligé de faire ça maid en le faisant on s’assure que
// c’est bien du XML valide qu’on envoie.
$xml = new \DOMDocument(); $xml = new \DOMDocument();
$xml->loadXml($task->getOsm()); $xml->loadXml($task->getOsm());
@ -409,7 +405,9 @@ class TaskController extends AbstractController
return $response; return $response;
} }
// Renvoie la liste des tâches du projet sous forme de CSV (ce qui devrait
// corresponddre à ce que l’on a pu importer)
#[Route('/download/{slug}.csv', name: 'app_task_csv')] #[Route('/download/{slug}.csv', name: 'app_task_csv')]
public function csv(Request $request, EntityManagerInterface $entityManager, $slug): Response public function csv(Request $request, EntityManagerInterface $entityManager, $slug): Response
{ {
@ -418,6 +416,7 @@ class TaskController extends AbstractController
if (!$project) { if (!$project) {
$this->addFlash('warning', 'Projet non trouvé !'); $this->addFlash('warning', 'Projet non trouvé !');
return $this->redirect($request->headers->get('referer')); return $this->redirect($request->headers->get('referer'));
} }
@ -434,9 +433,9 @@ class TaskController extends AbstractController
'status', 'status',
] ]
); );
foreach($project->getTasks() as $task) {
foreach ($project->getTasks() as $task) {
fputcsv( fputcsv(
$output,
$output,
[ [
$task->getName(), $task->getName(),
$task->getDescription(), $task->getDescription(),
@ -458,33 +457,4 @@ class TaskController extends AbstractController
return $response; return $response;
} }
#[Route('/{slug}/changesets', name: 'app_task_changesets')]
public function changesets(OpenStreetMapClient $osmClient, EntityManagerInterface $entityManager, $slug): Response
{
$repository = $entityManager->getRepository(Task::class);
$task = $repository->findOneBySlug($slug);
if (!$task) {
$this->addFlash('warning', 'Tâche non trouvée !');
return $this->redirect($request->headers->get('referer'));
}
$task->setChangesetsResult(
json_encode(
$osmClient->getChangesets(
$task->getDoneBy()->getUsername(),
$task->getStartAt(),
$task->getFinishAt()
)
)
);
$entityManager->persist($task);
$entityManager->flush();
return $this->redirectToRoute('app_task_show', ['slug' => $task->getSlug()]);
}
} }

+ 2
- 3
src/DataFixtures/AppFixtures.php View File

@ -2,18 +2,17 @@
namespace App\DataFixtures; namespace App\DataFixtures;
use App\Entity\Project;
use App\Entity\Tag; use App\Entity\Tag;
use App\Entity\Task;
use App\Entity\User; use App\Entity\User;
use Doctrine\Bundle\FixturesBundle\Fixture; use Doctrine\Bundle\FixturesBundle\Fixture;
use Doctrine\Persistence\ObjectManager; use Doctrine\Persistence\ObjectManager;
// Les données ci-dessous n’ont servi que lors du premier import pour gérer
// l’historique mais n’ont strictement aucun intérêt au delà.
class AppFixtures extends Fixture class AppFixtures extends Fixture
{ {
public function load(ObjectManager $manager): void public function load(ObjectManager $manager): void
{ {
$user = new User(); $user = new User();
$user->setUsername('Ptigrouick'); $user->setUsername('Ptigrouick');
$user->setOsmId(195175); $user->setOsmId(195175);


+ 3
- 0
src/Entity/Comment.php View File

@ -7,6 +7,9 @@ use Doctrine\DBAL\Types\Types;
use Doctrine\ORM\Mapping as ORM; use Doctrine\ORM\Mapping as ORM;
use Gedmo\Mapping\Annotation as Gedmo; use Gedmo\Mapping\Annotation as Gedmo;
// Les commentaires sont rédigés par les utilisateurs et associés aux tâches.
// Ils sont égelement générés automatiquements lors des changements d’états des
// tâches.
#[ORM\Entity(repositoryClass: CommentRepository::class)] #[ORM\Entity(repositoryClass: CommentRepository::class)]
class Comment class Comment
{ {


+ 3
- 2
src/Entity/Project.php View File

@ -9,6 +9,8 @@ use Doctrine\DBAL\Types\Types;
use Doctrine\ORM\Mapping as ORM; use Doctrine\ORM\Mapping as ORM;
use Gedmo\Mapping\Annotation as Gedmo; use Gedmo\Mapping\Annotation as Gedmo;
// Les projets sont des ensembles de tâches avec des paramètres de
// configuration pour la télécommande JOSM.
#[ORM\Entity(repositoryClass: ProjectRepository::class)] #[ORM\Entity(repositoryClass: ProjectRepository::class)]
class Project class Project
{ {
@ -269,11 +271,10 @@ class Project
$generatedAt = new \DateTimeImmutable($data['osm3s']['timestamp_osm_base']); $generatedAt = new \DateTimeImmutable($data['osm3s']['timestamp_osm_base']);
$now = new \DateTimeImmutable('now'); $now = new \DateTimeImmutable('now');
$minimum = new \DateInterval('PT5M');
$minimum = new \DateInterval('PT5M');
$isOutdated = ($generatedAt->add($minimum) < $now); $isOutdated = ($generatedAt->add($minimum) < $now);
return $isOutdated; return $isOutdated;
} }
} }

+ 2
- 0
src/Entity/Tag.php View File

@ -7,6 +7,8 @@ use Doctrine\Common\Collections\ArrayCollection;
use Doctrine\Common\Collections\Collection; use Doctrine\Common\Collections\Collection;
use Doctrine\ORM\Mapping as ORM; use Doctrine\ORM\Mapping as ORM;
// On utilise les tags pour étiqueter les projets, histoire de les classer le
// jour où il y en aura beaucoup.
#[ORM\Entity(repositoryClass: TagRepository::class)] #[ORM\Entity(repositoryClass: TagRepository::class)]
class Tag class Tag
{ {


+ 9
- 8
src/Entity/Task.php View File

@ -9,17 +9,19 @@ use Doctrine\DBAL\Types\Types;
use Doctrine\ORM\Mapping as ORM; use Doctrine\ORM\Mapping as ORM;
use Gedmo\Mapping\Annotation as Gedmo; use Gedmo\Mapping\Annotation as Gedmo;
// Les tâches sont l’unité de base du travail à faire. Elles sont groupées dans
// des projets et peut être appropriés par les utilisateurs selon leur état.
#[ORM\Entity(repositoryClass: TaskRepository::class)] #[ORM\Entity(repositoryClass: TaskRepository::class)]
class Task class Task
{ {
const STATUS_TODO = 'todo';
const STATUS_DOING = 'doing';
const STATUS_DONE = 'done';
public const STATUS_TODO = 'todo';
public const STATUS_DOING = 'doing';
public const STATUS_DONE = 'done';
const TRANSITION_START = 'start';
const TRANSITION_FINISH = 'finish';
const TRANSITION_CANCEL = 'cancel';
const TRANSITION_RESET = 'reset';
public const TRANSITION_START = 'start';
public const TRANSITION_FINISH = 'finish';
public const TRANSITION_CANCEL = 'cancel';
public const TRANSITION_RESET = 'reset';
#[ORM\Id] #[ORM\Id]
#[ORM\GeneratedValue] #[ORM\GeneratedValue]
@ -349,5 +351,4 @@ class Task
return $this; return $this;
} }
} }

+ 2
- 0
src/Entity/User.php View File

@ -8,6 +8,8 @@ use Doctrine\Common\Collections\Collection;
use Doctrine\ORM\Mapping as ORM; use Doctrine\ORM\Mapping as ORM;
use Symfony\Component\Security\Core\User\UserInterface; use Symfony\Component\Security\Core\User\UserInterface;
// Les utilisateurs sont générés automatiquements via l’oauth OSM et peuvent
// manipuler des projets et des tâches.
#[ORM\Entity(repositoryClass: UserRepository::class)] #[ORM\Entity(repositoryClass: UserRepository::class)]
#[ORM\UniqueConstraint(name: 'UNIQ_IDENTIFIER_USERNAME', fields: ['username'])] #[ORM\UniqueConstraint(name: 'UNIQ_IDENTIFIER_USERNAME', fields: ['username'])]
class User implements UserInterface class User implements UserInterface


+ 15
- 3
src/EventSubscriber/TaskLifecycleSubscriber.php View File

@ -15,9 +15,10 @@ use Symfony\Component\Workflow\Event\TransitionEvent;
use Symfony\Component\Workflow\WorkflowInterface; use Symfony\Component\Workflow\WorkflowInterface;
use Twig\Environment; use Twig\Environment;
// Surveille la vie des tâches et réalise automatiquement des actions selon les
// circonstances
class TaskLifecycleSubscriber implements EventSubscriberInterface class TaskLifecycleSubscriber implements EventSubscriberInterface
{ {
public function __construct( public function __construct(
protected EntityManagerInterface $entityManager, protected EntityManagerInterface $entityManager,
protected Environment $twig, protected Environment $twig,
@ -27,6 +28,7 @@ class TaskLifecycleSubscriber implements EventSubscriberInterface
) { ) {
} }
// Cas où la tâche change d’état
public function onTransition(Event $event): void public function onTransition(Event $event): void
{ {
$transition = $event->getTransition(); $transition = $event->getTransition();
@ -57,6 +59,7 @@ class TaskLifecycleSubscriber implements EventSubscriberInterface
$this->entityManager->flush(); $this->entityManager->flush();
} }
// Bloque certaines transitions
public function onGuard(Event $event): void public function onGuard(Event $event): void
{ {
$transition = $event->getTransition(); $transition = $event->getTransition();
@ -74,8 +77,9 @@ class TaskLifecycleSubscriber implements EventSubscriberInterface
} }
$event->setBlocked(false); $event->setBlocked(false);
}
}
// Cas où on commence une tâche
public function onStart(Event $event): void public function onStart(Event $event): void
{ {
$task = $event->getSubject(); $task = $event->getSubject();
@ -83,6 +87,7 @@ class TaskLifecycleSubscriber implements EventSubscriberInterface
$task->setStartAt(new \DateTimeImmutable('now')); $task->setStartAt(new \DateTimeImmutable('now'));
} }
// Cas où on termine une tâche
public function onFinish(Event $event): void public function onFinish(Event $event): void
{ {
$task = $event->getSubject(); $task = $event->getSubject();
@ -90,6 +95,7 @@ class TaskLifecycleSubscriber implements EventSubscriberInterface
$task->setFinishAt(new \DateTimeImmutable('now')); $task->setFinishAt(new \DateTimeImmutable('now'));
} }
// Cas où une tâche est terminée
public function onDone(Event $event): void public function onDone(Event $event): void
{ {
$task = $event->getSubject(); $task = $event->getSubject();
@ -97,8 +103,15 @@ class TaskLifecycleSubscriber implements EventSubscriberInterface
$task->setDoneBy($user); $task->setDoneBy($user);
// Essaie de trouver les changesets correspondants à la tâche
$task->setChangesetsResult( $task->setChangesetsResult(
json_encode($this->osmClient->getChangesets( json_encode($this->osmClient->getChangesets(
// On part du principe que les changesets correspondants sont
// ceux de l’utilisateur entre le moment où la tâche a été
// commencée et celui où elle a été terminée
// TODO C’est pas très précis, on doit pouvoir faire mieux
// (hashatg avec un id spécifique à l’app dans le commentaire
// du changeset, c’est moche mais ça marcherait)
$user->getUsername(), $user->getUsername(),
$task->getStartAt(), $task->getStartAt(),
$task->getFinishAt() $task->getFinishAt()
@ -116,5 +129,4 @@ class TaskLifecycleSubscriber implements EventSubscriberInterface
EnteredEvent::getName('task_lifecycle', Task::STATUS_DONE) => 'onDone', EnteredEvent::getName('task_lifecycle', Task::STATUS_DONE) => 'onDone',
]; ];
} }
} }

+ 0
- 2
src/Form/CommentType.php View File

@ -3,8 +3,6 @@
namespace App\Form; namespace App\Form;
use App\Entity\Comment; use App\Entity\Comment;
use App\Entity\Task;
use Symfony\Bridge\Doctrine\Form\Type\EntityType;
use Symfony\Component\Form\AbstractType; use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface; use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\OptionsResolver\OptionsResolver; use Symfony\Component\OptionsResolver\OptionsResolver;


+ 1
- 2
src/Form/CsvType.php View File

@ -5,7 +5,6 @@ namespace App\Form;
use Symfony\Component\Form\AbstractType; use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\Extension\Core\Type\FileType; use Symfony\Component\Form\Extension\Core\Type\FileType;
use Symfony\Component\Form\FormBuilderInterface; use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\OptionsResolver\OptionsResolver;
use Symfony\Component\Validator\Constraints\File; use Symfony\Component\Validator\Constraints\File;
class CsvType extends AbstractType class CsvType extends AbstractType
@ -24,7 +23,7 @@ class CsvType extends AbstractType
'text/plain', 'text/plain',
], ],
'mimeTypesMessage' => 'Type MIME inattendu', 'mimeTypesMessage' => 'Type MIME inattendu',
])
]),
], ],
'help' => 'Fichier CSV classique (encodé en UTF8, séparé par des virgules, entouré avec des doubles guillemets, échappé avec des barres obliques et avec des retours chariots UNIX) contenant une ligne de noms de colonnes « name,description,osm,geojson,status » et enfin dans la colonne « status » la valeur « todo », « doing » ou « done ».', 'help' => 'Fichier CSV classique (encodé en UTF8, séparé par des virgules, entouré avec des doubles guillemets, échappé avec des barres obliques et avec des retours chariots UNIX) contenant une ligne de noms de colonnes « name,description,osm,geojson,status » et enfin dans la colonne « status » la valeur « todo », « doing » ou « done ».',
]) ])


+ 0
- 2
src/Form/TaskLifecycleType.php View File

@ -2,12 +2,10 @@
namespace App\Form; namespace App\Form;
use App\Entity\Task;
use Symfony\Component\Form\AbstractType; use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\Extension\Core\Type\ChoiceType; use Symfony\Component\Form\Extension\Core\Type\ChoiceType;
use Symfony\Component\OptionsResolver\OptionsResolver; use Symfony\Component\OptionsResolver\OptionsResolver;
use Symfony\Component\Workflow\WorkflowInterface; use Symfony\Component\Workflow\WorkflowInterface;
use Symfony\Component\DependencyInjection\Attribute\Target;
class TaskLifecycleType extends AbstractType class TaskLifecycleType extends AbstractType
{ {


+ 2
- 3
src/Form/TaskType.php View File

@ -3,7 +3,6 @@
namespace App\Form; namespace App\Form;
use App\Entity\Task; use App\Entity\Task;
use App\Form\TaskLifecycleType;
use Symfony\Component\Form\AbstractType; use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\Extension\Core\Type\TextareaType; use Symfony\Component\Form\Extension\Core\Type\TextareaType;
use Symfony\Component\Form\FormBuilderInterface; use Symfony\Component\Form\FormBuilderInterface;
@ -24,7 +23,7 @@ class TaskType extends AbstractType
'label' => 'GeoJSON', 'label' => 'GeoJSON',
'required' => false, 'required' => false,
'help_html' => true, 'help_html' => true,
'help' => 'Ce qu’il faut dessinner sur la carte au format <a href="https://fr.wikipedia.org/wiki/GeoJSON" target="_blank">GeoJSON</a> et mettre à disposition à l’export sous forme de <a href="https://fr.wikipedia.org/wiki/GPX_(format_de_fichier)" target="_blank">GPX</a> pour l’import dans JOSM. Outil pratique&nbsp;: <a href="ttps://geojson.io/" target="_blank">geojson.io</a>.',
'help' => 'Ce qu’il faut dessiner sur la carte au format <a href="https://fr.wikipedia.org/wiki/GeoJSON" target="_blank">GeoJSON</a> et mettre à disposition à l’export sous forme de <a href="https://fr.wikipedia.org/wiki/GPX_(format_de_fichier)" target="_blank">GPX</a> pour l’import dans JOSM. Outil pratique&nbsp;: <a href="ttps://geojson.io/" target="_blank">geojson.io</a>.',
]) ])
->add('osm', TextareaType::class, [ ->add('osm', TextareaType::class, [
'label' => 'OSM', 'label' => 'OSM',
@ -33,7 +32,7 @@ class TaskType extends AbstractType
'help' => 'XML décrivant ce qu’il faut charger dans la feuille de données de JOSM (cf <a href="https://wiki.openstreetmap.org/wiki/FR:OSM_XML" target="_blank">FR:OSM XML - OpenStreetMap Wiki</a>)', 'help' => 'XML décrivant ce qu’il faut charger dans la feuille de données de JOSM (cf <a href="https://wiki.openstreetmap.org/wiki/FR:OSM_XML" target="_blank">FR:OSM XML - OpenStreetMap Wiki</a>)',
]) ])
->add('status', TaskLifecycleType::class, [ ->add('status', TaskLifecycleType::class, [
'label' => 'État'
'label' => 'État',
]) ])
->add('urgent', null, [ ->add('urgent', null, [
'label' => 'Urgence', 'label' => 'Urgence',


+ 2
- 3
src/Repository/CommentRepository.php View File

@ -2,9 +2,8 @@
namespace App\Repository; namespace App\Repository;
use App\Entity\Task;
use App\Entity\Project;
use App\Entity\Comment; use App\Entity\Comment;
use App\Entity\Project;
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository; use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
use Doctrine\Persistence\ManagerRegistry; use Doctrine\Persistence\ManagerRegistry;
@ -30,7 +29,7 @@ class CommentRepository extends ServiceEntityRepository
->orderBy('c.createdAt', 'DESC') ->orderBy('c.createdAt', 'DESC')
->getQuery() ->getQuery()
->getResult() ->getResult()
;
;
} }
// public function findOneBySomeField($value): ?Comment // public function findOneBySomeField($value): ?Comment


+ 5
- 6
src/Repository/TaskRepository.php View File

@ -9,7 +9,7 @@ use Doctrine\Persistence\ManagerRegistry;
use Knp\Component\Pager\PaginatorInterface; use Knp\Component\Pager\PaginatorInterface;
use Symfony\Component\HttpFoundation\Request; use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\RequestStack; use Symfony\Component\HttpFoundation\RequestStack;
/** /**
* @extends ServiceEntityRepository<Task> * @extends ServiceEntityRepository<Task>
*/ */
@ -76,7 +76,7 @@ class TaskRepository extends ServiceEntityRepository
->setMaxResults(1) ->setMaxResults(1)
->getQuery() ->getQuery()
->getOneOrNullResult() ->getOneOrNullResult()
;
;
} }
public function findRandomByProject(Project $project, $status = null) public function findRandomByProject(Project $project, $status = null)
@ -92,11 +92,10 @@ class TaskRepository extends ServiceEntityRepository
$tasks = $qb->getQuery() $tasks = $qb->getQuery()
->getResult() ->getResult()
;
;
shuffle($tasks); shuffle($tasks);
return reset($tasks);
}
return reset($tasks);
}
} }

+ 7
- 8
src/Security/OpenStreetMapAuthenticator.php View File

@ -2,7 +2,7 @@
namespace App\Security; namespace App\Security;
use App\Entity\User;
use App\Entity\User;
use Doctrine\ORM\EntityManagerInterface; use Doctrine\ORM\EntityManagerInterface;
use KnpU\OAuth2ClientBundle\Client\ClientRegistry; use KnpU\OAuth2ClientBundle\Client\ClientRegistry;
use KnpU\OAuth2ClientBundle\Security\Authenticator\OAuth2Authenticator; use KnpU\OAuth2ClientBundle\Security\Authenticator\OAuth2Authenticator;
@ -17,19 +17,18 @@ use Symfony\Component\Security\Http\Authenticator\Passport\Passport;
use Symfony\Component\Security\Http\Authenticator\Passport\SelfValidatingPassport; use Symfony\Component\Security\Http\Authenticator\Passport\SelfValidatingPassport;
use Symfony\Component\Security\Http\EntryPoint\AuthenticationEntryPointInterface; use Symfony\Component\Security\Http\EntryPoint\AuthenticationEntryPointInterface;
class OpenStreetMapAuthenticator extends OAuth2Authenticator implements AuthenticationEntrypointInterface
class OpenStreetMapAuthenticator extends OAuth2Authenticator implements AuthenticationEntryPointInterface
{ {
public function __construct( public function __construct(
private ClientRegistry $clientRegistry, private ClientRegistry $clientRegistry,
private EntityManagerInterface $entityManager, private EntityManagerInterface $entityManager,
private RouterInterface $router, private RouterInterface $router,
)
{
) {
} }
public function supports(Request $request): ?bool public function supports(Request $request): ?bool
{ {
return $request->attributes->get('_route') === 'app_osm_callback';
return 'app_osm_callback' === $request->attributes->get('_route');
} }
public function authenticate(Request $request): Passport public function authenticate(Request $request): Passport
@ -41,7 +40,7 @@ class OpenStreetMapAuthenticator extends OAuth2Authenticator implements Authenti
$session->set('access_token', $accessToken); $session->set('access_token', $accessToken);
return new SelfValidatingPassport( return new SelfValidatingPassport(
new UserBadge($accessToken->getToken(), function() use ($accessToken, $client) {
new UserBadge($accessToken->getToken(), function () use ($accessToken, $client) {
$resourceOwner = $client->fetchUserFromToken($accessToken); $resourceOwner = $client->fetchUserFromToken($accessToken);
$existingUser = $this->entityManager->getRepository(User::class)->findOneBy(['osmId' => $resourceOwner->getId()]); $existingUser = $this->entityManager->getRepository(User::class)->findOneBy(['osmId' => $resourceOwner->getId()]);
@ -72,8 +71,8 @@ class OpenStreetMapAuthenticator extends OAuth2Authenticator implements Authenti
{ {
return null; return null;
} }
public function start(Request $request, AuthenticationException $authException = null): Response
public function start(Request $request, ?AuthenticationException $authException = null): Response
{ {
return new RedirectResponse( return new RedirectResponse(
'/', '/',


+ 7
- 7
src/Service/GeoJsonManager.php View File

@ -10,13 +10,13 @@ use Symfony\Component\Workflow\WorkflowInterface;
class GeoJsonManager class GeoJsonManager
{ {
public function __construct( public function __construct(
private UrlGeneratorInterface $router, private UrlGeneratorInterface $router,
private WorkflowInterface $taskLifecycleStateMachine, private WorkflowInterface $taskLifecycleStateMachine,
) { ) {
} }
// Enrichit le geojson d’une tâche avec des propriétés calcuées
private function getFullGeoJson(Task $task) private function getFullGeoJson(Task $task)
{ {
if (!$task->hasGeojson()) { if (!$task->hasGeojson()) {
@ -31,12 +31,12 @@ class GeoJsonManager
if (!isset($data['features']) or empty($data['features'])) { if (!isset($data['features']) or empty($data['features'])) {
return null; return null;
}
}
foreach($data['features'] as $index => $feature) {
foreach ($data['features'] as $index => $feature) {
if (!isset($feature['properties'])) { if (!isset($feature['properties'])) {
continue; continue;
}
}
$feature['properties'] = array_merge($feature['properties'], [ $feature['properties'] = array_merge($feature['properties'], [
'name' => $task->getName(), 'name' => $task->getName(),
'url' => $this->router->generate('app_task_show', ['slug' => $task->getSlug()], UrlGeneratorInterface::ABSOLUTE_URL), 'url' => $this->router->generate('app_task_show', ['slug' => $task->getSlug()], UrlGeneratorInterface::ABSOLUTE_URL),
@ -44,11 +44,12 @@ class GeoJsonManager
]); ]);
$data['features'][$index] = $feature; $data['features'][$index] = $feature;
} }
return $data; return $data;
} }
public function generateGeoJson($entity)
// Produit le geojson d’un projet ou d’une tâche de façon transparente pour simplifier
public function generateGeoJson($entity)
{ {
$geoJsons = []; $geoJsons = [];
@ -70,5 +71,4 @@ class GeoJsonManager
return $geoJsons; return $geoJsons;
} }
} }

+ 5
- 5
src/Service/OpenStreetMapClient.php View File

@ -6,8 +6,9 @@ use KnpU\OAuth2ClientBundle\Client\ClientRegistry;
use Symfony\Component\HttpFoundation\RequestStack; use Symfony\Component\HttpFoundation\RequestStack;
use Symfony\Contracts\HttpClient\HttpClientInterface; use Symfony\Contracts\HttpClient\HttpClientInterface;
class OpenStreetMapClient {
// Fournit un accès à l’API OSM
class OpenStreetMapClient
{
public function __construct( public function __construct(
private ClientRegistry $clientRegistry, private ClientRegistry $clientRegistry,
private HttpClientInterface $client, private HttpClientInterface $client,
@ -27,12 +28,12 @@ class OpenStreetMapClient {
*/ */
$token = $accessToken->getToken(); $token = $accessToken->getToken();
$response = $this->client->request($method, 'https://api.openstreetmap.org/api/0.6/' . $path, [
$response = $this->client->request($method, 'https://api.openstreetmap.org/api/0.6/'.$path, [
'auth_bearer' => $token, 'auth_bearer' => $token,
'query' => $params, 'query' => $params,
]); ]);
$isStatusCodeOk = ($response->getStatusCode() === 200);
$isStatusCodeOk = (200 === $response->getStatusCode());
if (!$isStatusCodeOk) { if (!$isStatusCodeOk) {
throw new \RuntimeException(); throw new \RuntimeException();
@ -80,5 +81,4 @@ class OpenStreetMapClient {
return $changesets; return $changesets;
} }
} }

+ 5
- 3
src/Service/OsmoseClient.php View File

@ -19,11 +19,13 @@ namespace App\Service;
use Symfony\Contracts\HttpClient\HttpClientInterface; use Symfony\Contracts\HttpClient\HttpClientInterface;
class OsmoseClient {
// Fournira un accès à l’API Osmose si nécessaire (histoire de vérifier qu’on a
// pas trop oublié de tenir compte de la validation côté JOSM lors de la
// contribution sur une tâche ?)
class OsmoseClient
{
public function __construct( public function __construct(
private HttpClientInterface $client, private HttpClientInterface $client,
) { ) {
} }
} }

+ 6
- 6
src/Service/OverpassClient.php View File

@ -4,8 +4,9 @@ namespace App\Service;
use Symfony\Contracts\HttpClient\HttpClientInterface; use Symfony\Contracts\HttpClient\HttpClientInterface;
class OverpassClient {
// Founit un accès à l’API Overpass
class OverpassClient
{
public function __construct( public function __construct(
private HttpClientInterface $client, private HttpClientInterface $client,
) { ) {
@ -15,12 +16,12 @@ class OverpassClient {
{ {
$response = $this->client->request('GET', 'https://overpass-api.de/api/interpreter', [ $response = $this->client->request('GET', 'https://overpass-api.de/api/interpreter', [
'query' => [ 'query' => [
'data' => $prefix . $query . $suffix,
'data' => $prefix.$query.$suffix,
], ],
]); ]);
$isStatusCodeOk = ($response->getStatusCode() === 200);
$isContentTypeJson = ($response->getHeaders()['content-type'][0] === 'application/json');
$isStatusCodeOk = (200 === $response->getStatusCode());
$isContentTypeJson = ('application/json' === $response->getHeaders()['content-type'][0]);
if (!$isStatusCodeOk or !$isContentTypeJson) { if (!$isStatusCodeOk or !$isContentTypeJson) {
throw new \RuntimeException(); throw new \RuntimeException();
@ -28,5 +29,4 @@ class OverpassClient {
return $response->getContent(); return $response->getContent();
} }
} }

+ 2
- 3
src/Service/SourceGenerator.php View File

@ -4,9 +4,9 @@ namespace App\Service;
use App\Entity\Task; use App\Entity\Task;
// Génère la source de changeset
class SourceGenerator class SourceGenerator
{ {
public function generate(Task $task): string public function generate(Task $task): string
{ {
$parts = []; $parts = [];
@ -16,7 +16,7 @@ class SourceGenerator
$parts[] = $project->getName(); $parts[] = $project->getName();
$parts[] = $task->getName(); $parts[] = $task->getName();
foreach(explode(' ', $project->getHashtags()) as $hashtag) {
foreach (explode(' ', $project->getHashtags()) as $hashtag) {
$parts[] = $hashtag; $parts[] = $hashtag;
} }
@ -24,5 +24,4 @@ class SourceGenerator
return $source; return $source;
} }
} }

+ 6
- 9
src/Service/TaskLifecycleManager.php View File

@ -6,9 +6,9 @@ use App\Entity\Project;
use App\Entity\Task; use App\Entity\Task;
use Symfony\Component\Workflow\WorkflowInterface; use Symfony\Component\Workflow\WorkflowInterface;
// Utilistaire pour récupérer facilement les stats de réalisation d’un projet ou d’une tâche
class TaskLifecycleManager class TaskLifecycleManager
{ {
public function __construct( public function __construct(
private WorkflowInterface $taskLifecycleStateMachine, private WorkflowInterface $taskLifecycleStateMachine,
) { ) {
@ -40,13 +40,12 @@ class TaskLifecycleManager
} }
$max = 0; $max = 0;
foreach ($project->getTasks() as $task)
{
$stats[$task->getStatus()]['value'] += 1;
$max += 1;
foreach ($project->getTasks() as $task) {
++$stats[$task->getStatus()]['value'];
++$max;
} }
if ($max === 0) {
if (0 === $max) {
return $stats; return $stats;
} }
@ -55,10 +54,8 @@ class TaskLifecycleManager
$data['percentage'] = ((float) $data['value'] * 100.0) / $max; $data['percentage'] = ((float) $data['value'] * 100.0) / $max;
return $data; return $data;
}, $stats);
}, $stats);
return $stats; return $stats;
} }
} }

+ 12
- 0
symfony.lock View File

@ -38,6 +38,18 @@
"migrations/.gitignore" "migrations/.gitignore"
] ]
}, },
"friendsofphp/php-cs-fixer": {
"version": "3.64",
"recipe": {
"repo": "github.com/symfony/recipes",
"branch": "main",
"version": "3.0",
"ref": "be2103eb4a20942e28a6dd87736669b757132435"
},
"files": [
".php-cs-fixer.dist.php"
]
},
"knplabs/knp-paginator-bundle": { "knplabs/knp-paginator-bundle": {
"version": "v6.4.0" "version": "v6.4.0"
}, },


+ 1
- 1
templates/_header.html.twig View File

@ -1,7 +1,7 @@
<header> <header>
<nav class="navbar navbar-expand-lg bg-body-tertiary"> <nav class="navbar navbar-expand-lg bg-body-tertiary">
<div class="container-fluid"> <div class="container-fluid">
<a class="navbar-brand" href="{{ path('app_home') }}">{{ title }}</a>
<a class="navbar-brand" href="{{ path('app_home') }}" title="{{ long_title }}">{{ title }}</a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbar" aria-controls="navbar" aria-expanded="false" aria-label="Toggle navigation"> <button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbar" aria-controls="navbar" aria-expanded="false" aria-label="Toggle navigation">
<span class="navbar-toggler-icon"></span> <span class="navbar-toggler-icon"></span>
</button> </button>


+ 22
- 0
templates/macro.html.twig View File

@ -19,6 +19,17 @@
{% endif %} {% endif %}
{% endmacro %} {% endmacro %}
{#
Produit le HTML géré par le contrôleur Stimulus `map` et permet d’afficher une
carte interactive
Où :
* `entity` peut être une instance de `Project` ou de `Task` pour représenter ce
qu’il faut mapper
* `overpassResult` est optionnel pour représenter ce qui est actuellement mappé
#}
{% macro map(entity, overpassResult='') %} {% macro map(entity, overpassResult='') %}
<div <div
id="map" id="map"
@ -41,6 +52,17 @@
</details> </details>
{% endmacro %} {% endmacro %}
{#
Génère le HTML qui va bien pour un bloc de texte que l’on peut copier dans le
presse papier en cliquant simplement sur un bouton.
Cf le controlleur Stimulus `clipboard`
TODO Rajouter un retour visuel pour signifier que la copie a bien eu lieu
serait pas du luxe…
#}
{% macro clipboard(text) %} {% macro clipboard(text) %}
<div class="input-group" data-controller="clipboard"> <div class="input-group" data-controller="clipboard">
<input type="text" readonly class="form-control" value="{{ text }}" data-clipboard-target="source" /> <input type="text" readonly class="form-control" value="{{ text }}" data-clipboard-target="source" />


Loading…
Cancel
Save