hotsheet 0.2.7 → 0.2.8
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +6 -0
- package/dist/cli.js +36 -3
- package/dist/client/app.global.js +3 -3
- package/dist/client/assets/glassbox-icon.png +0 -0
- package/dist/client/styles.css +1 -1
- package/package.json +9 -4
package/README.md
CHANGED
|
@@ -335,6 +335,12 @@ npm link # Symlink for global 'hotsheet' command
|
|
|
335
335
|
|
|
336
336
|
---
|
|
337
337
|
|
|
338
|
+
## See Also
|
|
339
|
+
|
|
340
|
+
- **[Glassbox](https://github.com/brianwestphal/glassbox)** — AI-powered code review tool. Runs locally, reviews your changes, and posts inline annotations. Pairs well with Hot Sheet for a complete local dev workflow.
|
|
341
|
+
|
|
342
|
+
---
|
|
343
|
+
|
|
338
344
|
## License
|
|
339
345
|
|
|
340
346
|
MIT
|
package/dist/cli.js
CHANGED
|
@@ -2285,6 +2285,29 @@ apiRoutes.get("/worklist-info", (c) => {
|
|
|
2285
2285
|
const skillCreated = consumeSkillsCreatedFlag();
|
|
2286
2286
|
return c.json({ prompt, skillCreated });
|
|
2287
2287
|
});
|
|
2288
|
+
var glassboxAvailable = null;
|
|
2289
|
+
apiRoutes.get("/glassbox/status", async (c) => {
|
|
2290
|
+
if (glassboxAvailable === null) {
|
|
2291
|
+
const { execFileSync } = await import("child_process");
|
|
2292
|
+
try {
|
|
2293
|
+
execFileSync("which", ["glassbox"], { stdio: "ignore" });
|
|
2294
|
+
glassboxAvailable = true;
|
|
2295
|
+
} catch {
|
|
2296
|
+
glassboxAvailable = false;
|
|
2297
|
+
}
|
|
2298
|
+
}
|
|
2299
|
+
return c.json({ available: glassboxAvailable });
|
|
2300
|
+
});
|
|
2301
|
+
apiRoutes.post("/glassbox/launch", async (c) => {
|
|
2302
|
+
if (!glassboxAvailable) return c.json({ error: "Glassbox not available" }, 404);
|
|
2303
|
+
const { spawn } = await import("child_process");
|
|
2304
|
+
spawn("glassbox", [], {
|
|
2305
|
+
cwd: process.cwd(),
|
|
2306
|
+
detached: true,
|
|
2307
|
+
stdio: "ignore"
|
|
2308
|
+
}).unref();
|
|
2309
|
+
return c.json({ ok: true });
|
|
2310
|
+
});
|
|
2288
2311
|
apiRoutes.get("/gitignore/status", async (c) => {
|
|
2289
2312
|
const { isGitRepo: isGitRepo2, isHotsheetGitignored: isHotsheetGitignored2 } = await Promise.resolve().then(() => (init_gitignore(), gitignore_exports));
|
|
2290
2313
|
const cwd = process.cwd();
|
|
@@ -2445,8 +2468,8 @@ pageRoutes.get("/", (c) => {
|
|
|
2445
2468
|
/* @__PURE__ */ jsx("div", { className: "header-controls", children: [
|
|
2446
2469
|
/* @__PURE__ */ jsx("div", { className: "search-box", children: /* @__PURE__ */ jsx("input", { type: "text", id: "search-input", placeholder: "Search tickets..." }) }),
|
|
2447
2470
|
/* @__PURE__ */ jsx("div", { className: "layout-toggle", id: "layout-toggle", children: [
|
|
2448
|
-
/* @__PURE__ */ jsx("button", { className: "layout-btn active", "data-layout": "list", title: "List view", children: raw('<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round"><line x1="
|
|
2449
|
-
/* @__PURE__ */ jsx("button", { className: "layout-btn", "data-layout": "columns", title: "Column view", children: raw('<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round"
|
|
2471
|
+
/* @__PURE__ */ jsx("button", { className: "layout-btn active", "data-layout": "list", title: "List view", children: raw('<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><line x1="8" y1="6" x2="21" y2="6"/><line x1="8" y1="12" x2="21" y2="12"/><line x1="8" y1="18" x2="21" y2="18"/><line x1="3" y1="6" x2="3.01" y2="6"/><line x1="3" y1="12" x2="3.01" y2="12"/><line x1="3" y1="18" x2="3.01" y2="18"/></svg>') }),
|
|
2472
|
+
/* @__PURE__ */ jsx("button", { className: "layout-btn", "data-layout": "columns", title: "Column view", children: raw('<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect width="18" height="18" x="3" y="3" rx="2"/><path d="M9 3v18"/><path d="M15 3v18"/></svg>') })
|
|
2450
2473
|
] }),
|
|
2451
2474
|
/* @__PURE__ */ jsx("div", { className: "sort-controls", children: /* @__PURE__ */ jsx("select", { id: "sort-select", children: [
|
|
2452
2475
|
/* @__PURE__ */ jsx("option", { value: "created:desc", children: "Newest First" }),
|
|
@@ -2459,7 +2482,8 @@ pageRoutes.get("/", (c) => {
|
|
|
2459
2482
|
/* @__PURE__ */ jsx("button", { className: "layout-btn active", "data-position": "side", title: "Detail panel on side", children: raw('<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round"><rect x="3" y="3" width="18" height="18" rx="2"/><line x1="15" y1="3" x2="15" y2="21"/></svg>') }),
|
|
2460
2483
|
/* @__PURE__ */ jsx("button", { className: "layout-btn", "data-position": "bottom", title: "Detail panel on bottom", children: raw('<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round"><rect x="3" y="3" width="18" height="18" rx="2"/><line x1="3" y1="15" x2="21" y2="15"/></svg>') })
|
|
2461
2484
|
] }),
|
|
2462
|
-
/* @__PURE__ */ jsx("button", { className: "
|
|
2485
|
+
/* @__PURE__ */ jsx("button", { className: "glassbox-btn", id: "glassbox-btn", title: "Open Glassbox", style: "display:none", children: raw('<img id="glassbox-icon" alt="Glassbox" />') }),
|
|
2486
|
+
/* @__PURE__ */ jsx("button", { className: "settings-btn", id: "settings-btn", title: "Settings", children: raw('<svg xmlns="http://www.w3.org/2000/svg" width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M12.22 2h-.44a2 2 0 0 0-2 2v.18a2 2 0 0 1-1 1.73l-.43.25a2 2 0 0 1-2 0l-.15-.08a2 2 0 0 0-2.73.73l-.22.38a2 2 0 0 0 .73 2.73l.15.1a2 2 0 0 1 1 1.72v.51a2 2 0 0 1-1 1.74l-.15.09a2 2 0 0 0-.73 2.73l.22.38a2 2 0 0 0 2.73.73l.15-.08a2 2 0 0 1 2 0l.43.25a2 2 0 0 1 1 1.73V20a2 2 0 0 0 2 2h.44a2 2 0 0 0 2-2v-.18a2 2 0 0 1 1-1.73l.43-.25a2 2 0 0 1 2 0l.15.08a2 2 0 0 0 2.73-.73l.22-.39a2 2 0 0 0-.73-2.73l-.15-.08a2 2 0 0 1-1-1.74v-.5a2 2 0 0 1 1-1.74l.15-.09a2 2 0 0 0 .73-2.73l-.22-.38a2 2 0 0 0-2.73-.73l-.15.08a2 2 0 0 1-2 0l-.43-.25a2 2 0 0 1-1-1.73V4a2 2 0 0 0-2-2z"/><circle cx="12" cy="12" r="3"/></svg>') })
|
|
2463
2487
|
] })
|
|
2464
2488
|
] }),
|
|
2465
2489
|
/* @__PURE__ */ jsx("div", { id: "backup-preview-banner", className: "backup-preview-banner", style: "display:none", children: [
|
|
@@ -2746,6 +2770,15 @@ async function startServer(port2, dataDir2, options) {
|
|
|
2746
2770
|
const js = readFileSync6(join9(distDir, "app.global.js"), "utf-8");
|
|
2747
2771
|
return c.text(js, 200, { "Content-Type": "application/javascript", "Cache-Control": "no-cache" });
|
|
2748
2772
|
});
|
|
2773
|
+
app.get("/static/assets/:filename", (c) => {
|
|
2774
|
+
const filename = c.req.param("filename");
|
|
2775
|
+
const filePath = join9(distDir, "assets", filename);
|
|
2776
|
+
if (!existsSync7(filePath)) return c.notFound();
|
|
2777
|
+
const content = readFileSync6(filePath);
|
|
2778
|
+
const ext = filename.split(".").pop();
|
|
2779
|
+
const mimeTypes = { png: "image/png", jpg: "image/jpeg", svg: "image/svg+xml" };
|
|
2780
|
+
return new Response(content, { headers: { "Content-Type": mimeTypes[ext || ""] || "application/octet-stream", "Cache-Control": "max-age=86400" } });
|
|
2781
|
+
});
|
|
2749
2782
|
app.route("/api", apiRoutes);
|
|
2750
2783
|
app.route("/api/backups", backupRoutes);
|
|
2751
2784
|
app.route("/", pageRoutes);
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
"use strict";(()=>{var Be=Object.defineProperty;var He=(e,t,s)=>t in e?Be(e,t,{enumerable:!0,configurable:!0,writable:!0,value:s}):e[t]=s;var ae=(e,t,s)=>He(e,typeof t!="symbol"?t+"":t,s);function oe(e){return e.replace(/&/g,"&").replace(/</g,"<").replace(/>/g,">").replace(/"/g,""")}function le(e){return e.replace(/&/g,"&").replace(/"/g,""").replace(/'/g,"'").replace(/</g,"<").replace(/>/g,">")}var T=class{constructor(t){ae(this,"__html");this.__html=t}toString(){return this.__html}};function I(e){return new T(e)}var Me=new Set(["area","base","br","col","embed","hr","img","input","link","meta","source","track","wbr"]);function V(e){return e==null||typeof e=="boolean"?"":e instanceof T?e.__html:typeof e=="string"?oe(e):typeof e=="number"?String(e):Array.isArray(e)?e.map(V).join(""):""}function Ne(e,t){if(t==null||t===!1)return"";if(t===!0)return` ${e}`;let s=e==="className"?"class":e==="htmlFor"?"for":e,a;return t instanceof T?a=t.__html:typeof t=="number"?a=String(t):typeof t=="string"?a=le(t):a="",` ${s}="${a}"`}function l(e,t){if(typeof e=="function")return e(t);let{children:s,...a}=t,i=Object.entries(a).map(([o,c])=>Ne(o,c)).join("");if(Me.has(e))return new T(`<${e}${i}>`);let r=s!=null?V(s):"";return new T(`<${e}${i}>${r}</${e}>`)}function S({children:e}){return new T(e!=null?V(e):"")}function y(e){let t=document.createElement("template");return t.innerHTML=e.toString(),t.content.firstElementChild}function re(e){document.getElementById("network-error-popup")?.remove();let t=y(l("div",{id:"network-error-popup",className:"error-popup",children:l("div",{className:"error-popup-content",children:[l("strong",{children:"Connection Error"}),l("p",{children:e}),l("button",{children:I("Dismiss")})]})}));t.querySelector("button").addEventListener("click",()=>t.remove()),document.body.appendChild(t)}async function d(e,t={}){try{return(await fetch("/api"+e,{headers:t.body!==void 0?{"Content-Type":"application/json"}:{},method:t.method,body:t.body!==void 0?JSON.stringify(t.body):void 0})).json()}catch(s){throw re("Unable to reach the server. It may have been stopped."),s}}async function ce(e,t){try{let s=new FormData;return s.append("file",t),(await fetch("/api"+e,{method:"POST",body:s})).json()}catch(s){throw re("Unable to reach the server. It may have been stopped."),s}}var Pe={detail_position:"side",detail_width:360,detail_height:300,trash_cleanup_days:3,verified_cleanup_days:30},n={tickets:[],selectedIds:new Set,lastClickedId:null,activeTicketId:null,view:"all",layout:"list",sortBy:"created",sortDir:"desc",search:"",settings:{...Pe},backupPreview:null},De={issue:"#6b7280",bug:"#ef4444",feature:"#22c55e",requirement_change:"#f97316",task:"#3b82f6",investigation:"#8b5cf6"},Ae={issue:"ISS",bug:"BUG",feature:"FEA",requirement_change:"REQ",task:"TSK",investigation:"INV"},$e={highest:"\u2B06\u2B06",high:"\u2B06",default:"\u2014",low:"\u2B07",lowest:"\u2B07\u2B07"},Re={highest:"#ef4444",high:"#f97316",default:"#6b7280",low:"#3b82f6",lowest:"#94a3b8"},qe={not_started:"\u25CB",started:"\u25D4",completed:"\u2713",verified:"svg",backlog:"\u25A1",archive:"\u25A0"},z='<svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round"><path d="M18 6 7 17l-5-5"/><path d="m22 10-9.5 9.5-2-2"/></svg>';function E(e){return De[e]||"#6b7280"}function L(e){return Ae[e]||"ISS"}function B(e){return $e[e]||"\u2014"}function _(e){return Re[e]||"#6b7280"}function j(e){return qe[e]||"\u25CB"}function A(e){n.activeTicketId=e,me(e)}function de(){n.selectedIds.clear(),n.activeTicketId=null;let e=new CustomEvent("hotsheet:render");document.dispatchEvent(e)}function H(){let e=n.view==="trash",t=!!n.backupPreview?.active,s=document.getElementById("detail-panel"),a=document.getElementById("detail-resize-handle");if(n.selectedIds.size===1&&!e){let i=Array.from(n.selectedIds)[0];s.style.display="flex",a&&(a.style.display=""),n.activeTicketId!==i&&(n.activeTicketId=i,t?Oe(i):me(i))}else n.activeTicketId!=null&&(n.activeTicketId=null),s.style.display="none",a&&(a.style.display="none")}function ue(e){let t=document.getElementById("detail-title"),s=document.getElementById("detail-details"),a=document.getElementById("detail-category"),i=document.getElementById("detail-priority"),r=document.getElementById("detail-status"),o=document.getElementById("detail-upnext"),c=document.querySelector(".upload-btn");t.readOnly=e,s.readOnly=e,a.disabled=e,i.disabled=e,r.disabled=e,o.disabled=e,c&&(c.style.display=e?"none":"")}function Oe(e){let t=n.backupPreview?.tickets.find(o=>o.id===e);if(!t||n.activeTicketId!==e)return;document.getElementById("detail-ticket-number").textContent=t.ticket_number,document.getElementById("detail-title").value=t.title,document.getElementById("detail-category").value=t.category,document.getElementById("detail-priority").value=t.priority,document.getElementById("detail-status").value=t.status,document.getElementById("detail-upnext").checked=t.up_next,document.getElementById("detail-details").value=t.details,ue(!0),document.getElementById("detail-attachments").innerHTML="";let s=document.getElementById("detail-notes-section"),a=document.getElementById("detail-notes"),i=pe(t.notes);i.length>0?(s.style.display="",a.innerHTML=l(S,{children:i.map(o=>l("div",{className:"note-entry",children:[o.created_at?l("div",{className:"note-timestamp",children:new Date(o.created_at).toLocaleString()}):null,l("div",{className:"note-text",children:o.text})]}))}).toString()):(s.style.display="none",a.innerHTML="");let r=document.getElementById("detail-meta");r.innerHTML=l(S,{children:[l("div",{children:["Created: ",new Date(t.created_at).toLocaleString()]}),l("div",{children:["Updated: ",new Date(t.updated_at).toLocaleString()]}),t.completed_at?l("div",{children:["Completed: ",new Date(t.completed_at).toLocaleString()]}):null,t.verified_at?l("div",{children:["Verified: ",new Date(t.verified_at).toLocaleString()]}):null]}).toString()}async function me(e){let t=await d(`/tickets/${e}`);if(n.activeTicketId!==e)return;ue(!1),document.getElementById("detail-ticket-number").textContent=t.ticket_number,document.getElementById("detail-title").value=t.title,document.getElementById("detail-category").value=t.category,document.getElementById("detail-priority").value=t.priority,document.getElementById("detail-status").value=t.status,document.getElementById("detail-upnext").checked=t.up_next,document.getElementById("detail-details").value=t.details;let s=document.getElementById("detail-attachments");t.attachments.length>0?s.innerHTML=l(S,{children:t.attachments.map(c=>l("div",{className:"attachment-item",children:[l("span",{className:"attachment-name",children:c.original_filename}),l("button",{className:"attachment-reveal","data-att-id":String(c.id),title:"Show in file manager",children:I('<svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M22 19a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h5l2 3h9a2 2 0 0 1 2 2z"/></svg>')}),l("button",{className:"attachment-delete","data-att-id":String(c.id),title:"Remove",children:I("×")})]}))}).toString():s.innerHTML="";let a=document.getElementById("detail-notes-section"),i=document.getElementById("detail-notes"),r=pe(t.notes);r.length>0?(a.style.display="",i.innerHTML=l(S,{children:r.map(c=>l("div",{className:"note-entry",children:[c.created_at?l("div",{className:"note-timestamp",children:new Date(c.created_at).toLocaleString()}):null,l("div",{className:"note-text",children:c.text})]}))}).toString()):(a.style.display="none",i.innerHTML="");let o=document.getElementById("detail-meta");o.innerHTML=l(S,{children:[l("div",{children:["Created: ",new Date(t.created_at).toLocaleString()]}),l("div",{children:["Updated: ",new Date(t.updated_at).toLocaleString()]}),t.completed_at?l("div",{children:["Completed: ",new Date(t.completed_at).toLocaleString()]}):null,t.verified_at?l("div",{children:["Verified: ",new Date(t.verified_at).toLocaleString()]}):null]}).toString()}function pe(e){if(!e||e==="")return[];try{let t=JSON.parse(e);if(Array.isArray(t))return t}catch{}return e.trim()?[{text:e,created_at:""}]:[]}async function $(){try{let e=await d("/stats"),t=document.getElementById("status-bar");t&&(t.textContent=`${e.total} tickets \xB7 ${e.open} open \xB7 ${e.up_next} up next`)}catch{}}function W(e){let t=document.getElementById("content-area");t.classList.remove("detail-side","detail-bottom"),t.classList.add(e==="bottom"?"detail-bottom":"detail-side")}function G(){let e=document.getElementById("detail-panel");n.settings.detail_position==="bottom"?(e.style.width="",e.style.height=`${n.settings.detail_height}px`):(e.style.height="",e.style.width=`${n.settings.detail_width}px`)}function fe(){let e=document.getElementById("detail-resize-handle"),t=document.getElementById("detail-panel"),s=document.getElementById("content-area"),a=!1;e.addEventListener("mousedown",i=>{i.preventDefault(),a=!0,document.body.style.cursor=n.settings.detail_position==="bottom"?"row-resize":"col-resize",document.body.style.userSelect="none"}),document.addEventListener("mousemove",i=>{if(!a)return;let r=s.getBoundingClientRect();if(n.settings.detail_position==="bottom"){let o=Math.max(150,Math.min(500,r.bottom-i.clientY));n.settings.detail_height=o,t.style.height=`${o}px`}else{let o=Math.max(250,Math.min(600,r.right-i.clientX));n.settings.detail_width=o,t.style.width=`${o}px`}}),document.addEventListener("mouseup",()=>{a&&(a=!1,document.body.style.cursor="",document.body.style.userSelect="",n.settings.detail_position==="bottom"?d("/settings",{method:"PATCH",body:{detail_height:String(n.settings.detail_height)}}):d("/settings",{method:"PATCH",body:{detail_width:String(n.settings.detail_width)}}))})}function R(e,t){let s=t.getBoundingClientRect(),a=e.getBoundingClientRect(),i=window.innerWidth,r=window.innerHeight,o=s.left;o+a.width>i-8&&(o=s.right-a.width),o<8&&(o=8);let c=s.bottom+4;c+a.height>r-8&&(c=s.top-a.height-4),c<8&&(c=8),e.style.left=`${o}px`,e.style.top=`${c}px`}function q(e,t){let s=y(l("div",{className:"dropdown-menu",style:"visibility:hidden;top:0;left:0",children:t.map(c=>l("button",{className:`dropdown-item${c.active?" active":""}`,"data-key":c.key,children:[c.color?l("span",{className:"dropdown-dot",style:`background-color:${c.color}`}):null,l("span",{className:"dropdown-label",children:c.label}),c.shortcut?l("kbd",{className:"dropdown-kbd",children:c.shortcut}):null]}))}));s.querySelectorAll(".dropdown-item").forEach((c,m)=>{c.addEventListener("click",()=>{t[m].action(),s.remove()})});function i(c){let m=t.find(u=>c.key.toLowerCase()===u.key.toLowerCase());m?(c.preventDefault(),c.stopPropagation(),m.action(),r()):c.key==="Escape"&&(c.preventDefault(),r())}function r(){s.remove(),document.removeEventListener("keydown",i,!0),document.removeEventListener("click",o)}function o(){r()}return document.addEventListener("keydown",i,!0),setTimeout(()=>{document.addEventListener("click",o)},0),s}function O(){document.querySelectorAll(".dropdown-menu").forEach(e=>{e.remove()})}var Y=null,M=!1,N=null,J="",h=[],U=[{key:"i",value:"issue",label:"Issue"},{key:"b",value:"bug",label:"Bug"},{key:"f",value:"feature",label:"Feature"},{key:"r",value:"requirement_change",label:"Req Change"},{key:"k",value:"task",label:"Task"},{key:"g",value:"investigation",label:"Investigation"}],X=[{key:"1",value:"highest",label:"Highest"},{key:"2",value:"high",label:"High"},{key:"3",value:"default",label:"Default"},{key:"4",value:"low",label:"Low"},{key:"5",value:"lowest",label:"Lowest"}];function Ue(){let e=document.activeElement;if(!e||!(e instanceof HTMLElement))return null;let t=e.closest(".ticket-row");if(!t)return null;if(t.classList.contains("draft-row"))return"draft";let s=t.dataset.id;return s!==void 0&&s!==""?parseInt(s,10):null}function Ke(e){e!=null&&(M=!0,e==="draft"?b():document.querySelector(`.ticket-row[data-id="${e}"] .ticket-title-input`)?.focus(),M=!1)}function K(){let e=n.view;return e!=="completed"&&e!=="verified"&&e!=="trash"&&e!=="backlog"&&e!=="archive"}function ve(){return n.view==="up-next"||n.view==="open"?[{status:"not_started",label:"Not Started"},{status:"started",label:"Started"}]:n.view==="non-verified"?[{status:"not_started",label:"Not Started"},{status:"started",label:"Started"},{status:"completed",label:"Completed"}]:[{status:"not_started",label:"Not Started"},{status:"started",label:"Started"},{status:"completed",label:"Completed"},{status:"verified",label:"Verified"}]}function v(){let e=!!n.backupPreview?.active;if(n.layout==="columns"&&K()){if(e){Ve();return}je();return}let t=n.view==="trash",s=e?null:Ue(),a=null;if(s!=null&&s!=="draft"){let o=document.querySelector(`.ticket-row[data-id="${s}"] .ticket-title-input`);o&&(a=o.value)}let i=document.getElementById("ticket-list"),r=i.scrollTop;if(i.innerHTML="",i.classList.remove("ticket-list-columns"),!t&&!e&&i.appendChild(Ee()),n.tickets.length===0){let o=t?"Trash is empty":e?"No tickets match this view":"";o&&i.appendChild(y(l("div",{className:"ticket-list-empty",children:o})))}for(let o of n.tickets)e?i.appendChild(Fe(o)):t?i.appendChild(Xe(o)):i.appendChild(Je(o));if(i.scrollTop=r,e){let o=document.getElementById("batch-toolbar");o&&(o.style.display="none"),P(),H()}else{let o=document.getElementById("batch-toolbar");if(o&&(o.style.display=""),s!=null&&s!=="draft"&&a!=null){let c=document.querySelector(`.ticket-row[data-id="${s}"] .ticket-title-input`);c&&c.value!==a&&(c.value=a)}Ke(s),C()}$()}function Fe(e){let t=n.selectedIds.has(e.id),s=e.status==="completed"||e.status==="verified",a=e.status==="verified",i=y(l("div",{className:`ticket-row${t?" selected":""}${s?" completed":""}${e.up_next?" up-next":""}`,"data-id":String(e.id),children:[l("span",{className:"ticket-checkbox-spacer"}),l("span",{className:`ticket-status-btn${a?" verified":""}`,style:"cursor:default",children:a?I(z):j(e.status)}),l("span",{className:"ticket-category-badge",style:`background-color:${E(e.category)};cursor:default`,title:e.category,children:L(e.category)}),l("span",{className:"ticket-number",children:e.ticket_number}),l("span",{className:"ticket-title-input",style:"cursor:default",children:e.title}),l("span",{className:"ticket-priority-indicator",style:`color:${_(e.priority)};cursor:default`,title:e.priority,children:B(e.priority)}),l("span",{className:`ticket-star${e.up_next?" active":""}`,style:"cursor:default",children:e.up_next?"\u2605":"\u2606"})]}));return i.addEventListener("click",()=>{n.selectedIds.clear(),n.selectedIds.add(e.id),n.lastClickedId=e.id,P(),H()}),i}function he(e){let t={scrollLeft:0,columns:{}},s=e.querySelector(".columns-container");return s&&(t.scrollLeft=s.scrollLeft,s.querySelectorAll(".column[data-status]").forEach(a=>{let i=a.dataset.status,r=a.querySelector(".column-body");r&&(t.columns[i]=r.scrollTop)})),t}function be(e,t){let s=e.querySelector(".columns-container");s&&(s.scrollLeft=t.scrollLeft,s.querySelectorAll(".column[data-status]").forEach(a=>{let i=a.dataset.status,r=a.querySelector(".column-body");r&&t.columns[i]!=null&&(r.scrollTop=t.columns[i])}))}function Ve(){let e=document.getElementById("ticket-list"),t=he(e);e.innerHTML="",e.classList.add("ticket-list-columns");let s=ve(),a=y(l("div",{className:"columns-container"}));for(let r of s){let o=n.tickets.filter(u=>u.status===r.status),c=y(l("div",{className:"column","data-status":r.status,children:[l("div",{className:"column-header",children:[l("span",{className:"column-title",children:r.label}),l("span",{className:"column-count",children:String(o.length)})]}),l("div",{className:"column-body"})]})),m=c.querySelector(".column-body");for(let u of o)m.appendChild(ze(u));a.appendChild(c)}e.appendChild(a),be(e,t);let i=document.getElementById("batch-toolbar");i&&(i.style.display="none"),$()}function ze(e){let t=n.selectedIds.has(e.id),s=y(l("div",{className:`column-card${t?" selected":""}${e.up_next?" up-next":""}`,"data-id":String(e.id),children:[l("div",{className:"column-card-header",children:[l("span",{className:"ticket-category-badge",style:`background-color:${E(e.category)}`,children:L(e.category)}),l("span",{className:"ticket-number",children:e.ticket_number}),l("span",{className:"ticket-priority-indicator",style:`color:${_(e.priority)};cursor:default`,children:B(e.priority)}),l("span",{className:`ticket-star${e.up_next?" active":""}`,style:"cursor:default",children:e.up_next?"\u2605":"\u2606"})]}),l("div",{className:"column-card-title",children:e.title})]}));return s.addEventListener("click",()=>{n.selectedIds.clear(),n.selectedIds.add(e.id),n.lastClickedId=e.id,ke(),H()}),s}function je(){let e=document.getElementById("ticket-list"),t=he(e);e.innerHTML="",e.classList.add("ticket-list-columns"),e.appendChild(Ee());let s=ve(),a=y(l("div",{className:"columns-container"}));for(let i of s){let r=n.tickets.filter(m=>m.status===i.status),o=y(l("div",{className:"column","data-status":i.status,children:[l("div",{className:"column-header",children:[l("span",{className:"column-title",children:i.label}),l("span",{className:"column-count",children:String(r.length)})]}),l("div",{className:"column-body"})]})),c=o.querySelector(".column-body");for(let m of r)c.appendChild(We(m));c.addEventListener("dragover",m=>{m.preventDefault(),m.dataTransfer.dropEffect="move",o.classList.add("column-drop-target")}),c.addEventListener("dragleave",m=>{let u=m.relatedTarget;(!u||!c.contains(u))&&o.classList.remove("column-drop-target")}),c.addEventListener("drop",m=>{m.preventDefault(),o.classList.remove("column-drop-target");let u=h;h=[],u.length!==0&&d("/tickets/batch",{method:"POST",body:{ids:u,action:"status",value:i.status}}).then(()=>{p()})}),a.appendChild(o)}e.appendChild(a),be(e,t),C(),$()}function We(e){let t=n.selectedIds.has(e.id),s=y(l("div",{className:`column-card${t?" selected":""}${e.up_next?" up-next":""}`,"data-id":String(e.id),children:[l("div",{className:"column-card-header",children:[l("span",{className:"ticket-category-badge",style:`background-color:${E(e.category)}`,children:L(e.category)}),l("span",{className:"ticket-number",children:e.ticket_number}),l("span",{className:"ticket-priority-indicator",style:`color:${_(e.priority)}`,children:B(e.priority)}),l("button",{className:`ticket-star${e.up_next?" active":""}`,title:e.up_next?"Remove from Up Next":"Add to Up Next",children:e.up_next?"\u2605":"\u2606"})]}),l("div",{className:"column-card-title",children:e.title})]})),a=s.querySelector(".ticket-priority-indicator");return a.addEventListener("click",i=>{i.stopPropagation(),Le(a,e)}),s.querySelector(".ticket-star").addEventListener("click",i=>{i.stopPropagation(),we(e)}),s.draggable=!0,s.addEventListener("dragstart",i=>{n.selectedIds.has(e.id)&&n.selectedIds.size>1?h=Array.from(n.selectedIds):h=[e.id],i.dataTransfer.setData("text/plain",JSON.stringify(h)),i.dataTransfer.effectAllowed="move"}),s.addEventListener("dragend",()=>{h=[]}),s.addEventListener("click",i=>{if(i.metaKey||i.ctrlKey)n.selectedIds.has(e.id)?n.selectedIds.delete(e.id):n.selectedIds.add(e.id),n.lastClickedId=e.id;else if(i.shiftKey&&n.lastClickedId!=null){let r=n.tickets.map(m=>m.id),o=r.indexOf(n.lastClickedId),c=r.indexOf(e.id);if(o!==-1&&c!==-1){let m=Math.min(o,c),u=Math.max(o,c);n.selectedIds.clear();for(let g=m;g<=u;g++)n.selectedIds.add(r[g])}}else n.selectedIds.clear(),n.selectedIds.add(e.id),n.lastClickedId=e.id;ke(),C()}),s}function ke(){document.querySelectorAll(".column-card[data-id]").forEach(e=>{let t=parseInt(e.dataset.id,10);n.selectedIds.has(t)?e.classList.add("selected"):e.classList.remove("selected")})}function Ee(){let e=Te(),t=n.view.startsWith("category:"),s=y(l("div",{className:"ticket-row draft-row",children:[l("span",{className:"ticket-checkbox-spacer"}),l("span",{className:"ticket-status-btn draft-placeholder",children:"\u25CB"}),l("span",{className:"ticket-category-badge draft-badge",style:`background-color:${E(e)}${t?"":";cursor:pointer;opacity:1"}`,children:L(e)}),l("span",{className:"ticket-number draft-number"}),l("input",{type:"text",className:"ticket-title-input draft-input",placeholder:"New ticket...",value:J}),l("span",{className:"ticket-priority-indicator draft-placeholder"}),l("span",{className:"ticket-star draft-placeholder"})]}));if(!t){let i=s.querySelector(".ticket-category-badge");i.addEventListener("click",r=>{r.stopPropagation(),Ye(i)})}let a=s.querySelector(".draft-input");return a.addEventListener("input",()=>{J=a.value}),a.addEventListener("keydown",async i=>{if(i.key==="Enter"&&a.value.trim()){i.preventDefault();let r=a.value.trim();J="",a.value="";let o=Ge();N&&!n.view.startsWith("category:")&&(o.category=N);let c=await d("/tickets",{method:"POST",body:{title:r,defaults:o}});c&&(n.selectedIds.clear(),n.selectedIds.add(c.id)),await p(),b()}else i.key==="ArrowDown"&&(i.preventDefault(),n.tickets.length>0&&document.querySelector(`.ticket-row[data-id="${n.tickets[0].id}"] .ticket-title-input`)?.focus())}),s}function b(){document.querySelector(".draft-row .draft-input")?.focus()}function Ge(){let e=n.view;return e==="up-next"?{up_next:!0}:e==="open"?{}:e==="completed"?{status:"completed"}:e==="backlog"?{status:"backlog"}:e==="archive"?{status:"archive"}:e.startsWith("category:")?{category:e.split(":")[1]}:e.startsWith("priority:")?{priority:e.split(":")[1]}:{}}function Te(){if(N)return N;let e=n.view;return e.startsWith("category:")?e.split(":")[1]:"issue"}function Ye(e){O();let s=navigator.platform.includes("Mac")?"\u2318":"Ctrl+",a=Te(),i=q(e,U.map(r=>({label:r.label,key:r.key,shortcut:`${s}${r.key.toUpperCase()}`,color:E(r.value),active:a===r.value,action:()=>{N=r.value,v(),b()}})));document.body.appendChild(i),R(i,e),i.style.visibility=""}function Je(e){let t=n.selectedIds.has(e.id),s=e.status==="completed"||e.status==="verified",a=e.status==="verified",i=y(l("div",{className:`ticket-row${t?" selected":""}${s?" completed":""}${e.up_next?" up-next":""}`,"data-id":String(e.id),children:[l("input",{type:"checkbox",className:"ticket-checkbox",checked:t}),l("button",{className:`ticket-status-btn${a?" verified":""}`,title:e.status.replace("_"," "),children:a?I(z):j(e.status)}),l("span",{className:"ticket-category-badge",style:`background-color:${E(e.category)}`,title:e.category,children:L(e.category)}),l("span",{className:"ticket-number",children:e.ticket_number}),l("input",{type:"text",className:"ticket-title-input",value:e.title}),l("span",{className:"ticket-priority-indicator",style:`color:${_(e.priority)}`,title:e.priority,children:B(e.priority)}),l("button",{className:`ticket-star${e.up_next?" active":""}`,title:e.up_next?"Remove from Up Next":"Add to Up Next",children:e.up_next?"\u2605":"\u2606"})]}));i.addEventListener("mousedown",u=>{let g=u.target;g.tagName!=="INPUT"&&g.tagName!=="BUTTON"&&(i.draggable=!0)}),i.addEventListener("mouseup",()=>{i.draggable=!1}),i.addEventListener("dragend",()=>{i.draggable=!1,h=[]}),i.addEventListener("dragstart",u=>{n.selectedIds.has(e.id)&&n.selectedIds.size>1?h=Array.from(n.selectedIds):h=[e.id],u.dataTransfer.setData("text/plain",JSON.stringify(h)),u.dataTransfer.effectAllowed="move"}),i.addEventListener("mousedown",u=>{(u.metaKey||u.ctrlKey||u.shiftKey)&&(u.preventDefault(),Ie(u,e)&&u.stopPropagation())});let r=i.querySelector(".ticket-checkbox");r.addEventListener("click",u=>u.stopPropagation()),r.addEventListener("change",()=>{r.checked?n.selectedIds.add(e.id):n.selectedIds.delete(e.id),n.lastClickedId=e.id,v()}),i.querySelector(".ticket-status-btn").addEventListener("click",u=>{u.stopPropagation(),tt(e)});let o=i.querySelector(".ticket-category-badge");o.addEventListener("click",u=>{u.stopPropagation(),it(o,e)});let c=i.querySelector(".ticket-title-input");c.addEventListener("focus",()=>{M||n.selectedIds.size===1&&n.selectedIds.has(e.id)||(n.selectedIds.clear(),n.selectedIds.add(e.id),n.lastClickedId=e.id,P(),C())}),c.addEventListener("input",()=>{st(e.id,{title:c.value})}),c.addEventListener("keydown",u=>{Qe(u,e,c)});let m=i.querySelector(".ticket-priority-indicator");return m.addEventListener("click",u=>{u.stopPropagation(),Le(m,e)}),i.querySelector(".ticket-star").addEventListener("click",u=>{u.stopPropagation(),we(e)}),i}function Xe(e){let t=n.selectedIds.has(e.id),s=e.deleted_at?new Date(e.deleted_at):null,a=y(l("div",{className:`ticket-row trash-row${t?" selected":""}`,"data-id":String(e.id),children:[l("input",{type:"checkbox",className:"ticket-checkbox",checked:t}),l("span",{className:"ticket-category-badge",style:`background-color:${E(e.category)}`,children:L(e.category)}),l("span",{className:"ticket-number",children:e.ticket_number}),l("span",{className:"ticket-title-input trash-title",style:"cursor:default",children:e.title}),l("span",{className:"ticket-number",title:s?`Deleted: ${s.toLocaleString()}`:"",children:s?s.toLocaleDateString():""}),l("button",{className:"btn btn-sm",title:"Restore from trash",children:"Restore"})]}));a.addEventListener("mousedown",r=>{(r.metaKey||r.ctrlKey||r.shiftKey)&&(r.preventDefault(),Ie(r,e)&&r.stopPropagation())});let i=a.querySelector(".ticket-checkbox");return i.addEventListener("click",r=>r.stopPropagation()),i.addEventListener("change",()=>{i.checked?n.selectedIds.add(e.id):n.selectedIds.delete(e.id),n.lastClickedId=e.id,v()}),a.querySelector(".trash-title").addEventListener("click",()=>{n.selectedIds.size===1&&n.selectedIds.has(e.id)||(n.selectedIds.clear(),n.selectedIds.add(e.id),n.lastClickedId=e.id,P(),C())}),a.querySelector(".btn").addEventListener("click",async r=>{r.stopPropagation(),await d(`/tickets/${e.id}/restore`,{method:"POST"}),p()}),a}function Ie(e,t){let s=e.metaKey||e.ctrlKey,a=e.shiftKey;if(s)n.selectedIds.has(t.id)?n.selectedIds.delete(t.id):n.selectedIds.add(t.id),n.lastClickedId=t.id,v();else if(a&&n.lastClickedId!=null){let i=n.tickets.map(c=>c.id),r=i.indexOf(n.lastClickedId),o=i.indexOf(t.id);if(r!==-1&&o!==-1){let c=Math.min(r,o),m=Math.max(r,o);n.selectedIds.clear();for(let u=c;u<=m;u++)n.selectedIds.add(i[u])}v()}else return!1;return!0}function Qe(e,t,s){if(e.key==="Enter")e.preventDefault(),b();else if(e.key==="Backspace"&&s.value==="")e.preventDefault(),nt(t.id);else if(e.key==="ArrowDown"&&e.shiftKey)e.preventDefault(),ye(t.id,1);else if(e.key==="ArrowUp"&&e.shiftKey)e.preventDefault(),ye(t.id,-1);else if(e.key==="ArrowDown")e.preventDefault(),Ze(t.id);else if(e.key==="ArrowUp")e.preventDefault(),et(t.id);else if((e.metaKey||e.ctrlKey)&&!e.altKey&&U.some(a=>a.key===e.key)){e.preventDefault();let a=U.find(i=>i.key===e.key);ge(t,"category",a.value)}else if(e.altKey&&!e.metaKey&&!e.ctrlKey&&X.some(a=>a.key===e.key)){e.preventDefault();let a=X.find(i=>i.key===e.key);ge(t,"priority",a.value)}}function Ze(e){let t=n.tickets.findIndex(s=>s.id===e);t<n.tickets.length-1&&document.querySelector(`.ticket-row[data-id="${n.tickets[t+1].id}"] .ticket-title-input`)?.focus()}function et(e){let t=n.tickets.findIndex(s=>s.id===e);t>0?document.querySelector(`.ticket-row[data-id="${n.tickets[t-1].id}"] .ticket-title-input`)?.focus():b()}function ye(e,t){let a=n.tickets.findIndex(o=>o.id===e)+t;if(a<0||a>=n.tickets.length)return;let i=n.tickets[a].id;n.selectedIds.add(e),n.selectedIds.has(i)?n.selectedIds.delete(e):n.selectedIds.add(i),M=!0,document.querySelector(`.ticket-row[data-id="${i}"] .ticket-title-input`)?.focus(),M=!1,P(),C()}async function tt(e){let s={not_started:"started",started:"completed",completed:"verified",verified:"not_started",backlog:"not_started",archive:"not_started"}[e.status]||"not_started",a=await d(`/tickets/${e.id}`,{method:"PATCH",body:{status:s}});Object.assign(e,a),v()}async function we(e){!e.up_next&&(e.status==="completed"||e.status==="verified")?await d(`/tickets/${e.id}`,{method:"PATCH",body:{status:"not_started",up_next:!0}}):await d(`/tickets/${e.id}/up-next`,{method:"POST"}),p()}async function ge(e,t,s){let a=await d(`/tickets/${e.id}`,{method:"PATCH",body:{[t]:s}});Object.assign(e,a),v()}async function nt(e){let t=n.tickets.findIndex(s=>s.id===e);if(await d(`/tickets/${e}`,{method:"DELETE"}),n.tickets=n.tickets.filter(s=>s.id!==e),n.selectedIds.delete(e),v(),t>0&&n.tickets.length>0){let s=Math.min(t-1,n.tickets.length-1);document.querySelector(`.ticket-row[data-id="${n.tickets[s].id}"] .ticket-title-input`)?.focus()}else b()}function st(e,t){Y&&clearTimeout(Y),Y=setTimeout(()=>{d(`/tickets/${e}`,{method:"PATCH",body:t})},300)}function it(e,t){O();let a=navigator.platform.includes("Mac")?"\u2318":"Ctrl+",i=q(e,U.map(r=>({label:r.label,key:r.key,shortcut:`${a}${r.key.toUpperCase()}`,color:E(r.value),active:t.category===r.value,action:async()=>{let o=await d(`/tickets/${t.id}`,{method:"PATCH",body:{category:r.value}});Object.assign(t,o),v()}})));document.body.appendChild(i),R(i,e),i.style.visibility=""}function Le(e,t){O();let s=q(e,X.map(a=>({label:a.label,key:a.key,shortcut:`Alt+${a.key}`,color:_(a.value),active:t.priority===a.value,action:async()=>{let i=await d(`/tickets/${t.id}`,{method:"PATCH",body:{priority:a.value}});Object.assign(t,i),v()}})));document.body.appendChild(s),R(s,e),s.style.visibility=""}function P(){document.querySelectorAll(".ticket-row[data-id]").forEach(e=>{let t=parseInt(e.dataset.id,10),s=e.querySelector(".ticket-checkbox");n.selectedIds.has(t)?(e.classList.add("selected"),s&&(s.checked=!0)):(e.classList.remove("selected"),s&&(s.checked=!1))})}function C(){let e=n.selectedIds.size,t=n.tickets.length,s=e>0,a=n.view==="trash",i=document.getElementById("batch-select-all");i.checked=t>0&&e===t,i.indeterminate=e>0&&e<t,document.getElementById("batch-count").textContent=s?`${e} selected`:"";let r=["batch-category","batch-priority","batch-status","batch-upnext","batch-delete"];for(let g of r){let k=document.getElementById(g);k.style.display=a?"none":"",a||(k.disabled=!s)}let o=document.getElementById("batch-restore"),c=document.getElementById("batch-empty-trash");if(a){let g=document.getElementById("batch-toolbar");o||(o=y(l("button",{id:"batch-restore",className:"btn btn-sm",children:"Restore"})),o.addEventListener("click",async()=>{await d("/tickets/batch",{method:"POST",body:{ids:Array.from(n.selectedIds),action:"restore"}}),n.selectedIds.clear(),p()}),g.insertBefore(o,document.getElementById("batch-count"))),o.disabled=!s,o.style.display="",c||(c=y(l("button",{id:"batch-empty-trash",className:"btn btn-sm btn-danger",children:"Empty Trash"})),c.addEventListener("click",async()=>{await d("/trash/empty",{method:"POST"}),n.selectedIds.clear(),p()}),g.insertBefore(c,document.getElementById("batch-count"))),c.disabled=t===0,c.style.display=""}else o&&(o.style.display="none"),c&&(c.style.display="none");let m=document.querySelector(".batch-star-icon"),u=document.getElementById("batch-upnext");if(!a&&m&&s){let g=n.tickets.filter(w=>n.selectedIds.has(w.id)),k=g.every(w=>w.up_next),D=g.every(w=>!w.up_next);k?(m.textContent="\u2605",u.classList.add("active"),u.classList.remove("mixed")):D?(m.textContent="\u2606",u.classList.remove("active","mixed")):(m.innerHTML=l("span",{className:"star-mixed-wrap",children:[l("span",{className:"star-mixed-fill",children:"\u2605"}),"\u2606"]}).toString(),u.classList.remove("active"),u.classList.add("mixed"))}else m&&(m.textContent="\u2606",u.classList.remove("active","mixed"));H()}async function p(){if(n.backupPreview?.active){at();return}let e=new URLSearchParams;n.view==="trash"?e.set("status","deleted"):n.view==="up-next"?e.set("up_next","true"):n.view==="open"?e.set("status","open"):n.view==="completed"?e.set("status","completed"):n.view==="non-verified"?e.set("status","non_verified"):n.view==="verified"?e.set("status","verified"):n.view==="backlog"?e.set("status","backlog"):n.view==="archive"?e.set("status","archive"):n.view.startsWith("category:")?e.set("category",n.view.split(":")[1]):n.view.startsWith("priority:")?e.set("priority",n.view.split(":")[1]):e.set("status","active"),n.search&&e.set("search",n.search),e.set("sort_by",n.sortBy),e.set("sort_dir",n.sortDir);let t=e.toString();n.tickets=await d(`/tickets${t?"?"+t:""}`),v()}function at(){let e=[...n.backupPreview?.tickets||[]];if(n.view==="trash")e=e.filter(t=>t.status==="deleted");else if(n.view==="up-next")e=e.filter(t=>t.up_next&&t.status!=="deleted");else if(n.view==="open")e=e.filter(t=>t.status==="not_started"||t.status==="started");else if(n.view==="completed")e=e.filter(t=>t.status==="completed");else if(n.view==="non-verified")e=e.filter(t=>t.status!=="verified"&&t.status!=="deleted"&&t.status!=="backlog"&&t.status!=="archive");else if(n.view==="verified")e=e.filter(t=>t.status==="verified");else if(n.view==="backlog")e=e.filter(t=>t.status==="backlog");else if(n.view==="archive")e=e.filter(t=>t.status==="archive");else if(n.view.startsWith("category:")){let t=n.view.split(":")[1];e=e.filter(s=>s.category===t&&s.status!=="deleted"&&s.status!=="backlog"&&s.status!=="archive")}else if(n.view.startsWith("priority:")){let t=n.view.split(":")[1];e=e.filter(s=>s.priority===t&&s.status!=="deleted"&&s.status!=="backlog"&&s.status!=="archive")}else e=e.filter(t=>t.status!=="deleted"&&t.status!=="backlog"&&t.status!=="archive");if(n.search){let t=n.search.toLowerCase();e=e.filter(s=>s.title.toLowerCase().includes(t)||s.ticket_number.toLowerCase().includes(t)||s.details&&s.details.toLowerCase().includes(t))}n.tickets=e,v()}function ot(e){let t=Date.now()-new Date(e).getTime(),s=Math.floor(t/6e4);if(s<1)return"just now";if(s<60)return`${s}m ago`;let a=Math.floor(s/60);return a<24?`${a}h ago`:`${Math.floor(a/24)}d ago`}function lt(e){return e<1024?`${e} B`:e<1024*1024?`${(e/1024).toFixed(0)} KB`:`${(e/(1024*1024)).toFixed(1)} MB`}function rt(e){return e==="5min"?"Recent (every 5 min)":e==="hourly"?"Hourly":e==="daily"?"Daily":e}async function Q(){var t;let e=document.getElementById("backup-list");if(e)try{let s=await d("/backups");if(s.backups.length===0){e.textContent="No backups yet. First backup will be created shortly.";return}let a={};for(let i of s.backups)(a[t=i.tier]||(a[t]=[])).push(i);e.innerHTML="";for(let i of["5min","hourly","daily"]){let r=a[i];if(!(!r||r.length===0)){e.appendChild(y(l("div",{className:"backup-tier-label",children:rt(i)})));for(let o of r){let c=y(l("div",{className:"backup-row","data-tier":o.tier,"data-filename":o.filename,children:[l("span",{className:"backup-row-time",children:ot(o.createdAt)}),l("span",{className:"backup-row-meta",children:[new Date(o.createdAt).toLocaleString()," \xB7 ",lt(o.sizeBytes)]})]}));c.addEventListener("click",()=>{ct(o.tier,o.filename,o.createdAt)}),e.appendChild(c)}}}}catch{e.textContent="Failed to load backups."}}async function ct(e,t,s){let a=document.getElementById("settings-overlay");a.style.display="none";let i=document.getElementById("backup-preview-banner"),r=document.getElementById("backup-preview-label");r.textContent="Loading backup preview...",i.style.display="flex";try{let o=await d(`/backups/preview/${e}/${t}`);n.backupPreview={active:!0,tickets:o.tickets,timestamp:s,tier:e,filename:t},n.selectedIds.clear(),n.activeTicketId=null,r.textContent=`Previewing backup from ${new Date(s).toLocaleString()} (${o.stats.total} tickets, ${o.stats.open} open) \u2014 read-only`,p()}catch{r.textContent="Failed to load backup preview.",setTimeout(()=>{i.style.display="none"},3e3)}}async function dt(){let e=document.getElementById("backup-preview-banner");e.style.display="none",n.backupPreview=null,n.selectedIds.clear(),n.activeTicketId=null,await d("/backups/preview/cleanup",{method:"POST"}),p()}async function ut(){if(!n.backupPreview)return;let e=document.getElementById("backup-restore-btn");e.textContent="Restoring...",e.disabled=!0;try{await d("/backups/restore",{method:"POST",body:{tier:n.backupPreview.tier,filename:n.backupPreview.filename}}),window.location.reload()}catch{e.textContent="Restore failed",e.disabled=!1,setTimeout(()=>{e.textContent="Restore This Backup"},3e3)}}function xe(){document.getElementById("backup-cancel-btn")?.addEventListener("click",()=>{dt()}),document.getElementById("backup-restore-btn")?.addEventListener("click",()=>{ut()});let e=document.getElementById("backup-now-btn");e?.addEventListener("click",async()=>{e.textContent="Backing up...",e.disabled=!0;try{(await d("/backups/now",{method:"POST"})).error?e.textContent="In progress...":(e.textContent="Done!",Q())}catch{e.textContent="Failed"}setTimeout(()=>{e.textContent="Backup Now",e.disabled=!1},1500)})}async function mt(){await pt(),ft(),await p(),Tt(),bt(),kt(),It(),wt(),Lt(),xt(),Ct(),yt(),xe(),ht(),fe(),Bt(),gt(),document.addEventListener("hotsheet:render",()=>v()),b()}async function pt(){try{let e=await d("/settings");(e.detail_position==="side"||e.detail_position==="bottom")&&(n.settings.detail_position=e.detail_position),e.detail_width&&(n.settings.detail_width=parseInt(e.detail_width,10)||360),e.detail_height&&(n.settings.detail_height=parseInt(e.detail_height,10)||300),e.trash_cleanup_days&&(n.settings.trash_cleanup_days=parseInt(e.trash_cleanup_days,10)||3),e.verified_cleanup_days&&(n.settings.verified_cleanup_days=parseInt(e.verified_cleanup_days,10)||30),(e.layout==="list"||e.layout==="columns")&&(n.layout=e.layout)}catch{}W(n.settings.detail_position),G()}async function ft(){try{let e=await d("/file-settings");if(e.appName){document.title=e.appName;let t=document.querySelector(".app-title h1");t&&(t.textContent=e.appName)}}catch{}}function yt(){let e=document.getElementById("settings-overlay"),t=document.getElementById("settings-close");document.getElementById("settings-btn").addEventListener("click",()=>{document.getElementById("settings-trash-days").value=String(n.settings.trash_cleanup_days),document.getElementById("settings-verified-days").value=String(n.settings.verified_cleanup_days),e.style.display="flex",Q(),d("/file-settings").then(f=>{document.getElementById("settings-app-name").value=f.appName||"",document.getElementById("settings-backup-dir").value=f.backupDir||""})}),t.addEventListener("click",()=>{e.style.display="none"}),e.addEventListener("click",f=>{f.target===e&&(e.style.display="none")});let a=document.getElementById("settings-trash-days"),i=null;a.addEventListener("input",()=>{i&&clearTimeout(i),i=setTimeout(()=>{let f=Math.max(1,parseInt(a.value,10)||3);a.value=String(f),n.settings.trash_cleanup_days=f,d("/settings",{method:"PATCH",body:{trash_cleanup_days:String(f)}})},500)});let r=document.getElementById("settings-verified-days"),o=null;r.addEventListener("input",()=>{o&&clearTimeout(o),o=setTimeout(()=>{let f=Math.max(1,parseInt(r.value,10)||30);r.value=String(f),n.settings.verified_cleanup_days=f,d("/settings",{method:"PATCH",body:{verified_cleanup_days:String(f)}})},500)});let c=document.getElementById("settings-app-name"),m=document.getElementById("settings-app-name-hint"),u=null;c.addEventListener("input",()=>{u&&clearTimeout(u),u=setTimeout(()=>{let f=c.value.trim();d("/file-settings",{method:"PATCH",body:{appName:f}}).then(()=>{let x=f||"Hot Sheet";document.title=x;let ie=document.querySelector(".app-title h1");ie&&(ie.textContent=x),m.textContent=f?"Saved. Restart the desktop app to update the title bar.":"Using default name."})},800)});let g=document.getElementById("check-updates-btn"),k=document.getElementById("check-updates-status");g.addEventListener("click",async()=>{let f=se();if(f){g.disabled=!0,g.textContent="Checking...",k.textContent="";try{let x=await f("check_for_update");x?(k.textContent=`Update available: v${x}`,_e(x)):k.textContent="Your software is up to date."}catch{k.textContent="Could not check for updates."}g.textContent="Check for Updates",g.disabled=!1}});let D=document.getElementById("settings-backup-dir"),w=document.getElementById("settings-backup-dir-hint"),F=null;D.addEventListener("input",()=>{F&&clearTimeout(F),F=setTimeout(()=>{let f=D.value.trim();d("/file-settings",{method:"PATCH",body:{backupDir:f}}).then(()=>{w.textContent=f?"Saved. New backups will use this location.":"Using default location inside the data directory."})},800)})}async function gt(){let e=se();if(!e)return;let t=document.getElementById("settings-updates-section");t&&(t.style.display="");let s=[0,3e3,1e4];for(let a of s){a>0&&await new Promise(i=>setTimeout(i,a));try{let i=await e("get_pending_update");if(i){_e(i);return}}catch{return}}}function se(){return window.__TAURI__?.core?.invoke??null}function _e(e){let t=document.getElementById("update-banner");if(!t)return;let s=document.getElementById("update-banner-label");s&&(s.textContent=`Update available: v${e}`),t.style.display="flex";let a=document.getElementById("update-install-btn");a?.addEventListener("click",async()=>{if(a){a.textContent="Installing...",a.disabled=!0;try{await se()?.("install_update"),s&&(s.textContent="Update installed! Restart the app to apply."),a.style.display="none"}catch{a.textContent="Install Failed",a.disabled=!1}}}),document.getElementById("update-banner-dismiss")?.addEventListener("click",()=>{t.style.display="none"})}function vt(){let e=document.getElementById("skills-banner");if(!e)return;e.style.display="flex",document.getElementById("skills-banner-dismiss")?.addEventListener("click",()=>{e.style.display="none"})}function ht(){let e=document.getElementById("copy-prompt-section"),t=document.getElementById("copy-prompt-btn"),s=document.getElementById("copy-prompt-label"),a=document.getElementById("copy-prompt-icon"),i="";d("/worklist-info").then(r=>{i=r.prompt,e.style.display="",r.skillCreated&&vt()}),t.addEventListener("click",()=>{i!==""&&navigator.clipboard.writeText(i).then(()=>{s.textContent="Copied!",a.innerHTML='<svg xmlns="http://www.w3.org/2000/svg" width="13" height="13" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M20 6L9 17l-5-5"/></svg>',setTimeout(()=>{s.textContent="Copy AI prompt",a.innerHTML='<svg xmlns="http://www.w3.org/2000/svg" width="13" height="13" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect x="9" y="9" width="13" height="13" rx="2"/><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"/></svg>'},1500)})})}function ne(){let e=document.getElementById("layout-toggle"),t=K(),s=e.querySelector('[data-layout="columns"]');s.disabled=!t,s.style.opacity=t?"":"0.3";let a=n.layout==="columns"&&!t?"list":n.layout;e.querySelectorAll(".layout-btn").forEach(i=>{i.classList.toggle("active",i.dataset.layout===a)})}function bt(){document.getElementById("layout-toggle").querySelectorAll(".layout-btn").forEach(t=>{t.addEventListener("click",()=>{let s=t.dataset.layout;s==="columns"&&!K()||(n.layout=s,ne(),v(),d("/settings",{method:"PATCH",body:{layout:s}}))})}),ne()}function Se(){document.getElementById("detail-position-toggle").querySelectorAll(".layout-btn").forEach(t=>{t.classList.toggle("active",t.dataset.position===n.settings.detail_position)})}function kt(){document.getElementById("detail-position-toggle").querySelectorAll(".layout-btn").forEach(t=>{t.addEventListener("click",()=>{let s=t.dataset.position;n.settings.detail_position=s,W(s),G(),Se(),d("/settings",{method:"PATCH",body:{detail_position:s}})})}),Se()}function Ce(e){return e==="up-next"?{action:"up_next",value:!0}:e==="open"?{action:"status",value:"not_started"}:e==="completed"?{action:"status",value:"completed"}:e==="verified"?{action:"status",value:"verified"}:e==="backlog"?{action:"status",value:"backlog"}:e==="archive"?{action:"status",value:"archive"}:e==="trash"?{action:"delete",value:null}:e.startsWith("category:")?{action:"category",value:e.split(":")[1]}:e.startsWith("priority:")?{action:"priority",value:e.split(":")[1]}:null}async function Et(e,t){let s=Ce(e);s&&(s.action==="delete"?await d("/tickets/batch",{method:"POST",body:{ids:t,action:"delete"}}):await d("/tickets/batch",{method:"POST",body:{ids:t,action:s.action,value:s.value}}),p())}function Tt(){let e=document.querySelectorAll(".sidebar-item[data-view]");e.forEach(t=>{t.addEventListener("click",()=>{e.forEach(a=>{a.classList.remove("active")}),t.classList.add("active"),n.view=t.dataset.view,n.selectedIds.clear(),ne(),p()});let s=t.dataset.view;Ce(s)&&(t.addEventListener("dragover",a=>{a.preventDefault(),a.dataTransfer.dropEffect="move",t.classList.add("drop-target")}),t.addEventListener("dragleave",()=>{t.classList.remove("drop-target")}),t.addEventListener("drop",a=>{a.preventDefault(),t.classList.remove("drop-target");let i=[...h];i.length!==0&&Et(s,i)}))})}function It(){let e=document.getElementById("sort-select");e.addEventListener("change",()=>{let[t,s]=e.value.split(":");n.sortBy=t,n.sortDir=s,p()})}var Z=null;function wt(){let e=document.getElementById("search-input");e.addEventListener("input",()=>{Z&&clearTimeout(Z),Z=setTimeout(()=>{n.search=e.value,p()},200)}),e.addEventListener("keydown",t=>{t.key==="Escape"&&(e.value="",n.search="",p())})}function Lt(){let e=document.getElementById("batch-category");e.addEventListener("change",async()=>{e.value&&(await d("/tickets/batch",{method:"POST",body:{ids:Array.from(n.selectedIds),action:"category",value:e.value}}),e.value="",p())});let t=document.getElementById("batch-priority");t.addEventListener("change",async()=>{t.value&&(await d("/tickets/batch",{method:"POST",body:{ids:Array.from(n.selectedIds),action:"priority",value:t.value}}),t.value="",p())});let s=document.getElementById("batch-status");s.addEventListener("change",async()=>{s.value&&(await d("/tickets/batch",{method:"POST",body:{ids:Array.from(n.selectedIds),action:"status",value:s.value}}),s.value="",p())}),document.getElementById("batch-upnext").addEventListener("click",async()=>{let a=n.tickets.filter(o=>n.selectedIds.has(o.id)),r=!a.every(o=>o.up_next);if(r){let o=a.filter(c=>c.status==="completed"||c.status==="verified");o.length>0&&await d("/tickets/batch",{method:"POST",body:{ids:o.map(c=>c.id),action:"status",value:"not_started"}})}await d("/tickets/batch",{method:"POST",body:{ids:Array.from(n.selectedIds),action:"up_next",value:r}}),p()}),document.getElementById("batch-delete").addEventListener("click",async()=>{await d("/tickets/batch",{method:"POST",body:{ids:Array.from(n.selectedIds),action:"delete"}}),n.selectedIds.clear(),p()}),document.getElementById("batch-select-all").addEventListener("change",a=>{if(a.target.checked)for(let r of n.tickets)n.selectedIds.add(r.id);else n.selectedIds.clear();v()})}var ee=null;function xt(){document.getElementById("detail-close").addEventListener("click",de);let e=["detail-title","detail-details"];for(let s of e){let a=document.getElementById(s);a.addEventListener("input",()=>{ee&&clearTimeout(ee),ee=setTimeout(()=>{if(n.activeTicketId==null)return;let i=s.replace("detail-","");d(`/tickets/${n.activeTicketId}`,{method:"PATCH",body:{[i]:a.value}}).then(()=>{p()})},300)})}let t=["detail-category","detail-priority","detail-status"];for(let s of t){let a=document.getElementById(s);a.addEventListener("change",async()=>{if(n.activeTicketId==null)return;let i=s.replace("detail-","");await d(`/tickets/${n.activeTicketId}`,{method:"PATCH",body:{[i]:a.value}}),p()})}document.getElementById("detail-upnext").addEventListener("change",async()=>{if(n.activeTicketId==null)return;let s=n.tickets.find(i=>i.id===n.activeTicketId);document.getElementById("detail-upnext").checked&&s&&(s.status==="completed"||s.status==="verified")?await d(`/tickets/${n.activeTicketId}`,{method:"PATCH",body:{status:"not_started",up_next:!0}}):await d(`/tickets/${n.activeTicketId}/up-next`,{method:"POST"}),p(),A(n.activeTicketId)}),document.getElementById("detail-file-input").addEventListener("change",async s=>{let a=s.target,i=a.files?.[0];!i||n.activeTicketId==null||(await ce(`/tickets/${n.activeTicketId}/attachments`,i),a.value="",A(n.activeTicketId),p())}),document.getElementById("detail-attachments").addEventListener("click",async s=>{let a=s.target,i=a.closest(".attachment-reveal");if(i){let c=i.dataset.attId;c&&d(`/attachments/${c}/reveal`,{method:"POST"});return}let r=a.closest(".attachment-delete");if(r===null)return;let o=r.dataset.attId;o===void 0||o===""||(await d(`/attachments/${o}`,{method:"DELETE"}),n.activeTicketId!=null&&A(n.activeTicketId))})}function St(e){if(!e||e==="")return[];try{let t=JSON.parse(e);if(Array.isArray(t))return t}catch{}return e.trim()?[{text:e,created_at:""}]:[]}function _t(e){let t=[];t.push(`${e.ticket_number}: ${e.title}`),e.details.trim()&&(t.push(""),t.push(e.details.trim()));let s=St(e.notes);if(s.length>0){t.push("");for(let a of s)t.push(`- ${a.text}`)}return t.join(`
|
|
2
|
-
`)}function
|
|
1
|
+
"use strict";(()=>{var Ge=Object.defineProperty;var Ye=(e,t,s)=>t in e?Ge(e,t,{enumerable:!0,configurable:!0,writable:!0,value:s}):e[t]=s;var $=(e,t,s)=>Ye(e,typeof t!="symbol"?t+"":t,s);function ve(e){return e.replace(/&/g,"&").replace(/</g,"<").replace(/>/g,">").replace(/"/g,""")}function he(e){return e.replace(/&/g,"&").replace(/"/g,""").replace(/'/g,"'").replace(/</g,"<").replace(/>/g,">")}var x=class{constructor(t){$(this,"__html");this.__html=t}toString(){return this.__html}};function S(e){return new x(e)}var Xe=new Set(["area","base","br","col","embed","hr","img","input","link","meta","source","track","wbr"]);function Z(e){return e==null||typeof e=="boolean"?"":e instanceof x?e.__html:typeof e=="string"?ve(e):typeof e=="number"?String(e):Array.isArray(e)?e.map(Z).join(""):""}function Qe(e,t){if(t==null||t===!1)return"";if(t===!0)return` ${e}`;let s=e==="className"?"class":e==="htmlFor"?"for":e,a;return t instanceof x?a=t.__html:typeof t=="number"?a=String(t):typeof t=="string"?a=he(t):a="",` ${s}="${a}"`}function l(e,t){if(typeof e=="function")return e(t);let{children:s,...a}=t,i=Object.entries(a).map(([r,c])=>Qe(r,c)).join("");if(Xe.has(e))return new x(`<${e}${i}>`);let o=s!=null?Z(s):"";return new x(`<${e}${i}>${o}</${e}>`)}function N({children:e}){return new x(e!=null?Z(e):"")}function y(e){let t=document.createElement("template");return t.innerHTML=e.toString(),t.content.firstElementChild}function be(e){document.getElementById("network-error-popup")?.remove();let t=y(l("div",{id:"network-error-popup",className:"error-popup",children:l("div",{className:"error-popup-content",children:[l("strong",{children:"Connection Error"}),l("p",{children:e}),l("button",{children:S("Dismiss")})]})}));t.querySelector("button").addEventListener("click",()=>t.remove()),document.body.appendChild(t)}async function p(e,t={}){try{return(await fetch("/api"+e,{headers:t.body!==void 0?{"Content-Type":"application/json"}:{},method:t.method,body:t.body!==void 0?JSON.stringify(t.body):void 0})).json()}catch(s){throw be("Unable to reach the server. It may have been stopped."),s}}async function ke(e,t){try{let s=new FormData;return s.append("file",t),(await fetch("/api"+e,{method:"POST",body:s})).json()}catch(s){throw be("Unable to reach the server. It may have been stopped."),s}}var Ze={detail_position:"side",detail_width:360,detail_height:300,trash_cleanup_days:3,verified_cleanup_days:30},n={tickets:[],selectedIds:new Set,lastClickedId:null,activeTicketId:null,view:"all",layout:"list",sortBy:"created",sortDir:"desc",search:"",settings:{...Ze},backupPreview:null},et={issue:"#6b7280",bug:"#ef4444",feature:"#22c55e",requirement_change:"#f97316",task:"#3b82f6",investigation:"#8b5cf6"},tt={issue:"ISS",bug:"BUG",feature:"FEA",requirement_change:"REQ",task:"TSK",investigation:"INV"},nt={highest:"\u2B06\u2B06",high:"\u2B06",default:"\u2014",low:"\u2B07",lowest:"\u2B07\u2B07"},st={highest:"#ef4444",high:"#f97316",default:"#6b7280",low:"#3b82f6",lowest:"#94a3b8"},at={not_started:"\u25CB",started:"\u25D4",completed:"\u2713",verified:"svg",backlog:"\u25A1",archive:"\u25A0"},ee='<svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round"><path d="M18 6 7 17l-5-5"/><path d="m22 10-9.5 9.5-2-2"/></svg>';function w(e){return et[e]||"#6b7280"}function M(e){return tt[e]||"ISS"}function R(e){return nt[e]||"\u2014"}function D(e){return st[e]||"#6b7280"}function te(e){return at[e]||"\u25CB"}function V(e){n.activeTicketId=e,se(e)}function Ee(){n.selectedIds.clear(),n.activeTicketId=null;let e=new CustomEvent("hotsheet:render");document.dispatchEvent(e)}function U(){let e=n.view==="trash",t=!!n.backupPreview?.active,s=document.getElementById("detail-panel"),a=document.getElementById("detail-resize-handle");if(n.selectedIds.size===1&&!e){let i=Array.from(n.selectedIds)[0];s.style.display="flex",a&&(a.style.display=""),n.activeTicketId!==i&&(n.activeTicketId=i,t?it(i):se(i))}else n.activeTicketId!=null&&(n.activeTicketId=null),s.style.display="none",a&&(a.style.display="none")}function Te(e){let t=document.getElementById("detail-title"),s=document.getElementById("detail-details"),a=document.getElementById("detail-category"),i=document.getElementById("detail-priority"),o=document.getElementById("detail-status"),r=document.getElementById("detail-upnext"),c=document.querySelector(".upload-btn");t.readOnly=e,s.readOnly=e,a.disabled=e,i.disabled=e,o.disabled=e,r.disabled=e,c&&(c.style.display=e?"none":"")}function it(e){let t=n.backupPreview?.tickets.find(r=>r.id===e);if(!t||n.activeTicketId!==e)return;document.getElementById("detail-ticket-number").textContent=t.ticket_number,document.getElementById("detail-title").value=t.title,document.getElementById("detail-category").value=t.category,document.getElementById("detail-priority").value=t.priority,document.getElementById("detail-status").value=t.status,document.getElementById("detail-upnext").checked=t.up_next,document.getElementById("detail-details").value=t.details,Te(!0),document.getElementById("detail-attachments").innerHTML="";let s=document.getElementById("detail-notes-section"),a=document.getElementById("detail-notes"),i=Ie(t.notes);i.length>0?(s.style.display="",a.innerHTML=l(N,{children:i.map(r=>l("div",{className:"note-entry",children:[r.created_at?l("div",{className:"note-timestamp",children:new Date(r.created_at).toLocaleString()}):null,l("div",{className:"note-text",children:r.text})]}))}).toString()):(s.style.display="none",a.innerHTML="");let o=document.getElementById("detail-meta");o.innerHTML=l(N,{children:[l("div",{children:["Created: ",new Date(t.created_at).toLocaleString()]}),l("div",{children:["Updated: ",new Date(t.updated_at).toLocaleString()]}),t.completed_at?l("div",{children:["Completed: ",new Date(t.completed_at).toLocaleString()]}):null,t.verified_at?l("div",{children:["Verified: ",new Date(t.verified_at).toLocaleString()]}):null]}).toString()}function ne(){n.activeTicketId!=null&&se(n.activeTicketId)}async function se(e){let t=await p(`/tickets/${e}`);if(n.activeTicketId!==e)return;Te(!1),document.getElementById("detail-ticket-number").textContent=t.ticket_number,document.getElementById("detail-title").value=t.title,document.getElementById("detail-category").value=t.category,document.getElementById("detail-priority").value=t.priority,document.getElementById("detail-status").value=t.status,document.getElementById("detail-upnext").checked=t.up_next,document.getElementById("detail-details").value=t.details;let s=document.getElementById("detail-attachments");t.attachments.length>0?s.innerHTML=l(N,{children:t.attachments.map(c=>l("div",{className:"attachment-item",children:[l("span",{className:"attachment-name",children:c.original_filename}),l("button",{className:"attachment-reveal","data-att-id":String(c.id),title:"Show in file manager",children:S('<svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M22 19a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h5l2 3h9a2 2 0 0 1 2 2z"/></svg>')}),l("button",{className:"attachment-delete","data-att-id":String(c.id),title:"Remove",children:S("×")})]}))}).toString():s.innerHTML="";let a=document.getElementById("detail-notes-section"),i=document.getElementById("detail-notes"),o=Ie(t.notes);o.length>0?(a.style.display="",i.innerHTML=l(N,{children:o.map(c=>l("div",{className:"note-entry",children:[c.created_at?l("div",{className:"note-timestamp",children:new Date(c.created_at).toLocaleString()}):null,l("div",{className:"note-text",children:c.text})]}))}).toString()):(a.style.display="none",i.innerHTML="");let r=document.getElementById("detail-meta");r.innerHTML=l(N,{children:[l("div",{children:["Created: ",new Date(t.created_at).toLocaleString()]}),l("div",{children:["Updated: ",new Date(t.updated_at).toLocaleString()]}),t.completed_at?l("div",{children:["Completed: ",new Date(t.completed_at).toLocaleString()]}):null,t.verified_at?l("div",{children:["Verified: ",new Date(t.verified_at).toLocaleString()]}):null]}).toString()}function Ie(e){if(!e||e==="")return[];try{let t=JSON.parse(e);if(Array.isArray(t))return t}catch{}return e.trim()?[{text:e,created_at:""}]:[]}async function z(){try{let e=await p("/stats"),t=document.getElementById("status-bar");t&&(t.textContent=`${e.total} tickets \xB7 ${e.open} open \xB7 ${e.up_next} up next`)}catch{}}function ae(e){let t=document.getElementById("content-area");t.classList.remove("detail-side","detail-bottom"),t.classList.add(e==="bottom"?"detail-bottom":"detail-side")}function ie(){let e=document.getElementById("detail-panel");n.settings.detail_position==="bottom"?(e.style.width="",e.style.height=`${n.settings.detail_height}px`):(e.style.height="",e.style.width=`${n.settings.detail_width}px`)}function we(){let e=document.getElementById("detail-resize-handle"),t=document.getElementById("detail-panel"),s=document.getElementById("content-area"),a=!1;e.addEventListener("mousedown",i=>{i.preventDefault(),a=!0,document.body.style.cursor=n.settings.detail_position==="bottom"?"row-resize":"col-resize",document.body.style.userSelect="none"}),document.addEventListener("mousemove",i=>{if(!a)return;let o=s.getBoundingClientRect();if(n.settings.detail_position==="bottom"){let r=Math.max(150,Math.min(500,o.bottom-i.clientY));n.settings.detail_height=r,t.style.height=`${r}px`}else{let r=Math.max(250,Math.min(600,o.right-i.clientX));n.settings.detail_width=r,t.style.width=`${r}px`}}),document.addEventListener("mouseup",()=>{a&&(a=!1,document.body.style.cursor="",document.body.style.userSelect="",n.settings.detail_position==="bottom"?p("/settings",{method:"PATCH",body:{detail_height:String(n.settings.detail_height)}}):p("/settings",{method:"PATCH",body:{detail_width:String(n.settings.detail_width)}}))})}function j(e,t){let s=t.getBoundingClientRect(),a=e.getBoundingClientRect(),i=window.innerWidth,o=window.innerHeight,r=s.left;r+a.width>i-8&&(r=s.right-a.width),r<8&&(r=8);let c=s.bottom+4;c+a.height>o-8&&(c=s.top-a.height-4),c<8&&(c=8),e.style.left=`${r}px`,e.style.top=`${c}px`}function W(e,t){let s=y(l("div",{className:"dropdown-menu",style:"visibility:hidden;top:0;left:0",children:t.map(c=>l("button",{className:`dropdown-item${c.active?" active":""}`,"data-key":c.key,children:[c.color?l("span",{className:"dropdown-dot",style:`background-color:${c.color}`}):null,l("span",{className:"dropdown-label",children:c.label}),c.shortcut?l("kbd",{className:"dropdown-kbd",children:c.shortcut}):null]}))}));s.querySelectorAll(".dropdown-item").forEach((c,u)=>{c.addEventListener("click",()=>{t[u].action(),s.remove()})});function i(c){let u=t.find(d=>c.key.toLowerCase()===d.key.toLowerCase());u?(c.preventDefault(),c.stopPropagation(),u.action(),o()):c.key==="Escape"&&(c.preventDefault(),o())}function o(){s.remove(),document.removeEventListener("keydown",i,!0),document.removeEventListener("click",r)}function r(){o()}return document.addEventListener("keydown",i,!0),setTimeout(()=>{document.addEventListener("click",r)},0),s}function J(){document.querySelectorAll(".dropdown-menu").forEach(e=>{e.remove()})}var oe=class{constructor(){$(this,"undoStack",[]);$(this,"redoStack",[])}push(t){console.log("[undo] push:",t.label,"stack depth:",this.undoStack.length+1,"before:",JSON.stringify(t.before),"after:",JSON.stringify(t.after)),this.undoStack.push(t),this.undoStack.length>1e3&&this.undoStack.shift(),this.redoStack=[]}coalesce(t){if(t.coalescingKey==null||t.coalescingKey==="")return!1;let s=this.peekUndo();return!s||s.coalescingKey!==t.coalescingKey||t.timestamp-s.timestamp>=5e3?!1:(s.after=t.after,!0)}popUndo(){let t=this.undoStack.pop();return t&&this.redoStack.push(t),t}popRedo(){let t=this.redoStack.pop();return t&&this.undoStack.push(t),t}peekUndo(){return this.undoStack[this.undoStack.length-1]}canUndo(){return this.undoStack.length>0}canRedo(){return this.redoStack.length>0}},h=new oe;function C(e){return{id:e.id,title:e.title,details:e.details,category:e.category,priority:e.priority,status:e.status,up_next:e.up_next}}async function T(e,t,s){let a=C(e),i=await p(`/tickets/${e.id}`,{method:"PATCH",body:t}),o=C(i);return h.push({label:s,timestamp:Date.now(),before:[a],after:[o]}),i}function G(e,t,s){let a=`${e.id}:${t}`,i=Date.now(),o={...C(e),[t]:s},r={label:`Edit ${t}`,timestamp:i,before:[C(e)],after:[o],coalescingKey:a};h.coalesce(r)||h.push(r)}async function b(e,t,s){let a=e.map(C);await p("/tickets/batch",{method:"POST",body:t});let i=a.map(o=>{let r={...o};return t.action==="category"?r.category=t.value:t.action==="priority"?r.priority=t.value:t.action==="status"?r.status=t.value:t.action==="up_next"?r.up_next=t.value:t.action==="delete"&&(r.status="deleted"),r});h.push({label:s,timestamp:Date.now(),before:a,after:i})}async function re(e,t,s){let a=e.map(C);for(let o of t)await p("/tickets/batch",{method:"POST",body:o});let i=a.map(o=>{let r={...o};for(let c of t)c.ids.includes(o.id)&&(c.action==="status"?r.status=c.value:c.action==="up_next"?r.up_next=c.value:c.action==="category"?r.category=c.value:c.action==="priority"?r.priority=c.value:c.action==="delete"&&(r.status="deleted"));return r});h.push({label:s,timestamp:Date.now(),before:a,after:i})}async function Le(e){let t=C(e);await p(`/tickets/${e.id}`,{method:"DELETE"});let s={...t,status:"deleted"};h.push({label:"Delete ticket",timestamp:Date.now(),before:[t],after:[s]})}async function xe(e){let t=C(e);await p(`/tickets/${e.id}/restore`,{method:"POST"});let s={...t,status:"not_started"};h.push({label:"Restore ticket",timestamp:Date.now(),before:[t],after:[s]})}async function Se(e){for(let t of e)t.status==="deleted"?await p(`/tickets/${t.id}`,{method:"DELETE"}):await p(`/tickets/${t.id}`,{method:"PATCH",body:{title:t.title,details:t.details,category:t.category,priority:t.priority,status:t.status,up_next:t.up_next}})}var _=!1;async function _e(){if(console.log("[undo] performUndo, inFlight:",_,"canUndo:",h.canUndo()),_){console.log("[undo] skipped \u2014 already in flight");return}let e=h.popUndo();if(!e){console.log("[undo] skipped \u2014 stack empty");return}console.log("[undo] applying before-state:",e.label,JSON.stringify(e.before)),_=!0;try{await Se(e.before),console.log("[undo] applySnapshots done, reloading tickets"),await m(),ne()}finally{_=!1}}async function Ce(){if(console.log("[undo] performRedo, inFlight:",_,"canRedo:",h.canRedo()),_){console.log("[undo] skipped \u2014 already in flight");return}let e=h.popRedo();if(!e){console.log("[undo] skipped \u2014 stack empty");return}console.log("[undo] applying after-state:",e.label,JSON.stringify(e.after)),_=!0;try{await Se(e.after),console.log("[undo] applySnapshots done, reloading tickets"),await m(),ne()}finally{_=!1}}function Be(){return h.canUndo()}function Me(){return h.canRedo()}var P=null;function de(){P&&(clearTimeout(P),P=null)}var q=!1,K=null,le="",k=[],Y=[{key:"i",value:"issue",label:"Issue"},{key:"b",value:"bug",label:"Bug"},{key:"f",value:"feature",label:"Feature"},{key:"r",value:"requirement_change",label:"Req Change"},{key:"k",value:"task",label:"Task"},{key:"g",value:"investigation",label:"Investigation"}],ce=[{key:"1",value:"highest",label:"Highest"},{key:"2",value:"high",label:"High"},{key:"3",value:"default",label:"Default"},{key:"4",value:"low",label:"Low"},{key:"5",value:"lowest",label:"Lowest"}];function ot(){let e=document.activeElement;if(!e||!(e instanceof HTMLElement))return null;let t=e.closest(".ticket-row");if(!t)return null;if(t.classList.contains("draft-row"))return"draft";let s=t.dataset.id;return s!==void 0&&s!==""?parseInt(s,10):null}function rt(e){e!=null&&(q=!0,e==="draft"?I():document.querySelector(`.ticket-row[data-id="${e}"] .ticket-title-input`)?.focus(),q=!1)}function X(){let e=n.view;return e!=="completed"&&e!=="verified"&&e!=="trash"&&e!=="backlog"&&e!=="archive"}function De(){return n.view==="up-next"||n.view==="open"?[{status:"not_started",label:"Not Started"},{status:"started",label:"Started"}]:n.view==="non-verified"?[{status:"not_started",label:"Not Started"},{status:"started",label:"Started"},{status:"completed",label:"Completed"}]:[{status:"not_started",label:"Not Started"},{status:"started",label:"Started"},{status:"completed",label:"Completed"},{status:"verified",label:"Verified"}]}function v(){let e=!!n.backupPreview?.active;if(n.layout==="columns"&&X()){if(e){ct();return}ut();return}let t=n.view==="trash",s=e?null:ot(),a=null;if(s!=null&&s!=="draft"){let r=document.querySelector(`.ticket-row[data-id="${s}"] .ticket-title-input`);r&&(a=r.value)}let i=document.getElementById("ticket-list"),o=i.scrollTop;if(i.innerHTML="",i.classList.remove("ticket-list-columns"),!t&&!e&&i.appendChild(Re()),n.tickets.length===0){let r=t?"Trash is empty":e?"No tickets match this view":"";r&&i.appendChild(y(l("div",{className:"ticket-list-empty",children:r})))}for(let r of n.tickets)e?i.appendChild(lt(r)):t?i.appendChild(yt(r)):i.appendChild(gt(r));if(i.scrollTop=o,e){let r=document.getElementById("batch-toolbar");r&&(r.style.display="none"),O(),U()}else{let r=document.getElementById("batch-toolbar");if(r&&(r.style.display=""),s!=null&&s!=="draft"&&a!=null){let c=document.querySelector(`.ticket-row[data-id="${s}"] .ticket-title-input`);c&&c.value!==a&&(c.value=a)}rt(s),A()}z()}function lt(e){let t=n.selectedIds.has(e.id),s=e.status==="completed"||e.status==="verified",a=e.status==="verified",i=y(l("div",{className:`ticket-row${t?" selected":""}${s?" completed":""}${e.up_next?" up-next":""}`,"data-id":String(e.id),children:[l("span",{className:"ticket-checkbox-spacer"}),l("span",{className:`ticket-status-btn${a?" verified":""}`,style:"cursor:default",children:a?S(ee):te(e.status)}),l("span",{className:"ticket-category-badge",style:`background-color:${w(e.category)};cursor:default`,title:e.category,children:M(e.category)}),l("span",{className:"ticket-number",children:e.ticket_number}),l("span",{className:"ticket-title-input",style:"cursor:default",children:e.title}),l("span",{className:"ticket-priority-indicator",style:`color:${D(e.priority)};cursor:default`,title:e.priority,children:R(e.priority)}),l("span",{className:`ticket-star${e.up_next?" active":""}`,style:"cursor:default",children:e.up_next?"\u2605":"\u2606"})]}));return i.addEventListener("click",()=>{n.selectedIds.clear(),n.selectedIds.add(e.id),n.lastClickedId=e.id,O(),U()}),i}function Pe(e){let t={scrollLeft:0,columns:{}},s=e.querySelector(".columns-container");return s&&(t.scrollLeft=s.scrollLeft,s.querySelectorAll(".column[data-status]").forEach(a=>{let i=a.dataset.status,o=a.querySelector(".column-body");o&&(t.columns[i]=o.scrollTop)})),t}function Ae(e,t){let s=e.querySelector(".columns-container");s&&(s.scrollLeft=t.scrollLeft,s.querySelectorAll(".column[data-status]").forEach(a=>{let i=a.dataset.status,o=a.querySelector(".column-body");o&&t.columns[i]!=null&&(o.scrollTop=t.columns[i])}))}function ct(){let e=document.getElementById("ticket-list"),t=Pe(e);e.innerHTML="",e.classList.add("ticket-list-columns");let s=De(),a=y(l("div",{className:"columns-container"}));for(let o of s){let r=n.tickets.filter(d=>d.status===o.status),c=y(l("div",{className:"column","data-status":o.status,children:[l("div",{className:"column-header",children:[l("span",{className:"column-title",children:o.label}),l("span",{className:"column-count",children:String(r.length)})]}),l("div",{className:"column-body"})]})),u=c.querySelector(".column-body");for(let d of r)u.appendChild(dt(d));a.appendChild(c)}e.appendChild(a),Ae(e,t);let i=document.getElementById("batch-toolbar");i&&(i.style.display="none"),z()}function dt(e){let t=n.selectedIds.has(e.id),s=y(l("div",{className:`column-card${t?" selected":""}${e.up_next?" up-next":""}`,"data-id":String(e.id),children:[l("div",{className:"column-card-header",children:[l("span",{className:"ticket-category-badge",style:`background-color:${w(e.category)}`,children:M(e.category)}),l("span",{className:"ticket-number",children:e.ticket_number}),l("span",{className:"ticket-priority-indicator",style:`color:${D(e.priority)};cursor:default`,children:R(e.priority)}),l("span",{className:`ticket-star${e.up_next?" active":""}`,style:"cursor:default",children:e.up_next?"\u2605":"\u2606"})]}),l("div",{className:"column-card-title",children:e.title})]}));return s.addEventListener("click",()=>{n.selectedIds.clear(),n.selectedIds.add(e.id),n.lastClickedId=e.id,$e(),U()}),s}function ut(){let e=document.getElementById("ticket-list"),t=Pe(e);e.innerHTML="",e.classList.add("ticket-list-columns"),e.appendChild(Re());let s=De(),a=y(l("div",{className:"columns-container"}));for(let i of s){let o=n.tickets.filter(u=>u.status===i.status),r=y(l("div",{className:"column","data-status":i.status,children:[l("div",{className:"column-header",children:[l("span",{className:"column-title",children:i.label}),l("span",{className:"column-count",children:String(o.length)})]}),l("div",{className:"column-body"})]})),c=r.querySelector(".column-body");for(let u of o)c.appendChild(pt(u));c.addEventListener("dragover",u=>{u.preventDefault(),u.dataTransfer.dropEffect="move",r.classList.add("column-drop-target")}),c.addEventListener("dragleave",u=>{let d=u.relatedTarget;(!d||!c.contains(d))&&r.classList.remove("column-drop-target")}),c.addEventListener("drop",u=>{u.preventDefault(),r.classList.remove("column-drop-target");let d=k;if(k=[],d.length===0)return;let f=n.tickets.filter(E=>d.includes(E.id));b(f,{ids:d,action:"status",value:i.status},"Change status").then(()=>{m()})}),a.appendChild(r)}e.appendChild(a),Ae(e,t),A(),z()}function pt(e){let t=n.selectedIds.has(e.id),s=y(l("div",{className:`column-card${t?" selected":""}${e.up_next?" up-next":""}`,"data-id":String(e.id),children:[l("div",{className:"column-card-header",children:[l("span",{className:"ticket-category-badge",style:`background-color:${w(e.category)}`,children:M(e.category)}),l("span",{className:"ticket-number",children:e.ticket_number}),l("span",{className:"ticket-priority-indicator",style:`color:${D(e.priority)}`,children:R(e.priority)}),l("button",{className:`ticket-star${e.up_next?" active":""}`,title:e.up_next?"Remove from Up Next":"Add to Up Next",children:e.up_next?"\u2605":"\u2606"})]}),l("div",{className:"column-card-title",children:e.title})]})),a=s.querySelector(".ticket-priority-indicator");return a.addEventListener("click",i=>{i.stopPropagation(),Oe(a,e)}),s.querySelector(".ticket-star").addEventListener("click",i=>{i.stopPropagation(),Ke(e)}),s.draggable=!0,s.addEventListener("dragstart",i=>{n.selectedIds.has(e.id)&&n.selectedIds.size>1?k=Array.from(n.selectedIds):k=[e.id],i.dataTransfer.setData("text/plain",JSON.stringify(k)),i.dataTransfer.effectAllowed="move"}),s.addEventListener("dragend",()=>{k=[]}),s.addEventListener("click",i=>{if(i.metaKey||i.ctrlKey)n.selectedIds.has(e.id)?n.selectedIds.delete(e.id):n.selectedIds.add(e.id),n.lastClickedId=e.id;else if(i.shiftKey&&n.lastClickedId!=null){let o=n.tickets.map(u=>u.id),r=o.indexOf(n.lastClickedId),c=o.indexOf(e.id);if(r!==-1&&c!==-1){let u=Math.min(r,c),d=Math.max(r,c);n.selectedIds.clear();for(let f=u;f<=d;f++)n.selectedIds.add(o[f])}}else n.selectedIds.clear(),n.selectedIds.add(e.id),n.lastClickedId=e.id;$e(),A()}),s}function $e(){document.querySelectorAll(".column-card[data-id]").forEach(e=>{let t=parseInt(e.dataset.id,10);n.selectedIds.has(t)?e.classList.add("selected"):e.classList.remove("selected")})}function Re(){let e=Ue(),t=n.view.startsWith("category:"),s=y(l("div",{className:"ticket-row draft-row",children:[l("span",{className:"ticket-checkbox-spacer"}),l("span",{className:"ticket-status-btn draft-placeholder",children:"\u25CB"}),l("span",{className:"ticket-category-badge draft-badge",style:`background-color:${w(e)}${t?"":";cursor:pointer;opacity:1"}`,children:M(e)}),l("span",{className:"ticket-number draft-number"}),l("input",{type:"text",className:"ticket-title-input draft-input",placeholder:"New ticket...",value:le}),l("span",{className:"ticket-priority-indicator draft-placeholder"}),l("span",{className:"ticket-star draft-placeholder"})]}));if(!t){let i=s.querySelector(".ticket-category-badge");i.addEventListener("click",o=>{o.stopPropagation(),ft(i)})}let a=s.querySelector(".draft-input");return a.addEventListener("input",()=>{le=a.value}),a.addEventListener("keydown",async i=>{if(i.key==="Enter"&&a.value.trim()){i.preventDefault();let o=a.value.trim();le="",a.value="";let r=mt();K&&!n.view.startsWith("category:")&&(r.category=K);let c=await p("/tickets",{method:"POST",body:{title:o,defaults:r}});c&&(n.selectedIds.clear(),n.selectedIds.add(c.id)),await m(),I()}else i.key==="ArrowDown"&&(i.preventDefault(),n.tickets.length>0&&document.querySelector(`.ticket-row[data-id="${n.tickets[0].id}"] .ticket-title-input`)?.focus())}),s}function I(){document.querySelector(".draft-row .draft-input")?.focus()}function mt(){let e=n.view;return e==="up-next"?{up_next:!0}:e==="open"?{}:e==="completed"?{status:"completed"}:e==="backlog"?{status:"backlog"}:e==="archive"?{status:"archive"}:e.startsWith("category:")?{category:e.split(":")[1]}:e.startsWith("priority:")?{priority:e.split(":")[1]}:{}}function Ue(){if(K)return K;let e=n.view;return e.startsWith("category:")?e.split(":")[1]:"issue"}function ft(e){J();let s=navigator.platform.includes("Mac")?"\u2318":"Ctrl+",a=Ue(),i=W(e,Y.map(o=>({label:o.label,key:o.key,shortcut:`${s}${o.key.toUpperCase()}`,color:w(o.value),active:a===o.value,action:()=>{K=o.value,v(),I()}})));document.body.appendChild(i),j(i,e),i.style.visibility=""}function gt(e){let t=n.selectedIds.has(e.id),s=e.status==="completed"||e.status==="verified",a=e.status==="verified",i=y(l("div",{className:`ticket-row${t?" selected":""}${s?" completed":""}${e.up_next?" up-next":""}`,"data-id":String(e.id),children:[l("input",{type:"checkbox",className:"ticket-checkbox",checked:t}),l("button",{className:`ticket-status-btn${a?" verified":""}`,title:e.status.replace("_"," "),children:a?S(ee):te(e.status)}),l("span",{className:"ticket-category-badge",style:`background-color:${w(e.category)}`,title:e.category,children:M(e.category)}),l("span",{className:"ticket-number",children:e.ticket_number}),l("input",{type:"text",className:"ticket-title-input",value:e.title}),l("span",{className:"ticket-priority-indicator",style:`color:${D(e.priority)}`,title:e.priority,children:R(e.priority)}),l("button",{className:`ticket-star${e.up_next?" active":""}`,title:e.up_next?"Remove from Up Next":"Add to Up Next",children:e.up_next?"\u2605":"\u2606"})]}));i.addEventListener("mousedown",d=>{let f=d.target;f.tagName!=="INPUT"&&f.tagName!=="BUTTON"&&(i.draggable=!0)}),i.addEventListener("mouseup",()=>{i.draggable=!1}),i.addEventListener("dragend",()=>{i.draggable=!1,k=[]}),i.addEventListener("dragstart",d=>{n.selectedIds.has(e.id)&&n.selectedIds.size>1?k=Array.from(n.selectedIds):k=[e.id],d.dataTransfer.setData("text/plain",JSON.stringify(k)),d.dataTransfer.effectAllowed="move"}),i.addEventListener("mousedown",d=>{(d.metaKey||d.ctrlKey||d.shiftKey)&&(d.preventDefault(),qe(d,e)&&d.stopPropagation())});let o=i.querySelector(".ticket-checkbox");o.addEventListener("click",d=>d.stopPropagation()),o.addEventListener("change",()=>{o.checked?n.selectedIds.add(e.id):n.selectedIds.delete(e.id),n.lastClickedId=e.id,v()}),i.querySelector(".ticket-status-btn").addEventListener("click",d=>{d.stopPropagation(),kt(e)});let r=i.querySelector(".ticket-category-badge");r.addEventListener("click",d=>{d.stopPropagation(),It(r,e)});let c=i.querySelector(".ticket-title-input");c.addEventListener("focus",()=>{q||n.selectedIds.size===1&&n.selectedIds.has(e.id)||(n.selectedIds.clear(),n.selectedIds.add(e.id),n.lastClickedId=e.id,O(),A())}),c.addEventListener("input",()=>{G(e,"title",c.value),Tt(e.id,{title:c.value})}),c.addEventListener("keydown",d=>{vt(d,e,c)});let u=i.querySelector(".ticket-priority-indicator");return u.addEventListener("click",d=>{d.stopPropagation(),Oe(u,e)}),i.querySelector(".ticket-star").addEventListener("click",d=>{d.stopPropagation(),Ke(e)}),i}function yt(e){let t=n.selectedIds.has(e.id),s=e.deleted_at?new Date(e.deleted_at):null,a=y(l("div",{className:`ticket-row trash-row${t?" selected":""}`,"data-id":String(e.id),children:[l("input",{type:"checkbox",className:"ticket-checkbox",checked:t}),l("span",{className:"ticket-category-badge",style:`background-color:${w(e.category)}`,children:M(e.category)}),l("span",{className:"ticket-number",children:e.ticket_number}),l("span",{className:"ticket-title-input trash-title",style:"cursor:default",children:e.title}),l("span",{className:"ticket-number",title:s?`Deleted: ${s.toLocaleString()}`:"",children:s?s.toLocaleDateString():""}),l("button",{className:"btn btn-sm",title:"Restore from trash",children:"Restore"})]}));a.addEventListener("mousedown",o=>{(o.metaKey||o.ctrlKey||o.shiftKey)&&(o.preventDefault(),qe(o,e)&&o.stopPropagation())});let i=a.querySelector(".ticket-checkbox");return i.addEventListener("click",o=>o.stopPropagation()),i.addEventListener("change",()=>{i.checked?n.selectedIds.add(e.id):n.selectedIds.delete(e.id),n.lastClickedId=e.id,v()}),a.querySelector(".trash-title").addEventListener("click",()=>{n.selectedIds.size===1&&n.selectedIds.has(e.id)||(n.selectedIds.clear(),n.selectedIds.add(e.id),n.lastClickedId=e.id,O(),A())}),a.querySelector(".btn").addEventListener("click",async o=>{o.stopPropagation(),await xe(e),m()}),a}function qe(e,t){let s=e.metaKey||e.ctrlKey,a=e.shiftKey;if(s)n.selectedIds.has(t.id)?n.selectedIds.delete(t.id):n.selectedIds.add(t.id),n.lastClickedId=t.id,v();else if(a&&n.lastClickedId!=null){let i=n.tickets.map(c=>c.id),o=i.indexOf(n.lastClickedId),r=i.indexOf(t.id);if(o!==-1&&r!==-1){let c=Math.min(o,r),u=Math.max(o,r);n.selectedIds.clear();for(let d=c;d<=u;d++)n.selectedIds.add(i[d])}v()}else return!1;return!0}function vt(e,t,s){if(e.key==="Enter")e.preventDefault(),I();else if(e.key==="Backspace"&&s.value==="")e.preventDefault(),Et(t.id);else if(e.key==="ArrowDown"&&e.shiftKey)e.preventDefault(),He(t.id,1);else if(e.key==="ArrowUp"&&e.shiftKey)e.preventDefault(),He(t.id,-1);else if(e.key==="ArrowDown")e.preventDefault(),ht(t.id);else if(e.key==="ArrowUp")e.preventDefault(),bt(t.id);else if((e.metaKey||e.ctrlKey)&&!e.altKey&&Y.some(a=>a.key===e.key)){e.preventDefault();let a=Y.find(i=>i.key===e.key);Ne(t,"category",a.value)}else if(e.altKey&&!e.metaKey&&!e.ctrlKey&&ce.some(a=>a.key===e.key)){e.preventDefault();let a=ce.find(i=>i.key===e.key);Ne(t,"priority",a.value)}}function ht(e){let t=n.tickets.findIndex(s=>s.id===e);t<n.tickets.length-1&&document.querySelector(`.ticket-row[data-id="${n.tickets[t+1].id}"] .ticket-title-input`)?.focus()}function bt(e){let t=n.tickets.findIndex(s=>s.id===e);t>0?document.querySelector(`.ticket-row[data-id="${n.tickets[t-1].id}"] .ticket-title-input`)?.focus():I()}function He(e,t){let a=n.tickets.findIndex(r=>r.id===e)+t;if(a<0||a>=n.tickets.length)return;let i=n.tickets[a].id;n.selectedIds.add(e),n.selectedIds.has(i)?n.selectedIds.delete(e):n.selectedIds.add(i),q=!0,document.querySelector(`.ticket-row[data-id="${i}"] .ticket-title-input`)?.focus(),q=!1,O(),A()}async function kt(e){let s={not_started:"started",started:"completed",completed:"verified",verified:"not_started",backlog:"not_started",archive:"not_started"}[e.status]||"not_started",a=await T(e,{status:s},"Change status");Object.assign(e,a),v()}async function Ke(e){!e.up_next&&(e.status==="completed"||e.status==="verified")?await T(e,{status:"not_started",up_next:!0},"Toggle up next"):await T(e,{up_next:!e.up_next},"Toggle up next"),m()}async function Ne(e,t,s){let a=await T(e,{[t]:s},`Change ${t}`);Object.assign(e,a),v()}async function Et(e){let t=n.tickets.findIndex(a=>a.id===e),s=n.tickets.find(a=>a.id===e);if(s?await Le(s):await p(`/tickets/${e}`,{method:"DELETE"}),n.tickets=n.tickets.filter(a=>a.id!==e),n.selectedIds.delete(e),v(),t>0&&n.tickets.length>0){let a=Math.min(t-1,n.tickets.length-1);document.querySelector(`.ticket-row[data-id="${n.tickets[a].id}"] .ticket-title-input`)?.focus()}else I()}function Tt(e,t){P&&clearTimeout(P),P=setTimeout(()=>{p(`/tickets/${e}`,{method:"PATCH",body:t})},300)}function It(e,t){J();let a=navigator.platform.includes("Mac")?"\u2318":"Ctrl+",i=W(e,Y.map(o=>({label:o.label,key:o.key,shortcut:`${a}${o.key.toUpperCase()}`,color:w(o.value),active:t.category===o.value,action:async()=>{let r=await T(t,{category:o.value},"Change category");Object.assign(t,r),v()}})));document.body.appendChild(i),j(i,e),i.style.visibility=""}function Oe(e,t){J();let s=W(e,ce.map(a=>({label:a.label,key:a.key,shortcut:`Alt+${a.key}`,color:D(a.value),active:t.priority===a.value,action:async()=>{let i=await T(t,{priority:a.value},"Change priority");Object.assign(t,i),v()}})));document.body.appendChild(s),j(s,e),s.style.visibility=""}function O(){document.querySelectorAll(".ticket-row[data-id]").forEach(e=>{let t=parseInt(e.dataset.id,10),s=e.querySelector(".ticket-checkbox");n.selectedIds.has(t)?(e.classList.add("selected"),s&&(s.checked=!0)):(e.classList.remove("selected"),s&&(s.checked=!1))})}function A(){let e=n.selectedIds.size,t=n.tickets.length,s=e>0,a=n.view==="trash",i=document.getElementById("batch-select-all");i.checked=t>0&&e===t,i.indeterminate=e>0&&e<t,document.getElementById("batch-count").textContent=s?`${e} selected`:"";let o=["batch-category","batch-priority","batch-status","batch-upnext","batch-delete"];for(let f of o){let E=document.getElementById(f);E.style.display=a?"none":"",a||(E.disabled=!s)}let r=document.getElementById("batch-restore"),c=document.getElementById("batch-empty-trash");if(a){let f=document.getElementById("batch-toolbar");r||(r=y(l("button",{id:"batch-restore",className:"btn btn-sm",children:"Restore"})),r.addEventListener("click",async()=>{await p("/tickets/batch",{method:"POST",body:{ids:Array.from(n.selectedIds),action:"restore"}}),n.selectedIds.clear(),m()}),f.insertBefore(r,document.getElementById("batch-count"))),r.disabled=!s,r.style.display="",c||(c=y(l("button",{id:"batch-empty-trash",className:"btn btn-sm btn-danger",children:"Empty Trash"})),c.addEventListener("click",async()=>{await p("/trash/empty",{method:"POST"}),n.selectedIds.clear(),m()}),f.insertBefore(c,document.getElementById("batch-count"))),c.disabled=t===0,c.style.display=""}else r&&(r.style.display="none"),c&&(c.style.display="none");let u=document.querySelector(".batch-star-icon"),d=document.getElementById("batch-upnext");if(!a&&u&&s){let f=n.tickets.filter(B=>n.selectedIds.has(B.id)),E=f.every(B=>B.up_next),F=f.every(B=>!B.up_next);E?(u.textContent="\u2605",d.classList.add("active"),d.classList.remove("mixed")):F?(u.textContent="\u2606",d.classList.remove("active","mixed")):(u.innerHTML=l("span",{className:"star-mixed-wrap",children:[l("span",{className:"star-mixed-fill",children:"\u2605"}),"\u2606"]}).toString(),d.classList.remove("active"),d.classList.add("mixed"))}else u&&(u.textContent="\u2606",d.classList.remove("active","mixed"));U()}async function m(){if(n.backupPreview?.active){wt();return}let e=new URLSearchParams;n.view==="trash"?e.set("status","deleted"):n.view==="up-next"?e.set("up_next","true"):n.view==="open"?e.set("status","open"):n.view==="completed"?e.set("status","completed"):n.view==="non-verified"?e.set("status","non_verified"):n.view==="verified"?e.set("status","verified"):n.view==="backlog"?e.set("status","backlog"):n.view==="archive"?e.set("status","archive"):n.view.startsWith("category:")?e.set("category",n.view.split(":")[1]):n.view.startsWith("priority:")?e.set("priority",n.view.split(":")[1]):e.set("status","active"),n.search&&e.set("search",n.search),e.set("sort_by",n.sortBy),e.set("sort_dir",n.sortDir);let t=e.toString();n.tickets=await p(`/tickets${t?"?"+t:""}`),v()}function wt(){let e=[...n.backupPreview?.tickets||[]];if(n.view==="trash")e=e.filter(t=>t.status==="deleted");else if(n.view==="up-next")e=e.filter(t=>t.up_next&&t.status!=="deleted");else if(n.view==="open")e=e.filter(t=>t.status==="not_started"||t.status==="started");else if(n.view==="completed")e=e.filter(t=>t.status==="completed");else if(n.view==="non-verified")e=e.filter(t=>t.status!=="verified"&&t.status!=="deleted"&&t.status!=="backlog"&&t.status!=="archive");else if(n.view==="verified")e=e.filter(t=>t.status==="verified");else if(n.view==="backlog")e=e.filter(t=>t.status==="backlog");else if(n.view==="archive")e=e.filter(t=>t.status==="archive");else if(n.view.startsWith("category:")){let t=n.view.split(":")[1];e=e.filter(s=>s.category===t&&s.status!=="deleted"&&s.status!=="backlog"&&s.status!=="archive")}else if(n.view.startsWith("priority:")){let t=n.view.split(":")[1];e=e.filter(s=>s.priority===t&&s.status!=="deleted"&&s.status!=="backlog"&&s.status!=="archive")}else e=e.filter(t=>t.status!=="deleted"&&t.status!=="backlog"&&t.status!=="archive");if(n.search){let t=n.search.toLowerCase();e=e.filter(s=>s.title.toLowerCase().includes(t)||s.ticket_number.toLowerCase().includes(t)||s.details&&s.details.toLowerCase().includes(t))}n.tickets=e,v()}function Lt(e){let t=Date.now()-new Date(e).getTime(),s=Math.floor(t/6e4);if(s<1)return"just now";if(s<60)return`${s}m ago`;let a=Math.floor(s/60);return a<24?`${a}h ago`:`${Math.floor(a/24)}d ago`}function xt(e){return e<1024?`${e} B`:e<1024*1024?`${(e/1024).toFixed(0)} KB`:`${(e/(1024*1024)).toFixed(1)} MB`}function St(e){return e==="5min"?"Recent (every 5 min)":e==="hourly"?"Hourly":e==="daily"?"Daily":e}async function ue(){var t;let e=document.getElementById("backup-list");if(e)try{let s=await p("/backups");if(s.backups.length===0){e.textContent="No backups yet. First backup will be created shortly.";return}let a={};for(let i of s.backups)(a[t=i.tier]||(a[t]=[])).push(i);e.innerHTML="";for(let i of["5min","hourly","daily"]){let o=a[i];if(!(!o||o.length===0)){e.appendChild(y(l("div",{className:"backup-tier-label",children:St(i)})));for(let r of o){let c=y(l("div",{className:"backup-row","data-tier":r.tier,"data-filename":r.filename,children:[l("span",{className:"backup-row-time",children:Lt(r.createdAt)}),l("span",{className:"backup-row-meta",children:[new Date(r.createdAt).toLocaleString()," \xB7 ",xt(r.sizeBytes)]})]}));c.addEventListener("click",()=>{_t(r.tier,r.filename,r.createdAt)}),e.appendChild(c)}}}}catch{e.textContent="Failed to load backups."}}async function _t(e,t,s){let a=document.getElementById("settings-overlay");a.style.display="none";let i=document.getElementById("backup-preview-banner"),o=document.getElementById("backup-preview-label");o.textContent="Loading backup preview...",i.style.display="flex";try{let r=await p(`/backups/preview/${e}/${t}`);n.backupPreview={active:!0,tickets:r.tickets,timestamp:s,tier:e,filename:t},n.selectedIds.clear(),n.activeTicketId=null,o.textContent=`Previewing backup from ${new Date(s).toLocaleString()} (${r.stats.total} tickets, ${r.stats.open} open) \u2014 read-only`,m()}catch{o.textContent="Failed to load backup preview.",setTimeout(()=>{i.style.display="none"},3e3)}}async function Ct(){let e=document.getElementById("backup-preview-banner");e.style.display="none",n.backupPreview=null,n.selectedIds.clear(),n.activeTicketId=null,await p("/backups/preview/cleanup",{method:"POST"}),m()}async function Bt(){if(!n.backupPreview)return;let e=document.getElementById("backup-restore-btn");e.textContent="Restoring...",e.disabled=!0;try{await p("/backups/restore",{method:"POST",body:{tier:n.backupPreview.tier,filename:n.backupPreview.filename}}),window.location.reload()}catch{e.textContent="Restore failed",e.disabled=!1,setTimeout(()=>{e.textContent="Restore This Backup"},3e3)}}function Fe(){document.getElementById("backup-cancel-btn")?.addEventListener("click",()=>{Ct()}),document.getElementById("backup-restore-btn")?.addEventListener("click",()=>{Bt()});let e=document.getElementById("backup-now-btn");e?.addEventListener("click",async()=>{e.textContent="Backing up...",e.disabled=!0;try{(await p("/backups/now",{method:"POST"})).error?e.textContent="In progress...":(e.textContent="Done!",ue())}catch{e.textContent="Failed"}setTimeout(()=>{e.textContent="Backup Now",e.disabled=!1},1500)})}async function Mt(){await Ht(),Nt(),await m(),Ot(),Ut(),qt(),Ft(),Vt(),zt(),jt(),Gt(),Dt(),Fe(),$t(),Rt(),we(),Yt(),Pt(),document.addEventListener("hotsheet:render",()=>v()),I()}async function Ht(){try{let e=await p("/settings");(e.detail_position==="side"||e.detail_position==="bottom")&&(n.settings.detail_position=e.detail_position),e.detail_width&&(n.settings.detail_width=parseInt(e.detail_width,10)||360),e.detail_height&&(n.settings.detail_height=parseInt(e.detail_height,10)||300),e.trash_cleanup_days&&(n.settings.trash_cleanup_days=parseInt(e.trash_cleanup_days,10)||3),e.verified_cleanup_days&&(n.settings.verified_cleanup_days=parseInt(e.verified_cleanup_days,10)||30),(e.layout==="list"||e.layout==="columns")&&(n.layout=e.layout)}catch{}ae(n.settings.detail_position),ie()}async function Nt(){try{let e=await p("/file-settings");if(e.appName){document.title=e.appName;let t=document.querySelector(".app-title h1");t&&(t.textContent=e.appName)}}catch{}}function Dt(){let e=document.getElementById("settings-overlay"),t=document.getElementById("settings-close");document.getElementById("settings-btn").addEventListener("click",()=>{document.getElementById("settings-trash-days").value=String(n.settings.trash_cleanup_days),document.getElementById("settings-verified-days").value=String(n.settings.verified_cleanup_days),e.style.display="flex",ue(),p("/file-settings").then(g=>{document.getElementById("settings-app-name").value=g.appName||"",document.getElementById("settings-backup-dir").value=g.backupDir||""})}),t.addEventListener("click",()=>{e.style.display="none"}),e.addEventListener("click",g=>{g.target===e&&(e.style.display="none")});let a=document.getElementById("settings-trash-days"),i=null;a.addEventListener("input",()=>{i&&clearTimeout(i),i=setTimeout(()=>{let g=Math.max(1,parseInt(a.value,10)||3);a.value=String(g),n.settings.trash_cleanup_days=g,p("/settings",{method:"PATCH",body:{trash_cleanup_days:String(g)}})},500)});let o=document.getElementById("settings-verified-days"),r=null;o.addEventListener("input",()=>{r&&clearTimeout(r),r=setTimeout(()=>{let g=Math.max(1,parseInt(o.value,10)||30);o.value=String(g),n.settings.verified_cleanup_days=g,p("/settings",{method:"PATCH",body:{verified_cleanup_days:String(g)}})},500)});let c=document.getElementById("settings-app-name"),u=document.getElementById("settings-app-name-hint"),d=null;c.addEventListener("input",()=>{d&&clearTimeout(d),d=setTimeout(()=>{let g=c.value.trim();p("/file-settings",{method:"PATCH",body:{appName:g}}).then(()=>{let H=g||"Hot Sheet";document.title=H;let ye=document.querySelector(".app-title h1");ye&&(ye.textContent=H),u.textContent=g?"Saved. Restart the desktop app to update the title bar.":"Using default name."})},800)});let f=document.getElementById("check-updates-btn"),E=document.getElementById("check-updates-status");f.addEventListener("click",async()=>{let g=ge();if(g){f.disabled=!0,f.textContent="Checking...",E.textContent="";try{let H=await g("check_for_update");H?(E.textContent=`Update available: v${H}`,document.getElementById("settings-overlay").style.display="none",We(H)):E.textContent="Your software is up to date."}catch{E.textContent="Could not check for updates."}f.textContent="Check for Updates",f.disabled=!1}});let F=document.getElementById("settings-backup-dir"),B=document.getElementById("settings-backup-dir-hint"),Q=null;F.addEventListener("input",()=>{Q&&clearTimeout(Q),Q=setTimeout(()=>{let g=F.value.trim();p("/file-settings",{method:"PATCH",body:{backupDir:g}}).then(()=>{B.textContent=g?"Saved. New backups will use this location.":"Using default location inside the data directory."})},800)})}async function Pt(){let e=ge();if(!e)return;let t=document.getElementById("settings-updates-section");t&&(t.style.display="");let s=[0,3e3,1e4];for(let a of s){a>0&&await new Promise(i=>setTimeout(i,a));try{let i=await e("get_pending_update");if(i){We(i);return}}catch{return}}}function ge(){return window.__TAURI__?.core?.invoke??null}function We(e){let t=document.getElementById("update-banner");if(!t)return;let s=document.getElementById("update-banner-label");s&&(s.textContent=`Update available: v${e}`),t.style.display="flex";let a=document.getElementById("update-install-btn");a?.addEventListener("click",async()=>{if(a){a.textContent="Installing...",a.disabled=!0;try{await ge()?.("install_update"),s&&(s.textContent="Update installed! Restart the app to apply."),a.style.display="none"}catch{a.textContent="Install Failed",a.disabled=!1}}}),document.getElementById("update-banner-dismiss")?.addEventListener("click",()=>{t.style.display="none"})}function At(){let e=document.getElementById("skills-banner");if(!e)return;e.style.display="flex",document.getElementById("skills-banner-dismiss")?.addEventListener("click",()=>{e.style.display="none"})}function $t(){let e=document.getElementById("copy-prompt-section"),t=document.getElementById("copy-prompt-btn"),s=document.getElementById("copy-prompt-label"),a=document.getElementById("copy-prompt-icon"),i="";p("/worklist-info").then(o=>{i=o.prompt,e.style.display="",o.skillCreated&&At()}),t.addEventListener("click",()=>{i!==""&&navigator.clipboard.writeText(i).then(()=>{s.textContent="Copied!",a.innerHTML='<svg xmlns="http://www.w3.org/2000/svg" width="13" height="13" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M20 6L9 17l-5-5"/></svg>',setTimeout(()=>{s.textContent="Copy AI prompt",a.innerHTML='<svg xmlns="http://www.w3.org/2000/svg" width="13" height="13" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect x="9" y="9" width="13" height="13" rx="2"/><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"/></svg>'},1500)})})}function Rt(){let e=document.getElementById("glassbox-btn"),t=document.getElementById("glassbox-icon");p("/glassbox/status").then(({available:s})=>{s&&(t.src="/static/assets/glassbox-icon.png",e.style.display="")}).catch(()=>{}),e.addEventListener("click",()=>{p("/glassbox/launch",{method:"POST"})})}function fe(){let e=document.getElementById("layout-toggle"),t=X(),s=e.querySelector('[data-layout="columns"]');s.disabled=!t,s.style.opacity=t?"":"0.3";let a=n.layout==="columns"&&!t?"list":n.layout;e.querySelectorAll(".layout-btn").forEach(i=>{i.classList.toggle("active",i.dataset.layout===a)})}function Ut(){document.getElementById("layout-toggle").querySelectorAll(".layout-btn").forEach(t=>{t.addEventListener("click",()=>{let s=t.dataset.layout;s==="columns"&&!X()||(n.layout=s,fe(),v(),p("/settings",{method:"PATCH",body:{layout:s}}))})}),fe()}function Ve(){document.getElementById("detail-position-toggle").querySelectorAll(".layout-btn").forEach(t=>{t.classList.toggle("active",t.dataset.position===n.settings.detail_position)})}function qt(){document.getElementById("detail-position-toggle").querySelectorAll(".layout-btn").forEach(t=>{t.addEventListener("click",()=>{let s=t.dataset.position;n.settings.detail_position=s,ae(s),ie(),Ve(),p("/settings",{method:"PATCH",body:{detail_position:s}})})}),Ve()}function Je(e){return e==="up-next"?{action:"up_next",value:!0}:e==="open"?{action:"status",value:"not_started"}:e==="completed"?{action:"status",value:"completed"}:e==="verified"?{action:"status",value:"verified"}:e==="backlog"?{action:"status",value:"backlog"}:e==="archive"?{action:"status",value:"archive"}:e==="trash"?{action:"delete",value:null}:e.startsWith("category:")?{action:"category",value:e.split(":")[1]}:e.startsWith("priority:")?{action:"priority",value:e.split(":")[1]}:null}async function Kt(e,t){let s=Je(e);if(!s)return;let a=n.tickets.filter(i=>t.includes(i.id));s.action==="delete"?await b(a,{ids:t,action:"delete"},"Delete tickets"):await b(a,{ids:t,action:s.action,value:s.value},`Change ${s.action}`),m()}function Ot(){let e=document.querySelectorAll(".sidebar-item[data-view]");e.forEach(t=>{t.addEventListener("click",()=>{e.forEach(a=>{a.classList.remove("active")}),t.classList.add("active"),n.view=t.dataset.view,n.selectedIds.clear(),fe(),m()});let s=t.dataset.view;Je(s)&&(t.addEventListener("dragover",a=>{a.preventDefault(),a.dataTransfer.dropEffect="move",t.classList.add("drop-target")}),t.addEventListener("dragleave",()=>{t.classList.remove("drop-target")}),t.addEventListener("drop",a=>{a.preventDefault(),t.classList.remove("drop-target");let i=[...k];i.length!==0&&Kt(s,i)}))})}function Ft(){let e=document.getElementById("sort-select");e.addEventListener("change",()=>{let[t,s]=e.value.split(":");n.sortBy=t,n.sortDir=s,m()})}var pe=null;function Vt(){let e=document.getElementById("search-input");e.addEventListener("input",()=>{pe&&clearTimeout(pe),pe=setTimeout(()=>{n.search=e.value,m()},200)}),e.addEventListener("keydown",t=>{t.key==="Escape"&&(e.value="",n.search="",m())})}function zt(){let e=document.getElementById("batch-category");e.addEventListener("change",async()=>{if(!e.value)return;let a=Array.from(n.selectedIds),i=n.tickets.filter(o=>n.selectedIds.has(o.id));await b(i,{ids:a,action:"category",value:e.value},"Batch change category"),e.value="",m()});let t=document.getElementById("batch-priority");t.addEventListener("change",async()=>{if(!t.value)return;let a=Array.from(n.selectedIds),i=n.tickets.filter(o=>n.selectedIds.has(o.id));await b(i,{ids:a,action:"priority",value:t.value},"Batch change priority"),t.value="",m()});let s=document.getElementById("batch-status");s.addEventListener("change",async()=>{if(!s.value)return;let a=Array.from(n.selectedIds),i=n.tickets.filter(o=>n.selectedIds.has(o.id));await b(i,{ids:a,action:"status",value:s.value},"Batch change status"),s.value="",m()}),document.getElementById("batch-upnext").addEventListener("click",async()=>{let a=n.tickets.filter(c=>n.selectedIds.has(c.id)),o=!a.every(c=>c.up_next),r=Array.from(n.selectedIds);if(o){let c=a.filter(u=>u.status==="completed"||u.status==="verified");if(c.length>0){let u=[{ids:c.map(d=>d.id),action:"status",value:"not_started"},{ids:r,action:"up_next",value:!0}];await re(a,u,"Batch toggle up next")}else await b(a,{ids:r,action:"up_next",value:!0},"Batch toggle up next")}else await b(a,{ids:r,action:"up_next",value:!1},"Batch toggle up next");m()}),document.getElementById("batch-delete").addEventListener("click",async()=>{let a=Array.from(n.selectedIds),i=n.tickets.filter(o=>n.selectedIds.has(o.id));await b(i,{ids:a,action:"delete"},"Batch delete"),n.selectedIds.clear(),m()}),document.getElementById("batch-select-all").addEventListener("change",a=>{if(a.target.checked)for(let o of n.tickets)n.selectedIds.add(o.id);else n.selectedIds.clear();v()})}var L=null;function jt(){document.getElementById("detail-close").addEventListener("click",Ee);let e=["detail-title","detail-details"];for(let s of e){let a=document.getElementById(s);a.addEventListener("input",()=>{let i=n.tickets.find(o=>o.id===n.activeTicketId);if(i){let o=s.replace("detail-","");G(i,o,a.value)}L&&clearTimeout(L),L=setTimeout(()=>{if(n.activeTicketId==null)return;let o=s.replace("detail-","");p(`/tickets/${n.activeTicketId}`,{method:"PATCH",body:{[o]:a.value}}).then(()=>{m()})},300)})}let t=["detail-category","detail-priority","detail-status"];for(let s of t){let a=document.getElementById(s);a.addEventListener("change",async()=>{if(n.activeTicketId==null)return;let i=n.tickets.find(r=>r.id===n.activeTicketId),o=s.replace("detail-","");i?await T(i,{[o]:a.value},`Change ${o}`):await p(`/tickets/${n.activeTicketId}`,{method:"PATCH",body:{[o]:a.value}}),m()})}document.getElementById("detail-upnext").addEventListener("change",async()=>{if(n.activeTicketId==null)return;let s=n.tickets.find(i=>i.id===n.activeTicketId),a=document.getElementById("detail-upnext");s?a.checked&&(s.status==="completed"||s.status==="verified")?await T(s,{status:"not_started",up_next:!0},"Toggle up next"):await T(s,{up_next:!s.up_next},"Toggle up next"):await p(`/tickets/${n.activeTicketId}/up-next`,{method:"POST"}),m(),V(n.activeTicketId)}),document.getElementById("detail-file-input").addEventListener("change",async s=>{let a=s.target,i=a.files?.[0];!i||n.activeTicketId==null||(await ke(`/tickets/${n.activeTicketId}/attachments`,i),a.value="",V(n.activeTicketId),m())}),document.getElementById("detail-attachments").addEventListener("click",async s=>{let a=s.target,i=a.closest(".attachment-reveal");if(i){let c=i.dataset.attId;c&&p(`/attachments/${c}/reveal`,{method:"POST"});return}let o=a.closest(".attachment-delete");if(o===null)return;let r=o.dataset.attId;r===void 0||r===""||(await p(`/attachments/${r}`,{method:"DELETE"}),n.activeTicketId!=null&&V(n.activeTicketId))})}function Wt(e){if(!e||e==="")return[];try{let t=JSON.parse(e);if(Array.isArray(t))return t}catch{}return e.trim()?[{text:e,created_at:""}]:[]}function Jt(e){let t=[];t.push(`${e.ticket_number}: ${e.title}`),e.details.trim()&&(t.push(""),t.push(e.details.trim()));let s=Wt(e.notes);if(s.length>0){t.push("");for(let a of s)t.push(`- ${a.text}`)}return t.join(`
|
|
2
|
+
`)}function ze(){console.log("[undo] triggerUndo called, canUndo:",Be()),L&&(clearTimeout(L),L=null),de(),_e().then(()=>console.log("[undo] performUndo completed")).catch(e=>console.error("[undo] performUndo error:",e))}function je(){console.log("[undo] triggerRedo called, canRedo:",Me()),L&&(clearTimeout(L),L=null),de(),Ce().then(()=>console.log("[undo] performRedo completed")).catch(e=>console.error("[undo] performRedo error:",e))}function Gt(){window.addEventListener("app:undo",ze),window.addEventListener("app:redo",je),document.addEventListener("keydown",e=>{(e.metaKey||e.ctrlKey)&&e.key.toLowerCase()==="z"&&(e.preventDefault(),e.stopPropagation(),e.shiftKey?je():ze())},!0),document.addEventListener("keydown",e=>{let t=e.target.tagName,s=t==="INPUT"||t==="TEXTAREA"||t==="SELECT",a=document.getElementById("settings-overlay");if(e.key==="Escape"&&a.style.display!=="none"){a.style.display="none";return}if(e.key==="Escape"){n.selectedIds.size>0&&(n.selectedIds.clear(),v());return}if((e.metaKey||e.ctrlKey)&&e.key==="a"&&!s){e.preventDefault(),n.selectedIds.clear();for(let i of n.tickets)n.selectedIds.add(i.id);v();return}if((e.metaKey||e.ctrlKey)&&e.key==="d"){if(n.selectedIds.size>0){e.preventDefault();let i=n.tickets.filter(u=>n.selectedIds.has(u.id)),r=!i.every(u=>u.up_next),c=Array.from(n.selectedIds);if(r){let u=i.filter(d=>d.status==="completed"||d.status==="verified");if(u.length>0){re(i,[{ids:u.map(d=>d.id),action:"status",value:"not_started"},{ids:c,action:"up_next",value:!0}],"Toggle up next").then(()=>{m()});return}}b(i,{ids:c,action:"up_next",value:r},"Toggle up next").then(()=>{m()})}return}if((e.metaKey||e.ctrlKey)&&e.key==="c"&&n.selectedIds.size>0&&!(s&&!e.altKey)){let i=!e.altKey&&window.getSelection();if(!(i&&!i.isCollapsed&&i.toString().trim()!=="")){e.preventDefault();let r=n.tickets.filter(c=>n.selectedIds.has(c.id)).map(Jt).join(`
|
|
3
3
|
|
|
4
|
-
`);navigator.clipboard.writeText(
|
|
4
|
+
`);navigator.clipboard.writeText(r);return}}if((e.metaKey||e.ctrlKey)&&e.key==="n"){e.preventDefault(),I();return}if((e.metaKey||e.ctrlKey)&&e.key==="f"){e.preventDefault(),document.getElementById("search-input").focus();return}if(e.key==="n"&&!s){e.preventDefault(),I();return}})}var me=0;function Yt(){async function e(){try{let t=await p(`/poll?version=${me}`);t.version>me&&(me=t.version,n.backupPreview?.active||m())}catch{await new Promise(t=>setTimeout(t,5e3))}setTimeout(e,100)}e()}Mt();})();
|
|
Binary file
|
package/dist/client/styles.css
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
*,*::before,*::after{box-sizing:border-box;margin:0;padding:0}:root{--bg: #ffffff;--bg-secondary: #f9fafb;--bg-hover: #f3f4f6;--bg-selected: #eff6ff;--border: #e5e7eb;--text: #111827;--text-secondary: #6b7280;--text-muted: #9ca3af;--accent: #3b82f6;--accent-hover: #2563eb;--danger: #ef4444;--star: #eab308;--radius: 6px;--shadow: 0 1px 3px rgba(0,0,0,0.08)}html,body{height:100%;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,sans-serif;font-size:14px;line-height:1.5;color:var(--text);background:var(--bg)}button{cursor:pointer;border:none;background:none;font:inherit;color:inherit}input,select,textarea{font:inherit;color:inherit}.app{display:flex;flex-direction:column;height:100vh}.app-header{display:flex;align-items:center;justify-content:space-between;padding:12px 20px;border-bottom:1px solid var(--border);background:var(--bg);gap:16px;flex-shrink:0}.app-header h1{font-size:18px;font-weight:700;white-space:nowrap}.header-controls{display:flex;align-items:center;gap:12px;flex:1;max-width:500px}.search-box{flex:1}.search-box input{width:100%;padding:6px 12px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg-secondary);outline:none}.search-box input:focus{border-color:var(--accent);background:var(--bg)}.layout-toggle{display:flex;border:1px solid var(--border);border-radius:var(--radius);overflow:hidden;flex-shrink:0}.layout-btn{display:flex;align-items:center;justify-content:center;padding:4px 8px;color:var(--text-secondary);background:var(--bg);border:none;border-right:1px solid var(--border);cursor:pointer}.layout-btn:last-child{border-right:none}.layout-btn:hover:not(:disabled){background:var(--bg-hover);color:var(--text)}.layout-btn.active{background:var(--bg-selected);color:var(--accent)}.layout-btn:disabled{cursor:default}.sort-controls select{padding:6px 8px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg);cursor:pointer}.settings-btn{font-size:18px;color:var(--text-muted);padding:4px;border-radius:var(--radius);flex-shrink:0}.settings-btn:hover{color:var(--text);background:var(--bg-hover)}.app-body{display:flex;flex:1;overflow:hidden}.content-area{flex:1;display:flex;overflow:hidden;min-width:0}.content-area.detail-side{flex-direction:row}.content-area.detail-bottom{flex-direction:column}.sidebar{width:180px;border-right:1px solid var(--border);background:var(--bg-secondary);overflow-y:auto;flex-shrink:0;padding:8px 0}.sidebar-copy-prompt{padding:8px 10px 4px}.copy-prompt-btn{display:flex;align-items:center;gap:6px;width:100%;padding:6px 8px;font-size:12px;color:var(--text-secondary);background:var(--bg);border:1px solid var(--border);border-radius:var(--radius);cursor:pointer;transition:all .15s}.copy-prompt-btn:hover{color:var(--accent);border-color:var(--accent);background:var(--bg-selected)}.copy-prompt-icon{display:flex;align-items:center;flex-shrink:0}.sidebar-section{margin-bottom:8px}.sidebar-label{padding:8px 16px 4px;font-size:11px;font-weight:600;text-transform:uppercase;color:var(--text-muted);letter-spacing:.5px}.sidebar-item{display:flex;align-items:center;gap:8px;width:100%;padding:6px 16px;text-align:left;font-size:13px;color:var(--text-secondary);border-radius:0}.sidebar-item:hover{background:var(--bg-hover);color:var(--text)}.sidebar-item.active{background:var(--bg-selected);color:var(--accent);font-weight:500}.sidebar-item.drop-target{background:var(--bg-selected);color:var(--accent);outline:2px dashed var(--accent);outline-offset:-2px}.sidebar-divider{height:1px;background:var(--border);margin:6px 16px}.cat-dot{display:inline-block;width:8px;height:8px;border-radius:50%;flex-shrink:0}.sidebar-stats{padding:12px 16px;font-size:12px;color:var(--text-muted);border-top:1px solid var(--border);margin-top:8px}.main-content{flex:1;display:flex;flex-direction:column;overflow:hidden;min-width:0;min-height:0}.batch-toolbar{display:flex;align-items:center;gap:8px;padding:6px 16px 6px 19px;background:var(--bg-secondary);border-bottom:1px solid var(--border);flex-shrink:0;flex-wrap:wrap;min-height:36px}.batch-toolbar select:disabled,.batch-toolbar .btn:disabled{opacity:.4;cursor:default;pointer-events:none}.batch-toolbar select{padding:4px 8px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg);font-size:12px}.batch-select-all{width:14px;height:14px;cursor:pointer;accent-color:var(--accent);flex-shrink:0}.batch-star-btn{font-size:16px;color:var(--text-muted);padding:2px 6px;border-radius:var(--radius);line-height:1}.batch-star-btn:hover:not(:disabled){color:var(--star);background:var(--bg-hover)}.batch-star-btn.active{color:var(--star)}.batch-star-btn.mixed{color:var(--star)}.batch-star-btn:disabled{opacity:.4;cursor:default;pointer-events:none}.star-mixed-wrap{position:relative;display:inline-block}.star-mixed-fill{position:absolute;left:0;top:0;overflow:hidden;width:50%}.batch-delete-btn.btn{display:inline-flex;align-items:center;justify-content:center;padding:4px 6px;border:none;background:none}.batch-count{font-size:12px;color:var(--text-muted);margin-left:auto;white-space:nowrap}.btn{padding:4px 10px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg);font-size:12px;cursor:pointer}.btn:hover{background:var(--bg-hover)}.btn-sm{padding:3px 8px;font-size:11px}.btn-danger{color:var(--danger)}.btn-danger:hover:not(:disabled){background:#fef2f2}.ticket-list{flex:1;overflow-y:auto;padding:4px 0}.ticket-list-empty,.ticket-list-loading{padding:40px 20px;text-align:center;color:var(--text-muted);font-size:14px}.ticket-row{display:flex;align-items:center;gap:8px;padding:4px 16px;border-bottom:1px solid rgba(0,0,0,0);border-left:3px solid rgba(0,0,0,0);min-height:36px}.ticket-row:hover{background:var(--bg-hover)}.ticket-row.selected{background:var(--bg-selected)}.ticket-row.completed .ticket-title-input{text-decoration:line-through;color:var(--text-muted)}.ticket-row.up-next{border-left:3px solid var(--star)}.ticket-checkbox{flex-shrink:0;width:14px;height:14px;cursor:pointer;accent-color:var(--accent)}.ticket-status-btn{flex-shrink:0;width:20px;height:20px;font-size:14px;display:flex;align-items:center;justify-content:center;border-radius:50%;color:var(--text-secondary)}.ticket-status-btn:hover{background:var(--bg-hover)}.ticket-status-btn.verified{color:#22c55e}.ticket-category-badge{flex-shrink:0;width:4em;padding:1px 0;border-radius:3px;font-size:10px;font-weight:600;color:#fff;text-transform:uppercase;text-align:center;cursor:pointer;letter-spacing:.3px}.ticket-category-badge:hover{opacity:.85}.ticket-number{flex-shrink:0;font-size:11px;color:var(--text-muted);font-family:"SF Mono",Monaco,"Cascadia Code",monospace;min-width:5em}.ticket-title-input{flex:1;border:none;background:rgba(0,0,0,0);padding:2px 0;outline:none;min-width:0}.ticket-title-input:focus{border-bottom:1px solid var(--accent)}.ticket-title-input::placeholder{color:var(--text-muted)}.ticket-priority-indicator{flex-shrink:0;font-size:12px;cursor:pointer;padding:2px 4px;border-radius:3px;min-width:24px;text-align:center}.ticket-priority-indicator:hover{background:var(--bg-hover)}.ticket-star{flex-shrink:0;font-size:16px;color:var(--text-muted);padding:0}.ticket-star.active{color:var(--star)}.ticket-star:hover{color:var(--star)}.trash-row .trash-title{flex:1;color:var(--text-muted);min-width:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.trash-row .btn{flex-shrink:0}.draft-row{border-bottom:1px dashed var(--border) !important;background:var(--bg-secondary)}.draft-row:hover{background:var(--bg-secondary)}.draft-row .draft-placeholder{opacity:.35;pointer-events:none}.draft-row .draft-badge{opacity:.35}.draft-row .draft-input::placeholder{color:var(--text-muted);font-style:italic}.ticket-checkbox-spacer{width:14px;flex-shrink:0}.detail-resize-handle{flex-shrink:0;background:rgba(0,0,0,0);position:relative;z-index:10}.detail-resize-handle:hover,.detail-resize-handle:active{background:var(--accent);opacity:.3}.detail-side .detail-resize-handle{width:4px;cursor:col-resize}.detail-bottom .detail-resize-handle{height:4px;cursor:row-resize}.detail-panel{border-left:1px solid var(--border);background:var(--bg);flex-shrink:0;display:flex;flex-direction:column;overflow:hidden}.detail-side .detail-panel{width:360px;border-left:1px solid var(--border);border-top:none}.detail-bottom .detail-panel{height:300px;width:auto;border-left:none;border-top:1px solid var(--border)}.detail-header{display:flex;align-items:center;justify-content:space-between;padding:12px 16px;border-bottom:1px solid var(--border);font-weight:600}.detail-ticket-number{font-family:"SF Mono",Monaco,"Cascadia Code",monospace;font-size:13px;color:var(--accent)}.detail-close{font-size:20px;color:var(--text-muted);padding:0 4px}.detail-close:hover{color:var(--text)}.detail-body{flex:1;overflow-y:auto;padding:16px}.detail-fields-row{display:contents}.detail-bottom .detail-body{display:flex;flex-wrap:wrap;gap:0 16px;align-content:flex-start}.detail-bottom .detail-body .detail-fields-row{display:flex;gap:16px;width:100%}.detail-bottom .detail-body .detail-fields-row .detail-field{flex:1;min-width:0}.detail-bottom .detail-body .detail-field-full{width:100%;flex-basis:100%}.detail-field{margin-bottom:16px}.detail-field label{display:block;font-size:12px;font-weight:500;color:var(--text-secondary);margin-bottom:4px}.detail-field input[type=text],.detail-field textarea,.detail-field select{width:100%;padding:6px 10px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg);outline:none}.detail-field input[type=text]:focus,.detail-field textarea:focus,.detail-field select:focus{border-color:var(--accent)}.detail-field textarea{resize:vertical;min-height:80px}.detail-upnext-label{display:flex !important;align-items:center;gap:8px;cursor:pointer;font-size:14px !important;font-weight:400 !important;color:var(--text) !important}.detail-upnext-label input[type=checkbox]{width:auto;accent-color:var(--star)}.detail-attachments{margin-bottom:8px}.attachment-item{display:flex;align-items:center;justify-content:space-between;padding:4px 8px;background:var(--bg-secondary);border-radius:var(--radius);margin-bottom:4px;font-size:12px}.attachment-name{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.attachment-reveal{color:var(--text-muted);padding:0 4px;cursor:pointer;border:none;background:none;display:flex;align-items:center}.attachment-reveal:hover{color:var(--accent)}.attachment-delete{color:var(--text-muted);font-size:16px;padding:0 4px;cursor:pointer;border:none;background:none}.attachment-delete:hover{color:var(--danger)}.upload-btn{cursor:pointer;display:inline-flex;align-items:center;gap:4px;margin-top:4px}.detail-meta{padding-top:12px;border-top:1px solid var(--border);font-size:11px;color:var(--text-muted)}.detail-meta div{margin-bottom:2px}.dropdown-menu{position:fixed;z-index:1000;background:var(--bg);border:1px solid var(--border);border-radius:var(--radius);box-shadow:var(--shadow);padding:4px 0;min-width:140px}.dropdown-item{display:flex;align-items:center;gap:8px;width:100%;padding:6px 12px;text-align:left;font-size:13px;text-transform:capitalize}.dropdown-item:hover{background:var(--bg-hover)}.dropdown-item.active{color:var(--accent);font-weight:500}.dropdown-label{flex:1}.dropdown-kbd{font-size:11px;color:var(--text-muted);padding:1px 5px;border:1px solid var(--border);border-radius:3px;background:var(--bg-secondary);font-family:inherit;flex-shrink:0}.dropdown-dot{display:inline-block;width:8px;height:8px;border-radius:50%;flex-shrink:0}.settings-overlay{position:fixed;inset:0;background:rgba(0,0,0,.3);display:flex;align-items:center;justify-content:center;z-index:2000}.settings-dialog{background:var(--bg);border-radius:8px;box-shadow:0 8px 32px rgba(0,0,0,.15);width:480px;max-width:90vw;max-height:85vh;overflow-y:auto}.settings-header{display:flex;align-items:center;justify-content:space-between;padding:16px 20px;border-bottom:1px solid var(--border)}.settings-header h2{font-size:16px;font-weight:600}.settings-body{padding:20px}.settings-field{margin-bottom:16px}.settings-field:last-child{margin-bottom:0}.settings-field label{display:block;font-size:13px;font-weight:500;color:var(--text-secondary);margin-bottom:6px}.settings-field select,.settings-field input[type=number],.settings-field input[type=text]{width:100%;padding:6px 10px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg);outline:none}.settings-field select:focus,.settings-field input[type=number]:focus,.settings-field input[type=text]:focus{border-color:var(--accent)}.settings-hint{display:block;font-size:11px;color:var(--text-secondary);margin-top:4px}.skills-banner{display:flex;align-items:center;justify-content:space-between;gap:12px;padding:8px 20px;background:#dbeafe;border-bottom:2px solid #3b82f6;font-size:13px;color:#1e3a5f;flex-shrink:0}@media(prefers-color-scheme: dark){.skills-banner{background:#1e293b;border-bottom-color:#3b82f6;color:#93c5fd}}.update-banner{display:flex;align-items:center;justify-content:space-between;gap:12px;padding:8px 20px;background:#dcfce7;border-bottom:2px solid #22c55e;font-size:13px;color:#166534;flex-shrink:0}@media(prefers-color-scheme: dark){.update-banner{background:#14532d;border-bottom-color:#22c55e;color:#86efac}}.update-banner-actions{display:flex;gap:8px;flex-shrink:0}.btn-accent{color:#fff;background:#22c55e;border-color:#22c55e}.btn-accent:hover:not(:disabled){background:#16a34a}.btn-accent:disabled{opacity:.7;cursor:default}.backup-preview-banner{display:flex;align-items:center;justify-content:space-between;padding:8px 20px;background:#fef3c7;border-bottom:2px solid #f59e0b;font-size:13px;font-weight:500;color:#92400e;flex-shrink:0}@media(prefers-color-scheme: dark){.backup-preview-banner{background:#451a03;border-bottom-color:#d97706;color:#fbbf24}}.backup-preview-actions{display:flex;gap:8px;flex-shrink:0}.settings-section{margin-top:20px;padding-top:16px;border-top:1px solid var(--border)}.settings-section h3{font-size:14px;font-weight:600;margin-bottom:12px}.settings-section-header{display:flex;align-items:center;justify-content:space-between}.settings-section-header h3{margin-bottom:0}.settings-section-header{margin-bottom:12px}.backup-list{max-height:300px;overflow-y:auto;font-size:13px;color:var(--text-muted)}.backup-tier-label{font-size:11px;font-weight:600;color:var(--text-muted);text-transform:uppercase;letter-spacing:.5px;padding:8px 0 4px;border-bottom:1px solid var(--border);margin-bottom:2px}.backup-tier-label:first-child{padding-top:0}.backup-row{display:flex;align-items:center;justify-content:space-between;padding:5px 8px;border-radius:var(--radius);cursor:pointer}.backup-row:hover{background:var(--bg-hover)}.backup-row-time{color:var(--text);font-weight:500}.backup-row-meta{font-size:11px;color:var(--text-muted)}.app-footer{display:flex;align-items:center;justify-content:space-between;padding:6px 20px;border-top:1px solid var(--border);background:var(--bg-secondary);font-size:12px;color:var(--text-muted);flex-shrink:0}.keyboard-hints{display:flex;gap:16px}.keyboard-hints kbd{display:inline-block;padding:1px 5px;border:1px solid var(--border);border-radius:3px;background:var(--bg);font-family:inherit;font-size:11px}.status-bar{font-size:12px}.ticket-list-columns{display:flex;flex-direction:column;overflow:hidden}.ticket-list-columns .draft-row{flex-shrink:0}.columns-container{display:flex;flex:1;gap:1px;background:var(--border);overflow-x:auto;min-height:0}.column{flex:1;min-width:180px;display:flex;flex-direction:column;background:var(--bg-secondary);overflow:hidden}.column.column-drop-target{background:var(--bg-selected)}.column.column-drop-target .column-body{outline:2px dashed var(--accent);outline-offset:-2px}.column-header{display:flex;align-items:center;justify-content:space-between;padding:8px 12px;font-size:12px;font-weight:600;color:var(--text-secondary);text-transform:uppercase;letter-spacing:.3px;border-bottom:1px solid var(--border);background:var(--bg);flex-shrink:0}.column-count{font-size:11px;font-weight:500;color:var(--text-muted);background:var(--bg-secondary);padding:1px 6px;border-radius:10px}.column-body{flex:1;overflow-y:auto;padding:8px;display:flex;flex-direction:column;gap:6px}.column-card{padding:8px 10px;background:var(--bg);border:1px solid var(--border);border-radius:var(--radius);cursor:grab;box-shadow:var(--shadow)}.column-card:hover{border-color:var(--accent)}.column-card.selected{border-color:var(--accent);background:var(--bg-selected)}.column-card.up-next{border-left:3px solid var(--star)}.column-card:active{cursor:grabbing}.column-card-header{display:flex;align-items:center;gap:6px;margin-bottom:4px}.column-card-title{font-size:13px;line-height:1.4;word-break:break-word}@media(max-width: 768px){.sidebar{width:140px}.detail-side .detail-panel{width:280px}.keyboard-hints{display:none}}@media(max-width: 600px){.sidebar{display:none}}.error-popup{position:fixed;inset:0;background:rgba(0,0,0,.3);display:flex;align-items:center;justify-content:center;z-index:3000}.error-popup-content{background:var(--bg);border:1px solid var(--danger);border-radius:8px;box-shadow:0 8px 32px rgba(0,0,0,.15);padding:24px;max-width:360px;text-align:center}.error-popup-content strong{display:block;font-size:16px;margin-bottom:8px;color:var(--danger)}.error-popup-content p{font-size:14px;color:var(--text-secondary);margin-bottom:16px}.error-popup-content button{padding:6px 16px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg);font-size:13px;cursor:pointer}.error-popup-content button:hover{background:var(--bg-hover)}.detail-notes{display:flex;flex-direction:column;gap:8px}.note-entry{padding:8px 10px;background:var(--bg-secondary);border-radius:var(--radius);border-left:3px solid var(--accent)}.note-timestamp{font-size:11px;color:var(--text-muted);margin-bottom:2px}.note-text{font-size:13px;color:var(--text);white-space:pre-wrap}
|
|
1
|
+
*,*::before,*::after{box-sizing:border-box;margin:0;padding:0}:root{--bg: #ffffff;--bg-secondary: #f9fafb;--bg-hover: #f3f4f6;--bg-selected: #eff6ff;--border: #e5e7eb;--text: #111827;--text-secondary: #6b7280;--text-muted: #9ca3af;--accent: #3b82f6;--accent-hover: #2563eb;--danger: #ef4444;--star: #eab308;--radius: 6px;--shadow: 0 1px 3px rgba(0,0,0,0.08)}html,body{height:100%;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,sans-serif;font-size:14px;line-height:1.5;color:var(--text);background:var(--bg)}button{cursor:pointer;border:none;background:none;font:inherit;color:inherit}input,select,textarea{font:inherit;color:inherit}.app{display:flex;flex-direction:column;height:100vh}.app-header{display:flex;align-items:center;justify-content:space-between;padding:12px 20px;border-bottom:1px solid var(--border);background:var(--bg);gap:16px;flex-shrink:0}.app-header h1{font-size:18px;font-weight:700;white-space:nowrap}.header-controls{display:flex;align-items:center;gap:12px;flex:1;max-width:500px}.search-box{flex:1}.search-box input{width:100%;padding:6px 12px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg-secondary);outline:none}.search-box input:focus{border-color:var(--accent);background:var(--bg)}.layout-toggle{display:flex;border:1px solid var(--border);border-radius:var(--radius);overflow:hidden;flex-shrink:0}.layout-btn{display:flex;align-items:center;justify-content:center;padding:4px 8px;color:var(--text-secondary);background:var(--bg);border:none;border-right:1px solid var(--border);cursor:pointer}.layout-btn:last-child{border-right:none}.layout-btn:hover:not(:disabled){background:var(--bg-hover);color:var(--text)}.layout-btn.active{background:var(--bg-selected);color:var(--accent)}.layout-btn:disabled{cursor:default}.sort-controls select{padding:6px 8px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg);cursor:pointer}.settings-btn{font-size:18px;color:var(--text-muted);padding:4px;border-radius:var(--radius);flex-shrink:0}.settings-btn:hover{color:var(--text);background:var(--bg-hover)}.glassbox-btn{padding:0;border:1px solid var(--border);border-radius:var(--radius);flex-shrink:0;cursor:pointer;overflow:hidden;line-height:0}.glassbox-btn img{height:28px;width:28px;display:block}.glassbox-btn:hover{border-color:var(--accent)}.app-body{display:flex;flex:1;overflow:hidden}.content-area{flex:1;display:flex;overflow:hidden;min-width:0}.content-area.detail-side{flex-direction:row}.content-area.detail-bottom{flex-direction:column}.sidebar{width:180px;border-right:1px solid var(--border);background:var(--bg-secondary);overflow-y:auto;flex-shrink:0;padding:8px 0}.sidebar-copy-prompt{padding:8px 10px 4px}.copy-prompt-btn{display:flex;align-items:center;gap:6px;width:100%;padding:6px 8px;font-size:12px;color:var(--text-secondary);background:var(--bg);border:1px solid var(--border);border-radius:var(--radius);cursor:pointer;transition:all .15s}.copy-prompt-btn:hover{color:var(--accent);border-color:var(--accent);background:var(--bg-selected)}.copy-prompt-icon{display:flex;align-items:center;flex-shrink:0}.sidebar-section{margin-bottom:8px}.sidebar-label{padding:8px 16px 4px;font-size:11px;font-weight:600;text-transform:uppercase;color:var(--text-muted);letter-spacing:.5px}.sidebar-item{display:flex;align-items:center;gap:8px;width:100%;padding:6px 16px;text-align:left;font-size:13px;color:var(--text-secondary);border-radius:0}.sidebar-item:hover{background:var(--bg-hover);color:var(--text)}.sidebar-item.active{background:var(--bg-selected);color:var(--accent);font-weight:500}.sidebar-item.drop-target{background:var(--bg-selected);color:var(--accent);outline:2px dashed var(--accent);outline-offset:-2px}.sidebar-divider{height:1px;background:var(--border);margin:6px 16px}.cat-dot{display:inline-block;width:8px;height:8px;border-radius:50%;flex-shrink:0}.sidebar-stats{padding:12px 16px;font-size:12px;color:var(--text-muted);border-top:1px solid var(--border);margin-top:8px}.main-content{flex:1;display:flex;flex-direction:column;overflow:hidden;min-width:0;min-height:0}.batch-toolbar{display:flex;align-items:center;gap:8px;padding:6px 16px 6px 19px;background:var(--bg-secondary);border-bottom:1px solid var(--border);flex-shrink:0;flex-wrap:wrap;min-height:36px}.batch-toolbar select:disabled,.batch-toolbar .btn:disabled{opacity:.4;cursor:default;pointer-events:none}.batch-toolbar select{padding:4px 8px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg);font-size:12px}.batch-select-all{width:14px;height:14px;cursor:pointer;accent-color:var(--accent);flex-shrink:0}.batch-star-btn{font-size:16px;color:var(--text-muted);padding:2px 6px;border-radius:var(--radius);line-height:1}.batch-star-btn:hover:not(:disabled){color:var(--star);background:var(--bg-hover)}.batch-star-btn.active{color:var(--star)}.batch-star-btn.mixed{color:var(--star)}.batch-star-btn:disabled{opacity:.4;cursor:default;pointer-events:none}.star-mixed-wrap{position:relative;display:inline-block}.star-mixed-fill{position:absolute;left:0;top:0;overflow:hidden;width:50%}.batch-delete-btn.btn{display:inline-flex;align-items:center;justify-content:center;padding:4px 6px;border:none;background:none}.batch-count{font-size:12px;color:var(--text-muted);margin-left:auto;white-space:nowrap}.btn{padding:4px 10px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg);font-size:12px;cursor:pointer}.btn:hover{background:var(--bg-hover)}.btn-sm{padding:3px 8px;font-size:11px}.btn-danger{color:var(--danger)}.btn-danger:hover:not(:disabled){background:#fef2f2}.ticket-list{flex:1;overflow-y:auto;padding:4px 0}.ticket-list-empty,.ticket-list-loading{padding:40px 20px;text-align:center;color:var(--text-muted);font-size:14px}.ticket-row{display:flex;align-items:center;gap:8px;padding:4px 16px;border-bottom:1px solid rgba(0,0,0,0);border-left:3px solid rgba(0,0,0,0);min-height:36px}.ticket-row:hover{background:var(--bg-hover)}.ticket-row.selected{background:var(--bg-selected)}.ticket-row.completed .ticket-title-input{text-decoration:line-through;color:var(--text-muted)}.ticket-row.up-next{border-left:3px solid var(--star)}.ticket-checkbox{flex-shrink:0;width:14px;height:14px;cursor:pointer;accent-color:var(--accent)}.ticket-status-btn{flex-shrink:0;width:20px;height:20px;font-size:14px;display:flex;align-items:center;justify-content:center;border-radius:50%;color:var(--text-secondary)}.ticket-status-btn:hover{background:var(--bg-hover)}.ticket-status-btn.verified{color:#22c55e}.ticket-category-badge{flex-shrink:0;width:4em;padding:1px 0;border-radius:3px;font-size:10px;font-weight:600;color:#fff;text-transform:uppercase;text-align:center;cursor:pointer;letter-spacing:.3px}.ticket-category-badge:hover{opacity:.85}.ticket-number{flex-shrink:0;font-size:11px;color:var(--text-muted);font-family:"SF Mono",Monaco,"Cascadia Code",monospace;min-width:5em}.ticket-title-input{flex:1;border:none;background:rgba(0,0,0,0);padding:2px 0;outline:none;min-width:0}.ticket-title-input:focus{border-bottom:1px solid var(--accent)}.ticket-title-input::placeholder{color:var(--text-muted)}.ticket-priority-indicator{flex-shrink:0;font-size:12px;cursor:pointer;padding:2px 4px;border-radius:3px;min-width:24px;text-align:center}.ticket-priority-indicator:hover{background:var(--bg-hover)}.ticket-star{flex-shrink:0;font-size:16px;color:var(--text-muted);padding:0}.ticket-star.active{color:var(--star)}.ticket-star:hover{color:var(--star)}.trash-row .trash-title{flex:1;color:var(--text-muted);min-width:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.trash-row .btn{flex-shrink:0}.draft-row{border-bottom:1px dashed var(--border) !important;background:var(--bg-secondary)}.draft-row:hover{background:var(--bg-secondary)}.draft-row .draft-placeholder{opacity:.35;pointer-events:none}.draft-row .draft-badge{opacity:.35}.draft-row .draft-input::placeholder{color:var(--text-muted);font-style:italic}.ticket-checkbox-spacer{width:14px;flex-shrink:0}.detail-resize-handle{flex-shrink:0;background:rgba(0,0,0,0);position:relative;z-index:10}.detail-resize-handle:hover,.detail-resize-handle:active{background:var(--accent);opacity:.3}.detail-side .detail-resize-handle{width:4px;cursor:col-resize}.detail-bottom .detail-resize-handle{height:4px;cursor:row-resize}.detail-panel{border-left:1px solid var(--border);background:var(--bg);flex-shrink:0;display:flex;flex-direction:column;overflow:hidden}.detail-side .detail-panel{width:360px;border-left:1px solid var(--border);border-top:none}.detail-bottom .detail-panel{height:300px;width:auto;border-left:none;border-top:1px solid var(--border)}.detail-header{display:flex;align-items:center;justify-content:space-between;padding:12px 16px;border-bottom:1px solid var(--border);font-weight:600}.detail-ticket-number{font-family:"SF Mono",Monaco,"Cascadia Code",monospace;font-size:13px;color:var(--accent)}.detail-close{font-size:20px;color:var(--text-muted);padding:0 4px}.detail-close:hover{color:var(--text)}.detail-body{flex:1;overflow-y:auto;padding:16px}.detail-fields-row{display:contents}.detail-bottom .detail-body{display:flex;flex-wrap:wrap;gap:0 16px;align-content:flex-start}.detail-bottom .detail-body .detail-fields-row{display:flex;gap:16px;width:100%}.detail-bottom .detail-body .detail-fields-row .detail-field{flex:1;min-width:0}.detail-bottom .detail-body .detail-field-full{width:100%;flex-basis:100%}.detail-field{margin-bottom:16px}.detail-field label{display:block;font-size:12px;font-weight:500;color:var(--text-secondary);margin-bottom:4px}.detail-field input[type=text],.detail-field textarea,.detail-field select{width:100%;padding:6px 10px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg);outline:none}.detail-field input[type=text]:focus,.detail-field textarea:focus,.detail-field select:focus{border-color:var(--accent)}.detail-field textarea{resize:vertical;min-height:80px}.detail-upnext-label{display:flex !important;align-items:center;gap:8px;cursor:pointer;font-size:14px !important;font-weight:400 !important;color:var(--text) !important}.detail-upnext-label input[type=checkbox]{width:auto;accent-color:var(--star)}.detail-attachments{margin-bottom:8px}.attachment-item{display:flex;align-items:center;justify-content:space-between;padding:4px 8px;background:var(--bg-secondary);border-radius:var(--radius);margin-bottom:4px;font-size:12px}.attachment-name{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.attachment-reveal{color:var(--text-muted);padding:0 4px;cursor:pointer;border:none;background:none;display:flex;align-items:center}.attachment-reveal:hover{color:var(--accent)}.attachment-delete{color:var(--text-muted);font-size:16px;padding:0 4px;cursor:pointer;border:none;background:none}.attachment-delete:hover{color:var(--danger)}.upload-btn{cursor:pointer;display:inline-flex;align-items:center;gap:4px;margin-top:4px}.detail-meta{padding-top:12px;border-top:1px solid var(--border);font-size:11px;color:var(--text-muted)}.detail-meta div{margin-bottom:2px}.dropdown-menu{position:fixed;z-index:1000;background:var(--bg);border:1px solid var(--border);border-radius:var(--radius);box-shadow:var(--shadow);padding:4px 0;min-width:140px}.dropdown-item{display:flex;align-items:center;gap:8px;width:100%;padding:6px 12px;text-align:left;font-size:13px;text-transform:capitalize}.dropdown-item:hover{background:var(--bg-hover)}.dropdown-item.active{color:var(--accent);font-weight:500}.dropdown-label{flex:1}.dropdown-kbd{font-size:11px;color:var(--text-muted);padding:1px 5px;border:1px solid var(--border);border-radius:3px;background:var(--bg-secondary);font-family:inherit;flex-shrink:0}.dropdown-dot{display:inline-block;width:8px;height:8px;border-radius:50%;flex-shrink:0}.settings-overlay{position:fixed;inset:0;background:rgba(0,0,0,.3);display:flex;align-items:center;justify-content:center;z-index:2000}.settings-dialog{background:var(--bg);border-radius:8px;box-shadow:0 8px 32px rgba(0,0,0,.15);width:480px;max-width:90vw;max-height:85vh;overflow-y:auto}.settings-header{display:flex;align-items:center;justify-content:space-between;padding:16px 20px;border-bottom:1px solid var(--border)}.settings-header h2{font-size:16px;font-weight:600}.settings-body{padding:20px}.settings-field{margin-bottom:16px}.settings-field:last-child{margin-bottom:0}.settings-field label{display:block;font-size:13px;font-weight:500;color:var(--text-secondary);margin-bottom:6px}.settings-field select,.settings-field input[type=number],.settings-field input[type=text]{width:100%;padding:6px 10px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg);outline:none}.settings-field select:focus,.settings-field input[type=number]:focus,.settings-field input[type=text]:focus{border-color:var(--accent)}.settings-hint{display:block;font-size:11px;color:var(--text-secondary);margin-top:4px}.skills-banner{display:flex;align-items:center;justify-content:space-between;gap:12px;padding:8px 20px;background:#dbeafe;border-bottom:2px solid #3b82f6;font-size:13px;color:#1e3a5f;flex-shrink:0}@media(prefers-color-scheme: dark){.skills-banner{background:#1e293b;border-bottom-color:#3b82f6;color:#93c5fd}}.update-banner{display:flex;align-items:center;justify-content:space-between;gap:12px;padding:8px 20px;background:#dcfce7;border-bottom:2px solid #22c55e;font-size:13px;color:#166534;flex-shrink:0}@media(prefers-color-scheme: dark){.update-banner{background:#14532d;border-bottom-color:#22c55e;color:#86efac}}.update-banner-actions{display:flex;gap:8px;flex-shrink:0}.btn-accent{color:#fff;background:#22c55e;border-color:#22c55e}.btn-accent:hover:not(:disabled){background:#16a34a}.btn-accent:disabled{opacity:.7;cursor:default}.backup-preview-banner{display:flex;align-items:center;justify-content:space-between;padding:8px 20px;background:#fef3c7;border-bottom:2px solid #f59e0b;font-size:13px;font-weight:500;color:#92400e;flex-shrink:0}@media(prefers-color-scheme: dark){.backup-preview-banner{background:#451a03;border-bottom-color:#d97706;color:#fbbf24}}.backup-preview-actions{display:flex;gap:8px;flex-shrink:0}.settings-section{margin-top:20px;padding-top:16px;border-top:1px solid var(--border)}.settings-section h3{font-size:14px;font-weight:600;margin-bottom:12px}.settings-section-header{display:flex;align-items:center;justify-content:space-between}.settings-section-header h3{margin-bottom:0}.settings-section-header{margin-bottom:12px}.backup-list{max-height:300px;overflow-y:auto;font-size:13px;color:var(--text-muted)}.backup-tier-label{font-size:11px;font-weight:600;color:var(--text-muted);text-transform:uppercase;letter-spacing:.5px;padding:8px 0 4px;border-bottom:1px solid var(--border);margin-bottom:2px}.backup-tier-label:first-child{padding-top:0}.backup-row{display:flex;align-items:center;justify-content:space-between;padding:5px 8px;border-radius:var(--radius);cursor:pointer}.backup-row:hover{background:var(--bg-hover)}.backup-row-time{color:var(--text);font-weight:500}.backup-row-meta{font-size:11px;color:var(--text-muted)}.app-footer{display:flex;align-items:center;justify-content:space-between;padding:6px 20px;border-top:1px solid var(--border);background:var(--bg-secondary);font-size:12px;color:var(--text-muted);flex-shrink:0}.keyboard-hints{display:flex;gap:16px}.keyboard-hints kbd{display:inline-block;padding:1px 5px;border:1px solid var(--border);border-radius:3px;background:var(--bg);font-family:inherit;font-size:11px}.status-bar{font-size:12px}.ticket-list-columns{display:flex;flex-direction:column;overflow:hidden}.ticket-list-columns .draft-row{flex-shrink:0}.columns-container{display:flex;flex:1;gap:1px;background:var(--border);overflow-x:auto;min-height:0}.column{flex:1;min-width:180px;display:flex;flex-direction:column;background:var(--bg-secondary);overflow:hidden}.column.column-drop-target{background:var(--bg-selected)}.column.column-drop-target .column-body{outline:2px dashed var(--accent);outline-offset:-2px}.column-header{display:flex;align-items:center;justify-content:space-between;padding:8px 12px;font-size:12px;font-weight:600;color:var(--text-secondary);text-transform:uppercase;letter-spacing:.3px;border-bottom:1px solid var(--border);background:var(--bg);flex-shrink:0}.column-count{font-size:11px;font-weight:500;color:var(--text-muted);background:var(--bg-secondary);padding:1px 6px;border-radius:10px}.column-body{flex:1;overflow-y:auto;padding:8px;display:flex;flex-direction:column;gap:6px}.column-card{padding:8px 10px;background:var(--bg);border:1px solid var(--border);border-radius:var(--radius);cursor:grab;box-shadow:var(--shadow)}.column-card:hover{border-color:var(--accent)}.column-card.selected{border-color:var(--accent);background:var(--bg-selected)}.column-card.up-next{border-left:3px solid var(--star)}.column-card:active{cursor:grabbing}.column-card-header{display:flex;align-items:center;gap:6px;margin-bottom:4px}.column-card-title{font-size:13px;line-height:1.4;word-break:break-word}@media(max-width: 768px){.sidebar{width:140px}.detail-side .detail-panel{width:280px}.keyboard-hints{display:none}}@media(max-width: 600px){.sidebar{display:none}}.error-popup{position:fixed;inset:0;background:rgba(0,0,0,.3);display:flex;align-items:center;justify-content:center;z-index:3000}.error-popup-content{background:var(--bg);border:1px solid var(--danger);border-radius:8px;box-shadow:0 8px 32px rgba(0,0,0,.15);padding:24px;max-width:360px;text-align:center}.error-popup-content strong{display:block;font-size:16px;margin-bottom:8px;color:var(--danger)}.error-popup-content p{font-size:14px;color:var(--text-secondary);margin-bottom:16px}.error-popup-content button{padding:6px 16px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg);font-size:13px;cursor:pointer}.error-popup-content button:hover{background:var(--bg-hover)}.detail-notes{display:flex;flex-direction:column;gap:8px}.note-entry{padding:8px 10px;background:var(--bg-secondary);border-radius:var(--radius);border-left:3px solid var(--accent)}.note-timestamp{font-size:11px;color:var(--text-muted);margin-bottom:2px}.note-text{font-size:13px;color:var(--text);white-space:pre-wrap}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "hotsheet",
|
|
3
|
-
"version": "0.2.
|
|
3
|
+
"version": "0.2.8",
|
|
4
4
|
"description": "A lightweight local project management tool. Create, categorize, and prioritize tickets with a fast bullet-list interface, then export an Up Next worklist for AI tools.",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"license": "MIT",
|
|
@@ -30,7 +30,10 @@
|
|
|
30
30
|
"scripts": {
|
|
31
31
|
"dev": "npm run build:client && tsx --tsconfig tsconfig.json src/cli.ts",
|
|
32
32
|
"build": "tsup",
|
|
33
|
-
"build:client": "mkdir -p dist/client && npx esbuild src/client/app.ts --bundle --format=iife --outfile=dist/client/app.global.js --target=es2020 --jsx=automatic --jsx-import-source=#jsx --alias:#jsx/jsx-runtime=./src/jsx-runtime.ts && npx sass src/client/styles.scss dist/client/styles.css --style compressed --no-source-map",
|
|
33
|
+
"build:client": "mkdir -p dist/client/assets && cp src/client/assets/* dist/client/assets/ && npx esbuild src/client/app.ts --bundle --format=iife --outfile=dist/client/app.global.js --target=es2020 --jsx=automatic --jsx-import-source=#jsx --alias:#jsx/jsx-runtime=./src/jsx-runtime.ts && npx sass src/client/styles.scss dist/client/styles.css --style compressed --no-source-map",
|
|
34
|
+
"test": "vitest run",
|
|
35
|
+
"test:watch": "vitest",
|
|
36
|
+
"test:coverage": "vitest run --coverage",
|
|
34
37
|
"lint": "eslint src/",
|
|
35
38
|
"clean": "rm -rf dist node_modules/.cache",
|
|
36
39
|
"release": "bash scripts/release.sh",
|
|
@@ -52,7 +55,9 @@
|
|
|
52
55
|
},
|
|
53
56
|
"devDependencies": {
|
|
54
57
|
"@eslint/js": "^9.39.3",
|
|
58
|
+
"@tauri-apps/cli": "^2",
|
|
55
59
|
"@types/node": "^22.0.0",
|
|
60
|
+
"@vitest/coverage-v8": "^4.1.0",
|
|
56
61
|
"esbuild": "^0.27.3",
|
|
57
62
|
"eslint": "^9.39.3",
|
|
58
63
|
"eslint-plugin-import-x": "^4.16.1",
|
|
@@ -62,7 +67,7 @@
|
|
|
62
67
|
"tsup": "^8.5.1",
|
|
63
68
|
"tsx": "^4.21.0",
|
|
64
69
|
"typescript": "^5.9.3",
|
|
65
|
-
"
|
|
66
|
-
"
|
|
70
|
+
"typescript-eslint": "^8.56.1",
|
|
71
|
+
"vitest": "^4.1.0"
|
|
67
72
|
}
|
|
68
73
|
}
|