base.html 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. <!DOCTYPE html>
  2. <html lang="fr" class="h-full bg-gray-50">
  3. <head>
  4. <meta charset="UTF-8">
  5. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  6. <title>{% block title %}Backup Manager{% endblock %} — {{ instance_name }}</title>
  7. <script src="https://cdn.tailwindcss.com"></script>
  8. <script type="text/tailwindcss">
  9. @layer components {
  10. /* Bouton principal (créer, lancer, enregistrer) */
  11. .btn-primary { @apply inline-flex items-center gap-1.5 bg-blue-600 hover:bg-blue-700 text-white font-medium rounded-lg transition; }
  12. /* Bouton secondaire (tester, éditer, sync, historique) */
  13. .btn-secondary { @apply inline-flex items-center gap-1.5 bg-white hover:bg-gray-50 text-gray-700 border border-gray-200 font-medium rounded transition; }
  14. /* Bouton icône / action légère */
  15. .btn-ghost { @apply inline-flex items-center gap-1 text-gray-500 hover:text-gray-800 hover:bg-gray-100 rounded transition; }
  16. /* Bouton destructeur */
  17. .btn-danger { @apply inline-flex items-center gap-1 text-red-400 hover:text-red-600 hover:bg-red-50 rounded transition; }
  18. /* Tailles */
  19. .btn-sm { @apply text-xs px-3 py-1.5; }
  20. .btn-md { @apply text-sm px-5 py-2; }
  21. .btn-icon-sm { @apply text-xs px-2 py-1; }
  22. }
  23. </script>
  24. </head>
  25. <body class="h-full flex flex-col">
  26. <nav class="bg-gray-900 text-white shadow-lg">
  27. <div class="max-w-7xl mx-auto px-6 py-3 flex items-center justify-between">
  28. <div class="flex items-center gap-3">
  29. <svg class="w-6 h-6 text-blue-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
  30. <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
  31. d="M5 12h14M5 12l4-4m-4 4l4 4M19 12l-4-4m4 4l-4 4"/>
  32. </svg>
  33. <a href="{{ url_for('jobs.index') }}" class="text-lg font-bold tracking-tight">Backup Manager</a>
  34. <span class="bg-blue-600 text-xs font-medium px-2 py-0.5 rounded">{{ instance_name }}</span>
  35. </div>
  36. <div class="flex items-center gap-4 text-sm">
  37. <a href="{{ url_for('jobs.index') }}"
  38. class="text-gray-300 hover:text-white transition {% if request.endpoint and request.endpoint.startswith('jobs.') and request.endpoint != 'jobs.job_new' %}text-white font-semibold{% endif %}">
  39. Dashboard
  40. </a>
  41. <a href="{{ url_for('jobs.archives') }}"
  42. class="text-gray-300 hover:text-white transition {% if request.endpoint == 'jobs.archives' %}text-white font-semibold{% endif %}">
  43. Archives
  44. </a>
  45. <a href="{{ url_for('cfg.settings') }}"
  46. class="text-gray-300 hover:text-white transition {% if request.endpoint and request.endpoint.startswith('cfg.') %}text-white font-semibold{% endif %}">
  47. Paramètres
  48. </a>
  49. <a href="{{ url_for('jobs.job_new') }}"
  50. class="bg-blue-600 hover:bg-blue-700 text-white px-3 py-1.5 rounded font-medium transition">
  51. + Nouveau job
  52. </a>
  53. </div>
  54. </div>
  55. </nav>
  56. <!-- Barre d'activité sticky -->
  57. <div id="activity-bar"
  58. class="hidden sticky top-0 z-40 bg-gray-900 border-b border-blue-700 text-white text-xs px-6 py-2 flex items-center gap-3 shadow">
  59. <span class="inline-block w-2 h-2 rounded-full bg-blue-400 animate-pulse shrink-0"></span>
  60. <span class="text-blue-300 font-medium shrink-0">En cours</span>
  61. <div id="activity-list" class="flex gap-4 flex-wrap flex-1 text-gray-200"></div>
  62. </div>
  63. <main class="flex-1 max-w-7xl mx-auto w-full px-6 py-8">
  64. {% with messages = get_flashed_messages(with_categories=true) %}
  65. {% if messages %}
  66. <div class="mb-6 space-y-2">
  67. {% for category, message in messages %}
  68. <div class="px-4 py-3 rounded-lg text-sm font-medium
  69. {% if category == 'error' %}bg-red-50 text-red-800 border border-red-200
  70. {% elif category == 'info' %}bg-blue-50 text-blue-800 border border-blue-200
  71. {% else %}bg-green-50 text-green-800 border border-green-200{% endif %}">
  72. {{ message }}
  73. </div>
  74. {% endfor %}
  75. </div>
  76. {% endif %}
  77. {% endwith %}
  78. {% block content %}{% endblock %}
  79. </main>
  80. <footer class="text-center text-xs text-gray-400 py-4">
  81. Backup Manager — instance <strong>{{ instance_name }}</strong>
  82. </footer>
  83. <script>
  84. (function() {
  85. const bar = document.getElementById('activity-bar');
  86. const list = document.getElementById('activity-list');
  87. const LABELS = { backup: 'Sauvegarde', restore: 'Restauration', push: 'Push', pull: 'Pull' };
  88. function elapsed(isoStr) {
  89. const sec = Math.round((Date.now() - new Date(isoStr).getTime()) / 1000);
  90. if (sec < 60) return sec + 's';
  91. return Math.floor(sec / 60) + 'min' + (sec % 60 ? (sec % 60) + 's' : '');
  92. }
  93. function render(items) {
  94. if (!items || items.length === 0) {
  95. bar.classList.add('hidden');
  96. return;
  97. }
  98. bar.classList.remove('hidden');
  99. list.innerHTML = items.map(it => {
  100. const label = LABELS[it.kind] || it.kind;
  101. const since = it.started_at ? ' · ' + elapsed(it.started_at) : '';
  102. const size = it.size_human ? ' · ' + it.size_human : '';
  103. return `<span class="bg-gray-800 rounded px-2 py-0.5">${label} <strong>${it.name}</strong>${since}${size}</span>`;
  104. }).join('');
  105. }
  106. function poll() {
  107. fetch('{{ url_for("api.api_running") }}', {
  108. headers: { 'X-BackupManager-Key': '{{ api_token }}' }
  109. })
  110. .then(r => r.ok ? r.json() : null)
  111. .then(data => render(data && data.running ? data.running : []))
  112. .catch(() => {});
  113. }
  114. poll();
  115. setInterval(poll, 5000);
  116. })();
  117. </script>
  118. </body>
  119. </html>