You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

198 lines
11 KiB

6 months ago
6 months ago
6 months ago
6 months ago
6 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
6 months ago
6 months ago
6 months ago
6 months ago
6 months ago
6 months ago
6 months ago
6 months ago
6 months ago
6 months ago
6 months ago
6 months ago
6 months ago
  1. {% extends 'base.html.twig' %}
  2. {% import 'macro.html.twig' as macro %}
  3. {% block breadcrumb %}
  4. <li class="breadcrumb-item"><a href="{{ path('app_project') }}">Projets</a></li>
  5. <li class="breadcrumb-item"><a href="{{ path('app_project_show', {'slug': project.slug}) }}">{{ project.name }}</a></li>
  6. {% endblock %}
  7. {% block page_title %}
  8. {{ project.name }}
  9. {% for tag in project.tags %}
  10. <span class="badge text-bg-info ms-2">{{ tag.name }}</span>
  11. {% endfor %}
  12. {% endblock %}
  13. {% block page_content %}
  14. <div class="row">
  15. <div class="col mb-3">
  16. <div class="btn-toolbar" role="toolbar">
  17. <div class="btn-group me-2">
  18. <a href="{{ path('app_project') }}" class="btn btn-secondary" title="Revenir aux projets">
  19. <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-house" viewBox="0 0 16 16">
  20. <path d="M8.707 1.5a1 1 0 0 0-1.414 0L.646 8.146a.5.5 0 0 0 .708.708L2 8.207V13.5A1.5 1.5 0 0 0 3.5 15h9a1.5 1.5 0 0 0 1.5-1.5V8.207l.646.647a.5.5 0 0 0 .708-.708L13 5.793V2.5a.5.5 0 0 0-.5-.5h-1a.5.5 0 0 0-.5.5v1.293zM13 7.207V13.5a.5.5 0 0 1-.5.5h-9a.5.5 0 0 1-.5-.5V7.207l5-5z"/>
  21. </svg>
  22. Projets
  23. </a>
  24. </div>
  25. {% if is_granted('IS_AUTHENTICATED_FULLY') %}
  26. {% if app.user is same as(project.createdBy) %}
  27. <div class="btn-group me-2">
  28. <a href="{{ path('app_project_update', {'slug': project.slug}) }}" class="btn btn-secondary" title="Modifier le projet">
  29. <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-pencil" viewBox="0 0 16 16">
  30. <path d="M12.146.146a.5.5 0 0 1 .708 0l3 3a.5.5 0 0 1 0 .708l-10 10a.5.5 0 0 1-.168.11l-5 2a.5.5 0 0 1-.65-.65l2-5a.5.5 0 0 1 .11-.168zM11.207 2.5 13.5 4.793 14.793 3.5 12.5 1.207zm1.586 3L10.5 3.207 4 9.707V10h.5a.5.5 0 0 1 .5.5v.5h.5a.5.5 0 0 1 .5.5v.5h.293zm-9.761 5.175-.106.106-1.528 3.821 3.821-1.528.106-.106A.5.5 0 0 1 5 12.5V12h-.5a.5.5 0 0 1-.5-.5V11h-.5a.5.5 0 0 1-.468-.325"/>
  31. </svg>
  32. Modifier
  33. </a>
  34. <a href="{{ path('app_project_remove', {'slug': project.slug}) }}" class="btn btn-secondary" title="Supprimer le projet">
  35. <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-trash" viewBox="0 0 16 16">
  36. <path d="M5.5 5.5A.5.5 0 0 1 6 6v6a.5.5 0 0 1-1 0V6a.5.5 0 0 1 .5-.5m2.5 0a.5.5 0 0 1 .5.5v6a.5.5 0 0 1-1 0V6a.5.5 0 0 1 .5-.5m3 .5a.5.5 0 0 0-1 0v6a.5.5 0 0 0 1 0z"/>
  37. <path d="M14.5 3a1 1 0 0 1-1 1H13v9a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V4h-.5a1 1 0 0 1-1-1V2a1 1 0 0 1 1-1H6a1 1 0 0 1 1-1h2a1 1 0 0 1 1 1h3.5a1 1 0 0 1 1 1zM4.118 4 4 4.059V13a1 1 0 0 0 1 1h6a1 1 0 0 0 1-1V4.059L11.882 4zM2.5 3h11V2h-11z"/>
  38. </svg>
  39. Supprimer
  40. </a>
  41. </div>
  42. {% endif %}
  43. {% if project.overpassQuery %}
  44. <div class="btn-group me-2">
  45. <a href="{{ path('app_project_overpass', {'slug': project.slug}) }}" class="btn btn-secondary" title="Requêter Overpass">
  46. <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-arrow-clockwise" viewBox="0 0 16 16">
  47. <path fill-rule="evenodd" d="M8 3a5 5 0 1 0 4.546 2.914.5.5 0 0 1 .908-.417A6 6 0 1 1 8 2z"/>
  48. <path d="M8 4.466V.534a.25.25 0 0 1 .41-.192l2.36 1.966c.12.1.12.284 0 .384L8.41 4.658A.25.25 0 0 1 8 4.466"/>
  49. </svg>
  50. Overpass
  51. </a>
  52. </div>
  53. {% endif %}
  54. <div class="btn-group me-2">
  55. <button type="button" class="btn btn-secondary" data-bs-toggle="modal" data-bs-target="#modal" title="Importer des tâches">
  56. <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-file-earmark-arrow-down" viewBox="0 0 16 16">
  57. <path d="M8.5 6.5a.5.5 0 0 0-1 0v3.793L6.354 9.146a.5.5 0 1 0-.708.708l2 2a.5.5 0 0 0 .708 0l2-2a.5.5 0 0 0-.708-.708L8.5 10.293z"/>
  58. <path d="M14 14V4.5L9.5 0H4a2 2 0 0 0-2 2v12a2 2 0 0 0 2 2h8a2 2 0 0 0 2-2M9.5 3A1.5 1.5 0 0 0 11 4.5h2V14a1 1 0 0 1-1 1H4a1 1 0 0 1-1-1V2a1 1 0 0 1 1-1h5.5z"/>
  59. </svg>
  60. Importer
  61. </button>
  62. <a href="{{ path('app_task_csv', {'slug': project.slug}) }}" class="btn btn-secondary" title="Exporter les tâches">
  63. <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-file-earmark-arrow-up" viewBox="0 0 16 16">
  64. <path d="M8.5 11.5a.5.5 0 0 1-1 0V7.707L6.354 8.854a.5.5 0 1 1-.708-.708l2-2a.5.5 0 0 1 .708 0l2 2a.5.5 0 0 1-.708.708L8.5 7.707z"/>
  65. <path d="M14 14V4.5L9.5 0H4a2 2 0 0 0-2 2v12a2 2 0 0 0 2 2h8a2 2 0 0 0 2-2M9.5 3A1.5 1.5 0 0 0 11 4.5h2V14a1 1 0 0 1-1 1H4a1 1 0 0 1-1-1V2a1 1 0 0 1 1-1h5.5z"/>
  66. </svg>
  67. Exporter
  68. </a>
  69. <a href="{{ path('app_task_create', {'slug': project.slug}) }}" class="btn btn-secondary" title="Créer une tâche">
  70. <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-plus" viewBox="0 0 16 16">
  71. <path d="M8 4a.5.5 0 0 1 .5.5v3h3a.5.5 0 0 1 0 1h-3v3a.5.5 0 0 1-1 0v-3h-3a.5.5 0 0 1 0-1h3v-3A.5.5 0 0 1 8 4"/>
  72. </svg>
  73. Créer
  74. </a>
  75. </div>
  76. {% endif %}
  77. {% if randomTask %}
  78. <div class="btn-group">
  79. <a href="{{ path('app_task_show', {'slug': randomTask.slug}) }}" class="btn btn-secondary" title="Piocher une tâche">
  80. <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-shuffle" viewBox="0 0 16 16">
  81. <path fill-rule="evenodd" d="M0 3.5A.5.5 0 0 1 .5 3H1c2.202 0 3.827 1.24 4.874 2.418.49.552.865 1.102 1.126 1.532.26-.43.636-.98 1.126-1.532C9.173 4.24 10.798 3 13 3v1c-1.798 0-3.173 1.01-4.126 2.082A9.6 9.6 0 0 0 7.556 8a9.6 9.6 0 0 0 1.317 1.918C9.828 10.99 11.204 12 13 12v1c-2.202 0-3.827-1.24-4.874-2.418A10.6 10.6 0 0 1 7 9.05c-.26.43-.636.98-1.126 1.532C4.827 11.76 3.202 13 1 13H.5a.5.5 0 0 1 0-1H1c1.798 0 3.173-1.01 4.126-2.082A9.6 9.6 0 0 0 6.444 8a9.6 9.6 0 0 0-1.317-1.918C4.172 5.01 2.796 4 1 4H.5a.5.5 0 0 1-.5-.5"/>
  82. <path d="M13 5.466V1.534a.25.25 0 0 1 .41-.192l2.36 1.966c.12.1.12.284 0 .384l-2.36 1.966a.25.25 0 0 1-.41-.192m0 9v-3.932a.25.25 0 0 1 .41-.192l2.36 1.966c.12.1.12.284 0 .384l-2.36 1.966a.25.25 0 0 1-.41-.192"/>
  83. </svg>
  84. Piocher
  85. </a>
  86. </div>
  87. {% endif %}
  88. </div>
  89. </div>
  90. </div>
  91. <div class="row">
  92. <div class="col mb-3">
  93. <p class="text-muted">{% include 'partials/_project-metadata.html.twig' %}</p>
  94. </div>
  95. </div>
  96. {% if project.description is not empty %}
  97. <div class="row">
  98. <div class="col mb-3 lead">{{ project.description|markdown_to_html }}</div>
  99. </div>
  100. {% endif %}
  101. {% if project.tasks is not empty %}
  102. <h2 class="mb-3">Carte</h2>
  103. <div class="row">
  104. <div class="col mb-3">
  105. {{ macro.map(project, project.overpassResult ? project.overpassResult : '') }}
  106. </div>
  107. </div>
  108. {% endif %}
  109. {% if project.tasks is not empty %}
  110. {% set dummy = tasks.setParam('_fragment', 'tasks') %}
  111. <h2 id="tasks" class="mb-3">Tâches</h2>
  112. <div class="row">
  113. <div class="col mb-3">
  114. <div class="progress-stacked">
  115. {% set stats = taskLifecycleManager.getProjectStats(project) %}
  116. {% for place, data in stats %}
  117. <div class="progress" role="progressbar" data-bs-toggle="tooltip" data-bs-title="{{ data.title }} {{ data.value ~ '/' ~ data.max }}" aria-label="{{ data.title }}" aria-valuenow="{{ data.value }}" aria-valuemin="0" aria-valuemax="{{ data.max }}" style="width:{{ data.percentage ~ '%' }}">
  118. <div class="progress-bar {{ 'bg-' ~ data.color }}">{{ data.percentage|format_number({fraction_digit: 0}) ~ '%' }}</div>
  119. </div>
  120. {% endfor %}
  121. </div>
  122. </div>
  123. </div>
  124. <div class="row">
  125. <div class="col">
  126. <table class="table table-sm table-hover">
  127. <thead>
  128. <tr>
  129. <th scope="col">{{ macro.paginated(tasks, 'Nom', 't.name') }}</th>
  130. <th scope="col">{{ macro.paginated(tasks, 'État', 't.status') }}</th>
  131. <th scope="col">{{ macro.paginated(tasks, 'Importance', 't.important') }}</th>
  132. <th scope="col">{{ macro.paginated(tasks, 'Urgence', 't.urgent') }}</th>
  133. </tr>
  134. </thead>
  135. <tbody>
  136. {% for task in tasks %}
  137. <tr class="{{ 'table-' ~ workflow_metadata(task, 'color', task.status) }}">
  138. <td><a href="{{ path('app_task_show', {'slug': task.slug}) }}">{{ task.name }}</a></td>
  139. <td>{{ workflow_metadata(task, 'title', task.status) }}</td>
  140. <td>{{ task.important }}</td>
  141. <td>{{ task.urgent }}</td>
  142. </tr>
  143. {% endfor %}
  144. </tbody>
  145. </table>
  146. </div>
  147. </div>
  148. {% endif %}
  149. {% if comments is not empty %}
  150. <details>
  151. <summary class="mb-3"><h2 class="d-inline">Commentaires <span class="badge text-bg-secondary">{{ comments|length }}</span></h2></summary>
  152. <div class="row">
  153. <div class="col mb-3">
  154. {% for comment in comments %}
  155. <blockquote class="blockquote">
  156. {{ comment.content|markdown_to_html }}
  157. </blockquote>
  158. <figcaption class="blockquote-footer">
  159. {{ comment.task.name }}
  160. {% include 'partials/_comment-metadata.html.twig' %}
  161. </figcaption>
  162. {% endfor %}
  163. </div>
  164. </div>
  165. </details>
  166. {% endif %}
  167. {% endblock %}
  168. {% block modals %}
  169. {{ form_start(csv_form) }}
  170. <div class="modal fade" id="modal" tabindex="-1" aria-labelledby="modalLabel" aria-hidden="true">
  171. <div class="modal-dialog">
  172. <div class="modal-content">
  173. <div class="modal-header">
  174. <h1 class="modal-title fs-5" id="modalLabel">Importer des tâches</h1>
  175. <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
  176. </div>
  177. <div class="modal-body">
  178. {{ form_row(csv_form.csv) }}
  179. </div>
  180. <div class="modal-footer">
  181. <button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Fermer</button>
  182. {{ form_widget(csv_form.submit) }}
  183. </div>
  184. </div>
  185. </div>
  186. </div>
  187. {{ form_end(csv_form) }}
  188. {% endblock %}