mirror of
https://github.com/github/awesome-copilot.git
synced 2026-06-14 11:54:54 +00:00
Add Canvas Extensions website page (#1900)
Generate extensions data, add the extensions listing route/navigation, and include install URL copy actions pinned to the build commit SHA. Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
This commit is contained in:
@@ -0,0 +1,446 @@
|
||||
import { CanvasError, createCanvas, joinSession } from "@github/copilot-sdk/extension";
|
||||
import http from "node:http";
|
||||
import fs from "node:fs";
|
||||
import os from "node:os";
|
||||
import path from "node:path";
|
||||
import { fileURLToPath } from "node:url";
|
||||
|
||||
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
||||
const EXTENSION_NAME = "accessibility-kanban";
|
||||
const STATE_FILE = "signalbox-accessibility-kanban-state.json";
|
||||
const COLUMNS = ["backlog", "plan", "ready", "implement", "done"];
|
||||
const VALID_COLUMNS = new Set(COLUMNS);
|
||||
|
||||
const defaultIssues = [
|
||||
{
|
||||
number: 39,
|
||||
title: "Add keyboard trap prevention for modal-like interactions",
|
||||
url: "https://github.com/sethjuarez/SignalBox/issues/39",
|
||||
labels: ["signalbox-mvp", "frontend", "accessibility"],
|
||||
column: "backlog",
|
||||
priority: "high",
|
||||
},
|
||||
{
|
||||
number: 38,
|
||||
title: "Ensure color contrast meets WCAG AA for all text",
|
||||
url: "https://github.com/sethjuarez/SignalBox/issues/38",
|
||||
labels: ["signalbox-mvp", "product-polish", "accessibility"],
|
||||
column: "backlog",
|
||||
priority: "high",
|
||||
},
|
||||
{
|
||||
number: 37,
|
||||
title: "Add aria-live region for form submission feedback",
|
||||
url: "https://github.com/sethjuarez/SignalBox/issues/37",
|
||||
labels: ["signalbox-mvp", "frontend", "accessibility"],
|
||||
column: "backlog",
|
||||
priority: "high",
|
||||
},
|
||||
{
|
||||
number: 36,
|
||||
title: "Add focus-visible outline to all interactive elements",
|
||||
url: "https://github.com/sethjuarez/SignalBox/issues/36",
|
||||
labels: ["signalbox-mvp", "frontend", "accessibility"],
|
||||
column: "backlog",
|
||||
priority: "high",
|
||||
},
|
||||
{
|
||||
number: 35,
|
||||
title: "Add aria-hidden to decorative SVG icons in AuthPage",
|
||||
url: "https://github.com/sethjuarez/SignalBox/issues/35",
|
||||
labels: ["signalbox-mvp", "frontend", "accessibility"],
|
||||
column: "backlog",
|
||||
priority: "medium",
|
||||
},
|
||||
{
|
||||
number: 20,
|
||||
title: "Audit and fix form field label association and aria-describedby",
|
||||
url: "https://github.com/sethjuarez/SignalBox/issues/20",
|
||||
labels: ["signalbox-mvp", "frontend", "product-polish", "accessibility"],
|
||||
column: "backlog",
|
||||
priority: "medium",
|
||||
},
|
||||
{
|
||||
number: 19,
|
||||
title: "Ensure consistent keyboard focus styles across the intake form",
|
||||
url: "https://github.com/sethjuarez/SignalBox/issues/19",
|
||||
labels: ["enhancement", "good first issue", "ready-for-implementation", "frontend", "accessibility"],
|
||||
column: "backlog",
|
||||
priority: "medium",
|
||||
},
|
||||
{
|
||||
number: 17,
|
||||
title: "Add accessible client-side validation errors to the intake form",
|
||||
url: "https://github.com/sethjuarez/SignalBox/issues/17",
|
||||
labels: ["enhancement", "good first issue", "ready-for-implementation", "frontend", "accessibility"],
|
||||
column: "backlog",
|
||||
priority: "medium",
|
||||
},
|
||||
{
|
||||
number: 16,
|
||||
title: "Improve page landmark and heading structure for screen reader navigation",
|
||||
url: "https://github.com/sethjuarez/SignalBox/issues/16",
|
||||
labels: ["good first issue", "signalbox-mvp", "frontend", "product-polish", "accessibility"],
|
||||
column: "backlog",
|
||||
priority: "medium",
|
||||
},
|
||||
];
|
||||
|
||||
// ─── State persistence ───
|
||||
|
||||
function copilotHome() {
|
||||
return process.env.COPILOT_HOME || path.join(os.homedir(), ".copilot");
|
||||
}
|
||||
|
||||
function getStatePath() {
|
||||
return path.join(copilotHome(), "extensions", EXTENSION_NAME, "artifacts", STATE_FILE);
|
||||
}
|
||||
|
||||
function defaultState() {
|
||||
return {
|
||||
repo: "sethjuarez/SignalBox",
|
||||
updatedAt: new Date().toISOString(),
|
||||
generation: Date.now(),
|
||||
columns: COLUMNS,
|
||||
issues: defaultIssues.map((issue, index) => ({ ...issue, order: index })),
|
||||
};
|
||||
}
|
||||
|
||||
function ensureStateDirectory() {
|
||||
fs.mkdirSync(path.dirname(getStatePath()), { recursive: true });
|
||||
}
|
||||
|
||||
function loadState() {
|
||||
try {
|
||||
return JSON.parse(fs.readFileSync(getStatePath(), "utf8"));
|
||||
} catch {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
function saveState(state) {
|
||||
ensureStateDirectory();
|
||||
fs.writeFileSync(getStatePath(), JSON.stringify({ ...state, updatedAt: new Date().toISOString() }, null, 2));
|
||||
}
|
||||
|
||||
function currentState() {
|
||||
const state = loadState();
|
||||
if (state) return state;
|
||||
const initial = defaultState();
|
||||
saveState(initial);
|
||||
return initial;
|
||||
}
|
||||
|
||||
// ─── Issue operations ───
|
||||
|
||||
function moveIssue(issueNumber, column) {
|
||||
if (!VALID_COLUMNS.has(column)) {
|
||||
throw new CanvasError("invalid_column", `Column must be one of: ${COLUMNS.join(", ")}`);
|
||||
}
|
||||
const state = currentState();
|
||||
const issue = state.issues.find((i) => i.number === issueNumber);
|
||||
if (!issue) {
|
||||
throw new CanvasError("not_found", `Issue #${issueNumber} not found on the board`);
|
||||
}
|
||||
const prevColumn = issue.column;
|
||||
issue.column = column;
|
||||
issue.order = state.issues.filter((i) => i.column === column).length;
|
||||
// Clear agent status when moved to done or backlog
|
||||
if (column === "done" || column === "backlog") {
|
||||
issue.agentActive = false;
|
||||
issue.agentStatus = column === "done" ? "Complete" : "";
|
||||
}
|
||||
saveState(state);
|
||||
broadcast("state", currentState());
|
||||
return { issue, prevColumn };
|
||||
}
|
||||
|
||||
function updateIssueStatus(issueNumber, status, logEntry) {
|
||||
const state = currentState();
|
||||
const issue = state.issues.find((i) => i.number === issueNumber);
|
||||
if (!issue) {
|
||||
throw new CanvasError("not_found", `Issue #${issueNumber} not found on the board`);
|
||||
}
|
||||
// Don't update agent status on issues that have been reset to backlog
|
||||
if (issue.column === "backlog") {
|
||||
return issue;
|
||||
}
|
||||
if (status !== undefined) issue.agentStatus = status;
|
||||
if (logEntry) {
|
||||
if (!issue.logs) issue.logs = [];
|
||||
issue.logs.push({ timestamp: new Date().toISOString(), message: logEntry });
|
||||
}
|
||||
issue.agentActive = true;
|
||||
saveState(state);
|
||||
broadcast("state", currentState());
|
||||
return issue;
|
||||
}
|
||||
|
||||
function clearAgentStatus(issueNumber) {
|
||||
const state = currentState();
|
||||
const issue = state.issues.find((i) => i.number === issueNumber);
|
||||
if (!issue) return;
|
||||
issue.agentActive = false;
|
||||
saveState(state);
|
||||
broadcast("state", currentState());
|
||||
}
|
||||
|
||||
function replaceIssues(issues) {
|
||||
const existing = currentState();
|
||||
const existingByNumber = new Map(existing.issues.map((i) => [i.number, i]));
|
||||
const next = {
|
||||
...existing,
|
||||
issues: issues
|
||||
.filter((i) => i && Number.isInteger(i.number) && i.title)
|
||||
.map((issue, idx) => {
|
||||
const prev = existingByNumber.get(issue.number);
|
||||
const labels = Array.isArray(issue.labels)
|
||||
? issue.labels.map((l) => (typeof l === "string" ? l : l.name)).filter(Boolean)
|
||||
: [];
|
||||
return {
|
||||
number: issue.number,
|
||||
title: issue.title,
|
||||
url: issue.url || `https://github.com/sethjuarez/SignalBox/issues/${issue.number}`,
|
||||
labels,
|
||||
column: VALID_COLUMNS.has(issue.column) ? issue.column : prev?.column || "backlog",
|
||||
priority: issue.priority || prev?.priority || "medium",
|
||||
order: Number.isInteger(issue.order) ? issue.order : prev?.order ?? idx,
|
||||
};
|
||||
}),
|
||||
};
|
||||
saveState(next);
|
||||
broadcast("state", currentState());
|
||||
return currentState();
|
||||
}
|
||||
|
||||
// ─── SSE ───
|
||||
|
||||
const sseClients = new Set();
|
||||
|
||||
function broadcast(event, data) {
|
||||
const msg = `event: ${event}\ndata: ${JSON.stringify(data)}\n\n`;
|
||||
for (const res of sseClients) res.write(msg);
|
||||
}
|
||||
|
||||
// ─── HTTP helpers ───
|
||||
|
||||
function readJson(req) {
|
||||
return new Promise((resolve, reject) => {
|
||||
let body = "";
|
||||
req.on("data", (c) => (body += c));
|
||||
req.on("end", () => resolve(body ? JSON.parse(body) : {}));
|
||||
req.on("error", reject);
|
||||
});
|
||||
}
|
||||
|
||||
function json(res, code, data) {
|
||||
res.writeHead(code, { "Content-Type": "application/json" });
|
||||
res.end(JSON.stringify(data));
|
||||
}
|
||||
|
||||
// ─── HTTP server ───
|
||||
|
||||
const server = http.createServer(async (req, res) => {
|
||||
const url = new URL(req.url, `http://${req.headers.host}`);
|
||||
|
||||
if (url.pathname === "/events") {
|
||||
res.writeHead(200, { "Content-Type": "text/event-stream", "Cache-Control": "no-cache", Connection: "keep-alive" });
|
||||
sseClients.add(res);
|
||||
req.on("close", () => sseClients.delete(res));
|
||||
res.write(`event: state\ndata: ${JSON.stringify(currentState())}\n\n`);
|
||||
return;
|
||||
}
|
||||
|
||||
if (req.method === "GET" && url.pathname === "/api/state") {
|
||||
json(res, 200, currentState());
|
||||
return;
|
||||
}
|
||||
|
||||
if (req.method === "POST" && url.pathname === "/api/move") {
|
||||
const input = await readJson(req);
|
||||
const { issue, prevColumn } = moveIssue(input.issue_number, input.column);
|
||||
|
||||
// When an issue moves INTO "plan", send a prompt to the agent
|
||||
if (input.column === "plan" && prevColumn !== "plan") {
|
||||
if (issue.number === 35) {
|
||||
// Fast path for demo — issue 35 is trivial, skip full analysis
|
||||
session.send({
|
||||
prompt: `The accessibility kanban board just moved issue #35 ("Add aria-hidden to decorative SVG icons in AuthPage") into the Plan column. This is a simple fix — just add aria-hidden="true" to the two decorative blur divs and the Microsoft logo SVG in src/components/AuthPage.tsx. Use the kanban_update_status tool to post a brief status update ("Analyzing..."), then after a moment post the plan summary, then move the issue to "ready" using kanban_move_issue. Keep it quick — no need to read the GitHub issue or deeply analyze the codebase. The plan is: add aria-hidden="true" to lines ~47-48 (decorative background circles) and the SVG element at lines ~6-17.`,
|
||||
});
|
||||
} else {
|
||||
session.send({
|
||||
prompt: `The accessibility kanban board just moved issue #${issue.number} ("${issue.title}") into the Plan column. Please start planning the implementation for this issue in a background agent. Read the issue details from GitHub, analyze the codebase to understand what needs to change, and produce a concrete implementation plan. When planning is complete, move the issue to "ready" on the canvas using the move_issue canvas action.`,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
json(res, 200, { issue, state: currentState() });
|
||||
return;
|
||||
}
|
||||
|
||||
if (req.method === "POST" && url.pathname === "/api/update-status") {
|
||||
const input = await readJson(req);
|
||||
const issue = updateIssueStatus(input.issue_number, input.status, input.log);
|
||||
if (input.done) clearAgentStatus(input.issue_number);
|
||||
json(res, 200, { issue, state: currentState() });
|
||||
return;
|
||||
}
|
||||
|
||||
if (req.method === "GET" && url.pathname.startsWith("/api/logs/")) {
|
||||
const num = parseInt(url.pathname.split("/").pop(), 10);
|
||||
const state = currentState();
|
||||
const issue = state.issues.find((i) => i.number === num);
|
||||
if (!issue) { json(res, 404, { error: "not found" }); return; }
|
||||
json(res, 200, { issue_number: num, title: issue.title, logs: issue.logs || [] });
|
||||
return;
|
||||
}
|
||||
|
||||
if (req.method === "POST" && url.pathname === "/api/reset") {
|
||||
const s = defaultState();
|
||||
saveState(s);
|
||||
broadcast("state", currentState());
|
||||
json(res, 200, currentState());
|
||||
return;
|
||||
}
|
||||
|
||||
if (url.pathname === "/") {
|
||||
res.writeHead(200, { "Content-Type": "text/html" });
|
||||
res.end(fs.readFileSync(path.join(__dirname, "public", "index.html"), "utf8"));
|
||||
return;
|
||||
}
|
||||
|
||||
res.writeHead(404);
|
||||
res.end("Not found");
|
||||
});
|
||||
|
||||
await new Promise((resolve) => server.listen(0, "127.0.0.1", resolve));
|
||||
function getPort() { return server.address().port; }
|
||||
|
||||
// ─── Canvas declaration ───
|
||||
|
||||
const canvas = createCanvas({
|
||||
id: "accessibility-kanban",
|
||||
displayName: "Accessibility Kanban",
|
||||
description: "Kanban board for triaging open SignalBox accessibility issues into backlog, plan, ready, implement, and done lanes. Moving an issue to plan triggers a background planning agent.",
|
||||
actions: [
|
||||
{
|
||||
name: "get_state",
|
||||
description: "Get the current Kanban board state including all issues and their columns.",
|
||||
inputSchema: { type: "object", properties: {}, additionalProperties: false },
|
||||
handler() {
|
||||
return currentState();
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "move_issue",
|
||||
description: "Move an issue to a different column on the Kanban board.",
|
||||
inputSchema: {
|
||||
type: "object",
|
||||
properties: {
|
||||
issue_number: { type: "number", description: "GitHub issue number" },
|
||||
column: { type: "string", enum: COLUMNS, description: "Target column" },
|
||||
},
|
||||
required: ["issue_number", "column"],
|
||||
additionalProperties: false,
|
||||
},
|
||||
handler({ input }) {
|
||||
const { issue } = moveIssue(input.issue_number, input.column);
|
||||
return { issue, state: currentState() };
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "refresh_issues",
|
||||
description: "Replace the board with fresh issue data supplied by the agent.",
|
||||
inputSchema: {
|
||||
type: "object",
|
||||
properties: {
|
||||
issues: {
|
||||
type: "array",
|
||||
items: {
|
||||
type: "object",
|
||||
properties: {
|
||||
number: { type: "number" },
|
||||
title: { type: "string" },
|
||||
url: { type: "string" },
|
||||
labels: { type: "array", items: { oneOf: [{ type: "string" }, { type: "object", properties: { name: { type: "string" } }, required: ["name"] }] } },
|
||||
column: { type: "string", enum: COLUMNS },
|
||||
priority: { type: "string" },
|
||||
order: { type: "number" },
|
||||
},
|
||||
required: ["number", "title"],
|
||||
additionalProperties: true,
|
||||
},
|
||||
},
|
||||
},
|
||||
required: ["issues"],
|
||||
additionalProperties: false,
|
||||
},
|
||||
handler({ input }) {
|
||||
return replaceIssues(input.issues);
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "reset_state",
|
||||
description: "Reset the board to the default issue list with everything in backlog.",
|
||||
inputSchema: { type: "object", properties: {}, additionalProperties: false },
|
||||
handler() {
|
||||
const s = defaultState();
|
||||
saveState(s);
|
||||
broadcast("state", currentState());
|
||||
return currentState();
|
||||
},
|
||||
},
|
||||
],
|
||||
open() {
|
||||
const state = currentState();
|
||||
broadcast("state", state);
|
||||
return {
|
||||
url: `http://127.0.0.1:${getPort()}`,
|
||||
title: "Accessibility Kanban",
|
||||
status: `${state.issues.length} issues across ${COLUMNS.length} columns`,
|
||||
};
|
||||
},
|
||||
});
|
||||
|
||||
// ─── Join session (tools + canvas) ───
|
||||
|
||||
const session = await joinSession({
|
||||
canvases: [canvas],
|
||||
tools: [
|
||||
{
|
||||
name: "kanban_move_issue",
|
||||
description: "Move an issue on the accessibility Kanban board to a new column (backlog, plan, ready, implement, done). Use after completing a planning or implementation step to advance the issue.",
|
||||
parameters: {
|
||||
type: "object",
|
||||
properties: {
|
||||
issue_number: { type: "number", description: "GitHub issue number" },
|
||||
column: { type: "string", enum: COLUMNS, description: "Target column to move the issue to" },
|
||||
},
|
||||
required: ["issue_number", "column"],
|
||||
},
|
||||
handler: async (args) => {
|
||||
const { issue } = moveIssue(args.issue_number, args.column);
|
||||
return JSON.stringify({ moved: true, issue, state: currentState() });
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "kanban_update_status",
|
||||
description: "Update the agent status line and log on a Kanban card. Use this to report progress while planning or implementing an issue. The status appears under the card title and a glow indicates active work.",
|
||||
parameters: {
|
||||
type: "object",
|
||||
properties: {
|
||||
issue_number: { type: "number", description: "GitHub issue number" },
|
||||
status: { type: "string", description: "Short status text shown on the card (e.g. 'Reading issue...', 'Analyzing codebase...', 'Plan complete')" },
|
||||
log: { type: "string", description: "Detailed log entry appended to the issue's agent log (viewable in modal)" },
|
||||
done: { type: "boolean", description: "Set true to stop the active glow (agent finished working)" },
|
||||
},
|
||||
required: ["issue_number", "status"],
|
||||
},
|
||||
handler: async (args) => {
|
||||
const issue = updateIssueStatus(args.issue_number, args.status, args.log);
|
||||
if (args.done) clearAgentStatus(args.issue_number);
|
||||
return JSON.stringify({ updated: true, issue });
|
||||
},
|
||||
},
|
||||
],
|
||||
});
|
||||
@@ -0,0 +1,9 @@
|
||||
{
|
||||
"name": "accessibility-kanban",
|
||||
"version": "1.0.0",
|
||||
"type": "module",
|
||||
"main": "extension.mjs",
|
||||
"dependencies": {
|
||||
"@github/copilot-sdk": "latest"
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,627 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="utf-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||
<title>Accessibility Kanban</title>
|
||||
<link rel="preconnect" href="https://fonts.googleapis.com" />
|
||||
<link href="https://fonts.googleapis.com/css2?family=DM+Sans:wght@400;500;600;700&family=IBM+Plex+Mono:wght@400;500&display=swap" rel="stylesheet" />
|
||||
<style>
|
||||
:root {
|
||||
--bg: #ffffff;
|
||||
--outer: #f1f5f9;
|
||||
--text: #111827;
|
||||
--muted: #6b7280;
|
||||
--meta: #94a3b8;
|
||||
--border: #e5e7eb;
|
||||
--coral: #ff7f50;
|
||||
--azure: #0ea5e9;
|
||||
--sage: #84cc16;
|
||||
--violet: #a78bfa;
|
||||
--radius: 6px;
|
||||
--font: 'DM Sans', sans-serif;
|
||||
--mono: 'IBM Plex Mono', monospace;
|
||||
}
|
||||
|
||||
* { margin: 0; padding: 0; box-sizing: border-box; }
|
||||
|
||||
body {
|
||||
font-family: var(--font);
|
||||
background: var(--outer);
|
||||
color: var(--text);
|
||||
height: 100vh;
|
||||
overflow: hidden;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
padding: 16px;
|
||||
}
|
||||
|
||||
.board-wrap {
|
||||
flex: 1;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
background: var(--bg);
|
||||
border-radius: 10px;
|
||||
overflow: hidden;
|
||||
border: 1px solid var(--border);
|
||||
}
|
||||
|
||||
header {
|
||||
padding: 10px 16px;
|
||||
border-bottom: 1px solid var(--border);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 6px;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
header .breadcrumb {
|
||||
font-size: 12px;
|
||||
color: var(--meta);
|
||||
font-weight: 400;
|
||||
}
|
||||
|
||||
header .breadcrumb .sep {
|
||||
margin: 0 5px;
|
||||
color: var(--border);
|
||||
}
|
||||
|
||||
header .breadcrumb .current {
|
||||
color: var(--text);
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
header .spacer { flex: 1; }
|
||||
|
||||
header .reset-btn {
|
||||
font-family: var(--mono);
|
||||
font-size: 9px;
|
||||
color: var(--meta);
|
||||
background: none;
|
||||
border: 1px solid var(--border);
|
||||
border-radius: 4px;
|
||||
padding: 3px 8px;
|
||||
cursor: pointer;
|
||||
transition: all 0.15s;
|
||||
text-transform: uppercase;
|
||||
letter-spacing: 0.5px;
|
||||
}
|
||||
|
||||
header .reset-btn:hover {
|
||||
border-color: var(--coral);
|
||||
color: var(--coral);
|
||||
}
|
||||
|
||||
.board {
|
||||
flex: 1;
|
||||
display: grid;
|
||||
grid-template-columns: repeat(5, 1fr);
|
||||
gap: 0;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.column {
|
||||
background: var(--bg);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
overflow: hidden;
|
||||
border-right: 1px solid var(--border);
|
||||
border-top: 2px solid var(--border);
|
||||
}
|
||||
|
||||
.column:last-child { border-right: none; }
|
||||
|
||||
.col-backlog { border-top-color: var(--meta); }
|
||||
.col-plan { border-top-color: var(--azure); }
|
||||
.col-ready { border-top-color: var(--sage); }
|
||||
.col-implement { border-top-color: var(--coral); }
|
||||
.col-done { border-top-color: var(--violet); }
|
||||
|
||||
.column-header {
|
||||
padding: 10px 12px 8px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 6px;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.column-header h2 {
|
||||
font-size: 10px;
|
||||
font-weight: 600;
|
||||
text-transform: uppercase;
|
||||
letter-spacing: 0.6px;
|
||||
color: var(--muted);
|
||||
}
|
||||
|
||||
.column-header .count {
|
||||
font-family: var(--mono);
|
||||
font-size: 9px;
|
||||
color: var(--meta);
|
||||
margin-left: auto;
|
||||
}
|
||||
|
||||
.card-list {
|
||||
flex: 1;
|
||||
overflow-y: auto;
|
||||
padding: 6px 8px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 4px;
|
||||
}
|
||||
|
||||
.card {
|
||||
background: var(--bg);
|
||||
border: 1px solid var(--border);
|
||||
border-radius: var(--radius);
|
||||
padding: 8px 10px;
|
||||
cursor: grab;
|
||||
transition: border-color 0.12s ease;
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.card:hover {
|
||||
border-color: var(--meta);
|
||||
}
|
||||
|
||||
.card.dragging {
|
||||
opacity: 0.35;
|
||||
}
|
||||
|
||||
.card .issue-title {
|
||||
font-size: 11px;
|
||||
font-weight: 400;
|
||||
line-height: 1.4;
|
||||
color: var(--text);
|
||||
display: -webkit-box;
|
||||
-webkit-line-clamp: 2;
|
||||
-webkit-box-orient: vertical;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.card .issue-title .num {
|
||||
font-family: var(--mono);
|
||||
font-size: 10px;
|
||||
color: var(--meta);
|
||||
margin-right: 4px;
|
||||
}
|
||||
|
||||
/* Agent status line */
|
||||
.card .agent-status {
|
||||
font-family: var(--mono);
|
||||
font-size: 9px;
|
||||
color: var(--azure);
|
||||
margin-top: 5px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 4px;
|
||||
}
|
||||
|
||||
.card .agent-status .pulse {
|
||||
width: 4px; height: 4px;
|
||||
border-radius: 50%;
|
||||
background: var(--azure);
|
||||
animation: pulse-dot 1.5s ease-in-out infinite;
|
||||
}
|
||||
|
||||
/* Glowing border for active agent work */
|
||||
.card.agent-active {
|
||||
border-color: rgba(14, 165, 233, 0.4);
|
||||
box-shadow:
|
||||
0 0 0 1px rgba(14, 165, 233, 0.15),
|
||||
0 0 12px rgba(14, 165, 233, 0.1);
|
||||
animation: glow-pulse 2.5s ease-in-out infinite;
|
||||
}
|
||||
|
||||
@keyframes glow-pulse {
|
||||
0%, 100% {
|
||||
box-shadow:
|
||||
0 0 0 1px rgba(14, 165, 233, 0.15),
|
||||
0 0 12px rgba(14, 165, 233, 0.1);
|
||||
}
|
||||
50% {
|
||||
box-shadow:
|
||||
0 0 0 1px rgba(14, 165, 233, 0.3),
|
||||
0 0 18px rgba(14, 165, 233, 0.15);
|
||||
}
|
||||
}
|
||||
|
||||
@keyframes pulse-dot {
|
||||
0%, 100% { opacity: 1; }
|
||||
50% { opacity: 0.3; }
|
||||
}
|
||||
|
||||
/* Log button */
|
||||
.card .log-btn {
|
||||
position: absolute;
|
||||
top: 6px;
|
||||
right: 6px;
|
||||
width: 16px; height: 16px;
|
||||
border-radius: 3px;
|
||||
border: none;
|
||||
background: transparent;
|
||||
color: var(--meta);
|
||||
font-size: 10px;
|
||||
cursor: pointer;
|
||||
display: none;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
transition: background 0.12s;
|
||||
}
|
||||
|
||||
.card .log-btn:hover {
|
||||
background: rgba(14,165,233,0.08);
|
||||
color: var(--azure);
|
||||
}
|
||||
|
||||
.card.agent-active .log-btn { display: flex; }
|
||||
.card.has-logs:hover .log-btn { display: flex; }
|
||||
|
||||
/* Drop target highlight */
|
||||
.column.drop-target {
|
||||
background: rgba(14,165,233,0.02);
|
||||
}
|
||||
|
||||
/* Ghost (drag preview) */
|
||||
.ghost {
|
||||
position: fixed;
|
||||
pointer-events: none;
|
||||
z-index: 9999;
|
||||
opacity: 0.85;
|
||||
transform: rotate(1.5deg) scale(1.01);
|
||||
box-shadow: 0 8px 20px rgba(0,0,0,0.12);
|
||||
}
|
||||
|
||||
/* Log Modal */
|
||||
.modal-overlay {
|
||||
display: none;
|
||||
position: fixed;
|
||||
inset: 0;
|
||||
background: rgba(0,0,0,0.3);
|
||||
z-index: 10000;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
backdrop-filter: blur(2px);
|
||||
}
|
||||
|
||||
.modal-overlay.open { display: flex; }
|
||||
|
||||
.modal {
|
||||
background: var(--bg);
|
||||
border-radius: 10px;
|
||||
width: 90%;
|
||||
max-width: 480px;
|
||||
max-height: 65vh;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
box-shadow: 0 20px 50px rgba(0,0,0,0.15);
|
||||
border: 1px solid var(--border);
|
||||
}
|
||||
|
||||
.modal-header {
|
||||
padding: 12px 16px;
|
||||
border-bottom: 1px solid var(--border);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
}
|
||||
|
||||
.modal-header h3 {
|
||||
font-size: 12px;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.modal-header button {
|
||||
border: none;
|
||||
background: none;
|
||||
font-size: 14px;
|
||||
cursor: pointer;
|
||||
color: var(--muted);
|
||||
padding: 2px 6px;
|
||||
border-radius: 3px;
|
||||
}
|
||||
|
||||
.modal-header button:hover { background: var(--border); }
|
||||
|
||||
.modal-body {
|
||||
flex: 1;
|
||||
overflow-y: auto;
|
||||
padding: 12px 16px;
|
||||
}
|
||||
|
||||
.log-entry {
|
||||
display: flex;
|
||||
gap: 10px;
|
||||
padding: 5px 0;
|
||||
border-bottom: 1px solid rgba(0,0,0,0.03);
|
||||
}
|
||||
|
||||
.log-entry:last-child { border-bottom: none; }
|
||||
|
||||
.log-entry .ts {
|
||||
font-family: var(--mono);
|
||||
font-size: 9px;
|
||||
color: var(--meta);
|
||||
white-space: nowrap;
|
||||
padding-top: 2px;
|
||||
min-width: 48px;
|
||||
}
|
||||
|
||||
.log-entry .msg {
|
||||
font-size: 11px;
|
||||
line-height: 1.5;
|
||||
color: var(--text);
|
||||
}
|
||||
|
||||
.empty-log {
|
||||
font-size: 11px;
|
||||
color: var(--meta);
|
||||
text-align: center;
|
||||
padding: 20px;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="board-wrap">
|
||||
<header>
|
||||
<div class="breadcrumb">
|
||||
SignalBox<span class="sep">/</span><span class="current">Accessibility Sprint</span>
|
||||
</div>
|
||||
<span class="spacer"></span>
|
||||
<button class="reset-btn" id="reset-btn">Reset</button>
|
||||
</header>
|
||||
<div class="board" id="board"></div>
|
||||
</div>
|
||||
|
||||
<!-- Log Modal -->
|
||||
<div class="modal-overlay" id="modal-overlay">
|
||||
<div class="modal">
|
||||
<div class="modal-header">
|
||||
<h3 id="modal-title">Agent Log</h3>
|
||||
<button id="modal-close">×</button>
|
||||
</div>
|
||||
<div class="modal-body" id="modal-body"></div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script>
|
||||
const COLUMNS = ["backlog", "plan", "ready", "implement", "done"];
|
||||
const COL_LABELS = { backlog: "Backlog", plan: "Plan", ready: "Ready", implement: "Implement", done: "Done" };
|
||||
|
||||
let state = { issues: [] };
|
||||
let dragState = null;
|
||||
|
||||
// ─── Rendering ───
|
||||
|
||||
function render() {
|
||||
const board = document.getElementById("board");
|
||||
board.innerHTML = "";
|
||||
|
||||
COLUMNS.forEach((col) => {
|
||||
const issues = state.issues
|
||||
.filter((i) => i.column === col)
|
||||
.sort((a, b) => (a.order || 0) - (b.order || 0));
|
||||
|
||||
const colEl = document.createElement("div");
|
||||
colEl.className = `column col-${col}`;
|
||||
colEl.dataset.column = col;
|
||||
|
||||
colEl.innerHTML = `
|
||||
<div class="column-header">
|
||||
<h2>${COL_LABELS[col]}</h2>
|
||||
<span class="count">${issues.length}</span>
|
||||
</div>
|
||||
<div class="card-list" data-column="${col}"></div>
|
||||
`;
|
||||
|
||||
const list = colEl.querySelector(".card-list");
|
||||
issues.forEach((issue) => {
|
||||
const card = document.createElement("div");
|
||||
const hasLogs = issue.logs && issue.logs.length > 0;
|
||||
let cls = "card";
|
||||
if (issue.agentActive) cls += " agent-active";
|
||||
if (hasLogs) cls += " has-logs";
|
||||
card.className = cls;
|
||||
card.dataset.issueNumber = issue.number;
|
||||
|
||||
let statusHtml = "";
|
||||
if (issue.agentStatus) {
|
||||
statusHtml = `<div class="agent-status">${issue.agentActive ? '<span class="pulse"></span>' : ''}${escHtml(issue.agentStatus)}</div>`;
|
||||
}
|
||||
|
||||
card.innerHTML = `
|
||||
<button class="log-btn" title="View agent log">▣</button>
|
||||
<div class="issue-title"><span class="num">#${issue.number}</span>${escHtml(issue.title)}</div>
|
||||
${statusHtml}
|
||||
`;
|
||||
|
||||
// Log button click
|
||||
const logBtn = card.querySelector(".log-btn");
|
||||
if (logBtn) {
|
||||
logBtn.addEventListener("click", (e) => {
|
||||
e.stopPropagation();
|
||||
openLogModal(issue);
|
||||
});
|
||||
}
|
||||
|
||||
// Pointer drag
|
||||
card.addEventListener("pointerdown", (e) => startDrag(e, card, issue));
|
||||
list.appendChild(card);
|
||||
});
|
||||
|
||||
board.appendChild(colEl);
|
||||
});
|
||||
}
|
||||
|
||||
function escHtml(s) {
|
||||
return s.replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">");
|
||||
}
|
||||
|
||||
// ─── Log Modal ───
|
||||
|
||||
function openLogModal(issue) {
|
||||
const overlay = document.getElementById("modal-overlay");
|
||||
const title = document.getElementById("modal-title");
|
||||
const body = document.getElementById("modal-body");
|
||||
|
||||
title.textContent = `#${issue.number} — Agent Log`;
|
||||
|
||||
if (!issue.logs || issue.logs.length === 0) {
|
||||
body.innerHTML = '<div class="empty-log">No agent activity logged yet.</div>';
|
||||
} else {
|
||||
body.innerHTML = issue.logs.map((l) => `
|
||||
<div class="log-entry">
|
||||
<span class="ts">${formatTime(l.timestamp)}</span>
|
||||
<span class="msg">${escHtml(l.message)}</span>
|
||||
</div>
|
||||
`).join("");
|
||||
body.scrollTop = body.scrollHeight;
|
||||
}
|
||||
|
||||
overlay.classList.add("open");
|
||||
}
|
||||
|
||||
function formatTime(ts) {
|
||||
try {
|
||||
const d = new Date(ts);
|
||||
return d.toLocaleTimeString([], { hour: "2-digit", minute: "2-digit" });
|
||||
} catch { return ""; }
|
||||
}
|
||||
|
||||
document.getElementById("modal-close").addEventListener("click", () => {
|
||||
document.getElementById("modal-overlay").classList.remove("open");
|
||||
});
|
||||
|
||||
document.getElementById("modal-overlay").addEventListener("click", (e) => {
|
||||
if (e.target === e.currentTarget) {
|
||||
e.currentTarget.classList.remove("open");
|
||||
}
|
||||
});
|
||||
|
||||
// ─── Pointer Drag & Drop ───
|
||||
|
||||
function startDrag(e, cardEl, issue) {
|
||||
if (e.button !== 0) return;
|
||||
e.preventDefault();
|
||||
cardEl.setPointerCapture(e.pointerId);
|
||||
|
||||
const rect = cardEl.getBoundingClientRect();
|
||||
const ghost = cardEl.cloneNode(true);
|
||||
ghost.className = "card ghost";
|
||||
ghost.style.width = rect.width + "px";
|
||||
ghost.style.left = rect.left + "px";
|
||||
ghost.style.top = rect.top + "px";
|
||||
document.body.appendChild(ghost);
|
||||
cardEl.classList.add("dragging");
|
||||
|
||||
dragState = {
|
||||
issue,
|
||||
ghost,
|
||||
cardEl,
|
||||
offsetX: e.clientX - rect.left,
|
||||
offsetY: e.clientY - rect.top,
|
||||
pointerId: e.pointerId,
|
||||
};
|
||||
|
||||
cardEl.addEventListener("pointermove", onDragMove);
|
||||
cardEl.addEventListener("pointerup", onDragEnd);
|
||||
}
|
||||
|
||||
function onDragMove(e) {
|
||||
if (!dragState) return;
|
||||
dragState.ghost.style.left = (e.clientX - dragState.offsetX) + "px";
|
||||
dragState.ghost.style.top = (e.clientY - dragState.offsetY) + "px";
|
||||
|
||||
// Highlight drop target
|
||||
document.querySelectorAll(".column").forEach((col) => {
|
||||
const r = col.getBoundingClientRect();
|
||||
if (e.clientX >= r.left && e.clientX <= r.right) {
|
||||
col.classList.add("drop-target");
|
||||
} else {
|
||||
col.classList.remove("drop-target");
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
function onDragEnd(e) {
|
||||
if (!dragState) return;
|
||||
const { issue, ghost, cardEl } = dragState;
|
||||
|
||||
cardEl.classList.remove("dragging");
|
||||
cardEl.releasePointerCapture(dragState.pointerId);
|
||||
cardEl.removeEventListener("pointermove", onDragMove);
|
||||
cardEl.removeEventListener("pointerup", onDragEnd);
|
||||
ghost.remove();
|
||||
|
||||
document.querySelectorAll(".column.drop-target").forEach((col) => {
|
||||
col.classList.remove("drop-target");
|
||||
});
|
||||
|
||||
// Find target column
|
||||
const targetCol = document.elementFromPoint(e.clientX, e.clientY)?.closest(".column");
|
||||
if (targetCol && targetCol.dataset.column !== issue.column) {
|
||||
moveIssue(issue.number, targetCol.dataset.column);
|
||||
}
|
||||
|
||||
dragState = null;
|
||||
}
|
||||
|
||||
// ─── API ───
|
||||
|
||||
async function moveIssue(issueNumber, column) {
|
||||
try {
|
||||
const resp = await fetch("/api/move", {
|
||||
method: "POST",
|
||||
headers: { "Content-Type": "application/json" },
|
||||
body: JSON.stringify({ issue_number: issueNumber, column }),
|
||||
});
|
||||
const data = await resp.json();
|
||||
state = data.state;
|
||||
render();
|
||||
} catch (err) {
|
||||
console.error("Move failed:", err);
|
||||
}
|
||||
}
|
||||
|
||||
// ─── SSE ───
|
||||
|
||||
function connectSSE() {
|
||||
const es = new EventSource("/events");
|
||||
es.addEventListener("state", (e) => {
|
||||
try {
|
||||
state = JSON.parse(e.data);
|
||||
render();
|
||||
} catch {}
|
||||
});
|
||||
es.addEventListener("connected", () => {});
|
||||
es.onerror = () => {
|
||||
es.close();
|
||||
setTimeout(connectSSE, 2000);
|
||||
};
|
||||
}
|
||||
|
||||
// ─── Init ───
|
||||
|
||||
async function init() {
|
||||
try {
|
||||
const resp = await fetch("/api/state");
|
||||
state = await resp.json();
|
||||
} catch {}
|
||||
render();
|
||||
connectSSE();
|
||||
|
||||
document.getElementById("reset-btn").addEventListener("click", async () => {
|
||||
if (!confirm("Reset all issues to backlog?")) return;
|
||||
try {
|
||||
const resp = await fetch("/api/reset", { method: "POST" });
|
||||
state = await resp.json();
|
||||
render();
|
||||
} catch (err) {
|
||||
console.error("Reset failed:", err);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
init();
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
||||
Reference in New Issue
Block a user