bg_remover_history.html 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. {% load static %}
  2. <!DOCTYPE html>
  3. <html lang="en">
  4. <head>
  5. <meta charset="UTF-8">
  6. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  7. <title>Task History | AI Background Remover</title>
  8. <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/css/bootstrap.min.css">
  9. <script src="https://unpkg.com/xlsx/dist/xlsx.full.min.js"></script>
  10. <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@fontsource/source-sans-3@5.0.12/index.css"
  11. integrity="sha256-tXJfXfp6Ewt1ilPzLDtQnJV4hclT9XuaZUKyUvmyr+Q=" crossorigin="anonymous">
  12. <!--end::Fonts--><!--begin::Third Party Plugin(OverlayScrollbars)-->
  13. <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/overlayscrollbars@2.3.0/styles/overlayscrollbars.min.css"
  14. integrity="sha256-dSokZseQNT08wYEWiz5iLI8QPlKxG+TswNRD8k35cpg=" crossorigin="anonymous">
  15. <!--end::Third Party Plugin(OverlayScrollbars)--><!--begin::Third Party Plugin(Bootstrap Icons)-->
  16. <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.11.0/font/bootstrap-icons.min.css"
  17. integrity="sha256-Qsx5lrStHZyR9REqhUF8iQt73X06c8LGIUPzpOhwRrI=" crossorigin="anonymous">
  18. <!--end::Third Party Plugin(Bootstrap Icons)--><!--begin::Required Plugin(AdminLTE)-->
  19. <link rel="stylesheet" href="{% static './css/adminlte.css' %}"><!--end::Required Plugin(AdminLTE)--><!-- apexcharts -->
  20. <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/apexcharts@3.37.1/dist/apexcharts.css"
  21. integrity="sha256-4MX+61mt9NVvvuPjUWdUdyfZfxSB1/Rf9WtqRHgG5S0=" crossorigin="anonymous">
  22. <link href="https://cdn.jsdelivr.net/npm/select2@4.1.0-rc.0/dist/css/select2.min.css" rel="stylesheet" />
  23. <link rel="stylesheet" href="{% static './css/select2-bootstrap4.min.css' %}">
  24. <link rel="stylesheet" href="{% static './css/custom.css' %}">
  25. <script src="https://cdn.tailwindcss.com"></script>
  26. <link href="https://fonts.googleapis.com/css2?family=Plus+Jakarta+Sans:wght@400;600;800&display=swap" rel="stylesheet">
  27. <style>
  28. body { font-family: 'Plus Jakarta Sans', sans-serif; background-color: #f8fafc; }
  29. .table-container { background: white; border-radius: 24px; border: 1px solid #e2e8f0; box-shadow: 0 10px 15px -3px rgba(0, 0, 0, 0.05); }
  30. .status-badge { font-weight: 700; font-size: 0.7rem; letter-spacing: 0.05em; padding: 6px 12px; border-radius: 10px; text-transform: uppercase; }
  31. .btn-new { background: #2563eb; color: white; padding: 10px 24px; border-radius: 14px; font-weight: 700; transition: all 0.3s; }
  32. .btn-new:hover { background: #1d4ed8; transform: translateY(-1px); box-shadow: 0 4px 12px rgba(37, 99, 235, 0.2); }
  33. </style>
  34. </head>
  35. <body class="layout-fixed sidebar-expand-lg sidebar-mini app-loaded sidebar-collapse">
  36. <div class="app-wrapper">
  37. {% include 'header.html' %}
  38. {% include 'sidebar.html' %}
  39. <main class="app-main">
  40. <div class="app-content pt-10 px-4">
  41. <div class="max-w-6xl mx-auto">
  42. <div class="mb-8 flex flex-col md:flex-row md:items-center justify-between gap-4">
  43. <div>
  44. <nav class="flex mb-2 text-[10px] font-bold uppercase tracking-[0.2em] text-blue-600">
  45. <a href="{% url 'remove_bg' %}" class="hover:text-blue-800">Image Background Remover</a>
  46. <span class="mx-2 text-gray-300">/</span>
  47. <span class="text-gray-400">Processing History</span>
  48. </nav>
  49. <h1 class="text-3xl font-black text-gray-900 tracking-tight">Batch History</h1>
  50. <p class="text-gray-500 font-medium">Manage and download your recently processed image archives.</p>
  51. </div>
  52. <a href="{% url 'remove_bg' %}" class="btn-new inline-flex items-center gap-2">
  53. <i class="bi bi-plus-lg"></i>
  54. New Batch Task
  55. </a>
  56. </div>
  57. <div class="table-container overflow-hidden">
  58. <table class="table table-hover align-middle mb-0">
  59. <thead class="bg-gray-50/50 border-b border-gray-100">
  60. <tr>
  61. <th class="px-6 py-4 text-[11px] font-black text-gray-400 uppercase tracking-widest">Date & Time</th>
  62. <th class="px-6 py-4 text-[11px] font-black text-gray-400 uppercase tracking-widest">Task Identifier</th>
  63. <th class="px-6 py-4 text-[11px] font-black text-gray-400 uppercase tracking-widest text-center">Status</th>
  64. <th class="px-6 py-4 text-[11px] font-black text-gray-400 uppercase tracking-widest text-end">Action</th>
  65. </tr>
  66. </thead>
  67. <tbody id="history-table-body" class="divide-y divide-gray-50">
  68. <tr>
  69. <td colspan="4" class="py-20 text-center">
  70. <div class="flex flex-col items-center">
  71. <div class="w-10 h-10 border-4 border-blue-600 border-t-transparent rounded-full animate-spin mb-4"></div>
  72. <p class="text-gray-400 font-bold uppercase tracking-widest text-xs">Syncing Tasks...</p>
  73. </div>
  74. </td>
  75. </tr>
  76. </tbody>
  77. </table>
  78. </div>
  79. <p class="mt-6 text-center text-xs text-gray-400 font-medium">
  80. <i class="bi bi-info-circle mr-1"></i>
  81. Tasks are automatically removed after 24 hours to optimize storage.
  82. </p>
  83. </div>
  84. </div>
  85. </main>
  86. {% include 'footer.html' %}
  87. </div>
  88. <script src="https://code.jquery.com/jquery-3.6.0.min.js"></script>
  89. <script src="https://cdn.jsdelivr.net/npm/@popperjs/core@2.11.8/dist/umd/popper.min.js"></script>
  90. <script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/js/bootstrap.min.js"></script>
  91. <script src="https://cdn.jsdelivr.net/npm/overlayscrollbars@2.3.0/browser/overlayscrollbars.browser.es6.min.js"></script>
  92. <script src="{% static './js/adminlte.js' %}"></script>
  93. <!-- <script>
  94. async function fetchHistory() {
  95. try {
  96. // Fetching from the API we built earlier
  97. const res = await fetch('{% url "bg_tasks_json" %}');
  98. const data = await res.json();
  99. const body = document.getElementById('history-table-body');
  100. if (data.length === 0) {
  101. body.innerHTML = `
  102. <tr>
  103. <td colspan="4" class="text-center py-20">
  104. <i class="bi bi-inbox text-4xl text-gray-200 mb-3 block"></i>
  105. <p class="text-gray-400 font-bold">No tasks found in your history.</p>
  106. </td>
  107. </tr>`;
  108. return;
  109. }
  110. body.innerHTML = data.map(t => {
  111. let badgeStyles = "";
  112. let statusLabel = t.status;
  113. if(t.status === 'COMPLETED') {
  114. badgeStyles = "bg-green-100 text-green-700 border border-green-200";
  115. } else if(t.status === 'PROCESSING') {
  116. badgeStyles = "bg-blue-100 text-blue-700 border border-blue-200 animate-pulse";
  117. statusLabel = "In Progress";
  118. } else {
  119. badgeStyles = "bg-red-100 text-red-700 border border-red-200";
  120. }
  121. return `
  122. <tr class="transition-colors hover:bg-gray-50/50">
  123. <td class="px-6 py-4">
  124. <div class="text-sm font-bold text-gray-800">${t.date}</div>
  125. </td>
  126. <td class="px-6 py-4">
  127. <div class="text-xs font-mono text-gray-400 tracking-tighter">${t.task_id}</div>
  128. </td>
  129. <td class="px-6 py-4 text-center">
  130. <span class="status-badge ${badgeStyles}">
  131. ${statusLabel}
  132. </span>
  133. </td>
  134. <td class="px-6 py-4 text-end">
  135. ${t.url ?
  136. `<a href="${t.url}" class="inline-flex items-center gap-2 bg-slate-900 text-white px-4 py-2 rounded-xl text-xs font-bold hover:bg-slate-800 transition-all">
  137. <i class="bi bi-download"></i> Download ZIP
  138. </a>` :
  139. `<span class="text-[11px] font-black text-gray-300 uppercase italic">Working...</span>`
  140. }
  141. </td>
  142. </tr>`;
  143. }).join('');
  144. } catch (err) {
  145. console.error("History fetch error:", err);
  146. }
  147. }
  148. // Auto-refresh every 5 seconds to update "Processing" tasks
  149. fetchHistory();
  150. // setInterval(fetchHistory, 5000);
  151. </script> -->
  152. <script>
  153. async function fetchHistory() {
  154. try {
  155. // Fetch tasks from your API (both BG and Caption)
  156. const res = await fetch('{% url "bg_tasks_json" %}');
  157. const data = await res.json();
  158. const body = document.getElementById('history-table-body');
  159. if (data.length === 0) {
  160. body.innerHTML = `
  161. <tr>
  162. <td colspan="4" class="text-center py-20">
  163. <i class="bi bi-inbox text-4xl text-gray-200 mb-3 block"></i>
  164. <p class="text-gray-400 font-bold">No tasks found in your history.</p>
  165. </td>
  166. </tr>`;
  167. return;
  168. }
  169. body.innerHTML = data.map(t => {
  170. let badgeStyles = "";
  171. let statusLabel = t.status;
  172. // Set badge colors
  173. if(t.status === 'COMPLETED') {
  174. badgeStyles = "bg-green-100 text-green-700 border border-green-200";
  175. } else if(t.status === 'PROCESSING') {
  176. badgeStyles = "bg-blue-100 text-blue-700 border border-blue-200 animate-pulse";
  177. statusLabel = "In Progress";
  178. } else {
  179. badgeStyles = "bg-red-100 text-red-700 border border-red-200";
  180. }
  181. console.log("t.task_type",t.task_type);
  182. // Detect task type (Background or Caption)
  183. const taskTypeLabel = t.task_type === "CAPTION" ? "AI Image Caption" : "Background Removal";
  184. // If Caption task, show sample caption or note
  185. const extraInfo = t.task_type === "CAPTION" && t.results?.length > 0
  186. ? `<div class="text-xs text-gray-600 mt-1">
  187. Example: "${t.results[0].caption || t.results[0].score}"
  188. </div>`
  189. : "";
  190. return `
  191. <tr class="transition-colors hover:bg-gray-50/50">
  192. <td class="px-6 py-4">
  193. <div class="text-sm font-bold text-gray-800">${t.date}</div>
  194. </td>
  195. <td class="px-6 py-4">
  196. <div class="text-xs font-mono text-gray-400 tracking-tighter">${t.task_id}</div>
  197. <div class="text-xs font-bold text-gray-500 mt-1">${taskTypeLabel}</div>
  198. </td>
  199. <td class="px-6 py-4 text-center">
  200. <span class="status-badge ${badgeStyles}">
  201. ${statusLabel}
  202. </span>
  203. ${extraInfo}
  204. </td>
  205. <td class="px-6 py-4 text-end">
  206. ${t.url ?
  207. `<a href="${t.url}" class="inline-flex items-center gap-2 bg-slate-900 text-white px-4 py-2 rounded-xl text-xs font-bold hover:bg-slate-800 transition-all">
  208. <i class="bi bi-download"></i> Download ZIP
  209. </a>` :
  210. `<span class="text-[11px] font-black text-gray-300 uppercase italic">Working...</span>`
  211. }
  212. </td>
  213. </tr>`;
  214. }).join('');
  215. } catch (err) {
  216. console.error("History fetch error:", err);
  217. }
  218. }
  219. // Initial fetch and optional auto-refresh
  220. fetchHistory();
  221. // setInterval(fetchHistory, 5000);
  222. </script>
  223. <script>
  224. const SELECTOR_SIDEBAR_WRAPPER = ".sidebar-wrapper";
  225. const Default = {
  226. scrollbarTheme: "os-theme-light",
  227. scrollbarAutoHide: "leave",
  228. scrollbarClickScroll: true,
  229. };
  230. document.addEventListener("DOMContentLoaded", function () {
  231. const sidebarWrapper = document.querySelector(SELECTOR_SIDEBAR_WRAPPER);
  232. if (
  233. sidebarWrapper &&
  234. typeof OverlayScrollbarsGlobal?.OverlayScrollbars !== "undefined"
  235. ) {
  236. OverlayScrollbarsGlobal.OverlayScrollbars(sidebarWrapper, {
  237. scrollbars: {
  238. theme: Default.scrollbarTheme,
  239. autoHide: Default.scrollbarAutoHide,
  240. clickScroll: Default.scrollbarClickScroll,
  241. },
  242. });
  243. }
  244. });
  245. </script>
  246. </body>
  247. </html>