@thxgg/steward 0.1.10 → 0.1.12
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/.output/nitro.json +1 -1
- package/.output/public/_nuxt/{DOvbLsAq.js → BFv4l3hn.js} +1 -1
- package/.output/public/_nuxt/{DbloiS5Y.js → BZ1iIOYp.js} +1 -1
- package/.output/public/_nuxt/{Cs5ptsBk.js → Bq6edYSd.js} +1 -1
- package/.output/public/_nuxt/BuQdImno.js +1 -0
- package/.output/public/_nuxt/{DcRwFvvS.js → C897Egk9.js} +1 -1
- package/.output/public/_nuxt/{C8LtDyY4.js → CMUOpExW.js} +1 -1
- package/.output/public/_nuxt/{CQgu_W_k.js → DE885CbX.js} +1 -1
- package/.output/public/_nuxt/{vr7VLA9A.js → DoNqd8jQ.js} +1 -1
- package/.output/public/_nuxt/{BPaqwWyl.js → DomrzX-T.js} +1 -1
- package/.output/public/_nuxt/{CZKCADv6.js → R2cvz8mH.js} +2 -2
- package/.output/public/_nuxt/builds/latest.json +1 -1
- package/.output/public/_nuxt/builds/meta/6f66fabf-cc26-482b-8adf-f8731dd68f83.json +1 -0
- package/.output/public/_nuxt/{CeO4HNxC.js → dOaEkD-3.js} +1 -1
- package/.output/public/_nuxt/{CshyynD6.js → kTT8NKtq.js} +1 -1
- package/.output/server/chunks/build/client.precomputed.mjs +1 -1
- package/.output/server/chunks/build/{default-DGOIWqjD.mjs → default-B5nw9_Xg.mjs} +22 -1
- package/.output/server/chunks/build/default-B5nw9_Xg.mjs.map +1 -0
- package/.output/server/chunks/build/server.mjs +1 -1
- package/.output/server/chunks/build/styles.mjs +2 -2
- package/.output/server/chunks/nitro/nitro.mjs +717 -653
- package/.output/server/chunks/routes/api/runtime.get.mjs +25 -0
- package/.output/server/chunks/routes/api/runtime.get.mjs.map +1 -0
- package/.output/server/package.json +1 -1
- package/README.md +9 -0
- package/bin/prd +117 -0
- package/dist/host/src/mcp.js +2 -0
- package/dist/server/utils/db.js +86 -1
- package/docs/MCP.md +14 -0
- package/package.json +1 -1
- package/.output/public/_nuxt/CzKPXRws.js +0 -1
- package/.output/public/_nuxt/builds/meta/7fda7510-94bc-443a-a338-a8d2af142ed9.json +0 -1
- package/.output/server/chunks/build/default-DGOIWqjD.mjs.map +0 -1
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { d as defineEventHandler, u as useRuntimeConfig, k as setHeader } from '../../nitro/nitro.mjs';
|
|
2
|
+
import { randomUUID } from 'node:crypto';
|
|
3
|
+
import 'node:http';
|
|
4
|
+
import 'node:https';
|
|
5
|
+
import 'node:events';
|
|
6
|
+
import 'node:buffer';
|
|
7
|
+
import 'node:fs';
|
|
8
|
+
import 'node:path';
|
|
9
|
+
import 'node:os';
|
|
10
|
+
import 'node:url';
|
|
11
|
+
|
|
12
|
+
const SERVER_INSTANCE_ID = randomUUID();
|
|
13
|
+
const SERVER_STARTED_AT = (/* @__PURE__ */ new Date()).toISOString();
|
|
14
|
+
const runtime_get = defineEventHandler((event) => {
|
|
15
|
+
const runtimeConfig = useRuntimeConfig(event);
|
|
16
|
+
setHeader(event, "Cache-Control", "no-store, no-cache, must-revalidate");
|
|
17
|
+
return {
|
|
18
|
+
buildId: runtimeConfig.app.buildId,
|
|
19
|
+
instanceId: SERVER_INSTANCE_ID,
|
|
20
|
+
startedAt: SERVER_STARTED_AT
|
|
21
|
+
};
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
export { runtime_get as default };
|
|
25
|
+
//# sourceMappingURL=runtime.get.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"runtime.get.mjs","sources":["../../../../../server/api/runtime.get.ts"],"names":[],"mappings":";;;;;;;;;;;AAEA,MAAA,qBAAA,UAAA,EAAA;AACA,MAAA,iBAAA,GAAA,iBAAA,IAAA,IAAA,EAAA,EAAA,WAAA,EAAA;AAEA,oBAAA,kBAAA,CAAA,CAAA,KAAA,KAAA;AACA,EAAA,MAAA,aAAA,GAAA,iBAAA,KAAA,CAAA;AAEA,EAAA,SAAA,CAAA,KAAA,EAAA,iBAAA,qCAAA,CAAA;AAEA,EAAA,OAAA;AAAA,IACA,OAAA,EAAA,cAAA,GAAA,CAAA,OAAA;AAAA,IACA,UAAA,EAAA,kBAAA;AAAA,IACA,SAAA,EAAA;AAAA,GACA;AACA,CAAA,CAAA;;;;"}
|
package/README.md
CHANGED
|
@@ -41,6 +41,15 @@ Add to your MCP client config:
|
|
|
41
41
|
}
|
|
42
42
|
```
|
|
43
43
|
|
|
44
|
+
Steward MCP requires a Node runtime with built-in sqlite support (`node:sqlite`) for `repos`, `prds`, and `state` APIs.
|
|
45
|
+
If you see `ERR_UNKNOWN_BUILTIN_MODULE: node:sqlite`, run with sqlite enabled:
|
|
46
|
+
|
|
47
|
+
```bash
|
|
48
|
+
NODE_OPTIONS=--experimental-sqlite npx -y @thxgg/steward mcp
|
|
49
|
+
```
|
|
50
|
+
|
|
51
|
+
Note: `execute` runs in a VM sandbox by design, so globals like `process` are intentionally not exposed.
|
|
52
|
+
|
|
44
53
|
### CLI
|
|
45
54
|
|
|
46
55
|
```bash
|
package/bin/prd
CHANGED
|
@@ -1,11 +1,128 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
+
import { spawnSync } from 'node:child_process'
|
|
2
3
|
import { existsSync } from 'node:fs'
|
|
3
4
|
import { dirname, resolve } from 'node:path'
|
|
4
5
|
import { fileURLToPath, pathToFileURL } from 'node:url'
|
|
5
6
|
|
|
7
|
+
const SQLITE_ENABLE_FLAG = '--experimental-sqlite'
|
|
8
|
+
const SQLITE_DISABLE_FLAG = '--no-experimental-sqlite'
|
|
9
|
+
const SQLITE_REEXEC_ENV = 'STEWARD_SQLITE_REEXEC'
|
|
10
|
+
|
|
6
11
|
const packageRoot = resolve(dirname(fileURLToPath(import.meta.url)), '..')
|
|
7
12
|
const entryPath = resolve(packageRoot, 'dist', 'host', 'src', 'index.js')
|
|
8
13
|
|
|
14
|
+
function requiresSqliteRuntime(command) {
|
|
15
|
+
return command === 'mcp' || command === 'ui'
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
function isSqliteRuntimeError(error) {
|
|
19
|
+
if (!(error instanceof Error)) {
|
|
20
|
+
return false
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
const code = error.code
|
|
24
|
+
return typeof code === 'string'
|
|
25
|
+
&& code === 'ERR_UNKNOWN_BUILTIN_MODULE'
|
|
26
|
+
&& error.message.includes('node:sqlite')
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
function sanitizeNodeOptions(rawNodeOptions = '') {
|
|
30
|
+
const options = rawNodeOptions
|
|
31
|
+
.split(/\s+/)
|
|
32
|
+
.filter(Boolean)
|
|
33
|
+
.filter((option) => option !== SQLITE_ENABLE_FLAG && option !== SQLITE_DISABLE_FLAG)
|
|
34
|
+
|
|
35
|
+
return options.join(' ')
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
function buildReexecEnv() {
|
|
39
|
+
const env = { ...process.env }
|
|
40
|
+
const sanitizedNodeOptions = sanitizeNodeOptions(env.NODE_OPTIONS)
|
|
41
|
+
|
|
42
|
+
if (sanitizedNodeOptions.length > 0) {
|
|
43
|
+
env.NODE_OPTIONS = sanitizedNodeOptions
|
|
44
|
+
} else {
|
|
45
|
+
delete env.NODE_OPTIONS
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
env[SQLITE_REEXEC_ENV] = '1'
|
|
49
|
+
return env
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
function canEnableSqliteWithFlag() {
|
|
53
|
+
const probe = spawnSync(
|
|
54
|
+
process.execPath,
|
|
55
|
+
[SQLITE_ENABLE_FLAG, '-e', "import('node:sqlite').then(() => process.exit(0)).catch(() => process.exit(1))"],
|
|
56
|
+
{
|
|
57
|
+
stdio: 'ignore',
|
|
58
|
+
env: buildReexecEnv()
|
|
59
|
+
}
|
|
60
|
+
)
|
|
61
|
+
|
|
62
|
+
return probe.status === 0
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
function reexecWithSqliteFlag() {
|
|
66
|
+
const result = spawnSync(
|
|
67
|
+
process.execPath,
|
|
68
|
+
[SQLITE_ENABLE_FLAG, fileURLToPath(import.meta.url), ...process.argv.slice(2)],
|
|
69
|
+
{
|
|
70
|
+
stdio: 'inherit',
|
|
71
|
+
env: buildReexecEnv()
|
|
72
|
+
}
|
|
73
|
+
)
|
|
74
|
+
|
|
75
|
+
if (result.error) {
|
|
76
|
+
throw result.error
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
process.exit(result.status ?? 1)
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
async function ensureSqliteRuntime(command) {
|
|
83
|
+
if (!requiresSqliteRuntime(command)) {
|
|
84
|
+
return
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
try {
|
|
88
|
+
await import('node:sqlite')
|
|
89
|
+
return
|
|
90
|
+
} catch (error) {
|
|
91
|
+
if (!isSqliteRuntimeError(error)) {
|
|
92
|
+
throw error
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
const alreadyReexecuted = process.env[SQLITE_REEXEC_ENV] === '1'
|
|
96
|
+
if (!alreadyReexecuted && canEnableSqliteWithFlag()) {
|
|
97
|
+
reexecWithSqliteFlag()
|
|
98
|
+
return
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
const message = error instanceof Error ? error.message : String(error)
|
|
102
|
+
const nodeOptions = process.env.NODE_OPTIONS || ''
|
|
103
|
+
const hasDisableFlag = process.execArgv.includes(SQLITE_DISABLE_FLAG)
|
|
104
|
+
|| nodeOptions.includes(SQLITE_DISABLE_FLAG)
|
|
105
|
+
|
|
106
|
+
console.error('Steward requires SQLite runtime support for repos/prds/state APIs.')
|
|
107
|
+
console.error(`Runtime: ${process.version} (${process.execPath})`)
|
|
108
|
+
if (nodeOptions) {
|
|
109
|
+
console.error(`NODE_OPTIONS: ${nodeOptions}`)
|
|
110
|
+
}
|
|
111
|
+
console.error(`Cause: ${message}`)
|
|
112
|
+
|
|
113
|
+
if (hasDisableFlag) {
|
|
114
|
+
console.error(`Hint: remove ${SQLITE_DISABLE_FLAG} from NODE_OPTIONS or runtime args and retry.`)
|
|
115
|
+
} else {
|
|
116
|
+
console.error(`Hint: use a Node runtime with sqlite support or launch with ${SQLITE_ENABLE_FLAG}.`)
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
process.exit(1)
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
const command = process.argv[2] || ''
|
|
124
|
+
await ensureSqliteRuntime(command)
|
|
125
|
+
|
|
9
126
|
if (!existsSync(entryPath)) {
|
|
10
127
|
console.error('Steward host runtime is not built.')
|
|
11
128
|
console.error('Run `npm run build:host` in this package before invoking `prd`.')
|
package/dist/host/src/mcp.js
CHANGED
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
|
|
2
2
|
import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';
|
|
3
3
|
import { z } from 'zod';
|
|
4
|
+
import { assertSqliteRuntimeSupport } from '../../server/utils/db.js';
|
|
4
5
|
import { execute } from './executor.js';
|
|
5
6
|
import { getExecuteToolDescription } from './help.js';
|
|
6
7
|
function serializeEnvelope(envelope) {
|
|
@@ -49,6 +50,7 @@ function buildUnexpectedErrorEnvelope(error) {
|
|
|
49
50
|
};
|
|
50
51
|
}
|
|
51
52
|
export async function runMcpServer() {
|
|
53
|
+
await assertSqliteRuntimeSupport();
|
|
52
54
|
const server = new McpServer({
|
|
53
55
|
name: 'steward',
|
|
54
56
|
version: '0.1.0'
|
package/dist/server/utils/db.js
CHANGED
|
@@ -3,7 +3,19 @@ import { dirname, join } from 'node:path';
|
|
|
3
3
|
import { homedir } from 'node:os';
|
|
4
4
|
const DEFAULT_DATA_HOME = process.env.XDG_DATA_HOME || join(homedir(), '.local', 'share');
|
|
5
5
|
const DEFAULT_DB_PATH = join(DEFAULT_DATA_HOME, 'prd', 'state.db');
|
|
6
|
+
const SQLITE_ENABLE_FLAG = '--experimental-sqlite';
|
|
7
|
+
const SQLITE_DISABLE_FLAG = '--no-experimental-sqlite';
|
|
6
8
|
let adapterPromise = null;
|
|
9
|
+
export class SqliteRuntimeError extends Error {
|
|
10
|
+
code;
|
|
11
|
+
details;
|
|
12
|
+
constructor(message, details) {
|
|
13
|
+
super(message);
|
|
14
|
+
this.name = 'SqliteRuntimeError';
|
|
15
|
+
this.code = 'SQLITE_RUNTIME_UNSUPPORTED';
|
|
16
|
+
this.details = details;
|
|
17
|
+
}
|
|
18
|
+
}
|
|
7
19
|
function coerceChanges(result) {
|
|
8
20
|
if (!result || typeof result !== 'object') {
|
|
9
21
|
return 0;
|
|
@@ -22,11 +34,84 @@ function resolveDbPath() {
|
|
|
22
34
|
}
|
|
23
35
|
return DEFAULT_DB_PATH;
|
|
24
36
|
}
|
|
37
|
+
function formatNodeRuntimeHint() {
|
|
38
|
+
const nodeOptions = process.env.NODE_OPTIONS || '';
|
|
39
|
+
const hasDisableFlag = process.execArgv.includes(SQLITE_DISABLE_FLAG)
|
|
40
|
+
|| nodeOptions.includes(SQLITE_DISABLE_FLAG);
|
|
41
|
+
if (hasDisableFlag) {
|
|
42
|
+
return `Remove ${SQLITE_DISABLE_FLAG} from NODE_OPTIONS or Node arguments.`;
|
|
43
|
+
}
|
|
44
|
+
return `Use Node.js with built-in sqlite support, or launch with ${SQLITE_ENABLE_FLAG}.`;
|
|
45
|
+
}
|
|
46
|
+
function mapSqliteImportError(error, dbPath) {
|
|
47
|
+
if (!(error instanceof Error)) {
|
|
48
|
+
return new SqliteRuntimeError('SQLite runtime support is unavailable in this process.', {
|
|
49
|
+
runtime: 'node',
|
|
50
|
+
dbPath,
|
|
51
|
+
nodeVersion: process.version,
|
|
52
|
+
execPath: process.execPath,
|
|
53
|
+
execArgv: [...process.execArgv],
|
|
54
|
+
nodeOptions: process.env.NODE_OPTIONS || '',
|
|
55
|
+
originalMessage: String(error)
|
|
56
|
+
});
|
|
57
|
+
}
|
|
58
|
+
const code = error.code;
|
|
59
|
+
const isMissingBuiltin = typeof code === 'string'
|
|
60
|
+
&& code === 'ERR_UNKNOWN_BUILTIN_MODULE'
|
|
61
|
+
&& error.message.includes('node:sqlite');
|
|
62
|
+
const isMissingModule = typeof code === 'string'
|
|
63
|
+
&& code === 'ERR_MODULE_NOT_FOUND'
|
|
64
|
+
&& error.message.includes('node:sqlite');
|
|
65
|
+
if (!isMissingBuiltin && !isMissingModule) {
|
|
66
|
+
return error;
|
|
67
|
+
}
|
|
68
|
+
return new SqliteRuntimeError(`SQLite runtime support is unavailable for Steward (${process.version}). ${formatNodeRuntimeHint()}`, {
|
|
69
|
+
runtime: 'node',
|
|
70
|
+
dbPath,
|
|
71
|
+
nodeVersion: process.version,
|
|
72
|
+
execPath: process.execPath,
|
|
73
|
+
execArgv: [...process.execArgv],
|
|
74
|
+
nodeOptions: process.env.NODE_OPTIONS || '',
|
|
75
|
+
originalCode: typeof code === 'string' ? code : undefined,
|
|
76
|
+
originalMessage: error.message
|
|
77
|
+
});
|
|
78
|
+
}
|
|
79
|
+
async function loadNodeSqlite(dbPath) {
|
|
80
|
+
try {
|
|
81
|
+
return await import('node:sqlite');
|
|
82
|
+
}
|
|
83
|
+
catch (error) {
|
|
84
|
+
throw mapSqliteImportError(error, dbPath);
|
|
85
|
+
}
|
|
86
|
+
}
|
|
25
87
|
export function getDbPath() {
|
|
26
88
|
return resolveDbPath();
|
|
27
89
|
}
|
|
90
|
+
export async function assertSqliteRuntimeSupport() {
|
|
91
|
+
const dbPath = resolveDbPath();
|
|
92
|
+
const isBunRuntime = typeof globalThis.Bun !== 'undefined';
|
|
93
|
+
if (isBunRuntime) {
|
|
94
|
+
try {
|
|
95
|
+
const bunModuleName = 'bun:sqlite';
|
|
96
|
+
await import(bunModuleName);
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
99
|
+
catch (error) {
|
|
100
|
+
const code = error instanceof Error
|
|
101
|
+
? error.code
|
|
102
|
+
: undefined;
|
|
103
|
+
throw new SqliteRuntimeError('SQLite runtime support is unavailable in Bun runtime.', {
|
|
104
|
+
runtime: 'bun',
|
|
105
|
+
dbPath,
|
|
106
|
+
originalCode: typeof code === 'string' ? code : undefined,
|
|
107
|
+
originalMessage: error instanceof Error ? error.message : String(error)
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
await loadNodeSqlite(dbPath);
|
|
112
|
+
}
|
|
28
113
|
async function createNodeAdapter(dbPath) {
|
|
29
|
-
const sqliteModule = await
|
|
114
|
+
const sqliteModule = await loadNodeSqlite(dbPath);
|
|
30
115
|
const db = new sqliteModule.DatabaseSync(dbPath);
|
|
31
116
|
return {
|
|
32
117
|
exec(sql) {
|
package/docs/MCP.md
CHANGED
|
@@ -46,6 +46,20 @@ Example MCP client config:
|
|
|
46
46
|
}
|
|
47
47
|
```
|
|
48
48
|
|
|
49
|
+
## Runtime Requirements
|
|
50
|
+
|
|
51
|
+
- `repos`, `prds`, and `state` APIs require sqlite runtime support.
|
|
52
|
+
- Steward uses Node's built-in `node:sqlite` module.
|
|
53
|
+
- `prd mcp` auto-retries with `--experimental-sqlite` when the runtime supports it.
|
|
54
|
+
|
|
55
|
+
If you still see `ERR_UNKNOWN_BUILTIN_MODULE: node:sqlite`, launch explicitly with:
|
|
56
|
+
|
|
57
|
+
```bash
|
|
58
|
+
NODE_OPTIONS=--experimental-sqlite npx -y @thxgg/steward mcp
|
|
59
|
+
```
|
|
60
|
+
|
|
61
|
+
If that still fails, the MCP host is using an incompatible Node runtime.
|
|
62
|
+
|
|
49
63
|
## Execute Contract
|
|
50
64
|
|
|
51
65
|
`execute` expects one input field:
|
package/package.json
CHANGED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
import{s as M,l as e,Y as _t,g as E,k as oe,o as u,z as g,E as T,c as k,n as H,F as Z,r as te,y as bt,C as ue,w as c,A as O,q as S,b as v,N as ge,p as ke,Z as xt,x as ce,B as ye,$ as Ct,j as Je,X as Te,a as d,t as L,a0 as kt,a1 as wt,m as ve,S as pe,d as G,a2 as St,O as Fe,P as Pe,V as Oe,a3 as Qe,W as $t,a4 as Et,a5 as Dt}from"./CZKCADv6.js";import{c as _e,l as Ie,e as Rt,r as Ye,f as Ze,P as ne,d as Be,u as we,o as Bt,B as de,b as et,a as ze,F as Le,k as J,j as W,m as At}from"./CQgu_W_k.js";import{x as Ft,V as Pt,y as Ve,a as je,u as Ot,g as It,c as be,d as Se,e as Lt,z as Mt,j as Tt,t as tt,k as zt,s as Vt,r as jt,m as qt,n as Kt,o as Ht,A as Ut,p as Nt,S as Gt,L as Wt,C as Xt,R as st,B as Jt,i as fe,F as Qt,G as qe,H as at,I as ot,J as nt,X as lt,h as $e,M as Yt,N as Zt,O as es,l as ts,v as ss,K as as}from"./C8LtDyY4.js";import{F as os,I as ns,a as Me}from"./CshyynD6.js";import{_ as ls}from"./vr7VLA9A.js";const rt=_e("monitor",[["rect",{width:"20",height:"14",x:"2",y:"3",rx:"2",key:"48i651"}],["line",{x1:"8",x2:"16",y1:"21",y2:"21",key:"1svkeh"}],["line",{x1:"12",x2:"12",y1:"17",y2:"21",key:"vw1qmm"}]]);const it=_e("moon",[["path",{d:"M20.985 12.486a9 9 0 1 1-9.473-9.472c.405-.022.617.46.402.803a6 6 0 0 0 8.268 8.268c.344-.215.825-.004.803.401",key:"kfwtm"}]]);const rs=_e("search",[["path",{d:"m21 21-4.34-4.34",key:"14j7rj"}],["circle",{cx:"11",cy:"11",r:"8",key:"4ej97u"}]]);const ut=_e("sun",[["circle",{cx:"12",cy:"12",r:"4",key:"4exip2"}],["path",{d:"M12 2v2",key:"tus03m"}],["path",{d:"M12 20v2",key:"1lh1kg"}],["path",{d:"m4.93 4.93 1.41 1.41",key:"149t6j"}],["path",{d:"m17.66 17.66 1.41 1.41",key:"ptbguv"}],["path",{d:"M2 12h2",key:"1t8f8n"}],["path",{d:"M20 12h2",key:"1q8mjw"}],["path",{d:"m6.34 17.66-1.41 1.41",key:"1m8zz5"}],["path",{d:"m19.07 4.93-1.41 1.41",key:"1shlcs"}]]);const is=_e("trash-2",[["path",{d:"M10 11v6",key:"nco0om"}],["path",{d:"M14 11v6",key:"outv1u"}],["path",{d:"M19 6v14a2 2 0 0 1-2 2H7a2 2 0 0 1-2-2V6",key:"miytrc"}],["path",{d:"M3 6h18",key:"d0wm0j"}],["path",{d:"M8 6V4a2 2 0 0 1 2-2h4a2 2 0 0 1 2 2v2",key:"e791ji"}]]);function us(a,t,s){const o=a.findIndex(l=>Ie(l,t)),r=a.findIndex(l=>Ie(l,s));if(o===-1||r===-1)return[];const[i,n]=[o,r].sort((l,f)=>l-f);return a.slice(i,n+1)}function ds(a){const t=M(()=>e(a)),s=M(()=>new Intl.Collator("en",{usage:"search",...t.value}));return{startsWith:(n,l)=>l.length===0?!0:(n=n.normalize("NFC"),l=l.normalize("NFC"),s.value.compare(n.slice(0,l.length),l)===0),endsWith:(n,l)=>l.length===0?!0:(n=n.normalize("NFC"),l=l.normalize("NFC"),s.value.compare(n.slice(-l.length),l)===0),contains:(n,l)=>{if(l.length===0)return!0;n=n.normalize("NFC"),l=l.normalize("NFC");let f=0;const p=l.length;for(;f+p<=n.length;f++){const y=n.slice(f,f+p);if(s.value.compare(l,y)===0)return!0}return!1}}}function cs(a){return M(()=>_t(a)?!!Rt(a)?.closest("form"):!0)}function ps(){return{ALT:"Alt",ARROW_DOWN:"ArrowDown",ARROW_LEFT:"ArrowLeft",ARROW_RIGHT:"ArrowRight",ARROW_UP:"ArrowUp",BACKSPACE:"Backspace",CAPS_LOCK:"CapsLock",CONTROL:"Control",DELETE:"Delete",END:"End",ENTER:"Enter",ESCAPE:"Escape",F1:"F1",F10:"F10",F11:"F11",F12:"F12",F2:"F2",F3:"F3",F4:"F4",F5:"F5",F6:"F6",F7:"F7",F8:"F8",F9:"F9",HOME:"Home",META:"Meta",PAGE_DOWN:"PageDown",PAGE_UP:"PageUp",SHIFT:"Shift",SPACE:" ",TAB:"Tab",CTRL:"Control",ASTERISK:"*",SPACE_CODE:"Space"}}function fs(a){const t=Ye("",1e3);return{search:t,handleTypeaheadSearch:(r,i)=>{t.value=t.value+r;{const n=Ft(),l=i.map(b=>({...b,textValue:b.value?.textValue??b.ref.textContent?.trim()??""})),f=l.find(b=>b.ref===n),p=l.map(b=>b.textValue),y=vs(p,t.value,f?.textValue),m=l.find(b=>b.textValue===y);return m&&m.ref.focus(),m?.ref}},resetTypeahead:()=>{t.value=""}}}function ms(a,t){return a.map((s,o)=>a[(t+o)%a.length])}function vs(a,t,s){const r=t.length>1&&Array.from(t).every(p=>p===t[0])?t[0]:t,i=s?a.indexOf(s):-1;let n=ms(a,Math.max(i,0));r.length===1&&(n=n.filter(p=>p!==s));const f=n.find(p=>p.toLowerCase().startsWith(r.toLowerCase()));return f!==s?f:void 0}var hs=E({inheritAttrs:!1,__name:"VisuallyHiddenInputBubble",props:{name:{type:String,required:!0},value:{type:null,required:!0},checked:{type:Boolean,required:!1,default:void 0},required:{type:Boolean,required:!1},disabled:{type:Boolean,required:!1},feature:{type:String,required:!1,default:"fully-hidden"}},setup(a){const t=a,{primitiveElement:s,currentElement:o}=Ve(),r=M(()=>t.checked??t.value);return oe(r,(i,n)=>{if(!o.value)return;const l=o.value,f=window.HTMLInputElement.prototype,y=Object.getOwnPropertyDescriptor(f,"value").set;if(y&&i!==n){const m=new Event("input",{bubbles:!0}),b=new Event("change",{bubbles:!0});y.call(l,i),l.dispatchEvent(m),l.dispatchEvent(b)}}),(i,n)=>(u(),g(Pt,T({ref_key:"primitiveElement",ref:s},{...t,...i.$attrs},{as:"input"}),null,16))}}),Ne=hs,gs=E({inheritAttrs:!1,__name:"VisuallyHiddenInput",props:{name:{type:String,required:!0},value:{type:null,required:!0},checked:{type:Boolean,required:!1,default:void 0},required:{type:Boolean,required:!1},disabled:{type:Boolean,required:!1},feature:{type:String,required:!1,default:"fully-hidden"}},setup(a){const t=a,s=M(()=>typeof t.value=="object"&&Array.isArray(t.value)&&t.value.length===0&&t.required),o=M(()=>typeof t.value=="string"||typeof t.value=="number"||typeof t.value=="boolean"||t.value===null||t.value===void 0?[{name:t.name,value:t.value}]:typeof t.value=="object"&&Array.isArray(t.value)?t.value.flatMap((r,i)=>typeof r=="object"?Object.entries(r).map(([n,l])=>({name:`${t.name}[${i}][${n}]`,value:l})):{name:`${t.name}[${i}]`,value:r}):t.value!==null&&typeof t.value=="object"&&!Array.isArray(t.value)?Object.entries(t.value).map(([r,i])=>({name:`${t.name}[${r}]`,value:i})):[]);return(r,i)=>(u(),k(Z,null,[H(" We render single input if it's required "),s.value?(u(),g(Ne,T({key:r.name},{...t,...r.$attrs},{name:r.name,value:r.value}),null,16,["name","value"])):(u(!0),k(Z,{key:1},te(o.value,n=>(u(),g(Ne,T({key:n.name},{ref_for:!0},{...t,...r.$attrs},{name:n.name,value:n.value}),null,16,["name","value"]))),128))],2112))}}),ys=gs;function _s(a,t,s){return a===void 0?!1:Array.isArray(a)?a.some(o=>he(o,t,s)):he(a,t,s)}function he(a,t,s){return a===void 0||t===void 0?!1:typeof a=="string"?a===t:typeof s=="function"?s(a,t):typeof s=="string"?a?.[s]===t?.[s]:Ie(a,t)}const[Ke,bs]=be("ListboxRoot");var xs=E({__name:"ListboxRoot",props:{modelValue:{type:null,required:!1},defaultValue:{type:null,required:!1},multiple:{type:Boolean,required:!1},orientation:{type:String,required:!1,default:"vertical"},dir:{type:String,required:!1},disabled:{type:Boolean,required:!1},selectionBehavior:{type:String,required:!1,default:"toggle"},highlightOnHover:{type:Boolean,required:!1},by:{type:[String,Function],required:!1},asChild:{type:Boolean,required:!1},as:{type:null,required:!1},name:{type:String,required:!1},required:{type:Boolean,required:!1}},emits:["update:modelValue","highlight","entryFocus","leave"],setup(a,{expose:t,emit:s}){const o=a,r=s,{multiple:i,highlightOnHover:n,orientation:l,disabled:f,selectionBehavior:p,dir:y}=bt(o),{getItems:m}=je({isProvider:!0}),{handleTypeaheadSearch:b}=fs(),{primitiveElement:z,currentElement:x}=Ve(),A=ps(),V=Ot(y),w=cs(x),K=S(),U=S(!1),X=S(!0),D=Ze(o,"modelValue",r,{defaultValue:o.defaultValue??(i.value?[]:void 0),passive:o.modelValue===void 0,deep:!0});function se(h){if(U.value=!0,o.multiple){const C=Array.isArray(D.value)?[...D.value]:[],$=C.findIndex(q=>he(q,h,o.by));o.selectionBehavior==="toggle"?($===-1?C.push(h):C.splice($,1),D.value=C):(D.value=[h],K.value=h)}else o.selectionBehavior==="toggle"&&he(D.value,h,o.by)?D.value=void 0:D.value=h;setTimeout(()=>{U.value=!1},1)}const j=S(null),ee=S(null),F=S(!1),P=S(!1),B=Be(),N=Be(),Y=Be();function ae(){return m().map(h=>h.ref).filter(h=>h.dataset.disabled!=="")}function Q(h,C=!0){if(!h)return;j.value=h,X.value&&j.value.focus(),C&&j.value.scrollIntoView({block:"nearest"});const $=m().find(q=>q.ref===h);r("highlight",$)}function De(h){if(F.value)Y.trigger(h);else{const C=m().find($=>he($.value,h,o.by));C&&(j.value=C.ref,Q(C.ref))}}function xe(h){j.value&&j.value.isConnected&&(h.preventDefault(),h.stopPropagation(),P.value||j.value.click())}function I(h){if(X.value){if(U.value=!0,F.value)N.trigger(h);else{const C=h.altKey||h.ctrlKey||h.metaKey;if(C&&h.key==="a"&&i.value){const $=m(),q=$.map(re=>re.value);D.value=[...q],h.preventDefault(),Q($[$.length-1].ref)}else if(!C){const $=b(h.key,m());$&&Q($)}}setTimeout(()=>{U.value=!1},1)}}function _(){P.value=!0}function R(){ue(()=>{P.value=!1})}function le(){ue(()=>{const h=new KeyboardEvent("keydown",{key:"PageUp"});He(h)})}function Re(h){const C=j.value;C?.isConnected&&(ee.value=C),j.value=null,r("leave",h)}function ht(h){const C=new CustomEvent("listbox.entryFocus",{bubbles:!1,cancelable:!0});if(h.currentTarget?.dispatchEvent(C),r("entryFocus",C),!C.defaultPrevented)if(ee.value)Q(ee.value);else{const $=ae()?.[0];Q($)}}function He(h){const C=It(h,l.value,V.value);if(!C)return;let $=ae();if(j.value){if(C==="last")$.reverse();else if(C==="prev"||C==="next"){C==="prev"&&$.reverse();const q=$.indexOf(j.value);$=$.slice(q+1)}gt(h,$[0])}if($.length){const q=!j.value&&C==="prev"?$.length-1:0;Q($[q])}if(F.value)return N.trigger(h)}function gt(h,C){if(!(F.value||o.selectionBehavior!=="replace"||!i.value||!Array.isArray(D.value)||(h.altKey||h.ctrlKey||h.metaKey)&&!h.shiftKey)&&h.shiftKey){const q=m().filter(me=>me.ref.dataset.disabled!=="");let re=q.find(me=>me.ref===C)?.value;if(h.key===A.END?re=q[q.length-1].value:h.key===A.HOME&&(re=q[0].value),!re||!K.value)return;const yt=us(q.map(me=>me.value),K.value,re);D.value=yt}}async function Ue(h){if(await ue(),F.value)B.trigger(h);else{const C=ae(),$=C.find(q=>q.dataset.state==="checked");$?Q($):C.length&&Q(C[0])}}return oe(D,()=>{U.value||ue(()=>{Ue()})},{immediate:!0,deep:!0}),t({highlightedElement:j,highlightItem:De,highlightFirstItem:le,highlightSelected:Ue,getItems:m}),bs({modelValue:D,onValueChange:se,multiple:i,orientation:l,dir:V,disabled:f,highlightOnHover:n,highlightedElement:j,isVirtual:F,virtualFocusHook:B,virtualKeydownHook:N,virtualHighlightHook:Y,by:o.by,firstValue:K,selectionBehavior:p,focusable:X,onLeave:Re,onEnter:ht,changeHighlight:Q,onKeydownEnter:xe,onKeydownNavigation:He,onKeydownTypeAhead:I,onCompositionStart:_,onCompositionEnd:R,highlightFirstItem:le}),(h,C)=>(u(),g(e(ne),{ref_key:"primitiveElement",ref:z,as:h.as,"as-child":h.asChild,dir:e(V),"data-disabled":e(f)?"":void 0,onPointerleave:Re,onFocusout:C[0]||(C[0]=async $=>{const q=$.relatedTarget||$.target;await ue(),j.value&&e(x)&&!e(x).contains(q)&&Re($)})},{default:c(()=>[O(h.$slots,"default",{modelValue:e(D)}),e(w)&&h.name?(u(),g(e(ys),{key:0,name:h.name,value:e(D),disabled:e(f),required:h.required},null,8,["name","value","disabled","required"])):H("v-if",!0)]),_:3},8,["as","as-child","dir","data-disabled"]))}}),Cs=xs,ks=E({__name:"ListboxContent",props:{asChild:{type:Boolean,required:!1},as:{type:null,required:!1}},setup(a){const{CollectionSlot:t}=je(),s=Ke(),o=Ye(!1,10);return(r,i)=>(u(),g(e(t),null,{default:c(()=>[v(e(ne),{role:"listbox",as:r.as,"as-child":r.asChild,tabindex:e(s).focusable.value?e(s).highlightedElement.value?"-1":"0":"-1","aria-orientation":e(s).orientation.value,"aria-multiselectable":!!e(s).multiple.value,"data-orientation":e(s).orientation.value,onMousedown:i[0]||(i[0]=ke(n=>o.value=!0,["left"])),onFocus:i[1]||(i[1]=n=>{e(o)||e(s).onEnter(n)}),onKeydown:[i[2]||(i[2]=ge(n=>{e(s).orientation.value==="vertical"&&(n.key==="ArrowLeft"||n.key==="ArrowRight")||e(s).orientation.value==="horizontal"&&(n.key==="ArrowUp"||n.key==="ArrowDown")||(n.preventDefault(),e(s).focusable.value&&e(s).onKeydownNavigation(n))},["down","up","left","right","home","end"])),ge(e(s).onKeydownEnter,["enter"]),e(s).onKeydownTypeAhead]},{default:c(()=>[O(r.$slots,"default")]),_:3},8,["as","as-child","tabindex","aria-orientation","aria-multiselectable","data-orientation","onKeydown"])]),_:3}))}}),ws=ks,Ss=E({__name:"ListboxFilter",props:{modelValue:{type:String,required:!1},autoFocus:{type:Boolean,required:!1},disabled:{type:Boolean,required:!1},asChild:{type:Boolean,required:!1},as:{type:null,required:!1,default:"input"}},emits:["update:modelValue"],setup(a,{emit:t}){const s=a,r=Ze(s,"modelValue",t,{defaultValue:"",passive:s.modelValue===void 0}),i=Ke(),{primitiveElement:n,currentElement:l}=Ve(),f=M(()=>s.disabled||i.disabled.value||!1),p=S();return xt(()=>p.value=i.highlightedElement.value?.id),ce(()=>{i.focusable.value=!1,setTimeout(()=>{s.autoFocus&&l.value?.focus()},1)}),ye(()=>{i.focusable.value=!0}),(y,m)=>(u(),g(e(ne),{ref_key:"primitiveElement",ref:n,as:y.as,"as-child":y.asChild,value:e(r),disabled:f.value?"":void 0,"data-disabled":f.value?"":void 0,"aria-disabled":f.value??void 0,"aria-activedescendant":p.value,type:"text",onKeydown:[ge(ke(e(i).onKeydownNavigation,["prevent"]),["down","up","home","end"]),ge(e(i).onKeydownEnter,["enter"])],onInput:m[0]||(m[0]=b=>{r.value=b.target.value,e(i).highlightFirstItem()}),onCompositionstart:e(i).onCompositionStart,onCompositionend:e(i).onCompositionEnd},{default:c(()=>[O(y.$slots,"default",{modelValue:e(r)})]),_:3},8,["as","as-child","value","disabled","data-disabled","aria-disabled","aria-activedescendant","onKeydown","onCompositionstart","onCompositionend"]))}}),$s=Ss;const[Es,Ds]=be("ListboxGroup");var Rs=E({__name:"ListboxGroup",props:{asChild:{type:Boolean,required:!1},as:{type:null,required:!1}},setup(a){const t=a,s=Se(void 0,"reka-listbox-group");return Ds({id:s}),(o,r)=>(u(),g(e(ne),T({role:"group"},t,{"aria-labelledby":e(s)}),{default:c(()=>[O(o.$slots,"default")]),_:3},16,["aria-labelledby"]))}}),Bs=Rs,As=E({__name:"ListboxGroupLabel",props:{for:{type:String,required:!1},asChild:{type:Boolean,required:!1},as:{type:null,required:!1,default:"div"}},setup(a){const t=a,s=Es({id:""});return(o,r)=>(u(),g(e(ne),T(t,{id:e(s).id}),{default:c(()=>[O(o.$slots,"default")]),_:3},16,["id"]))}}),Fs=As;const Ps="listbox.select",[Ao,Os]=be("ListboxItem");var Is=E({__name:"ListboxItem",props:{value:{type:null,required:!0},disabled:{type:Boolean,required:!1},asChild:{type:Boolean,required:!1},as:{type:null,required:!1,default:"div"}},emits:["select"],setup(a,{emit:t}){const s=a,o=t,r=Se(void 0,"reka-listbox-item"),{CollectionItem:i}=je(),{forwardRef:n,currentElement:l}=Lt(),f=Ke(),p=M(()=>l.value===f.highlightedElement.value),y=M(()=>_s(f.modelValue.value,s.value,f.by)),m=M(()=>f.disabled.value||s.disabled);async function b(x){o("select",x),!x?.defaultPrevented&&!m.value&&x&&(f.onValueChange(s.value),f.changeHighlight(l.value))}function z(x){const A={originalEvent:x,value:s.value};Mt(Ps,b,A)}return Os({isSelected:y}),(x,A)=>(u(),g(e(i),{value:x.value},{default:c(()=>[Ct([p.value,y.value],()=>v(e(ne),T({id:e(r)},x.$attrs,{ref:e(n),role:"option",tabindex:e(f).focusable.value?p.value?"0":"-1":-1,"aria-selected":y.value,as:x.as,"as-child":x.asChild,disabled:m.value?"":void 0,"data-disabled":m.value?"":void 0,"data-highlighted":p.value?"":void 0,"data-state":y.value?"checked":"unchecked",onClick:z,onKeydown:ge(ke(z,["prevent"]),["space"]),onPointermove:A[0]||(A[0]=()=>{e(f).highlightedElement.value!==e(l)&&e(f).highlightOnHover.value&&!e(f).focusable.value&&e(f).changeHighlight(e(l),!1)})}),{default:c(()=>[O(x.$slots,"default")]),_:3},16,["id","tabindex","aria-selected","as","as-child","disabled","data-disabled","data-highlighted","data-state","onKeydown"]),A,1)]),_:3},8,["value"]))}}),Ls=Is;const Ms={class:"flex items-center gap-2 truncate"},Ts={class:"truncate"},zs={key:0,class:"absolute top-full right-0 z-[9999] mt-1 w-[280px] rounded-md border bg-popover p-1 text-popover-foreground shadow-md"},Vs={class:"max-h-[200px] overflow-y-auto"},js=["aria-selected","onClick"],qs={class:"flex flex-1 flex-col gap-0.5 overflow-hidden text-left"},Ks={class:"truncate font-medium"},Hs={class:"truncate text-xs text-muted-foreground"},Us=["onClick"],Ns={key:1,class:"px-2 py-1.5 text-sm text-muted-foreground"},Gs={key:2,class:"my-1 h-px bg-border"},Ws={class:"space-y-2"},Xs={class:"flex gap-2"},Js={key:0,class:"text-sm text-destructive"},Qs={key:0,class:"space-y-2 rounded-md border p-3"},Ys={class:"flex items-center gap-2 text-sm"},Zs=["disabled"],ea={class:"flex-1 truncate font-mono text-xs text-muted-foreground"},ta={class:"max-h-[200px] overflow-y-auto"},sa={key:0,class:"py-2 text-center text-sm text-muted-foreground"},aa={key:1,class:"py-2 text-center text-sm text-muted-foreground"},oa=["onClick"],na={class:"truncate"},la={key:0},ra={key:1},ia=E({__name:"RepoSelector",setup(a,{expose:t}){const s=Je(),o=Te(),{repos:r,currentRepo:i,currentRepoId:n,selectRepo:l,addRepo:f,removeRepo:p}=we(),{showSuccess:y}=et(),m=S(!1),b=S(""),z=S(null);Bt(z,()=>{m.value&&(m.value=!1,b.value="")}),oe(()=>o.fullPath,()=>{m.value=!1,b.value=""});async function x(I){l(I),m.value=!1,b.value="";try{const R=(await $fetch(`/api/repos/${I}/prds`))?.[0];R?s.push(`/${I}/${R.slug}`):s.push("/")}catch{s.push("/")}}function A(){m.value=!1,P.value=!0}const V=S(!1),w=S(""),K=S(""),U=S([]),X=S(!1),D=M(()=>!!K.value&&K.value!==w.value);async function se(I){X.value=!0;try{const _=await $fetch("/api/browse",{query:{path:I||w.value||void 0}});w.value=_.current,K.value=_.parent,U.value=_.directories}catch{}finally{X.value=!1}}function j(){V.value=!0,se(B.value||void 0)}function ee(I){B.value=I,V.value=!1,N.value=null}function F(){D.value&&se(K.value)}const P=S(!1),B=S(""),N=S(null),Y=S(!1),ae=M(()=>{if(!b.value)return r.value;const I=b.value.toLowerCase();return r.value?.filter(_=>_.name.toLowerCase().includes(I)||_.path.toLowerCase().includes(I))??[]});async function Q(){if(!B.value.trim()){N.value="Please enter a repository path";return}Y.value=!0,N.value=null;try{const I=await f(B.value.trim());if(P.value=!1,B.value="",y("Repository added",I?.name||"Successfully added repository"),I?.id)try{const R=(await $fetch(`/api/repos/${I.id}/prds`))?.[0];R&&s.push(`/${I.id}/${R.slug}`)}catch{}}catch(I){if(I instanceof Error){const _=I;N.value=_.data?.message||I.message}else N.value="Failed to add repository"}finally{Y.value=!1}}async function De(I,_){I.stopPropagation(),await p(_)}function xe(){P.value=!1,B.value="",N.value=null}return t({openAddDialog:()=>{P.value=!0}}),(I,_)=>(u(),k("div",{ref_key:"dropdownRef",ref:z,class:"relative"},[d("button",{type:"button",class:"inline-flex h-8 w-[200px] items-center justify-between gap-2 rounded-md border border-input bg-background px-3 text-sm font-normal ring-offset-background transition-colors hover:bg-accent hover:text-accent-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2",onClick:_[0]||(_[0]=R=>m.value=!e(m))},[d("span",Ms,[v(e(os),{class:"size-4 shrink-0 text-muted-foreground"}),d("span",Ts,L(e(i)?.name??"Select repository"),1)]),v(e(Tt),{class:"size-4 shrink-0 opacity-50"})]),e(m)?(u(),k("div",zs,[kt(d("input",{"onUpdate:modelValue":_[1]||(_[1]=R=>ve(b)?b.value=R:null),type:"text",placeholder:"Search repositories...",class:"mb-1 h-9 w-full rounded-md border border-input bg-background px-3 text-sm placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring"},null,512),[[wt,e(b)]]),d("div",Vs,[e(ae)?.length?(u(!0),k(Z,{key:0},te(e(ae),R=>(u(),k("div",{key:R.id,class:"group flex w-full items-center gap-2 rounded-sm px-2 py-1.5 text-sm hover:bg-accent hover:text-accent-foreground cursor-pointer",role:"option","aria-selected":e(n)===R.id,onClick:le=>x(R.id)},[v(e(tt),{class:pe(["size-4 shrink-0",e(n)===R.id?"opacity-100":"opacity-0"])},null,8,["class"]),d("div",qs,[d("span",Ks,L(R.name),1),d("span",Hs,L(R.path),1)]),d("button",{type:"button",class:"ml-auto opacity-0 group-hover:opacity-100 p-1 hover:bg-destructive/10 rounded transition-opacity",title:"Remove repository",onClick:le=>De(le,R.id)},[v(e(is),{class:"size-3.5 text-destructive"})],8,Us)],8,js))),128)):(u(),k("div",Ns," No repositories found. ")),e(ae)?.length?(u(),k("div",Gs)):H("",!0),d("button",{type:"button",class:"flex w-full items-center gap-2 rounded-sm px-2 py-1.5 text-sm text-muted-foreground hover:bg-accent hover:text-accent-foreground",onClick:A},[v(e(zt),{class:"size-4"}),_[5]||(_[5]=d("span",null,"Add repository...",-1))])])])):H("",!0),v(e(Vt),{open:e(P),"onUpdate:open":_[4]||(_[4]=R=>P.value=R)},{default:c(()=>[v(e(jt),{side:"right",class:"flex h-full flex-col",onEscapeKeyDown:xe},{default:c(()=>[v(e(qt),{class:"px-6"},{default:c(()=>[v(e(Kt),null,{default:c(()=>[..._[6]||(_[6]=[G("Add Repository",-1)])]),_:1}),v(e(Ht),null,{default:c(()=>[..._[7]||(_[7]=[G(" Enter the absolute path to a repository containing PRD documents. ",-1)])]),_:1})]),_:1}),d("form",{id:"add-repo-form",class:"min-h-0 flex-1 space-y-4 overflow-y-auto px-6",onSubmit:ke(Q,["prevent"])},[d("div",Ws,[_[8]||(_[8]=d("label",{for:"repo-path",class:"text-sm font-medium"}," Repository Path ",-1)),d("div",Xs,[v(e(ns),{id:"repo-path",modelValue:e(B),"onUpdate:modelValue":_[2]||(_[2]=R=>ve(B)?B.value=R:null),placeholder:"/path/to/your/project","aria-invalid":!!e(N),disabled:e(Y),class:"flex-1"},null,8,["modelValue","aria-invalid","disabled"]),v(e(de),{type:"button",variant:"outline",size:"icon",disabled:e(Y),onClick:j},{default:c(()=>[v(e(Me),{class:"size-4"})]),_:1},8,["disabled"])]),e(N)?(u(),k("p",Js,L(e(N)),1)):H("",!0),_[9]||(_[9]=d("p",{class:"text-xs text-muted-foreground"},[G(" The repository should contain a "),d("code",{class:"rounded bg-muted px-1"},"docs/prd/"),G(" directory with markdown files. ")],-1))]),e(V)?(u(),k("div",Qs,[d("div",Ys,[d("button",{type:"button",class:"inline-flex items-center gap-1 rounded px-1.5 py-0.5 text-xs text-muted-foreground hover:bg-accent hover:text-foreground disabled:pointer-events-none disabled:opacity-50",disabled:!e(D),onClick:F},[v(e(Ut),{class:"size-3"}),_[10]||(_[10]=G(" Up ",-1))],8,Zs),d("span",ea,L(e(w)),1),v(e(de),{type:"button",variant:"ghost",size:"sm",class:"h-7 text-xs",onClick:_[3]||(_[3]=R=>ee(e(w)))},{default:c(()=>[..._[11]||(_[11]=[G(" Select ",-1)])]),_:1})]),d("div",ta,[e(X)?(u(),k("div",sa," Loading... ")):e(U).length?(u(!0),k(Z,{key:2},te(e(U),R=>(u(),k("button",{key:R.path,type:"button",class:"flex w-full items-center gap-2 rounded px-2 py-1 text-sm hover:bg-accent",onClick:le=>se(R.path)},[v(e(Me),{class:"size-4 text-muted-foreground"}),d("span",na,L(R.name),1)],8,oa))),128)):(u(),k("div",aa," No subdirectories "))])])):H("",!0)],32),v(e(Nt),{class:"px-6 pb-6"},{default:c(()=>[v(e(de),{type:"button",variant:"outline",disabled:e(Y),onClick:xe},{default:c(()=>[..._[12]||(_[12]=[G(" Cancel ",-1)])]),_:1},8,["disabled"]),v(e(de),{type:"submit",form:"add-repo-form",disabled:e(Y)},{default:c(()=>[e(Y)?(u(),k("span",la,"Adding...")):(u(),k("span",ra,"Add Repository"))]),_:1},8,["disabled"])]),_:1})]),_:1})]),_:1},8,["open"])],512))}}),ua=Object.assign(ia,{__name:"LayoutRepoSelector"}),da={class:"flex h-full w-64 flex-col border-r border-border bg-background"},ca={class:"p-2"},pa={key:0,class:"flex items-center justify-center py-8"},fa={key:1,class:"px-2 py-8 text-center"},ma={key:2,class:"px-2 py-8 text-center"},va={key:3,class:"px-2 py-8 text-center"},ha={key:4,class:"space-y-1"},ga={class:"flex-1 truncate"},ya=E({__name:"Sidebar",setup(a){const t=Te(),{prds:s,prdsStatus:o,refreshPrds:r}=ze(),{currentRepoId:i}=we(),n=M(()=>t.params.prd);function l(f){return n.value===f}return(f,p)=>{const y=ls;return u(),k("aside",da,[v(e(Gt),{class:"flex-1"},{default:c(()=>[d("div",ca,[p[5]||(p[5]=d("h2",{class:"flex h-10 items-center px-2 text-sm font-medium text-muted-foreground"},"Documents",-1)),e(o)==="pending"?(u(),k("div",pa,[v(e(Wt),{class:"size-5 animate-spin text-muted-foreground"})])):e(i)?e(o)==="error"?(u(),k("div",ma,[v(e(Xt),{class:"mx-auto size-8 text-destructive/50"}),p[2]||(p[2]=d("p",{class:"mt-2 text-sm text-muted-foreground"}," Failed to load PRDs ",-1)),v(e(de),{variant:"ghost",size:"sm",class:"mt-2",onClick:e(r)},{default:c(()=>[v(e(st),{class:"mr-1 size-3"}),p[1]||(p[1]=G(" Retry ",-1))]),_:1},8,["onClick"])])):e(s)?.length?(u(),k("nav",ha,[(u(!0),k(Z,null,te(e(s),m=>(u(),g(y,{key:m.slug,to:`/${e(i)}/${m.slug}`,class:pe(["group flex items-center gap-3 rounded-md px-3 py-2 text-sm transition-colors",[l(m.slug)?"bg-accent text-accent-foreground":"text-muted-foreground hover:bg-accent/50 hover:text-foreground"]])},{default:c(()=>[v(e(Le),{class:"size-4 shrink-0"}),d("span",ga,L(m.name),1),m.hasState&&m.taskCount?(u(),g(e(Jt),{key:0,variant:"secondary",class:"shrink-0 text-xs"},{default:c(()=>[G(L(m.completedCount??0)+"/"+L(m.taskCount),1)]),_:2},1024)):H("",!0)]),_:2},1032,["to","class"]))),128))])):(u(),k("div",va,[v(e(Le),{class:"mx-auto size-8 text-muted-foreground/50"}),p[3]||(p[3]=d("p",{class:"mt-2 text-sm text-muted-foreground"}," No PRDs found ",-1)),p[4]||(p[4]=d("p",{class:"mt-1 text-xs text-muted-foreground/70"}," Add .md files to docs/prd/ ",-1))])):(u(),k("div",fa,[...p[0]||(p[0]=[d("p",{class:"text-sm text-muted-foreground"}," Select a repository to view PRDs ",-1)])]))])]),_:1})])}}}),_a=Object.assign(ya,{__name:"LayoutSidebar"}),ba=E({__name:"Command",props:{modelValue:{default:""},defaultValue:{},multiple:{type:Boolean},orientation:{},dir:{},disabled:{type:Boolean},selectionBehavior:{},highlightOnHover:{type:Boolean},by:{},asChild:{type:Boolean},as:{},name:{},required:{type:Boolean},class:{},defaultSearch:{default:""}},emits:["update:modelValue","highlight","entryFocus","leave"],setup(a,{emit:t}){const s=a,o=t,r=J(s,"class"),i=fe(r,o),n=S(new Map),l=S(new Map),{contains:f}=ds({sensitivity:"base"}),p=St({search:s.defaultSearch||"",filtered:{count:0,items:new Map,groups:new Set}});oe(()=>s.defaultSearch,m=>{m!==void 0&&(p.search=m)});function y(){if(!p.search){p.filtered.count=n.value.size;return}p.filtered.groups=new Set;let m=0;for(const[b,z]of n.value){const x=f(z,p.search);p.filtered.items.set(b,x?1:0),x&&m++}for(const[b,z]of l.value)for(const x of z)if(p.filtered.items.get(x)>0){p.filtered.groups.add(b);break}p.filtered.count=m}return oe(()=>p.search,()=>{y()}),oe(()=>n.value.size,()=>{p.search&&y()}),Ga({allItems:n,allGroups:l,filterState:p}),(m,b)=>(u(),g(e(Cs),T({"data-slot":"command"},e(i),{class:e(W)("bg-popover text-popover-foreground flex h-full w-full flex-col overflow-hidden rounded-md",s.class)}),{default:c(()=>[O(m.$slots,"default")]),_:3},16,["class"]))}}),xa=Object.assign(ba,{__name:"UiCommand"}),Ca=E({__name:"Dialog",props:{open:{type:Boolean},defaultOpen:{type:Boolean},modal:{type:Boolean}},emits:["update:open"],setup(a,{emit:t}){const r=fe(a,t);return(i,n)=>(u(),g(e(Qt),T({"data-slot":"dialog"},e(r)),{default:c(l=>[O(i.$slots,"default",Fe(Pe(l)))]),_:3},16))}}),dt=Object.assign(Ca,{__name:"UiDialog"}),ka=E({__name:"DialogClose",props:{asChild:{type:Boolean},as:{}},setup(a){const t=a;return(s,o)=>(u(),g(e(qe),T({"data-slot":"dialog-close"},t),{default:c(()=>[O(s.$slots,"default")]),_:3},16))}});Object.assign(ka,{__name:"UiDialogClose"});const wa=E({__name:"DialogOverlay",props:{forceMount:{type:Boolean},asChild:{type:Boolean},as:{},class:{}},setup(a){const t=a,s=J(t,"class");return(o,r)=>(u(),g(e(at),T({"data-slot":"dialog-overlay"},e(s),{class:e(W)("data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 fixed inset-0 z-50 bg-black/80",t.class)}),{default:c(()=>[O(o.$slots,"default")]),_:3},16,["class"]))}}),Sa=Object.assign(wa,{__name:"UiDialogOverlay"}),$a=E({inheritAttrs:!1,__name:"DialogContent",props:{forceMount:{type:Boolean},disableOutsidePointerEvents:{type:Boolean},asChild:{type:Boolean},as:{},class:{},showCloseButton:{type:Boolean,default:!0}},emits:["escapeKeyDown","pointerDownOutside","focusOutside","interactOutside","openAutoFocus","closeAutoFocus"],setup(a,{emit:t}){const s=a,o=t,r=J(s,"class"),i=fe(r,o);return(n,l)=>(u(),g(e(ot),null,{default:c(()=>[v(Sa),v(e(nt),T({"data-slot":"dialog-content"},{...n.$attrs,...e(i)},{class:e(W)("bg-background data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 fixed top-[50%] left-[50%] z-50 grid w-full max-w-[calc(100%-2rem)] translate-x-[-50%] translate-y-[-50%] gap-4 rounded-lg border p-6 shadow-lg duration-200 sm:max-w-lg",s.class)}),{default:c(()=>[O(n.$slots,"default"),a.showCloseButton?(u(),g(e(qe),{key:0,"data-slot":"dialog-close",class:"ring-offset-background focus:ring-ring data-[state=open]:bg-accent data-[state=open]:text-muted-foreground absolute top-4 right-4 rounded-xs opacity-70 transition-opacity hover:opacity-100 focus:ring-2 focus:ring-offset-2 focus:outline-hidden disabled:pointer-events-none [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4"},{default:c(()=>[v(e(lt)),l[0]||(l[0]=d("span",{class:"sr-only"},"Close",-1))]),_:1})):H("",!0)]),_:3},16,["class"])]),_:3}))}}),ct=Object.assign($a,{__name:"UiDialogContent"}),Ea=E({__name:"DialogDescription",props:{asChild:{type:Boolean},as:{},class:{}},setup(a){const t=a,s=J(t,"class"),o=$e(s);return(r,i)=>(u(),g(e(Yt),T({"data-slot":"dialog-description"},e(o),{class:e(W)("text-muted-foreground text-sm",t.class)}),{default:c(()=>[O(r.$slots,"default")]),_:3},16,["class"]))}}),pt=Object.assign(Ea,{__name:"UiDialogDescription"}),Da=E({__name:"DialogFooter",props:{class:{}},setup(a){const t=a;return(s,o)=>(u(),k("div",{"data-slot":"dialog-footer",class:pe(e(W)("flex flex-col-reverse gap-2 sm:flex-row sm:justify-end",t.class))},[O(s.$slots,"default")],2))}});Object.assign(Da,{__name:"UiDialogFooter"});const Ra=E({__name:"DialogHeader",props:{class:{}},setup(a){const t=a;return(s,o)=>(u(),k("div",{"data-slot":"dialog-header",class:pe(e(W)("flex flex-col gap-2 text-center sm:text-left",t.class))},[O(s.$slots,"default")],2))}}),ft=Object.assign(Ra,{__name:"UiDialogHeader"}),Ba=E({inheritAttrs:!1,__name:"DialogScrollContent",props:{forceMount:{type:Boolean},disableOutsidePointerEvents:{type:Boolean},asChild:{type:Boolean},as:{},class:{}},emits:["escapeKeyDown","pointerDownOutside","focusOutside","interactOutside","openAutoFocus","closeAutoFocus"],setup(a,{emit:t}){const s=a,o=t,r=J(s,"class"),i=fe(r,o);return(n,l)=>(u(),g(e(ot),null,{default:c(()=>[v(e(at),{class:"fixed inset-0 z-50 grid place-items-center overflow-y-auto bg-black/80 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0"},{default:c(()=>[v(e(nt),T({class:e(W)("relative z-50 grid w-full max-w-lg my-8 gap-4 border border-border bg-background p-6 shadow-lg duration-200 sm:rounded-lg md:w-full",s.class)},{...n.$attrs,...e(i)},{onPointerDownOutside:l[0]||(l[0]=f=>{const p=f.detail.originalEvent,y=p.target;(p.offsetX>y.clientWidth||p.offsetY>y.clientHeight)&&f.preventDefault()})}),{default:c(()=>[O(n.$slots,"default"),v(e(qe),{class:"absolute top-4 right-4 p-0.5 transition-colors rounded-md hover:bg-secondary"},{default:c(()=>[v(e(lt),{class:"w-4 h-4"}),l[1]||(l[1]=d("span",{class:"sr-only"},"Close",-1))]),_:1})]),_:3},16,["class"])]),_:3})]),_:3}))}});Object.assign(Ba,{__name:"UiDialogScrollContent"});const Aa=E({__name:"DialogTitle",props:{asChild:{type:Boolean},as:{},class:{}},setup(a){const t=a,s=J(t,"class"),o=$e(s);return(r,i)=>(u(),g(e(Zt),T({"data-slot":"dialog-title"},e(o),{class:e(W)("text-lg leading-none font-semibold",t.class)}),{default:c(()=>[O(r.$slots,"default")]),_:3},16,["class"]))}}),mt=Object.assign(Aa,{__name:"UiDialogTitle"}),Fa=E({__name:"DialogTrigger",props:{asChild:{type:Boolean},as:{}},setup(a){const t=a;return(s,o)=>(u(),g(e(es),T({"data-slot":"dialog-trigger"},t),{default:c(()=>[O(s.$slots,"default")]),_:3},16))}});Object.assign(Fa,{__name:"UiDialogTrigger"});const Pa=E({__name:"CommandDialog",props:{open:{type:Boolean},defaultOpen:{type:Boolean},modal:{type:Boolean},title:{default:"Command Palette"},description:{default:"Search for a command to run..."},defaultSearch:{default:""}},emits:["update:open"],setup(a,{emit:t}){const r=fe(a,t);return(i,n)=>(u(),g(e(dt),Fe(Pe(e(r))),{default:c(l=>[v(e(ct),{class:"overflow-hidden p-0","show-close-button":!1},{default:c(()=>[v(e(ft),{class:"sr-only"},{default:c(()=>[v(e(mt),null,{default:c(()=>[G(L(a.title),1)]),_:1}),v(e(pt),null,{default:c(()=>[G(L(a.description),1)]),_:1})]),_:1}),v(xa,{"default-search":a.defaultSearch},{default:c(()=>[O(i.$slots,"default",Fe(Pe(l)))]),_:2},1032,["default-search"])]),_:2},1024)]),_:3},16))}}),Oa=Object.assign(Pa,{__name:"UiCommandDialog"}),Ia=E({__name:"CommandEmpty",props:{asChild:{type:Boolean},as:{},class:{}},setup(a){const t=a,s=J(t,"class"),{filterState:o}=Ee(),r=M(()=>!!o.search&&o.filtered.count===0);return(i,n)=>r.value?(u(),g(e(ne),T({key:0,"data-slot":"command-empty"},e(s),{class:e(W)("py-6 text-center text-sm",t.class)}),{default:c(()=>[O(i.$slots,"default")]),_:3},16,["class"])):H("",!0)}}),La=Object.assign(Ia,{__name:"UiCommandEmpty"}),Ma=E({__name:"CommandGroup",props:{asChild:{type:Boolean},as:{},class:{},heading:{}},setup(a){const t=a,s=J(t,"class"),{allGroups:o,filterState:r}=Ee(),i=Se(),n=M(()=>r.search?r.filtered.groups.has(i):!0);return Xa({id:i}),ce(()=>{o.value.has(i)||o.value.set(i,new Set)}),ye(()=>{o.value.delete(i)}),(l,f)=>(u(),g(e(Bs),T(e(s),{id:e(i),"data-slot":"command-group",class:e(W)("text-foreground overflow-hidden p-1",t.class),hidden:n.value?void 0:!0}),{default:c(()=>[a.heading?(u(),g(e(Fs),{key:0,"data-slot":"command-group-heading",class:"px-2 py-1.5 text-xs font-medium text-muted-foreground"},{default:c(()=>[G(L(a.heading),1)]),_:1})):H("",!0),O(l.$slots,"default")]),_:3},16,["id","class","hidden"]))}}),Ae=Object.assign(Ma,{__name:"UiCommandGroup"}),Ta={"data-slot":"command-input-wrapper",class:"flex h-9 items-center gap-2 border-b px-3"},za=E({inheritAttrs:!1,__name:"CommandInput",props:{modelValue:{},autoFocus:{type:Boolean},disabled:{type:Boolean},asChild:{type:Boolean},as:{},class:{}},setup(a){const t=a,s=J(t,"class"),o=$e(s),{filterState:r}=Ee(),i=S(null);return ce(()=>{ue(()=>{const n=i.value?.$el;if(n&&r.search){const l=r.search.length;n.setSelectionRange(l,l)}})}),(n,l)=>(u(),k("div",Ta,[v(e(rs),{class:"size-4 shrink-0 opacity-50"}),v(e($s),T({ref_key:"inputRef",ref:i},{...e(o),...n.$attrs},{modelValue:e(r).search,"onUpdate:modelValue":l[0]||(l[0]=f=>e(r).search=f),"data-slot":"command-input","auto-focus":"",class:e(W)("placeholder:text-muted-foreground flex h-10 w-full rounded-md bg-transparent py-3 text-sm outline-hidden disabled:cursor-not-allowed disabled:opacity-50",t.class)}),null,16,["modelValue","class"])]))}}),Va=Object.assign(za,{__name:"UiCommandInput"}),ja=E({__name:"CommandItem",props:{value:{},disabled:{type:Boolean},asChild:{type:Boolean},as:{},class:{}},emits:["select"],setup(a,{emit:t}){const s=a,o=t,r=J(s,"class"),i=fe(r,o),n=Se(),{filterState:l,allItems:f,allGroups:p}=Ee(),y=Wa(),m=M(()=>{if(l.search){const x=l.filtered.items.get(n);return x===void 0?!0:x>0}else return!0}),b=S(),z=At(b);return ce(()=>{if(!(z.value instanceof HTMLElement))return;f.value.set(n,z.value.textContent??s.value?.toString()??"");const x=y?.id;x&&(p.value.has(x)?p.value.get(x)?.add(n):p.value.set(x,new Set([n])))}),ye(()=>{f.value.delete(n)}),(x,A)=>m.value?(u(),g(e(Ls),T({key:0},e(i),{id:e(n),ref_key:"itemRef",ref:b,"data-slot":"command-item",class:e(W)("data-[highlighted]:bg-accent data-[highlighted]:text-accent-foreground [&_svg:not([class*='text-'])]:text-muted-foreground relative flex cursor-default items-center gap-2 rounded-sm px-2 py-1.5 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",s.class),onSelect:A[0]||(A[0]=()=>{e(l).search=""})}),{default:c(()=>[O(x.$slots,"default")]),_:3},16,["id","class"])):H("",!0)}}),ie=Object.assign(ja,{__name:"UiCommandItem"}),qa={role:"presentation"},Ka=E({__name:"CommandList",props:{asChild:{type:Boolean},as:{},class:{}},setup(a){const t=a,s=J(t,"class"),o=$e(s);return(r,i)=>(u(),g(e(ws),T({"data-slot":"command-list"},e(o),{class:e(W)("max-h-[300px] scroll-py-1 overflow-x-hidden overflow-y-auto scrollbar-none",t.class)}),{default:c(()=>[d("div",qa,[O(r.$slots,"default")])]),_:3},16,["class"]))}}),Ha=Object.assign(Ka,{__name:"UiCommandList"}),Ua=E({__name:"CommandSeparator",props:{orientation:{},decorative:{type:Boolean},asChild:{type:Boolean},as:{},class:{}},setup(a){const t=a,s=J(t,"class");return(o,r)=>(u(),g(e(ts),T({"data-slot":"command-separator"},e(s),{class:e(W)("bg-border -mx-1 h-px",t.class)}),{default:c(()=>[O(o.$slots,"default")]),_:3},16,["class"]))}}),Ge=Object.assign(Ua,{__name:"UiCommandSeparator"}),Na=E({__name:"CommandShortcut",props:{class:{}},setup(a){const t=a;return(s,o)=>(u(),k("span",{"data-slot":"command-shortcut",class:pe(e(W)("text-muted-foreground ml-auto text-xs tracking-widest",t.class))},[O(s.$slots,"default")],2))}});Object.assign(Na,{__name:"UiCommandShortcut"});const[Ee,Ga]=be("Command"),[Wa,Xa]=be("CommandGroup"),Ja={class:"flex-1"},Qa={class:"flex-1"},Ya={class:"ml-auto flex items-center gap-1"},Za={class:"inline-flex h-5 min-w-5 items-center justify-center rounded border border-border bg-muted px-1.5 font-mono text-[10px] text-muted-foreground"},eo={class:"flex-1"},to={class:"ml-auto flex items-center gap-1"},so={class:"inline-flex h-5 min-w-5 items-center justify-center rounded border border-border bg-muted px-1.5 font-mono text-[10px] text-muted-foreground"},ao={class:"ml-auto flex items-center gap-1"},oo={class:"inline-flex h-5 min-w-5 items-center justify-center rounded border border-border bg-muted px-1.5 font-mono text-[10px] text-muted-foreground"},no=E({__name:"CommandPalette",props:{open:{type:Boolean,default:!1},openModifiers:{},filter:{default:""},filterModifiers:{}},emits:$t(["openShortcutsHelp"],["update:open","update:filter"]),setup(a,{emit:t}){const s=Oe(a,"open"),o=Oe(a,"filter"),r=t,i=Je(),{themeMode:n,cycleThemeMode:l}=Qe(),{prds:f}=ze(),{repos:p,currentRepoId:y,selectRepo:m,refreshGitRepos:b}=we(),{showSuccess:z,showError:x}=et(),A=S("document");{const F=localStorage.getItem("prd-viewer-tab");(F==="document"||F==="board")&&(A.value=F)}function V(F){y.value&&(i.push(`/${y.value}/${F}`),s.value=!1)}function w(F){m(F),i.push("/"),s.value=!1}function K(){l(),s.value=!1}function U(){const F=A.value==="document"?"board":"document";A.value=F,localStorage.setItem("prd-viewer-tab",F),window.dispatchEvent(new StorageEvent("storage",{key:"prd-viewer-tab",newValue:F})),s.value=!1}function X(){s.value=!1,r("openShortcutsHelp")}const D=S(!1);async function se(){if(!(!y.value||D.value)){D.value=!0,s.value=!1;try{const F=await b(y.value);z(`Discovered ${F.discovered} git repositories`)}catch{x("Failed to refresh git repos")}finally{D.value=!1}}}oe(s,F=>{F||(o.value="")});const j=M(()=>navigator.platform.toUpperCase().indexOf("MAC")>=0),ee=M(()=>j.value?"⌘":"Ctrl");return(F,P)=>(u(),g(e(Oa),{open:s.value,"onUpdate:open":P[0]||(P[0]=B=>s.value=B),"default-search":o.value},{default:c(()=>[v(e(Va),{placeholder:"Type a command or search..."}),v(e(Ha),null,{default:c(()=>[v(e(La),null,{default:c(()=>[...P[1]||(P[1]=[G("No results found.",-1)])]),_:1}),e(f)?.length?(u(),g(e(Ae),{key:0,heading:"Documents"},{default:c(()=>[(u(!0),k(Z,null,te(e(f),B=>(u(),g(e(ie),{key:B.slug,value:`PRD: ${B.name}`,onSelect:N=>V(B.slug)},{default:c(()=>[v(e(Le),{class:"size-4"}),d("span",null,L(B.name),1)]),_:2},1032,["value","onSelect"]))),128))]),_:1})):H("",!0),e(f)?.length&&e(p)?.length?(u(),g(e(Ge),{key:1})):H("",!0),e(p)?.length?(u(),g(e(Ae),{key:2,heading:"Repositories"},{default:c(()=>[(u(!0),k(Z,null,te(e(p),B=>(u(),g(e(ie),{key:B.id,value:`repo-${B.id} ${B.name}`,onSelect:N=>w(B.id)},{default:c(()=>[v(e(Me),{class:"size-4"}),d("span",Ja,L(B.name),1),B.id===e(y)?(u(),g(e(tt),{key:0,class:"size-4 text-primary"})):H("",!0)]),_:2},1032,["value","onSelect"]))),128))]),_:1})):H("",!0),e(p)?.length?(u(),g(e(Ge),{key:3})):H("",!0),v(e(Ae),{heading:"Actions"},{default:c(()=>[v(e(ie),{value:"cycle-theme light dark system",onSelect:K},{default:c(()=>[e(n)==="system"?(u(),g(e(rt),{key:0,class:"size-4"})):e(n)==="light"?(u(),g(e(ut),{key:1,class:"size-4"})):(u(),g(e(it),{key:2,class:"size-4"})),d("span",Qa,"Cycle theme mode ("+L(e(n))+")",1),d("div",Ya,[d("kbd",Za,L(e(ee)),1),P[2]||(P[2]=d("kbd",{class:"inline-flex h-5 min-w-5 items-center justify-center rounded border border-border bg-muted px-1.5 font-mono text-[10px] text-muted-foreground"},".",-1))])]),_:1}),v(e(ie),{value:`switch-tab ${e(A)==="document"?"task board":"document"}`,onSelect:U},{default:c(()=>[v(e(ss),{class:"size-4"}),d("span",eo,"Switch to "+L(e(A)==="document"?"Task Board":"Document"),1),d("div",to,[d("kbd",so,L(e(ee)),1),P[3]||(P[3]=d("kbd",{class:"inline-flex h-5 min-w-5 items-center justify-center rounded border border-border bg-muted px-1.5 font-mono text-[10px] text-muted-foreground"},"\\",-1))])]),_:1},8,["value"]),v(e(ie),{value:"keyboard shortcuts help",onSelect:X},{default:c(()=>[v(e(as),{class:"size-4"}),P[5]||(P[5]=d("span",{class:"flex-1"},"Keyboard shortcuts",-1)),d("div",ao,[d("kbd",oo,L(e(ee)),1),P[4]||(P[4]=d("kbd",{class:"inline-flex h-5 min-w-5 items-center justify-center rounded border border-border bg-muted px-1.5 font-mono text-[10px] text-muted-foreground"},"/",-1))])]),_:1}),e(y)?(u(),g(e(ie),{key:0,value:"refresh git repos rescan discover",disabled:e(D),onSelect:se},{default:c(()=>[v(e(st),{class:pe(["size-4",{"animate-spin":e(D)}])},null,8,["class"]),P[6]||(P[6]=d("span",{class:"flex-1"},"Refresh git repos",-1))]),_:1},8,["disabled"])):H("",!0)]),_:1})]),_:1})]),_:1},8,["open","default-search"]))}}),lo=Object.assign(no,{__name:"CommandPalette"}),ro={class:"space-y-6"},io={class:"mb-3 text-sm font-medium text-muted-foreground"},uo={class:"space-y-2"},co={class:"text-sm"},po={class:"flex items-center gap-1"},fo=E({__name:"ShortcutsHelp",props:{open:{type:Boolean,default:!1},openModifiers:{}},emits:["update:open"],setup(a){const t=Oe(a,"open"),s=M(()=>navigator.platform.toUpperCase().indexOf("MAC")>=0),o=M(()=>s.value?"⌘":"Ctrl"),r=M(()=>[{name:"Navigation",shortcuts:[{keys:[o.value,"K"],description:"Open command palette"},{keys:[o.value,"J"],description:"Quick jump to document"}]},{name:"Actions",shortcuts:[{keys:[o.value,"."],description:"Cycle theme mode (light/dark/system)"},{keys:[o.value,"\\"],description:"Switch between Document and Task Board"},{keys:[o.value,","],description:"Add repository"}]},{name:"Help",shortcuts:[{keys:[o.value,"/"],description:"Show this help"},{keys:["Esc"],description:"Close dialog or palette"}]}]);return(i,n)=>(u(),g(e(dt),{open:t.value,"onUpdate:open":n[0]||(n[0]=l=>t.value=l)},{default:c(()=>[v(e(ct),{class:"max-w-md"},{default:c(()=>[v(e(ft),null,{default:c(()=>[v(e(mt),null,{default:c(()=>[...n[1]||(n[1]=[G("Keyboard Shortcuts",-1)])]),_:1}),v(e(pt),null,{default:c(()=>[...n[2]||(n[2]=[G(" Quick reference for available keyboard shortcuts. ",-1)])]),_:1})]),_:1}),d("div",ro,[(u(!0),k(Z,null,te(e(r),l=>(u(),k("div",{key:l.name},[d("h3",io,L(l.name),1),d("div",uo,[(u(!0),k(Z,null,te(l.shortcuts,f=>(u(),k("div",{key:f.description,class:"flex items-center justify-between"},[d("span",co,L(f.description),1),d("div",po,[(u(!0),k(Z,null,te(f.keys,(p,y)=>(u(),k("kbd",{key:y,class:"inline-flex h-5 min-w-5 items-center justify-center rounded border border-border bg-muted px-1.5 font-mono text-xs font-medium text-muted-foreground"},L(p),1))),128))])]))),128))])]))),128))])]),_:1})]),_:1},8,["open"]))}}),mo=Object.assign(fo,{__name:"ShortcutsHelp"});function vt(){const a=document.activeElement;if(!a)return!1;const t=a.tagName.toLowerCase();return t==="input"||t==="textarea"||a.getAttribute("contenteditable")==="true"||a.getAttribute("role")==="textbox"}function vo(a){const t=a.toLowerCase().split("+"),s=t.pop()||"";return{meta:t.includes("meta"),ctrl:t.includes("ctrl"),shift:t.includes("shift"),alt:t.includes("alt"),key:s}}function ho(a,t){const s=a.key.toLowerCase();let o=s===t.key;return t.key==="\\"&&(o=s==="\\"||a.code==="Backslash"),t.key==="/"&&(o=s==="/"||a.code==="Slash"),t.key==="."&&(o=s==="."||a.code==="Period"),t.key===","&&(o=s===","||a.code==="Comma"),o&&a.metaKey===t.meta&&a.ctrlKey===t.ctrl&&a.shiftKey===t.shift&&a.altKey===t.alt}const Ce=[];let We=!1;function go(a){for(const t of Ce)if(ho(a,t.combo)){if(!t.allowInInput&&vt())continue;a.preventDefault(),a.stopPropagation(),t.handler();return}}function Xe(){We||(document.addEventListener("keydown",go,{capture:!0}),We=!0)}function yo(){ce(()=>{Xe()});function a(s,o,r){const n={combo:vo(s),handler:o,allowInInput:r?.allowInInput??!1};Ce.push(n),Xe(),ye(()=>{const l=Ce.indexOf(n);l>-1&&Ce.splice(l,1)})}function t(){return vt()}return{onShortcut:a,isInputFocused:t}}function _o(a){const t=S(null),s=S(!1),o=S(null);function r(){if(!t.value)try{const n=new EventSource("/api/watch");n.onopen=()=>{s.value=!0,o.value=null},n.onmessage=l=>{try{const f=JSON.parse(l.data);a(f)}catch{}},n.onerror=()=>{s.value=!1,o.value="Connection lost",setTimeout(()=>{t.value===n&&(t.value=null,r())},5e3)},t.value=n}catch{o.value="Failed to connect"}}function i(){t.value&&(t.value.close(),t.value=null,s.value=!1)}return ce(()=>{r()}),ye(()=>{i()}),{isConnected:s,error:o,connect:r,disconnect:i}}const bo={class:"min-h-screen bg-background text-foreground"},xo={class:"fixed top-0 left-0 right-0 z-50 h-14 border-b border-border bg-background/95 backdrop-blur-sm supports-[backdrop-filter]:bg-background/80"},Co={class:"flex h-full items-center justify-between px-4 md:px-6"},ko={class:"flex items-center gap-3"},wo={class:"flex h-screen pt-14"},So={class:"flex-1 overflow-auto"},Fo=E({__name:"default",setup(a){const{themeMode:t,cycleThemeMode:s}=Qe(),{refreshPrds:o}=ze(),{currentRepoId:r}=we(),i=Te(),n=S(null);Et("fileChangeEvent",n);const l=S(!1),f=S(""),p=S(!1),y=S(null),{onShortcut:m}=yo();m("Meta+k",()=>{l.value=!0}),m("Ctrl+k",()=>{l.value=!0}),m("Meta+j",()=>{f.value="PRD: ",l.value=!0}),m("Ctrl+j",()=>{f.value="PRD: ",l.value=!0});function b(){const w=(localStorage.getItem("prd-viewer-tab")||"document")==="document"?"board":"document";localStorage.setItem("prd-viewer-tab",w),window.dispatchEvent(new StorageEvent("storage",{key:"prd-viewer-tab",newValue:w}))}m("Meta+\\",b),m("Ctrl+\\",b);function z(){s()}m("Meta+.",z),m("Ctrl+.",z);function x(){y.value?.openAddDialog()}m("Meta+,",x),m("Ctrl+,",x);function A(){p.value=!0}return m("Meta+/",A),m("Ctrl+/",A),m("Meta+Shift+/",A),m("Ctrl+Shift+/",A),_o(V=>{if(V.type==="connected"||!V.category)return;const w=V.category;if(V.repoId!==r.value)return;(w==="prd"||w==="tasks")&&o();const K=i.params.prd;if(K){const U=w==="prd"&&V.path?.includes(`/${K}.`),X=(w==="tasks"||w==="progress")&&V.path?.includes(`/${K}/`);(U||X)&&(n.value={category:w,path:V.path,timestamp:Date.now()})}}),(V,w)=>{const K=ua,U=Dt,X=_a;return u(),k("div",bo,[v(lo,{open:e(l),"onUpdate:open":w[0]||(w[0]=D=>ve(l)?l.value=D:null),filter:e(f),"onUpdate:filter":w[1]||(w[1]=D=>ve(f)?f.value=D:null),onOpenShortcutsHelp:A},null,8,["open","filter"]),v(mo,{open:e(p),"onUpdate:open":w[2]||(w[2]=D=>ve(p)?p.value=D:null)},null,8,["open"]),d("header",xo,[d("div",Co,[w[5]||(w[5]=d("div",{class:"flex items-center gap-4"},[d("h1",{class:"text-lg font-semibold tracking-tight"}," PRD Viewer ")],-1)),v(U,null,{fallback:c(()=>[...w[4]||(w[4]=[d("div",{class:"flex items-center gap-3"},[d("div",{class:"h-8 w-[200px] animate-pulse rounded-md bg-muted"}),d("div",{class:"size-9 animate-pulse rounded-md bg-muted"})],-1)])]),default:c(()=>[d("div",ko,[v(K,{ref_key:"repoSelectorRef",ref:y},null,512),v(e(de),{variant:"ghost",size:"icon",class:"size-9",onClick:z},{default:c(()=>[e(t)==="system"?(u(),g(e(rt),{key:0,class:"size-4"})):e(t)==="light"?(u(),g(e(ut),{key:1,class:"size-4"})):(u(),g(e(it),{key:2,class:"size-4"})),w[3]||(w[3]=d("span",{class:"sr-only"},"Cycle theme mode",-1))]),_:1})])]),_:1})])]),d("div",wo,[v(U,null,{fallback:c(()=>[...w[6]||(w[6]=[d("aside",{class:"flex h-full w-64 flex-col border-r border-border bg-background"},[d("div",{class:"flex h-12 items-center border-b border-border px-4"},[d("div",{class:"h-4 w-20 animate-pulse rounded bg-muted"})]),d("div",{class:"flex-1 p-2 space-y-2"},[d("div",{class:"h-9 animate-pulse rounded-md bg-muted"}),d("div",{class:"h-9 animate-pulse rounded-md bg-muted"}),d("div",{class:"h-9 animate-pulse rounded-md bg-muted"})])],-1)])]),default:c(()=>[v(X)]),_:1}),d("main",So,[O(V.$slots,"default")])])])}}});export{Fo as default};
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"id":"7fda7510-94bc-443a-a338-a8d2af142ed9","timestamp":1771867324818,"prerendered":[]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"default-DGOIWqjD.mjs","sources":["../../../../node_modules/.cache/nuxt/.nuxt/dist/server/_nuxt/default-DGOIWqjD.js"],"names":[],"mappings":"","x_google_ignoreList":[0]}
|