Files
task_flow/templates/stats.html

362 lines
12 KiB
HTML

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>TaskFlow Stats</title>
<style>
:root {
color-scheme: light;
}
body {
margin: 0;
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif;
background: #f5f7fb;
color: #1f2937;
}
.page {
max-width: 1200px;
margin: 0 auto;
padding: 24px;
}
.title {
margin: 0 0 20px 0;
font-size: 28px;
font-weight: 700;
}
.details-card {
background: #ffffff;
border-radius: 16px;
padding: 20px;
box-shadow: 0 8px 24px rgba(15, 23, 42, 0.08);
margin-bottom: 24px;
}
.details-grid {
display: grid;
grid-template-columns: repeat(2, minmax(0, 1fr));
gap: 12px 20px;
}
.details-item {
background: #f8fafc;
border-radius: 12px;
padding: 12px 14px;
}
.details-label {
font-size: 12px;
color: #64748b;
margin-bottom: 6px;
text-transform: uppercase;
letter-spacing: 0.04em;
}
.details-value {
font-size: 15px;
font-weight: 600;
word-break: break-word;
}
.board {
display: grid;
grid-template-columns: repeat(3, minmax(0, 1fr));
gap: 20px;
}
.column {
background: #ffffff;
border-radius: 16px;
padding: 16px;
box-shadow: 0 8px 24px rgba(15, 23, 42, 0.08);
min-height: 280px;
}
.column-title {
margin: 0 0 14px 0;
font-size: 18px;
font-weight: 700;
}
.task-list {
display: flex;
flex-direction: column;
gap: 10px;
}
.task-card {
width: 100%;
border: 1px solid #e2e8f0;
background: #f8fafc;
border-radius: 12px;
padding: 12px;
text-align: left;
cursor: pointer;
transition: transform 0.15s ease, box-shadow 0.15s ease, border-color 0.15s ease;
}
.task-card:hover {
transform: translateY(-1px);
box-shadow: 0 8px 18px rgba(15, 23, 42, 0.08);
border-color: #94a3b8;
}
.task-card.active {
border-color: #2563eb;
box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.12);
background: #eff6ff;
}
.task-title {
font-size: 15px;
font-weight: 700;
margin-bottom: 8px;
}
.task-meta {
font-size: 12px;
color: #64748b;
}
.empty {
color: #94a3b8;
font-size: 14px;
padding: 8px 2px;
}
@media (max-width: 900px) {
.board {
grid-template-columns: 1fr;
}
.details-grid {
grid-template-columns: 1fr;
}
}
</style>
</head>
<body>
<div class="page">
<h1 class="title">TaskFlow Stats</h1>
<section class="details-card">
<div class="details-grid">
<div class="details-item">
<div class="details-label">Title</div>
<div class="details-value" id="selected-title">{{ selected_task.title if selected_task else "No task selected" }}</div>
</div>
<div class="details-item">
<div class="details-label">Status</div>
<div class="details-value" id="selected-status">{{ selected_task.status if selected_task else "—" }}</div>
</div>
<div class="details-item">
<div class="details-label">Start datetime</div>
<div class="details-value" id="selected-started-at">{{ selected_task.started_at if selected_task and selected_task.started_at else "—" }}</div>
</div>
<div class="details-item">
<div class="details-label">Done datetime</div>
<div class="details-value" id="selected-done-at">{{ selected_task.done_at if selected_task and selected_task.done_at else "—" }}</div>
</div>
<div class="details-item">
<div class="details-label">Cycle time</div>
<div class="details-value" id="selected-cycle-time">{{ selected_task.cycle_time if selected_task else "—" }}</div>
</div>
<div class="details-item">
<div class="details-label">Created at</div>
<div class="details-value" id="selected-created-at">{{ selected_task.created_at if selected_task else "—" }}</div>
</div>
</div>
</section>
<section class="board">
<div class="column">
<h2 class="column-title">Backlog</h2>
<div class="task-list" id="backlog-list">
<div class="empty">Loading tasks...</div>
</div>
</div>
<div class="column">
<h2 class="column-title">In Progress</h2>
<div class="task-list" id="in-progress-list">
<div class="empty">Loading tasks...</div>
</div>
</div>
<div class="column">
<h2 class="column-title">Done</h2>
<div class="task-list" id="done-list">
<div class="empty">Loading tasks...</div>
</div>
</div>
</section>
</div>
<script>
const selectedTitle = document.getElementById("selected-title");
const selectedStatus = document.getElementById("selected-status");
const selectedStartedAt = document.getElementById("selected-started-at");
const selectedDoneAt = document.getElementById("selected-done-at");
const selectedCycleTime = document.getElementById("selected-cycle-time");
const selectedCreatedAt = document.getElementById("selected-created-at");
const boardLists = {
backlog: document.getElementById("backlog-list"),
in_progress: document.getElementById("in-progress-list"),
done: document.getElementById("done-list"),
};
const detailFields = [
selectedTitle,
selectedStatus,
selectedStartedAt,
selectedDoneAt,
selectedCycleTime,
selectedCreatedAt,
];
function setDetailsLoading() {
detailFields.forEach((field) => {
field.textContent = "Loading...";
});
}
function renderTaskDetails(task) {
selectedTitle.textContent = task.title || "No task selected";
selectedStatus.textContent = task.status || "—";
selectedStartedAt.textContent = task.started_at || "—";
selectedDoneAt.textContent = task.done_at || "—";
selectedCycleTime.textContent = task.cycle_time || "—";
selectedCreatedAt.textContent = task.created_at || "—";
}
function renderTaskDetailsError() {
selectedTitle.textContent = "Failed to load task";
selectedStatus.textContent = "—";
selectedStartedAt.textContent = "—";
selectedDoneAt.textContent = "—";
selectedCycleTime.textContent = "—";
selectedCreatedAt.textContent = "—";
}
function createTaskCard(task) {
const card = document.createElement("button");
card.type = "button";
card.className = "task-card";
card.dataset.taskId = task.id;
const title = document.createElement("div");
title.className = "task-title";
title.textContent = task.title;
const meta = document.createElement("div");
meta.className = "task-meta";
meta.textContent = `${task.display_date_label}: ${task.display_date_value || "—"}`;
card.appendChild(title);
card.appendChild(meta);
card.addEventListener("click", () => {
setActiveCard(card);
loadTaskDetails(task.id);
});
return card;
}
function renderTaskList(listElement, tasks) {
listElement.replaceChildren();
if (!tasks.length) {
const empty = document.createElement("div");
empty.className = "empty";
empty.textContent = "No tasks";
listElement.appendChild(empty);
return;
}
tasks.forEach((task) => {
listElement.appendChild(createTaskCard(task));
});
}
function renderBoard(board) {
renderTaskList(boardLists.backlog, board.backlog_tasks);
renderTaskList(boardLists.in_progress, board.in_progress_tasks);
renderTaskList(boardLists.done, board.done_tasks);
}
function setActiveCard(activeCard) {
document.querySelectorAll(".task-card").forEach((card) => {
card.classList.toggle("active", card === activeCard);
});
}
async function loadTaskDetails(taskId) {
setDetailsLoading();
try {
const response = await fetch(`/ui_data/tasks/${taskId}`);
if (!response.ok) {
throw new Error("Failed to load task details");
}
const task = await response.json();
renderTaskDetails(task);
} catch (error) {
renderTaskDetailsError();
}
}
function getFirstTaskId(board) {
const groups = [board.backlog_tasks, board.in_progress_tasks, board.done_tasks];
for (const group of groups) {
if (group.length) {
return group[0].id;
}
}
return null;
}
async function loadBoard() {
try {
const response = await fetch("/ui_data/stats/board");
if (!response.ok) {
throw new Error("Failed to load board");
}
const board = await response.json();
renderBoard(board);
const firstTaskId = getFirstTaskId(board);
if (firstTaskId === null) {
return;
}
const firstCard = document.querySelector(`.task-card[data-task-id="${firstTaskId}"]`);
if (firstCard !== null) {
setActiveCard(firstCard);
}
await loadTaskDetails(firstTaskId);
} catch (error) {
Object.values(boardLists).forEach((listElement) => {
listElement.replaceChildren();
const empty = document.createElement("div");
empty.className = "empty";
empty.textContent = "Failed to load tasks";
listElement.appendChild(empty);
});
renderTaskDetailsError();
}
}
void loadBoard();
</script>
</body>
</html>