@hienlh/ppm 0.5.20 → 0.6.0
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/CHANGELOG.md +6 -0
- package/bun.lock +45 -0
- package/dist/web/assets/{api-client-BxCvlogn.js → api-client-ANLU-Irq.js} +1 -1
- package/dist/web/assets/chat-tab-C24nbKz1.js +7 -0
- package/dist/web/assets/{code-editor-0YVgeS1c.js → code-editor-DjIL6ta3.js} +1 -1
- package/dist/web/assets/{diff-viewer-CtEmKn4e.js → diff-viewer-BnvcXY3g.js} +1 -1
- package/dist/web/assets/{git-graph-DycoowxO.js → git-graph-iAf_zaqe.js} +1 -1
- package/dist/web/assets/index-BwLVvoev.js +21 -0
- package/dist/web/assets/index-CP_2zE5O.css +2 -0
- package/dist/web/assets/{input-Bzyi1GeB.js → input-DV4tynJq.js} +1 -1
- package/dist/web/assets/{jsx-runtime-Bzk8w7Zh.js → jsx-runtime-B4BJKQ1u.js} +1 -1
- package/dist/web/assets/{markdown-renderer-LHjvxp5Q.js → markdown-renderer-CIfiE3o8.js} +2 -2
- package/dist/web/assets/react-WvgCEYPV.js +1 -0
- package/dist/web/assets/{rotate-ccw-ZqeedZLA.js → rotate-ccw-BesidNnx.js} +1 -1
- package/dist/web/assets/settings-store-BGF8--S9.js +1 -0
- package/dist/web/assets/settings-tab-B_QwULcp.js +1 -0
- package/dist/web/assets/sqlite-viewer-DpGb3i2g.js +16 -0
- package/dist/web/assets/tab-store-L0a7ao4c.js +1 -0
- package/dist/web/assets/{terminal-tab-B2QEABNU.js → terminal-tab-4-DINw_B.js} +1 -1
- package/dist/web/assets/{use-monaco-theme-BFv4d2_j.js → use-monaco-theme-RFoGvnp0.js} +2 -2
- package/dist/web/index.html +9 -8
- package/dist/web/sw.js +1 -1
- package/docs/codebase-summary.md +96 -61
- package/docs/deployment-guide.md +16 -14
- package/docs/design-guidelines.md +5 -2
- package/docs/project-overview-pdr.md +20 -17
- package/docs/project-roadmap.md +35 -23
- package/docs/system-architecture.md +27 -18
- package/package.json +4 -1
- package/src/cli/commands/init.ts +7 -2
- package/src/cli/commands/restart.ts +6 -0
- package/src/index.ts +9 -1
- package/src/providers/claude-agent-sdk.ts +59 -28
- package/src/server/index.ts +10 -2
- package/src/server/routes/chat.ts +19 -0
- package/src/server/routes/project-scoped.ts +2 -0
- package/src/server/routes/sqlite.ts +75 -0
- package/src/server/routes/tunnel.ts +17 -2
- package/src/server/ws/chat.ts +33 -1
- package/src/services/config.service.ts +182 -58
- package/src/services/db.service.ts +303 -0
- package/src/services/push-notification.service.ts +23 -37
- package/src/services/session-log.service.ts +12 -24
- package/src/services/sqlite.service.ts +144 -0
- package/src/web/components/chat/chat-history-bar.tsx +68 -8
- package/src/web/components/chat/chat-tab.tsx +10 -1
- package/src/web/components/chat/file-picker.tsx +1 -1
- package/src/web/components/chat/slash-command-picker.tsx +1 -1
- package/src/web/components/explorer/file-tree.tsx +3 -1
- package/src/web/components/layout/draggable-tab.tsx +50 -4
- package/src/web/components/layout/editor-panel.tsx +1 -0
- package/src/web/components/layout/mobile-nav.tsx +2 -2
- package/src/web/components/layout/project-bar.tsx +40 -17
- package/src/web/components/layout/tab-bar.tsx +16 -1
- package/src/web/components/layout/tab-content.tsx +5 -0
- package/src/web/components/sqlite/sqlite-data-grid.tsx +165 -0
- package/src/web/components/sqlite/sqlite-query-editor.tsx +97 -0
- package/src/web/components/sqlite/sqlite-table-list.tsx +48 -0
- package/src/web/components/sqlite/sqlite-viewer.tsx +117 -0
- package/src/web/components/sqlite/use-sqlite.ts +97 -0
- package/src/web/hooks/use-chat.ts +12 -0
- package/src/web/stores/tab-store.ts +1 -0
- package/dist/web/assets/chat-tab-D_LO6cRM.js +0 -7
- package/dist/web/assets/index-82E_pIrH.css +0 -2
- package/dist/web/assets/index-y49eIXuR.js +0 -21
- package/dist/web/assets/settings-store-DikslxSJ.js +0 -1
- package/dist/web/assets/settings-tab-Dt9Sv1zx.js +0 -1
- package/dist/web/assets/tab-store-BNgVKR5w.js +0 -1
- /package/dist/web/assets/{utils-EM9hC5pN.js → utils-C2KxHr1H.js} +0 -0
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
import { useState, useCallback } from "react";
|
|
2
|
+
import CodeMirror from "@uiw/react-codemirror";
|
|
3
|
+
import { sql, SQLite } from "@codemirror/lang-sql";
|
|
4
|
+
import { Play, Loader2 } from "lucide-react";
|
|
5
|
+
import type { QueryResult } from "./use-sqlite";
|
|
6
|
+
|
|
7
|
+
interface Props {
|
|
8
|
+
onExecute: (sql: string) => void;
|
|
9
|
+
result: QueryResult | null;
|
|
10
|
+
error: string | null;
|
|
11
|
+
loading: boolean;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export function SqliteQueryEditor({ onExecute, result, error, loading }: Props) {
|
|
15
|
+
const [query, setQuery] = useState("SELECT * FROM ");
|
|
16
|
+
|
|
17
|
+
const handleExecute = useCallback(() => {
|
|
18
|
+
const trimmed = query.trim();
|
|
19
|
+
if (!trimmed) return;
|
|
20
|
+
onExecute(trimmed);
|
|
21
|
+
}, [query, onExecute]);
|
|
22
|
+
|
|
23
|
+
const handleKeyDown = useCallback((e: React.KeyboardEvent) => {
|
|
24
|
+
if ((e.metaKey || e.ctrlKey) && e.key === "Enter") {
|
|
25
|
+
e.preventDefault();
|
|
26
|
+
handleExecute();
|
|
27
|
+
}
|
|
28
|
+
}, [handleExecute]);
|
|
29
|
+
|
|
30
|
+
return (
|
|
31
|
+
<div className="flex flex-col h-full overflow-hidden">
|
|
32
|
+
{/* Editor area */}
|
|
33
|
+
<div className="flex items-start gap-1 border-b border-border bg-background" onKeyDown={handleKeyDown}>
|
|
34
|
+
<div className="flex-1 max-h-[120px] overflow-auto">
|
|
35
|
+
<CodeMirror
|
|
36
|
+
value={query}
|
|
37
|
+
onChange={setQuery}
|
|
38
|
+
extensions={[sql({ dialect: SQLite })]}
|
|
39
|
+
basicSetup={{ lineNumbers: false, foldGutter: false, highlightActiveLine: false }}
|
|
40
|
+
className="text-xs [&_.cm-editor]:!outline-none [&_.cm-scroller]:!overflow-auto"
|
|
41
|
+
/>
|
|
42
|
+
</div>
|
|
43
|
+
<button
|
|
44
|
+
type="button"
|
|
45
|
+
onClick={handleExecute}
|
|
46
|
+
disabled={loading}
|
|
47
|
+
className="shrink-0 m-1 p-1.5 rounded bg-primary text-primary-foreground hover:bg-primary/90 disabled:opacity-50 transition-colors"
|
|
48
|
+
title="Execute (Cmd+Enter)"
|
|
49
|
+
>
|
|
50
|
+
{loading ? <Loader2 className="size-3.5 animate-spin" /> : <Play className="size-3.5" />}
|
|
51
|
+
</button>
|
|
52
|
+
</div>
|
|
53
|
+
|
|
54
|
+
{/* Results area */}
|
|
55
|
+
<div className="flex-1 overflow-auto text-xs">
|
|
56
|
+
{error && (
|
|
57
|
+
<div className="px-3 py-2 text-destructive bg-destructive/5">{error}</div>
|
|
58
|
+
)}
|
|
59
|
+
|
|
60
|
+
{result && result.changeType === "modify" && (
|
|
61
|
+
<div className="px-3 py-2 text-green-500">
|
|
62
|
+
Query executed. {result.rowsAffected} row(s) affected.
|
|
63
|
+
</div>
|
|
64
|
+
)}
|
|
65
|
+
|
|
66
|
+
{result && result.changeType === "select" && result.rows.length > 0 && (
|
|
67
|
+
<table className="w-full border-collapse">
|
|
68
|
+
<thead className="sticky top-0 bg-muted">
|
|
69
|
+
<tr>
|
|
70
|
+
{result.columns.map((col) => (
|
|
71
|
+
<th key={col} className="px-2 py-1 text-left font-medium text-muted-foreground border-b border-border whitespace-nowrap">
|
|
72
|
+
{col}
|
|
73
|
+
</th>
|
|
74
|
+
))}
|
|
75
|
+
</tr>
|
|
76
|
+
</thead>
|
|
77
|
+
<tbody>
|
|
78
|
+
{result.rows.map((row, i) => (
|
|
79
|
+
<tr key={i} className="hover:bg-muted/30 border-b border-border/50">
|
|
80
|
+
{result.columns.map((col) => (
|
|
81
|
+
<td key={col} className="px-2 py-1 max-w-[300px] truncate" title={row[col] == null ? "NULL" : String(row[col])}>
|
|
82
|
+
{row[col] == null ? <span className="text-muted-foreground/40 italic">NULL</span> : String(row[col])}
|
|
83
|
+
</td>
|
|
84
|
+
))}
|
|
85
|
+
</tr>
|
|
86
|
+
))}
|
|
87
|
+
</tbody>
|
|
88
|
+
</table>
|
|
89
|
+
)}
|
|
90
|
+
|
|
91
|
+
{result && result.changeType === "select" && result.rows.length === 0 && (
|
|
92
|
+
<div className="px-3 py-2 text-muted-foreground">No results</div>
|
|
93
|
+
)}
|
|
94
|
+
</div>
|
|
95
|
+
</div>
|
|
96
|
+
);
|
|
97
|
+
}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import { Table, RefreshCw } from "lucide-react";
|
|
2
|
+
import type { TableInfo } from "./use-sqlite";
|
|
3
|
+
|
|
4
|
+
interface Props {
|
|
5
|
+
tables: TableInfo[];
|
|
6
|
+
selectedTable: string | null;
|
|
7
|
+
onSelect: (name: string) => void;
|
|
8
|
+
onRefresh: () => void;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
export function SqliteTableList({ tables, selectedTable, onSelect, onRefresh }: Props) {
|
|
12
|
+
return (
|
|
13
|
+
<div className="w-48 shrink-0 flex flex-col bg-background overflow-hidden">
|
|
14
|
+
<div className="flex items-center justify-between px-3 py-2 border-b border-border">
|
|
15
|
+
<span className="text-xs font-medium text-muted-foreground uppercase tracking-wider">Tables</span>
|
|
16
|
+
<button
|
|
17
|
+
type="button"
|
|
18
|
+
onClick={onRefresh}
|
|
19
|
+
className="text-muted-foreground hover:text-foreground transition-colors"
|
|
20
|
+
title="Refresh tables"
|
|
21
|
+
>
|
|
22
|
+
<RefreshCw className="size-3" />
|
|
23
|
+
</button>
|
|
24
|
+
</div>
|
|
25
|
+
<div className="flex-1 overflow-y-auto">
|
|
26
|
+
{tables.map((t) => (
|
|
27
|
+
<button
|
|
28
|
+
key={t.name}
|
|
29
|
+
type="button"
|
|
30
|
+
onClick={() => onSelect(t.name)}
|
|
31
|
+
className={`w-full flex items-center gap-2 px-3 py-1.5 text-left text-xs transition-colors ${
|
|
32
|
+
selectedTable === t.name
|
|
33
|
+
? "bg-muted text-foreground"
|
|
34
|
+
: "text-muted-foreground hover:bg-muted/50 hover:text-foreground"
|
|
35
|
+
}`}
|
|
36
|
+
>
|
|
37
|
+
<Table className="size-3 shrink-0" />
|
|
38
|
+
<span className="truncate flex-1">{t.name}</span>
|
|
39
|
+
<span className="text-[10px] opacity-60">{t.rowCount}</span>
|
|
40
|
+
</button>
|
|
41
|
+
))}
|
|
42
|
+
{tables.length === 0 && (
|
|
43
|
+
<p className="px-3 py-4 text-xs text-muted-foreground text-center">No tables found</p>
|
|
44
|
+
)}
|
|
45
|
+
</div>
|
|
46
|
+
</div>
|
|
47
|
+
);
|
|
48
|
+
}
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
import { useState } from "react";
|
|
2
|
+
import { Database, Loader2, AlertCircle } from "lucide-react";
|
|
3
|
+
import { useSqlite } from "./use-sqlite";
|
|
4
|
+
import { SqliteTableList } from "./sqlite-table-list";
|
|
5
|
+
import { SqliteDataGrid } from "./sqlite-data-grid";
|
|
6
|
+
import { SqliteQueryEditor } from "./sqlite-query-editor";
|
|
7
|
+
|
|
8
|
+
interface SqliteViewerProps {
|
|
9
|
+
metadata?: Record<string, unknown>;
|
|
10
|
+
tabId?: string;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export function SqliteViewer({ metadata }: SqliteViewerProps) {
|
|
14
|
+
const filePath = metadata?.filePath as string | undefined;
|
|
15
|
+
const projectName = metadata?.projectName as string | undefined;
|
|
16
|
+
const [queryPanelOpen, setQueryPanelOpen] = useState(false);
|
|
17
|
+
|
|
18
|
+
if (!filePath || !projectName) {
|
|
19
|
+
return (
|
|
20
|
+
<div className="flex items-center justify-center h-full text-text-secondary text-sm">
|
|
21
|
+
<Database className="size-5 mr-2" /> No database file selected.
|
|
22
|
+
</div>
|
|
23
|
+
);
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
return (
|
|
27
|
+
<SqliteViewerInner
|
|
28
|
+
projectName={projectName}
|
|
29
|
+
dbPath={filePath}
|
|
30
|
+
queryPanelOpen={queryPanelOpen}
|
|
31
|
+
onToggleQueryPanel={() => setQueryPanelOpen((v) => !v)}
|
|
32
|
+
/>
|
|
33
|
+
);
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
function SqliteViewerInner({
|
|
37
|
+
projectName, dbPath, queryPanelOpen, onToggleQueryPanel,
|
|
38
|
+
}: {
|
|
39
|
+
projectName: string; dbPath: string; queryPanelOpen: boolean; onToggleQueryPanel: () => void;
|
|
40
|
+
}) {
|
|
41
|
+
const sqlite = useSqlite(projectName, dbPath);
|
|
42
|
+
|
|
43
|
+
if (sqlite.error && sqlite.tables.length === 0) {
|
|
44
|
+
return (
|
|
45
|
+
<div className="flex flex-col items-center justify-center h-full gap-3 text-text-secondary">
|
|
46
|
+
<AlertCircle className="size-10 text-destructive" />
|
|
47
|
+
<p className="text-sm">{sqlite.error}</p>
|
|
48
|
+
</div>
|
|
49
|
+
);
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
if (sqlite.loading && sqlite.tables.length === 0) {
|
|
53
|
+
return (
|
|
54
|
+
<div className="flex items-center justify-center h-full gap-2 text-text-secondary">
|
|
55
|
+
<Loader2 className="size-5 animate-spin" />
|
|
56
|
+
<span className="text-sm">Loading database...</span>
|
|
57
|
+
</div>
|
|
58
|
+
);
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
return (
|
|
62
|
+
<div className="flex h-full w-full overflow-hidden">
|
|
63
|
+
{/* Left sidebar — table list */}
|
|
64
|
+
<SqliteTableList
|
|
65
|
+
tables={sqlite.tables}
|
|
66
|
+
selectedTable={sqlite.selectedTable}
|
|
67
|
+
onSelect={sqlite.selectTable}
|
|
68
|
+
onRefresh={sqlite.refreshTables}
|
|
69
|
+
/>
|
|
70
|
+
|
|
71
|
+
{/* Main content area */}
|
|
72
|
+
<div className="flex-1 flex flex-col overflow-hidden border-l border-border">
|
|
73
|
+
{/* Toolbar */}
|
|
74
|
+
<div className="flex items-center gap-2 px-3 py-1.5 border-b border-border bg-background shrink-0">
|
|
75
|
+
<Database className="size-3.5 text-muted-foreground" />
|
|
76
|
+
<span className="text-xs text-muted-foreground truncate">{dbPath}</span>
|
|
77
|
+
<span className="text-xs text-muted-foreground">
|
|
78
|
+
{sqlite.selectedTable && `/ ${sqlite.selectedTable}`}
|
|
79
|
+
</span>
|
|
80
|
+
<div className="ml-auto">
|
|
81
|
+
<button
|
|
82
|
+
type="button"
|
|
83
|
+
onClick={onToggleQueryPanel}
|
|
84
|
+
className={`px-2 py-1 rounded text-xs transition-colors ${queryPanelOpen ? "bg-muted text-foreground" : "text-muted-foreground hover:text-foreground"}`}
|
|
85
|
+
>
|
|
86
|
+
SQL
|
|
87
|
+
</button>
|
|
88
|
+
</div>
|
|
89
|
+
</div>
|
|
90
|
+
|
|
91
|
+
{/* Data grid */}
|
|
92
|
+
<div className={`flex-1 overflow-hidden ${queryPanelOpen ? "max-h-[60%]" : ""}`}>
|
|
93
|
+
<SqliteDataGrid
|
|
94
|
+
tableData={sqlite.tableData}
|
|
95
|
+
schema={sqlite.schema}
|
|
96
|
+
loading={sqlite.loading}
|
|
97
|
+
page={sqlite.page}
|
|
98
|
+
onPageChange={sqlite.setPage}
|
|
99
|
+
onCellUpdate={sqlite.updateCell}
|
|
100
|
+
/>
|
|
101
|
+
</div>
|
|
102
|
+
|
|
103
|
+
{/* Query editor (collapsible) */}
|
|
104
|
+
{queryPanelOpen && (
|
|
105
|
+
<div className="border-t border-border h-[40%] shrink-0">
|
|
106
|
+
<SqliteQueryEditor
|
|
107
|
+
onExecute={sqlite.executeQuery}
|
|
108
|
+
result={sqlite.queryResult}
|
|
109
|
+
error={sqlite.queryError}
|
|
110
|
+
loading={sqlite.queryLoading}
|
|
111
|
+
/>
|
|
112
|
+
</div>
|
|
113
|
+
)}
|
|
114
|
+
</div>
|
|
115
|
+
</div>
|
|
116
|
+
);
|
|
117
|
+
}
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
import { useState, useEffect, useCallback } from "react";
|
|
2
|
+
import { api, projectUrl } from "@/lib/api-client";
|
|
3
|
+
|
|
4
|
+
export interface TableInfo { name: string; rowCount: number }
|
|
5
|
+
export interface ColumnInfo { cid: number; name: string; type: string; notnull: boolean; pk: boolean; dflt_value: string | null }
|
|
6
|
+
export interface QueryResult { columns: string[]; rows: Record<string, unknown>[]; rowsAffected: number; changeType: "select" | "modify" }
|
|
7
|
+
interface TableData { columns: string[]; rows: Record<string, unknown>[]; total: number; page: number; limit: number }
|
|
8
|
+
|
|
9
|
+
export function useSqlite(projectName: string, dbPath: string) {
|
|
10
|
+
const [tables, setTables] = useState<TableInfo[]>([]);
|
|
11
|
+
const [selectedTable, setSelectedTable] = useState<string | null>(null);
|
|
12
|
+
const [tableData, setTableData] = useState<TableData | null>(null);
|
|
13
|
+
const [schema, setSchema] = useState<ColumnInfo[]>([]);
|
|
14
|
+
const [loading, setLoading] = useState(false);
|
|
15
|
+
const [error, setError] = useState<string | null>(null);
|
|
16
|
+
const [page, setPage] = useState(1);
|
|
17
|
+
const [queryResult, setQueryResult] = useState<QueryResult | null>(null);
|
|
18
|
+
const [queryError, setQueryError] = useState<string | null>(null);
|
|
19
|
+
const [queryLoading, setQueryLoading] = useState(false);
|
|
20
|
+
|
|
21
|
+
const base = `${projectUrl(projectName)}/sqlite`;
|
|
22
|
+
const qs = `path=${encodeURIComponent(dbPath)}`;
|
|
23
|
+
|
|
24
|
+
// Fetch tables on mount
|
|
25
|
+
const fetchTables = useCallback(async () => {
|
|
26
|
+
setLoading(true);
|
|
27
|
+
setError(null);
|
|
28
|
+
try {
|
|
29
|
+
const data = await api.get<TableInfo[]>(`${base}/tables?${qs}`);
|
|
30
|
+
setTables(data);
|
|
31
|
+
if (data.length > 0 && !selectedTable) setSelectedTable(data[0].name);
|
|
32
|
+
} catch (e) {
|
|
33
|
+
setError((e as Error).message);
|
|
34
|
+
} finally {
|
|
35
|
+
setLoading(false);
|
|
36
|
+
}
|
|
37
|
+
}, [base, qs]); // eslint-disable-line react-hooks/exhaustive-deps
|
|
38
|
+
|
|
39
|
+
useEffect(() => { fetchTables(); }, [fetchTables]);
|
|
40
|
+
|
|
41
|
+
// Fetch table data when selection or page changes
|
|
42
|
+
const fetchTableData = useCallback(async () => {
|
|
43
|
+
if (!selectedTable) return;
|
|
44
|
+
setLoading(true);
|
|
45
|
+
try {
|
|
46
|
+
const [data, cols] = await Promise.all([
|
|
47
|
+
api.get<TableData>(`${base}/data?${qs}&table=${encodeURIComponent(selectedTable)}&page=${page}&limit=100`),
|
|
48
|
+
api.get<ColumnInfo[]>(`${base}/schema?${qs}&table=${encodeURIComponent(selectedTable)}`),
|
|
49
|
+
]);
|
|
50
|
+
setTableData(data);
|
|
51
|
+
setSchema(cols);
|
|
52
|
+
} catch (e) {
|
|
53
|
+
setError((e as Error).message);
|
|
54
|
+
} finally {
|
|
55
|
+
setLoading(false);
|
|
56
|
+
}
|
|
57
|
+
}, [base, qs, selectedTable, page]); // eslint-disable-line react-hooks/exhaustive-deps
|
|
58
|
+
|
|
59
|
+
useEffect(() => { fetchTableData(); }, [fetchTableData]);
|
|
60
|
+
|
|
61
|
+
const selectTable = useCallback((name: string) => {
|
|
62
|
+
setSelectedTable(name);
|
|
63
|
+
setPage(1);
|
|
64
|
+
setQueryResult(null);
|
|
65
|
+
}, []);
|
|
66
|
+
|
|
67
|
+
const executeQuery = useCallback(async (sql: string) => {
|
|
68
|
+
setQueryLoading(true);
|
|
69
|
+
setQueryError(null);
|
|
70
|
+
try {
|
|
71
|
+
const result = await api.post<QueryResult>(`${base}/query`, { path: dbPath, sql });
|
|
72
|
+
setQueryResult(result);
|
|
73
|
+
if (result.changeType === "modify") fetchTableData(); // Refresh table after modification
|
|
74
|
+
} catch (e) {
|
|
75
|
+
setQueryError((e as Error).message);
|
|
76
|
+
} finally {
|
|
77
|
+
setQueryLoading(false);
|
|
78
|
+
}
|
|
79
|
+
}, [base, dbPath, fetchTableData]);
|
|
80
|
+
|
|
81
|
+
const updateCell = useCallback(async (rowid: number, column: string, value: unknown) => {
|
|
82
|
+
if (!selectedTable) return;
|
|
83
|
+
try {
|
|
84
|
+
await api.put(`${base}/cell`, { path: dbPath, table: selectedTable, rowid, column, value });
|
|
85
|
+
fetchTableData(); // Refresh
|
|
86
|
+
} catch (e) {
|
|
87
|
+
setError((e as Error).message);
|
|
88
|
+
}
|
|
89
|
+
}, [base, dbPath, selectedTable, fetchTableData]);
|
|
90
|
+
|
|
91
|
+
return {
|
|
92
|
+
tables, selectedTable, selectTable, tableData, schema,
|
|
93
|
+
loading, error, page, setPage,
|
|
94
|
+
queryResult, queryError, queryLoading, executeQuery,
|
|
95
|
+
updateCell, refreshTables: fetchTables, refreshData: fetchTableData,
|
|
96
|
+
};
|
|
97
|
+
}
|
|
@@ -23,6 +23,8 @@ interface UseChatReturn {
|
|
|
23
23
|
pendingApproval: ApprovalRequest | null;
|
|
24
24
|
/** Context window usage % from last completed query (0–100) */
|
|
25
25
|
contextWindowPct: number | null;
|
|
26
|
+
/** Updated session title from SDK summary (set after stream completes) */
|
|
27
|
+
sessionTitle: string | null;
|
|
26
28
|
sendMessage: (content: string) => void;
|
|
27
29
|
respondToApproval: (requestId: string, approved: boolean, data?: unknown) => void;
|
|
28
30
|
cancelStreaming: () => void;
|
|
@@ -42,6 +44,7 @@ export function useChat(sessionId: string | null, providerId = "claude-sdk", pro
|
|
|
42
44
|
const [thinkingWarningThreshold, setThinkingWarningThreshold] = useState(15);
|
|
43
45
|
const [pendingApproval, setPendingApproval] = useState<ApprovalRequest | null>(null);
|
|
44
46
|
const [contextWindowPct, setContextWindowPct] = useState<number | null>(null);
|
|
47
|
+
const [sessionTitle, setSessionTitle] = useState<string | null>(null);
|
|
45
48
|
const [isConnected, setIsConnected] = useState(false);
|
|
46
49
|
const streamingContentRef = useRef("");
|
|
47
50
|
const streamingEventsRef = useRef<ChatEvent[]>([]);
|
|
@@ -61,6 +64,12 @@ export function useChat(sessionId: string | null, providerId = "claude-sdk", pro
|
|
|
61
64
|
// Ignore keepalive pings
|
|
62
65
|
if ((data as any).type === "ping") return;
|
|
63
66
|
|
|
67
|
+
// Handle title updates from SDK summary
|
|
68
|
+
if ((data as any).type === "title_updated") {
|
|
69
|
+
setSessionTitle((data as any).title ?? null);
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
|
|
64
73
|
// Handle streaming status updates (connecting → streaming → idle)
|
|
65
74
|
if ((data as any).type === "streaming_status") {
|
|
66
75
|
const s = (data as any).status ?? "idle";
|
|
@@ -88,6 +97,7 @@ export function useChat(sessionId: string | null, providerId = "claude-sdk", pro
|
|
|
88
97
|
// Handle connected event (new session)
|
|
89
98
|
if ((data as any).type === "connected") {
|
|
90
99
|
setIsConnected(true);
|
|
100
|
+
if ((data as any).sessionTitle) setSessionTitle((data as any).sessionTitle);
|
|
91
101
|
return;
|
|
92
102
|
}
|
|
93
103
|
|
|
@@ -95,6 +105,7 @@ export function useChat(sessionId: string | null, providerId = "claude-sdk", pro
|
|
|
95
105
|
if ((data as any).type === "status") {
|
|
96
106
|
setIsConnected(true);
|
|
97
107
|
const status = data as any;
|
|
108
|
+
if (status.sessionTitle) setSessionTitle(status.sessionTitle);
|
|
98
109
|
if (status.isStreaming) {
|
|
99
110
|
isStreamingRef.current = true;
|
|
100
111
|
setIsStreaming(true);
|
|
@@ -475,6 +486,7 @@ export function useChat(sessionId: string | null, providerId = "claude-sdk", pro
|
|
|
475
486
|
thinkingWarningThreshold,
|
|
476
487
|
pendingApproval,
|
|
477
488
|
contextWindowPct,
|
|
489
|
+
sessionTitle,
|
|
478
490
|
sendMessage,
|
|
479
491
|
respondToApproval,
|
|
480
492
|
cancelStreaming,
|
|
@@ -1,7 +0,0 @@
|
|
|
1
|
-
const __vite__mapDeps=(i,m=__vite__mapDeps,d=(m.f||(m.f=["assets/api-client-BxCvlogn.js","assets/jsx-runtime-Bzk8w7Zh.js"])))=>i.map(i=>d[i]);
|
|
2
|
-
import{i as e,r as t,s as n,t as r}from"./jsx-runtime-Bzk8w7Zh.js";import"./input-Bzyi1GeB.js";import{a as i,t as a}from"./tab-store-BNgVKR5w.js";import{t as o}from"./rotate-ccw-ZqeedZLA.js";import{n as s}from"./settings-store-DikslxSJ.js";import{r as c,t as l}from"./utils-EM9hC5pN.js";import{i as u,r as d,t as f}from"./api-client-BxCvlogn.js";import{C as p,D as m,F as h,M as g,N as _,O as v,P as y,S as b,b as x,g as S,k as C,r as w,t as T,v as E,w as D,x as O}from"./index-y49eIXuR.js";import{t as k}from"./markdown-renderer-LHjvxp5Q.js";var A=t(`activity`,[[`path`,{d:`M22 12h-2.48a2 2 0 0 0-1.93 1.46l-2.35 8.36a.25.25 0 0 1-.48 0L9.24 2.18a.25.25 0 0 0-.48 0l-2.35 8.36A2 2 0 0 1 4.49 12H2`,key:`169zse`}]]),j=t(`arrow-up`,[[`path`,{d:`m5 12 7-7 7 7`,key:`hav0vg`}],[`path`,{d:`M12 19V5`,key:`x0mq9r`}]]),M=t(`bot`,[[`path`,{d:`M12 8V4H8`,key:`hb8ula`}],[`rect`,{width:`16`,height:`12`,x:`4`,y:`8`,rx:`2`,key:`enze0r`}],[`path`,{d:`M2 14h2`,key:`vft8re`}],[`path`,{d:`M20 14h2`,key:`4cs60a`}],[`path`,{d:`M15 13v2`,key:`1xurst`}],[`path`,{d:`M9 13v2`,key:`rq6x2g`}]]),N=t(`circle-x`,[[`circle`,{cx:`12`,cy:`12`,r:`10`,key:`1mglay`}],[`path`,{d:`m15 9-6 6`,key:`1uzhvr`}],[`path`,{d:`m9 9 6 6`,key:`z0biqf`}]]),P=t(`code`,[[`path`,{d:`m16 18 6-6-6-6`,key:`eg8j8`}],[`path`,{d:`m8 6-6 6 6 6`,key:`ppft3o`}]]),F=t(`columns-2`,[[`rect`,{width:`18`,height:`18`,x:`3`,y:`3`,rx:`2`,key:`afitv7`}],[`path`,{d:`M12 3v18`,key:`108xh3`}]]),I=t(`globe`,[[`circle`,{cx:`12`,cy:`12`,r:`10`,key:`1mglay`}],[`path`,{d:`M12 2a14.5 14.5 0 0 0 0 20 14.5 14.5 0 0 0 0-20`,key:`13o1zl`}],[`path`,{d:`M2 12h20`,key:`9i4pu4`}]]),L=t(`history`,[[`path`,{d:`M3 12a9 9 0 1 0 9-9 9.75 9.75 0 0 0-6.74 2.74L3 8`,key:`1357e3`}],[`path`,{d:`M3 3v5h5`,key:`1xhq8a`}],[`path`,{d:`M12 7v5l4 2`,key:`1fdv2h`}]]),R=t(`image`,[[`rect`,{width:`18`,height:`18`,x:`3`,y:`3`,rx:`2`,ry:`2`,key:`1m3agn`}],[`circle`,{cx:`9`,cy:`9`,r:`2`,key:`af1f0g`}],[`path`,{d:`m21 15-3.086-3.086a2 2 0 0 0-2.828 0L6 21`,key:`1xmnt7`}]]),z=t(`list-todo`,[[`path`,{d:`M13 5h8`,key:`a7qcls`}],[`path`,{d:`M13 12h8`,key:`h98zly`}],[`path`,{d:`M13 19h8`,key:`c3s6r1`}],[`path`,{d:`m3 17 2 2 4-4`,key:`1jhpwq`}],[`rect`,{x:`3`,y:`4`,width:`6`,height:`6`,rx:`1`,key:`cif1o7`}]]),ee=t(`paperclip`,[[`path`,{d:`m16 6-8.414 8.586a2 2 0 0 0 2.829 2.829l8.414-8.586a4 4 0 1 0-5.657-5.657l-8.379 8.551a6 6 0 1 0 8.485 8.485l8.379-8.551`,key:`1miecu`}]]),B=t(`settings-2`,[[`path`,{d:`M14 17H5`,key:`gfn3mx`}],[`path`,{d:`M19 7h-9`,key:`6i9tg`}],[`circle`,{cx:`17`,cy:`17`,r:`3`,key:`18b49y`}],[`circle`,{cx:`7`,cy:`7`,r:`3`,key:`dfmy0x`}]]),V=t(`shield-alert`,[[`path`,{d:`M20 13c0 5-3.5 7.5-7.66 8.95a1 1 0 0 1-.67-.01C7.5 20.5 4 18 4 13V6a1 1 0 0 1 1-1c2 0 4.5-1.2 6.24-2.72a1.17 1.17 0 0 1 1.52 0C14.51 3.81 17 5 19 5a1 1 0 0 1 1 1z`,key:`oel41y`}],[`path`,{d:`M12 8v4`,key:`1got3b`}],[`path`,{d:`M12 16h.01`,key:`1drbdi`}]]),te=t(`sparkles`,[[`path`,{d:`M11.017 2.814a1 1 0 0 1 1.966 0l1.051 5.558a2 2 0 0 0 1.594 1.594l5.558 1.051a1 1 0 0 1 0 1.966l-5.558 1.051a2 2 0 0 0-1.594 1.594l-1.051 5.558a1 1 0 0 1-1.966 0l-1.051-5.558a2 2 0 0 0-1.594-1.594l-5.558-1.051a1 1 0 0 1 0-1.966l5.558-1.051a2 2 0 0 0 1.594-1.594z`,key:`1s2grr`}],[`path`,{d:`M20 2v4`,key:`1rf3ol`}],[`path`,{d:`M22 4h-4`,key:`gwowj6`}],[`circle`,{cx:`4`,cy:`20`,r:`2`,key:`6kqj1y`}]]),H=t(`square`,[[`rect`,{width:`18`,height:`18`,x:`3`,y:`3`,rx:`2`,key:`afitv7`}]]),ne=t(`upload`,[[`path`,{d:`M12 3v12`,key:`1x0j5s`}],[`path`,{d:`m17 8-5-5-5 5`,key:`7q97r8`}],[`path`,{d:`M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4`,key:`ih7n3h`}]]),U=n(e(),1),re=3e4,ie=1e3,ae=class{ws=null;url;handlers=[];reconnectAttempts=0;reconnectTimer=null;intentionalClose=!1;constructor(e){this.url=e}connect(){this.intentionalClose=!1,this.cleanup();let e=window.location.protocol===`https:`?`wss:`:`ws:`,t=this.url.startsWith(`ws`)?this.url:`${e}//${window.location.host}${this.url}`;this.ws=new WebSocket(t),this.ws.onopen=()=>{this.reconnectAttempts=0;try{this.ws?.send(JSON.stringify({type:`ready`}))}catch{}},this.ws.onmessage=e=>{for(let t of this.handlers)t(e)},this.ws.onclose=()=>{this.intentionalClose||this.scheduleReconnect()},this.ws.onerror=()=>{this.ws?.close()}}disconnect(){this.intentionalClose=!0,this.cleanup(),this.reconnectTimer&&=(clearTimeout(this.reconnectTimer),null)}send(e){this.ws?.readyState===WebSocket.OPEN&&this.ws.send(e)}onMessage(e){return this.handlers.push(e),()=>{this.handlers=this.handlers.filter(t=>t!==e)}}get isConnected(){return this.ws?.readyState===WebSocket.OPEN}cleanup(){this.ws&&=(this.ws.onopen=null,this.ws.onclose=null,this.ws.onmessage=null,this.ws.onerror=null,(this.ws.readyState===WebSocket.OPEN||this.ws.readyState===WebSocket.CONNECTING)&&this.ws.close(),null)}scheduleReconnect(){let e=Math.min(ie*2**this.reconnectAttempts,re);this.reconnectAttempts++,this.reconnectTimer=setTimeout(()=>this.connect(),e)}};function W({url:e,onMessage:t,autoConnect:n=!0}){let r=(0,U.useRef)(null);return(0,U.useEffect)(()=>{let i=new ae(e);return r.current=i,t&&i.onMessage(t),n&&i.connect(),()=>{i.disconnect(),r.current=null}},[e,n]),{send:(0,U.useCallback)(e=>{r.current?.send(e)},[]),connect:(0,U.useCallback)(()=>{r.current?.connect()},[]),disconnect:(0,U.useCallback)(()=>{r.current?.disconnect()},[])}}function oe(e,t=`claude-sdk`,n=``){let[r,i]=(0,U.useState)([]),[a,o]=(0,U.useState)(!1),[s,c]=(0,U.useState)(!1),[l,f]=(0,U.useState)(`idle`),[p,m]=(0,U.useState)(0),[h,g]=(0,U.useState)(15),[_,v]=(0,U.useState)(null),[y,b]=(0,U.useState)(null),[x,S]=(0,U.useState)(!1),C=(0,U.useRef)(``),w=(0,U.useRef)([]),T=(0,U.useRef)(!1),E=(0,U.useRef)(null),D=(0,U.useRef)(()=>{}),O=(0,U.useRef)(null),k=(0,U.useCallback)(e=>{let t;try{t=JSON.parse(e.data)}catch{return}if(t.type===`ping`)return;if(t.type===`streaming_status`){let e=t.status??`idle`;if(f(e),m(e===`connecting`?t.elapsed??0:0),e===`connecting`){let e=t.effort,n=t.thinkingBudget,r=15;n&&n>0?r=Math.max(15,Math.round(n/500)):e===`high`?r=30:e===`low`&&(r=10),g(r)}return}if(t.type===`connected`){S(!0);return}if(t.type===`status`){S(!0);let e=t;e.isStreaming&&(T.current=!0,c(!0)),e.pendingApproval&&v({requestId:e.pendingApproval.requestId,tool:e.pendingApproval.tool,input:e.pendingApproval.input}),O.current?.();return}let n=(e,t)=>{let n=w.current.findIndex(e=>e.type===`tool_use`&&(e.tool===`Agent`||e.tool===`Task`)&&e.toolUseId===t);if(n===-1)return!1;let r=w.current[n];if(r.type!==`tool_use`)return!1;let i=[...r.children??[],e];return w.current[n]={...r,children:i},!0},r=()=>{let e=C.current,t=[...w.current];i(n=>{let r=n[n.length-1];return r?.role===`assistant`&&!r.id.startsWith(`final-`)?[...n.slice(0,-1),{...r,content:e,events:t}]:[...n,{id:`streaming-${Date.now()}`,role:`assistant`,content:e,events:t,timestamp:new Date().toISOString()}]})};switch(t.type){case`text`:{let e=t.parentToolUseId;if(e&&n(t,e)){r();break}C.current+=t.content,w.current.push(t),r();break}case`thinking`:{let e=t.parentToolUseId;if(e&&n(t,e)){r();break}w.current.push(t),r();break}case`tool_use`:{let e=t.parentToolUseId;if(e&&n(t,e)){r();break}w.current.push(t),r();break}case`tool_result`:{let e=t.parentToolUseId;if(e&&n(t,e)){r();break}w.current.push(t),r();break}case`approval_request`:w.current.push(t),v({requestId:t.requestId,tool:t.tool,input:t.input});break;case`error`:{w.current.push(t);let e=[...w.current];i(n=>{let r=n[n.length-1];return r?.role===`assistant`?[...n.slice(0,-1),{...r,events:e}]:[...n,{id:`error-${Date.now()}`,role:`system`,content:t.message,events:[t],timestamp:new Date().toISOString()}]}),T.current=!1,c(!1),f(`idle`);break}case`done`:{if(!T.current)break;t.contextWindowPct!=null&&b(t.contextWindowPct);let e=C.current,n=[...w.current];i(t=>{let r=t[t.length-1];return r?.role===`assistant`?[...t.slice(0,-1),{...r,id:`final-${Date.now()}`,content:e||r.content,events:n.length>0?n:r.events}]:t}),C.current=``,w.current=[],T.current=!1,c(!1),f(`idle`);break}}},[]),{send:A,connect:j}=W({url:e&&n?`/ws/project/${encodeURIComponent(n)}/chat/${e}`:``,onMessage:k,autoConnect:!!e&&!!n});D.current=A,(0,U.useEffect)(()=>{let r=!1;return c(!1),v(null),C.current=``,w.current=[],S(!1),e&&n?(o(!0),fetch(`${u(n)}/chat/sessions/${e}/messages?providerId=${t}`,{headers:{Authorization:`Bearer ${d()}`}}).then(e=>e.json()).then(e=>{r||T.current||(e.ok&&Array.isArray(e.data)&&e.data.length>0?i(e.data):i([]))}).catch(()=>{!r&&!T.current&&i([])}).finally(()=>{r||o(!1)})):i([]),()=>{r=!0}},[e,t,n]);let M=(0,U.useCallback)(e=>{if(e.trim()){if(T.current){let e=C.current,t=[...w.current];i(n=>{let r=n[n.length-1];return r?.role===`assistant`?[...n.slice(0,-1),{...r,id:`final-${Date.now()}`,content:e||r.content,events:t.length>0?t:r.events}]:n}),A(JSON.stringify({type:`cancel`}))}i(t=>[...t,{id:`user-${Date.now()}`,role:`user`,content:e,timestamp:new Date().toISOString()}]),C.current=``,w.current=[],E.current=null,T.current=!0,c(!0),f(`connecting`),v(null),A(JSON.stringify({type:`message`,content:e}))}},[A]),N=(0,U.useCallback)((e,t,n)=>{if(A(JSON.stringify({type:`approval_response`,requestId:e,approved:t,data:n})),t&&n){let t=w.current.find(t=>t.type===`approval_request`&&t.requestId===e&&t.tool===`AskUserQuestion`);if(t){let e=t.input;e&&typeof e==`object`&&(e.answers=n)}i(e=>[...e])}v(null)},[A]),P=(0,U.useCallback)(()=>{if(!T.current)return;A(JSON.stringify({type:`cancel`}));let e=C.current,t=[...w.current];i(n=>{let r=n[n.length-1];return r?.role===`assistant`?[...n.slice(0,-1),{...r,id:`final-${Date.now()}`,content:e||r.content,events:t.length>0?t:r.events}]:n}),C.current=``,w.current=[],E.current=null,T.current=!1,c(!1),v(null)},[A]),F=(0,U.useCallback)(()=>{S(!1),j(),O.current?.()},[j]),I=(0,U.useCallback)(()=>{!e||!n||(o(!0),fetch(`${u(n)}/chat/sessions/${e}/messages?providerId=${t}`,{headers:{Authorization:`Bearer ${d()}`}}).then(e=>e.json()).then(e=>{e.ok&&Array.isArray(e.data)&&e.data.length>0&&(i(e.data),C.current=``,w.current=[])}).catch(()=>{}).finally(()=>o(!1)))},[e,t,n]);return O.current=I,{messages:r,messagesLoading:a,isStreaming:s,streamingStatus:l,connectingElapsed:p,thinkingWarningThreshold:h,pendingApproval:_,contextWindowPct:y,sendMessage:M,respondToApproval:N,cancelStreaming:P,reconnect:F,refetchMessages:I,isConnected:x}}var se=3e4;function ce(e,t=`claude-sdk`){let[n,r]=(0,U.useState)({}),[i,a]=(0,U.useState)(!1),[o,s]=(0,U.useState)(null),c=(0,U.useRef)(null),l=(0,U.useCallback)((n=!1)=>{if(!e)return;a(!0);let i=n?`&refresh=1`:``;fetch(`${u(e)}/chat/usage?providerId=${t}${i}`,{headers:{Authorization:`Bearer ${d()}`}}).then(e=>e.json()).then(e=>{e.ok&&e.data&&(r(t=>({...t,...e.data})),e.data.lastFetchedAt&&s(e.data.lastFetchedAt))}).catch(()=>{}).finally(()=>a(!1))},[e,t]);return(0,U.useEffect)(()=>(l(),c.current=setInterval(()=>l(),se),()=>{c.current&&clearInterval(c.current)}),[l]),{usageInfo:n,usageLoading:i,lastFetchedAt:o,refreshUsage:(0,U.useCallback)(()=>l(!0),[l])}}var le={damping:.7,stiffness:.05,mass:1.25},ue=70,de=1e3/60,G=350,K=!1;globalThis.document?.addEventListener(`mousedown`,()=>{K=!0}),globalThis.document?.addEventListener(`mouseup`,()=>{K=!1}),globalThis.document?.addEventListener(`click`,()=>{K=!1});var fe=(e={})=>{let[t,n]=(0,U.useState)(!1),[r,i]=(0,U.useState)(e.initial!==!1),[a,o]=(0,U.useState)(!1),s=(0,U.useRef)(null);s.current=e;let c=(0,U.useCallback)(()=>{if(!K)return!1;let e=window.getSelection();if(!e||!e.rangeCount)return!1;let t=e.getRangeAt(0);return t.commonAncestorContainer.contains(g.current)||g.current?.contains(t.commonAncestorContainer)},[]),l=(0,U.useCallback)(e=>{d.isAtBottom=e,i(e)},[]),u=(0,U.useCallback)(e=>{d.escapedFromLock=e,n(e)},[]),d=(0,U.useMemo)(()=>{let n;return{escapedFromLock:t,isAtBottom:r,resizeDifference:0,accumulated:0,velocity:0,listeners:new Set,get scrollTop(){return g.current?.scrollTop??0},set scrollTop(e){g.current&&(g.current.scrollTop=e,d.ignoreScrollToTop=g.current.scrollTop)},get targetScrollTop(){return!g.current||!_.current?0:g.current.scrollHeight-1-g.current.clientHeight},get calculatedTargetScrollTop(){if(!g.current||!_.current)return 0;let{targetScrollTop:t}=this;if(!e.targetScrollTop)return t;if(n?.targetScrollTop===t)return n.calculatedScrollTop;let r=Math.max(Math.min(e.targetScrollTop(t,{scrollElement:g.current,contentElement:_.current}),t),0);return n={targetScrollTop:t,calculatedScrollTop:r},requestAnimationFrame(()=>{n=void 0}),r},get scrollDifference(){return this.calculatedTargetScrollTop-this.scrollTop},get isNearBottom(){return this.scrollDifference<=ue}}},[]),f=(0,U.useCallback)((e={})=>{typeof e==`string`&&(e={animation:e}),e.preserveScrollPosition||l(!0);let t=Date.now()+(Number(e.wait)||0),n=Y(s.current,e.animation),{ignoreEscapes:r=!1}=e,i,a=d.calculatedTargetScrollTop;e.duration instanceof Promise?e.duration.finally(()=>{i=Date.now()}):i=t+(e.duration??0);let o=async()=>{let e=new Promise(requestAnimationFrame).then(()=>{if(!d.isAtBottom)return d.animation=void 0,!1;let{scrollTop:l}=d,u=performance.now(),p=(u-(d.lastTick??u))/de;if(d.animation||={behavior:n,promise:e,ignoreEscapes:r},d.animation.behavior===n&&(d.lastTick=u),c()||t>Date.now())return o();if(l<Math.min(a,d.calculatedTargetScrollTop)){if(d.animation?.behavior===n){if(n===`instant`)return d.scrollTop=d.calculatedTargetScrollTop,o();d.velocity=(n.damping*d.velocity+n.stiffness*d.scrollDifference)/n.mass,d.accumulated+=d.velocity*p,d.scrollTop+=d.accumulated,d.scrollTop!==l&&(d.accumulated=0)}return o()}return i>Date.now()?(a=d.calculatedTargetScrollTop,o()):(d.animation=void 0,d.scrollTop<d.calculatedTargetScrollTop?f({animation:Y(s.current,s.current.resize),ignoreEscapes:r,duration:Math.max(0,i-Date.now())||void 0}):d.isAtBottom)});return e.then(e=>(requestAnimationFrame(()=>{d.animation||(d.lastTick=void 0,d.velocity=0)}),e))};return e.wait!==!0&&(d.animation=void 0),d.animation?.behavior===n?d.animation.promise:o()},[l,c,d]),p=(0,U.useCallback)(()=>{u(!0),l(!1)},[u,l]),m=(0,U.useCallback)(({target:e})=>{if(e!==g.current)return;let{scrollTop:t,ignoreScrollToTop:n}=d,{lastScrollTop:r=t}=d;d.lastScrollTop=t,d.ignoreScrollToTop=void 0,n&&n>t&&(r=n),o(d.isNearBottom),setTimeout(()=>{if(d.resizeDifference||t===n)return;if(c()){u(!0),l(!1);return}let e=t>r,i=t<r;if(d.animation?.ignoreEscapes){d.scrollTop=r;return}i&&(u(!0),l(!1)),e&&u(!1),!d.escapedFromLock&&d.isNearBottom&&l(!0)},1)},[u,l,c,d]),h=(0,U.useCallback)(({target:e,deltaY:t})=>{let n=e;for(;![`scroll`,`auto`].includes(getComputedStyle(n).overflow);){if(!n.parentElement)return;n=n.parentElement}n===g.current&&t<0&&g.current.scrollHeight>g.current.clientHeight&&!d.animation?.ignoreEscapes&&(u(!0),l(!1))},[u,l,d]),g=q(e=>{g.current?.removeEventListener(`scroll`,m),g.current?.removeEventListener(`wheel`,h),e?.addEventListener(`scroll`,m,{passive:!0}),e?.addEventListener(`wheel`,h,{passive:!0})},[]),_=q(e=>{if(d.resizeObserver?.disconnect(),!e)return;let t;d.resizeObserver=new ResizeObserver(([e])=>{let{height:n}=e.contentRect,r=n-(t??n);if(d.resizeDifference=r,d.scrollTop>d.targetScrollTop&&(d.scrollTop=d.targetScrollTop),o(d.isNearBottom),r>=0){let e=Y(s.current,t?s.current.resize:s.current.initial);f({animation:e,wait:!0,preserveScrollPosition:!0,duration:e===`instant`?void 0:G})}else d.isNearBottom&&(u(!1),l(!0));t=n,requestAnimationFrame(()=>{setTimeout(()=>{d.resizeDifference===r&&(d.resizeDifference=0)},1)})}),d.resizeObserver?.observe(e)},[]);return{contentRef:_,scrollRef:g,scrollToBottom:f,stopScroll:p,isAtBottom:r||a,isNearBottom:a,escapedFromLock:t,state:d}};function q(e,t){let n=(0,U.useCallback)(t=>(n.current=t,e(t)),t);return n}var J=new Map;function Y(...e){let t={...le},n=!1;for(let r of e){if(r===`instant`){n=!0;continue}typeof r==`object`&&(n=!1,t.damping=r.damping??t.damping,t.stiffness=r.stiffness??t.stiffness,t.mass=r.mass??t.mass)}let r=JSON.stringify(t);return J.has(r)||J.set(r,Object.freeze(t)),n?`instant`:J.get(r)}var pe=(0,U.createContext)(null),me=typeof window<`u`?U.useLayoutEffect:U.useEffect;function X({instance:e,children:t,resize:n,initial:r,mass:i,damping:a,stiffness:o,targetScrollTop:s,contextRef:c,...l}){let u=(0,U.useRef)(null),d=fe({mass:i,damping:a,stiffness:o,resize:n,initial:r,targetScrollTop:U.useCallback((e,t)=>(y?.targetScrollTop??s)?.(e,t)??e,[s])}),{scrollRef:f,contentRef:p,scrollToBottom:m,stopScroll:h,isAtBottom:g,escapedFromLock:_,state:v}=e??d,y=(0,U.useMemo)(()=>({scrollToBottom:m,stopScroll:h,scrollRef:f,isAtBottom:g,escapedFromLock:_,contentRef:p,state:v,get targetScrollTop(){return u.current},set targetScrollTop(e){u.current=e}}),[m,g,p,f,h,_,v]);return(0,U.useImperativeHandle)(c,()=>y,[y]),me(()=>{f.current&&getComputedStyle(f.current).overflow===`visible`&&(f.current.style.overflow=`auto`)},[]),U.createElement(pe.Provider,{value:y},U.createElement(`div`,{...l},typeof t==`function`?t(y):t))}(function(e){function t({children:e,scrollClassName:t,...n}){let r=he();return U.createElement(`div`,{ref:r.scrollRef,style:{height:`100%`,width:`100%`,scrollbarGutter:`stable both-edges`},className:t},U.createElement(`div`,{...n,ref:r.contentRef},typeof e==`function`?e(r):e))}e.Content=t})(X||={});function he(){let e=(0,U.useContext)(pe);if(!e)throw Error(`use-stick-to-bottom component context must be used within a StickToBottom component`);return e}var Z=r();function ge(e){let t=e.type===`approval_request`;return{toolName:e.type===`tool_use`?e.tool:t?e.tool??`Tool`:`Tool`,input:e.type===`tool_use`?e.input:t?e.input??{}:{}}}function _e({tool:e,result:t,completed:n,projectName:r}){let[i,a]=(0,U.useState)(!1);if(e.type===`error`)return(0,Z.jsxs)(`div`,{className:`flex items-center gap-2 rounded bg-red-500/10 border border-red-500/20 px-2 py-1.5 text-xs text-red-400`,children:[(0,Z.jsx)(_,{className:`size-3`}),(0,Z.jsx)(`span`,{children:e.message})]});let{toolName:o,input:s}=ge(e),c=t?.type===`tool_result`,l=c&&!!t.isError,u=o===`AskUserQuestion`&&!!s?.answers,d=(o===`Agent`||o===`Task`)&&e.type===`tool_use`,f=d?e.children:void 0,p=f&&f.length>0;return(0,Z.jsxs)(`div`,{className:`rounded border text-xs ${d?`border-accent/30 bg-accent/5`:`border-border bg-background`}`,children:[(0,Z.jsxs)(`button`,{onClick:()=>a(!i),className:`flex items-center gap-2 px-2 py-1.5 w-full text-left hover:bg-surface transition-colors min-w-0`,children:[i?(0,Z.jsx)(h,{className:`size-3 shrink-0`}):(0,Z.jsx)(y,{className:`size-3 shrink-0`}),l?(0,Z.jsx)(N,{className:`size-3 text-red-400 shrink-0`}):c||u||n?(0,Z.jsx)(g,{className:`size-3 text-green-400 shrink-0`}):(0,Z.jsx)(D,{className:`size-3 text-yellow-400 shrink-0 animate-spin`}),(0,Z.jsx)(`span`,{className:`truncate text-text-primary`,children:(0,Z.jsx)(ve,{name:o,input:s})}),p&&(0,Z.jsxs)(`span`,{className:`ml-auto text-[10px] text-text-subtle shrink-0`,children:[f.length,` steps`]})]}),i&&(0,Z.jsxs)(`div`,{className:`px-2 pb-2 space-y-1.5`,children:[(e.type===`tool_use`||e.type===`approval_request`)&&(0,Z.jsx)(ye,{name:o,input:s,projectName:r}),p&&(0,Z.jsx)(Ce,{events:f,projectName:r}),c&&(0,Z.jsx)(xe,{toolName:o,output:t.output})]})]})}function ve({name:e,input:t}){let n=e=>String(e??``);switch(e){case`Read`:case`Write`:case`Edit`:case`MultiEdit`:case`NotebookEdit`:return(0,Z.jsxs)(Z.Fragment,{children:[e,` `,(0,Z.jsx)(`span`,{className:`text-text-subtle`,children:l(n(t.file_path))})]});case`Bash`:return(0,Z.jsxs)(Z.Fragment,{children:[e,` `,(0,Z.jsx)(`span`,{className:`font-mono text-text-subtle`,children:Q(n(t.command),60)})]});case`Glob`:return(0,Z.jsxs)(Z.Fragment,{children:[e,` `,(0,Z.jsx)(`span`,{className:`font-mono text-text-subtle`,children:n(t.pattern)})]});case`Grep`:return(0,Z.jsxs)(Z.Fragment,{children:[e,` `,(0,Z.jsx)(`span`,{className:`font-mono text-text-subtle`,children:Q(n(t.pattern),40)})]});case`WebSearch`:return(0,Z.jsxs)(Z.Fragment,{children:[(0,Z.jsx)(O,{className:`size-3 inline`}),` `,e,` `,(0,Z.jsx)(`span`,{className:`text-text-subtle`,children:Q(n(t.query),50)})]});case`WebFetch`:return(0,Z.jsxs)(Z.Fragment,{children:[(0,Z.jsx)(I,{className:`size-3 inline`}),` `,e,` `,(0,Z.jsx)(`span`,{className:`text-text-subtle`,children:Q(n(t.url),50)})]});case`ToolSearch`:return(0,Z.jsxs)(Z.Fragment,{children:[(0,Z.jsx)(O,{className:`size-3 inline`}),` `,e,` `,(0,Z.jsx)(`span`,{className:`text-text-subtle`,children:Q(n(t.query),50)})]});case`Agent`:case`Task`:return(0,Z.jsxs)(Z.Fragment,{children:[(0,Z.jsx)(M,{className:`size-3 inline`}),` `,e,` `,(0,Z.jsx)(`span`,{className:`text-text-subtle`,children:Q(n(t.description||t.prompt),60)})]});case`TodoWrite`:{let n=t.todos??[],r=n.filter(e=>e.status===`completed`).length;return(0,Z.jsxs)(Z.Fragment,{children:[(0,Z.jsx)(z,{className:`size-3 inline`}),` `,e,` `,(0,Z.jsxs)(`span`,{className:`text-text-subtle`,children:[r,`/`,n.length,` done`]})]})}case`AskUserQuestion`:{let n=t.questions??[],r=!!t.answers;return(0,Z.jsxs)(Z.Fragment,{children:[e,` `,(0,Z.jsxs)(`span`,{className:`text-text-subtle`,children:[n.length,` question`,n.length===1?``:`s`,r?` ✓`:``]})]})}default:return(0,Z.jsx)(Z.Fragment,{children:e})}}function ye({name:e,input:t,projectName:n}){let r=e=>String(e??``),{openTab:o}=a(),s=e=>{n&&o({type:`editor`,title:l(e),metadata:{filePath:e,projectName:n},projectId:n,closable:!0})},c=(e,t,r)=>{o({type:`git-diff`,title:`Diff ${l(e)}`,metadata:{filePath:e,projectName:n,original:t,modified:r},projectId:n??null,closable:!0})};switch(e){case`Bash`:return(0,Z.jsxs)(`div`,{className:`space-y-1`,children:[!!t.description&&(0,Z.jsx)(`p`,{className:`text-text-subtle italic`,children:r(t.description)}),(0,Z.jsx)(`pre`,{className:`font-mono text-text-secondary overflow-x-auto whitespace-pre-wrap break-all`,children:r(t.command)})]});case`Read`:case`Write`:case`Edit`:case`MultiEdit`:case`NotebookEdit`:{let n=r(t.file_path);return(0,Z.jsxs)(`div`,{className:`space-y-1`,children:[(0,Z.jsxs)(`button`,{type:`button`,className:`font-mono text-text-secondary break-all hover:text-primary hover:underline text-left flex items-center gap-1`,onClick:()=>s(n),title:`Open file in editor`,children:[(0,Z.jsx)(i,{className:`size-3 shrink-0`}),n]}),e===`Edit`&&(!!t.old_string||!!t.new_string)&&(0,Z.jsxs)(`button`,{type:`button`,className:`text-text-subtle hover:text-primary hover:underline text-left flex items-center gap-1`,onClick:()=>c(n,r(t.old_string),r(t.new_string)),title:`View diff in new tab`,children:[(0,Z.jsx)(F,{className:`size-3 shrink-0`}),`View Diff`]}),e===`Write`&&!!t.content&&(0,Z.jsx)(`pre`,{className:`font-mono text-text-subtle overflow-x-auto max-h-32 whitespace-pre-wrap`,children:Q(r(t.content),300)})]})}case`Glob`:return(0,Z.jsxs)(`p`,{className:`font-mono text-text-secondary`,children:[r(t.pattern),t.path?` in ${r(t.path)}`:``]});case`Grep`:return(0,Z.jsxs)(`div`,{className:`space-y-0.5`,children:[(0,Z.jsxs)(`p`,{className:`font-mono text-text-secondary`,children:[`/`,r(t.pattern),`/`]}),!!t.path&&(0,Z.jsxs)(`p`,{className:`text-text-subtle`,children:[`in `,r(t.path)]})]});case`TodoWrite`:return(0,Z.jsx)(be,{todos:t.todos??[]});case`Agent`:case`Task`:return(0,Z.jsxs)(`div`,{className:`space-y-1`,children:[!!t.description&&(0,Z.jsx)(`p`,{className:`text-text-secondary font-medium`,children:r(t.description)}),!!t.subagent_type&&(0,Z.jsxs)(`p`,{className:`text-text-subtle`,children:[`Type: `,r(t.subagent_type)]}),!!t.prompt&&(0,Z.jsx)(we,{content:r(t.prompt),maxHeight:`max-h-48`})]});case`ToolSearch`:return(0,Z.jsxs)(`div`,{className:`space-y-0.5`,children:[(0,Z.jsx)(`p`,{className:`font-mono text-text-secondary`,children:r(t.query)}),!!t.max_results&&(0,Z.jsxs)(`p`,{className:`text-text-subtle`,children:[`Max results: `,r(t.max_results)]})]});case`WebFetch`:return(0,Z.jsxs)(`div`,{className:`space-y-0.5`,children:[(0,Z.jsxs)(`a`,{href:r(t.url),target:`_blank`,rel:`noopener noreferrer`,className:`font-mono text-primary hover:underline break-all flex items-center gap-1`,children:[(0,Z.jsx)(I,{className:`size-3 shrink-0`}),r(t.url)]}),!!t.prompt&&(0,Z.jsx)(`p`,{className:`text-text-subtle`,children:Q(r(t.prompt),100)})]});case`AskUserQuestion`:{let e=t.questions??[],n=t.answers??{};return(0,Z.jsx)(`div`,{className:`space-y-2`,children:e.map((e,t)=>(0,Z.jsxs)(`div`,{className:`space-y-0.5`,children:[(0,Z.jsxs)(`p`,{className:`text-text-primary font-medium`,children:[e.header?`${e.header}: `:``,e.question]}),(0,Z.jsx)(`div`,{className:`flex flex-wrap gap-1`,children:e.options.map((t,r)=>(0,Z.jsx)(`span`,{className:`inline-block rounded px-1.5 py-0.5 text-xs border ${(n[e.question]??``).split(`, `).includes(t.label)?`border-accent bg-accent/20 text-text-primary`:`border-border text-text-subtle`}`,children:t.label},r))}),n[e.question]&&(0,Z.jsxs)(`p`,{className:`text-accent text-xs`,children:[`Answer: `,n[e.question]]})]},t))})}default:return(0,Z.jsx)(`pre`,{className:`overflow-x-auto text-text-secondary font-mono whitespace-pre-wrap break-all`,children:JSON.stringify(t,null,2)})}}function be({todos:e}){return(0,Z.jsx)(`div`,{className:`space-y-0.5`,children:e.map((e,t)=>(0,Z.jsxs)(`div`,{className:`flex items-start gap-1.5`,children:[(0,Z.jsx)(`span`,{className:`shrink-0 mt-0.5 ${e.status===`completed`?`text-green-400`:e.status===`in_progress`?`text-yellow-400`:`text-text-subtle`}`,children:e.status===`completed`?`✓`:e.status===`in_progress`?`▶`:`○`}),(0,Z.jsx)(`span`,{className:e.status===`completed`?`line-through text-text-subtle`:`text-text-secondary`,children:e.content})]},t))})}function xe({toolName:e,output:t}){let[n,r]=(0,U.useState)(!1),i=(0,U.useMemo)(()=>{if(e!==`Agent`&&e!==`Task`)return null;try{let e=JSON.parse(t);if(Array.isArray(e)){let t=e.filter(e=>e.type===`text`&&e.text).map(e=>e.text).join(`
|
|
3
|
-
|
|
4
|
-
`);if(t)return t}if(typeof e==`string`)return e}catch{if(t&&!t.startsWith(`[{`))return t}return null},[e,t]);return i?(0,Z.jsxs)(`div`,{className:`border-t border-border pt-1.5 space-y-1`,children:[(0,Z.jsx)(we,{content:i,maxHeight:`max-h-60`}),(0,Z.jsxs)(`button`,{type:`button`,onClick:()=>r(!n),className:`flex items-center gap-1 text-[10px] text-text-subtle hover:text-text-secondary transition-colors`,children:[(0,Z.jsx)(P,{className:`size-3`}),n?`Hide`:`Show`,` raw`]}),n&&(0,Z.jsx)(`pre`,{className:`overflow-x-auto text-text-subtle font-mono max-h-40 whitespace-pre-wrap break-all text-[10px]`,children:t})]}):(0,Z.jsx)(Se,{output:t})}function Se({output:e}){let t=e.split(`
|
|
5
|
-
`).length,n=t>3||e.length>200,[r,i]=(0,U.useState)(n);return(0,Z.jsxs)(`div`,{className:`border-t border-border pt-1.5`,children:[n&&(0,Z.jsxs)(`button`,{type:`button`,onClick:()=>i(!r),className:`flex items-center gap-1 text-[10px] text-text-subtle hover:text-text-secondary transition-colors mb-1`,children:[r?(0,Z.jsx)(y,{className:`size-3`}):(0,Z.jsx)(h,{className:`size-3`}),`Output (`,t,` lines)`]}),(0,Z.jsx)(`pre`,{className:`overflow-x-auto text-text-subtle font-mono whitespace-pre-wrap break-all ${r?`max-h-16 overflow-hidden`:`max-h-60`}`,children:e})]})}function Ce({events:e,projectName:t}){let n=[],r=``;for(let t of e)if(t.type===`text`)r+=t.content;else if(t.type===`tool_use`)r&&=(n.push({kind:`text`,content:r}),``),n.push({kind:`tool`,tool:t});else if(t.type===`tool_result`){let e=t.toolUseId,r=e?n.find(t=>t.kind===`tool`&&t.tool.type===`tool_use`&&t.tool.toolUseId===e&&!t.result):n.findLast(e=>e.kind===`tool`&&!e.result);r&&(r.result=t)}return r&&n.push({kind:`text`,content:r}),(0,Z.jsx)(`div`,{className:`border-l-2 border-accent/20 pl-2 space-y-1 mt-1`,children:n.map((e,n)=>e.kind===`text`?(0,Z.jsx)(`div`,{className:`text-text-secondary text-[11px]`,children:(0,Z.jsx)(we,{content:e.content,maxHeight:`max-h-24`})},`st-${n}`):(0,Z.jsx)(_e,{tool:e.tool,result:e.result,completed:!!e.result,projectName:t},`sc-${n}`))})}function we({content:e,maxHeight:t=`max-h-48`}){return(0,Z.jsx)(k,{content:e,className:`text-text-secondary overflow-auto ${t}`})}function Q(e,t=50){return e?e.length>t?e.slice(0,t)+`…`:e:``}function Te({messages:e,messagesLoading:t,pendingApproval:n,onApprovalResponse:r,isStreaming:i,streamingStatus:a,connectingElapsed:o,thinkingWarningThreshold:s,projectName:c,onFork:l}){return t?(0,Z.jsxs)(`div`,{className:`flex flex-col items-center justify-center h-full gap-3 text-text-secondary`,children:[(0,Z.jsx)(M,{className:`size-10 text-text-subtle animate-pulse`}),(0,Z.jsx)(`p`,{className:`text-sm`,children:`Loading messages...`})]}):e.length===0&&!i?(0,Z.jsxs)(`div`,{className:`flex flex-col items-center justify-center h-full gap-3 text-text-secondary`,children:[(0,Z.jsx)(M,{className:`size-10 text-text-subtle`}),(0,Z.jsx)(`p`,{className:`text-sm`,children:`Send a message to start the conversation`})]}):(0,Z.jsxs)(X,{className:`flex-1 overflow-y-auto`,resize:`smooth`,initial:`instant`,children:[(0,Z.jsxs)(X.Content,{className:`p-4 space-y-4`,children:[e.filter(e=>{let t=e.content&&e.content.trim().length>0,n=e.events&&e.events.length>0;return t||n}).map(e=>(0,Z.jsx)(De,{message:e,isStreaming:i&&e.id.startsWith(`streaming-`),projectName:c,onFork:e.role===`user`&&l?()=>l(e.content):void 0},e.id)),n&&(n.tool===`AskUserQuestion`?(0,Z.jsx)(He,{approval:n,onRespond:r}):(0,Z.jsx)(Ve,{approval:n,onRespond:r})),i&&(0,Z.jsx)(ze,{lastMessage:e[e.length-1],streamingStatus:a,elapsed:o,warningThreshold:s})]}),(0,Z.jsx)(Ee,{})]})}function Ee(){let{isAtBottom:e,scrollToBottom:t}=he();return e?null:(0,Z.jsxs)(`button`,{onClick:()=>t(),className:`absolute bottom-2 left-1/2 -translate-x-1/2 flex items-center gap-1 px-3 py-1 rounded-full bg-surface-elevated border border-border text-xs text-text-secondary hover:text-foreground shadow-lg transition-all`,children:[(0,Z.jsx)(h,{className:`size-3`}),`Scroll to bottom`]})}function De({message:e,isStreaming:t,projectName:n,onFork:r}){return e.role===`user`?(0,Z.jsx)(Ne,{content:e.content,projectName:n,onFork:r}):e.role===`system`?(0,Z.jsxs)(`div`,{className:`flex items-center gap-2 rounded-lg bg-red-500/10 border border-red-500/20 px-3 py-2 text-sm text-red-400`,children:[(0,Z.jsx)(_,{className:`size-4 shrink-0`}),(0,Z.jsx)(`p`,{children:e.content})]}):(0,Z.jsx)(`div`,{className:`flex flex-col gap-2`,children:e.events&&e.events.length>0?(0,Z.jsx)(Ie,{events:e.events,isStreaming:t,projectName:n}):e.content&&(0,Z.jsx)(`div`,{className:`text-sm text-text-primary`,children:(0,Z.jsx)(Be,{content:e.content,projectName:n})})})}var Oe=new Set([`.png`,`.jpg`,`.jpeg`,`.gif`,`.webp`]);function ke(e){let t=e.match(/^\[Attached file: (.+?)\]\n\n?/);if(t)return{files:[t[1]],text:e.slice(t[0].length)};let n=e.match(/^\[Attached files:\n([\s\S]+?)\]\n\n?/);return n?{files:n[1].split(`
|
|
6
|
-
`).map(e=>e.trim()).filter(Boolean),text:e.slice(n[0].length)}:{files:[],text:e}}function Ae(e,t){let n=l(e);return`/api/project/${encodeURIComponent(t??`_`)}/chat/uploads/${encodeURIComponent(n)}`}function je(e){let t=e.lastIndexOf(`.`);return t===-1?!1:Oe.has(e.slice(t).toLowerCase())}function Me(e){return e.toLowerCase().endsWith(`.pdf`)}function Ne({content:e,projectName:t,onFork:n}){let{files:r,text:i}=(0,U.useMemo)(()=>ke(e),[e]);return(0,Z.jsx)(`div`,{className:`flex justify-end group/user`,children:(0,Z.jsxs)(`div`,{className:`rounded-lg bg-primary/10 px-3 py-2 text-sm text-text-primary max-w-[85%] space-y-2 relative`,children:[r.length>0&&(0,Z.jsx)(`div`,{className:`flex flex-wrap gap-2`,children:r.map((e,n)=>je(e)?(0,Z.jsx)(Pe,{src:Ae(e,t),alt:l(e)||`image`},n):Me(e)?(0,Z.jsx)(Fe,{src:Ae(e,t),filename:l(e)||`document.pdf`,mimeType:`application/pdf`},n):(0,Z.jsxs)(`div`,{className:`flex items-center gap-1.5 rounded-md border border-border bg-background/50 px-2 py-1 text-xs text-text-secondary`,children:[(0,Z.jsx)(C,{className:`size-3.5 shrink-0`}),(0,Z.jsx)(`span`,{className:`truncate max-w-40`,children:l(e)})]},n))}),i&&(0,Z.jsx)(`p`,{className:`whitespace-pre-wrap break-words`,children:i}),n&&(0,Z.jsx)(`button`,{onClick:n,title:`Retry from this message (fork session)`,className:`absolute -left-8 top-1/2 -translate-y-1/2 opacity-0 group-hover/user:opacity-100 transition-opacity size-6 flex items-center justify-center rounded bg-surface border border-border text-text-subtle hover:text-text-primary hover:bg-surface-elevated`,children:(0,Z.jsx)(o,{className:`size-3`})})]})})}function Pe({src:e,alt:t}){let[n,r]=(0,U.useState)(null),[i,a]=(0,U.useState)(!1);return(0,U.useEffect)(()=>{let t,n=d();return fetch(e,{headers:n?{Authorization:`Bearer ${n}`}:{}}).then(e=>{if(!e.ok)throw Error(`Failed to load`);return e.blob()}).then(e=>{let n=URL.createObjectURL(e);t=n,r(n)}).catch(()=>a(!0)),()=>{t&&URL.revokeObjectURL(t)}},[e]),i?(0,Z.jsxs)(`div`,{className:`flex items-center gap-1.5 rounded-md border border-border bg-background/50 px-2 py-1 text-xs text-text-secondary`,children:[(0,Z.jsx)(R,{className:`size-3.5 shrink-0`}),(0,Z.jsx)(`span`,{className:`truncate max-w-40`,children:t})]}):n?(0,Z.jsx)(`a`,{href:n,target:`_blank`,rel:`noopener noreferrer`,className:`block`,children:(0,Z.jsx)(`img`,{src:n,alt:t,className:`rounded-md max-h-48 max-w-full object-contain border border-border`})}):(0,Z.jsx)(`div`,{className:`rounded-md bg-surface border border-border h-24 w-32 animate-pulse`})}function Fe({src:e,filename:t,mimeType:n}){let[r,i]=(0,U.useState)(!1);return(0,Z.jsxs)(`button`,{type:`button`,onClick:(0,U.useCallback)(async()=>{i(!0);try{let t=d(),r=await fetch(e,{headers:t?{Authorization:`Bearer ${t}`}:{}});if(!r.ok)throw Error(`Failed to load`);let i=await r.blob(),a=URL.createObjectURL(new Blob([i],{type:n}));window.open(a,`_blank`),setTimeout(()=>URL.revokeObjectURL(a),6e4)}catch{window.open(e,`_blank`)}finally{i(!1)}},[e,n]),disabled:r,className:`flex items-center gap-1.5 rounded-md border border-border bg-background/50 px-2 py-1 text-xs text-text-secondary hover:bg-surface hover:text-text-primary transition-colors cursor-pointer disabled:opacity-50`,children:[(0,Z.jsx)(C,{className:`size-3.5 shrink-0 text-red-400`}),(0,Z.jsx)(`span`,{className:`truncate max-w-40`,children:t}),r&&(0,Z.jsx)(`span`,{className:`animate-spin text-[10px]`,children:`...`})]})}function Ie({events:e,isStreaming:t,projectName:n}){let r=[],i=``,a=``;for(let t=0;t<e.length;t++){let n=e[t];if(n.type===`thinking`){i&&=(r.push({kind:`text`,content:i}),``),a+=n.content;continue}a&&=(r.push({kind:`thinking`,content:a}),``),n.type===`text`?i+=n.content:n.type===`tool_use`?(i&&=(r.push({kind:`text`,content:i}),``),r.push({kind:`tool`,tool:n})):n.type===`tool_result`||(i&&=(r.push({kind:`text`,content:i}),``),r.push({kind:`tool`,tool:n}))}a&&r.push({kind:`thinking`,content:a}),i&&r.push({kind:`text`,content:i});let o=e.filter(e=>e.type===`tool_result`);for(let e of o){let t=e.toolUseId;if(t){let n=r.find(e=>e.kind===`tool`&&e.tool.type===`tool_use`&&e.tool.toolUseId===t);if(n){n.result=e;continue}}let n=r.find(e=>e.kind===`tool`&&!e.result);n&&(n.result=e)}for(let e=0;e<r.length;e++){let n=r[e];if(n.kind===`tool`&&!n.result){let i=!1;if(n.tool.type===`tool_use`&&n.tool.tool===`Read`){let t=n.tool.input?.file_path;t&&(i=r.slice(e+1).some(e=>e.kind===`tool`&&e.result&&e.tool.type===`tool_use`&&e.tool.tool===`Edit`&&e.tool.input?.file_path===t))}n.completed=i||!t}}return(0,Z.jsx)(Z.Fragment,{children:r.map((e,i)=>{if(e.kind===`thinking`)return(0,Z.jsx)(Le,{content:e.content,isStreaming:t&&i===r.length-1},`think-${i}`);if(e.kind===`text`){let a=t&&i===r.length-1;return(0,Z.jsx)(`div`,{className:`text-sm text-text-primary`,children:(0,Z.jsx)(Re,{content:e.content,animate:a,projectName:n})},`text-${i}`)}return(0,Z.jsx)(_e,{tool:e.tool,result:e.result,completed:e.completed,projectName:n},`tool-${i}`)})})}function Le({content:e,isStreaming:t}){let[n,r]=(0,U.useState)(t);return(0,U.useEffect)(()=>{!t&&e.length>0&&r(!1)},[t,e.length]),(0,Z.jsxs)(`div`,{className:`rounded border border-border/50 bg-surface/30 text-xs`,children:[(0,Z.jsxs)(`button`,{onClick:()=>r(!n),className:`flex items-center gap-2 px-2 py-1.5 w-full text-left hover:bg-surface transition-colors text-text-subtle`,children:[t?(0,Z.jsx)(D,{className:`size-3 animate-spin`}):(0,Z.jsx)(y,{className:`size-3 transition-transform ${n?`rotate-90`:``}`}),(0,Z.jsxs)(`span`,{children:[`Thinking`,t?`...`:``]}),!t&&(0,Z.jsx)(`span`,{className:`text-text-subtle/50 ml-auto`,children:e.length>100?`${Math.round(e.length/4)} tokens`:``})]}),n&&(0,Z.jsx)(`div`,{className:`px-2 pb-2 text-text-subtle/80 whitespace-pre-wrap max-h-60 overflow-y-auto text-[11px] leading-relaxed`,children:e})]})}function Re({content:e,animate:t,projectName:n}){return(0,Z.jsxs)(Z.Fragment,{children:[(0,Z.jsx)(Be,{content:e,projectName:n}),t&&(0,Z.jsx)(`span`,{className:`text-text-subtle text-sm animate-pulse`,children:`Thinking...`})]})}function ze({lastMessage:e,streamingStatus:t,elapsed:n,warningThreshold:r=15}){let i=!e||e.role!==`assistant`,a=e?.events?.length?e.events[e.events.length-1].type===`tool_result`:!1;return!i&&!a?null:(0,Z.jsxs)(`div`,{className:`flex flex-col gap-1 text-sm`,children:[(0,Z.jsxs)(`div`,{className:`flex items-center gap-2 text-text-subtle`,children:[(0,Z.jsx)(D,{className:`size-3 animate-spin`}),(0,Z.jsxs)(`span`,{children:[`Thinking`,i&&(n??0)>0&&(0,Z.jsxs)(`span`,{className:`text-text-subtle/60`,children:[`... (`,n,`s)`]})]})]}),i&&(n??0)>=r&&(0,Z.jsx)(`p`,{className:`text-xs text-yellow-500/80 ml-5`,children:`Taking longer than usual — may be rate-limited or API slow. Try sending a new message to retry.`})]})}function Be({content:e,projectName:t}){return(0,Z.jsx)(k,{content:e,projectName:t,codeActions:!0})}function Ve({approval:e,onRespond:t}){return(0,Z.jsxs)(`div`,{className:`rounded-lg border-2 border-yellow-500/40 bg-yellow-500/10 p-3 space-y-2`,children:[(0,Z.jsxs)(`div`,{className:`flex items-center gap-2 text-yellow-400 text-sm font-medium`,children:[(0,Z.jsx)(V,{className:`size-4`}),(0,Z.jsx)(`span`,{children:`Tool Approval Required`})]}),(0,Z.jsx)(`div`,{className:`text-xs text-text-primary`,children:(0,Z.jsx)(`span`,{className:`font-medium`,children:e.tool})}),(0,Z.jsx)(`pre`,{className:`text-xs font-mono text-text-secondary overflow-x-auto bg-background rounded p-2 border border-border`,children:JSON.stringify(e.input,null,2)}),(0,Z.jsxs)(`div`,{className:`flex gap-2`,children:[(0,Z.jsx)(`button`,{onClick:()=>t(e.requestId,!0),className:`px-4 py-1.5 rounded bg-green-600 text-white text-xs font-medium hover:bg-green-500 transition-colors`,children:`Allow`}),(0,Z.jsx)(`button`,{onClick:()=>t(e.requestId,!1),className:`px-4 py-1.5 rounded bg-red-600 text-white text-xs font-medium hover:bg-red-500 transition-colors`,children:`Deny`})]})]})}function He({approval:e,onRespond:t}){let n=e.input.questions??[],[r,i]=(0,U.useState)({}),[a,o]=(0,U.useState)({}),s=(e,t,n)=>{o(t=>({...t,[e]:!1})),i(r=>{if(!n)return{...r,[e]:t};let i=r[e]??``,a=i?i.split(`, `):[],o=a.indexOf(t);return o>=0?a.splice(o,1):a.push(t),{...r,[e]:a.join(`, `)}})},c=e=>{o(t=>({...t,[e]:!0})),i(t=>({...t,[e]:``}))},l=(e,t)=>{i(n=>({...n,[e]:t}))},u=n.every(e=>r[e.question]?.trim());return(0,Z.jsxs)(`div`,{className:`rounded-lg border-2 border-accent/40 bg-accent/5 p-3 space-y-3`,children:[n.map((e,t)=>(0,Z.jsxs)(`div`,{className:`space-y-1.5`,children:[(0,Z.jsxs)(`p`,{className:`text-sm text-text-primary font-medium`,children:[e.header?`${e.header}: `:``,e.question]}),e.multiSelect&&(0,Z.jsx)(`p`,{className:`text-xs text-text-subtle`,children:`Select multiple`}),(0,Z.jsxs)(`div`,{className:`flex flex-col gap-1`,children:[e.options.map((t,n)=>(0,Z.jsxs)(`button`,{onClick:()=>s(e.question,t.label,e.multiSelect),className:`text-left rounded px-2.5 py-1.5 text-xs border transition-colors ${!a[e.question]&&(r[e.question]??``).split(`, `).includes(t.label)?`border-accent bg-accent/20 text-text-primary`:`border-border bg-background text-text-secondary hover:bg-surface-elevated`}`,children:[(0,Z.jsx)(`span`,{className:`font-medium`,children:t.label}),t.description&&(0,Z.jsxs)(`span`,{className:`text-text-subtle ml-1.5`,children:[`— `,t.description]})]},n)),a[e.question]?(0,Z.jsx)(`input`,{type:`text`,autoFocus:!0,placeholder:`Type your answer...`,value:r[e.question]??``,onChange:t=>l(e.question,t.target.value),className:`rounded px-2.5 py-1.5 text-xs border border-accent bg-accent/10 text-text-primary outline-none placeholder:text-text-subtle`}):(0,Z.jsx)(`button`,{onClick:()=>c(e.question),className:`text-left rounded px-2.5 py-1.5 text-xs border border-dashed border-border text-text-subtle hover:bg-surface-elevated transition-colors`,children:`Other — type your own answer`})]})]},t)),(0,Z.jsxs)(`div`,{className:`flex gap-2 pt-1`,children:[(0,Z.jsx)(`button`,{onClick:()=>t(e.requestId,!0,r),disabled:!u,className:`px-4 py-1.5 rounded bg-accent text-white text-xs font-medium hover:bg-accent/80 transition-colors disabled:opacity-40 disabled:cursor-not-allowed`,children:`Submit`}),(0,Z.jsx)(`button`,{onClick:()=>t(e.requestId,!1),className:`px-4 py-1.5 rounded bg-surface-elevated text-text-secondary text-xs hover:bg-surface transition-colors`,children:`Skip`})]})]})}var Ue=new Set([`image/png`,`image/jpeg`,`image/gif`,`image/webp`]),We=new Set([`application/pdf`]),Ge=[`text/`,`application/json`,`application/xml`,`application/javascript`,`application/typescript`,`application/x-yaml`,`application/toml`,`application/x-sh`],Ke=new Set(`.ts,.tsx,.js,.jsx,.mjs,.cjs,.py,.rb,.go,.rs,.java,.kt,.swift,.c,.cpp,.h,.hpp,.cs,.json,.yaml,.yml,.toml,.xml,.md,.mdx,.txt,.csv,.tsv,.html,.css,.scss,.less,.sass,.sh,.bash,.zsh,.fish,.sql,.graphql,.gql,.env,.ini,.cfg,.conf,.dockerfile,.makefile,.vue,.svelte,.astro,.ipynb`.split(`,`));function qe(e){return Ue.has(e.type)}function Je(e){if(Ue.has(e.type)||We.has(e.type)||Ge.some(t=>e.type.startsWith(t)))return!0;let t=Ye(e.name);return!!(t&&Ke.has(t))}function Ye(e){let t=e.lastIndexOf(`.`);return t===-1?``:e.slice(t).toLowerCase()}function Xe({attachments:e,onRemove:t}){return e.length===0?null:(0,Z.jsx)(`div`,{className:`flex flex-wrap gap-1.5 px-2 md:px-4 pt-2`,children:e.map(e=>(0,Z.jsxs)(`div`,{className:`flex items-center gap-1.5 rounded-md border border-border bg-surface px-2 py-1 text-xs text-text-secondary max-w-48`,children:[e.previewUrl?(0,Z.jsx)(`img`,{src:e.previewUrl,alt:e.name,className:`size-5 rounded object-cover shrink-0`}):e.isImage?(0,Z.jsx)(R,{className:`size-3.5 shrink-0 text-text-subtle`}):(0,Z.jsx)(C,{className:`size-3.5 shrink-0 text-text-subtle`}),(0,Z.jsx)(`span`,{className:`truncate`,children:e.name}),e.status===`uploading`?(0,Z.jsx)(D,{className:`size-3 shrink-0 animate-spin text-text-subtle`}):e.status===`error`?(0,Z.jsx)(`span`,{className:`text-red-500 shrink-0`,title:`Upload failed`,children:`!`}):null,(0,Z.jsx)(`button`,{type:`button`,onClick:()=>t(e.id),className:`shrink-0 rounded-sm p-0.5 hover:bg-border/50 transition-colors`,"aria-label":`Remove ${e.name}`,children:(0,Z.jsx)(E,{className:`size-3`})})]},e.id))})}function Ze(e){let t=[];function n(e){for(let r of e)t.push(r),r.children&&n(r.children)}return n(e),t}function Qe({items:e,filter:t,onSelect:n,onClose:r,visible:i}){let[a,o]=(0,U.useState)(0),s=(0,U.useRef)(null),c=(()=>{if(!t)return e.slice(0,50);let n=t.toLowerCase();return e.filter(e=>e.path.toLowerCase().includes(n)||e.name.toLowerCase().includes(n)).slice(0,50)})();(0,U.useEffect)(()=>{o(0)},[t]),(0,U.useEffect)(()=>{let e=s.current;e&&e.children[a]?.scrollIntoView({block:`nearest`})},[a]);let l=(0,U.useCallback)(e=>{if(!i||c.length===0)return!1;switch(e.key){case`ArrowUp`:return e.preventDefault(),o(e=>e>0?e-1:c.length-1),!0;case`ArrowDown`:return e.preventDefault(),o(e=>e<c.length-1?e+1:0),!0;case`Enter`:case`Tab`:return e.preventDefault(),c[a]&&n(c[a]),!0;case`Escape`:return e.preventDefault(),r(),!0}return!1},[i,c,a,n,r]);return(0,U.useEffect)(()=>{if(!i)return;let e=e=>{l(e)};return document.addEventListener(`keydown`,e,!0),()=>document.removeEventListener(`keydown`,e,!0)},[i,l]),!i||c.length===0?null:(0,Z.jsx)(`div`,{className:`max-h-52 overflow-y-auto border-b border-border bg-surface`,children:(0,Z.jsx)(`div`,{ref:s,className:`py-1`,children:c.map((e,t)=>(0,Z.jsxs)(`button`,{className:`flex items-center gap-2 w-full px-3 py-1.5 text-left transition-colors ${t===a?`bg-primary/10 text-primary`:`hover:bg-surface-hover text-text-primary`}`,onMouseEnter:()=>o(t),onClick:()=>n(e),children:[(0,Z.jsx)(`span`,{className:`shrink-0`,children:e.type===`directory`?(0,Z.jsx)(m,{className:`size-4 text-amber-500`}):(0,Z.jsx)(v,{className:`size-4 text-blue-400`})}),(0,Z.jsx)(`span`,{className:`text-sm truncate`,children:e.path})]},e.path))})})}function $e({onSend:e,isStreaming:t,onCancel:n,disabled:r,projectName:i,onSlashStateChange:a,onSlashItemsLoaded:o,slashSelected:s,onFileStateChange:l,onFileItemsLoaded:p,fileSelected:m,externalFiles:h,initialValue:g}){let[_,v]=(0,U.useState)(g??``),[y,b]=(0,U.useState)([]),x=(0,U.useRef)(null),S=(0,U.useRef)(null),C=(0,U.useRef)(null),w=(0,U.useRef)([]),T=(0,U.useRef)([]);(0,U.useEffect)(()=>{g&&(v(g),setTimeout(()=>{let e=x.current;e&&(e.focus(),e.selectionStart=e.selectionEnd=e.value.length)},50))},[g]),(0,U.useEffect)(()=>{if(!i){w.current=[],o?.([]);return}f.get(`${u(i)}/chat/slash-items`).then(e=>{w.current=e,o?.(e)}).catch(()=>{w.current=[],o?.([])})},[i]),(0,U.useEffect)(()=>{if(!i){T.current=[],p?.([]);return}f.get(`${u(i)}/files/tree?depth=5`).then(e=>{let t=Ze(e);T.current=t,p?.(t)}).catch(()=>{T.current=[],p?.([])})},[i]),(0,U.useEffect)(()=>{if(!s)return;let e=x.current,t=e?.selectionStart??_.length,n=_.slice(0,t),r=_.slice(t),i=n.replace(/(?:^|\s)\/\S*$/,e=>`${e.startsWith(`/`)?``:e[0]}/${s.name} `);v(i+r),a?.(!1,``),l?.(!1,``),e&&(e.focus(),setTimeout(()=>{e.selectionStart=e.selectionEnd=i.length},0))},[s]),(0,U.useEffect)(()=>{if(!m)return;let e=x.current;if(!e)return;let t=e.selectionStart,n=_.slice(0,t),r=_.slice(t),i=n.match(/@(\S*)$/);if(i){let t=n.length-i[0].length;v(n.slice(0,t)+`@${m.path} `+r);let a=t+m.path.length+2;setTimeout(()=>{e.selectionStart=e.selectionEnd=a,e.focus()},0)}else{let t=_+`@${m.path} `;v(t),setTimeout(()=>{e.selectionStart=e.selectionEnd=t.length,e.focus()},0)}l?.(!1,``)},[m]),(0,U.useEffect)(()=>{!h||h.length===0||D(h)},[h]);let E=(0,U.useCallback)(async e=>{if(!i)return null;try{let t=new FormData;t.append(`files`,e);let n={},r=d();r&&(n.Authorization=`Bearer ${r}`);let a=await(await fetch(`${u(i)}/chat/upload`,{method:`POST`,headers:n,body:t})).json();return a.ok&&Array.isArray(a.data)&&a.data.length>0?a.data[0].path:null}catch{return null}},[i]),D=(0,U.useCallback)(e=>{for(let t of e){if(!Je(t)){v(e=>e+(e.length>0&&!e.endsWith(` `)?` `:``)+t.name);continue}let e=c(),n=qe(t),r=n?URL.createObjectURL(t):void 0,i={id:e,name:t.name,file:t,isImage:n,previewUrl:r,status:`uploading`};b(e=>[...e,i]),E(t).then(t=>{b(n=>n.map(n=>n.id===e?{...n,serverPath:t??void 0,status:t?`ready`:`error`}:n))})}(S.current??x.current)?.focus()},[E]),O=(0,U.useCallback)(e=>{b(t=>{let n=t.find(t=>t.id===e);return n?.previewUrl&&URL.revokeObjectURL(n.previewUrl),t.filter(t=>t.id!==e)})},[]),k=(0,U.useCallback)(()=>{let t=_.trim(),n=y.filter(e=>e.status===`ready`);if(!(!t&&n.length===0)&&!r){a?.(!1,``),l?.(!1,``),e(t,n),v(``);for(let e of y)e.previewUrl&&URL.revokeObjectURL(e.previewUrl);b([]),x.current&&(x.current.style.height=`auto`),S.current&&(S.current.style.height=`auto`)}},[_,y,r,e,a,l]),A=(0,U.useCallback)(e=>{e.key===`Enter`&&!e.shiftKey&&(e.preventDefault(),k())},[k]),M=(0,U.useCallback)((e,t)=>{let n=e.slice(0,t),r=n.match(/(?:^|\s)\/(\S*)$/);if(r&&w.current.length>0){a?.(!0,r[1]??``),l?.(!1,``);return}let i=n.match(/@(\S*)$/);if(i&&T.current.length>0){l?.(!0,i[1]??``),a?.(!1,``);return}a?.(!1,``),l?.(!1,``)},[a,l]),N=(0,U.useCallback)(e=>{v(e),setTimeout(()=>{M(e,x.current?.selectionStart??e.length)},0)},[M]),P=(0,U.useCallback)(e=>{let t=e?.target??x.current;t&&(t.style.height=`auto`,t.style.height=Math.min(t.scrollHeight,160)+`px`)},[]),F=(0,U.useCallback)(e=>{let t=e.clipboardData?.items;if(!t)return;let n=[];for(let e of t)if(e.kind===`file`){let t=e.getAsFile();t&&n.push(t)}n.length>0&&(e.preventDefault(),D(n))},[D]),I=(0,U.useCallback)(e=>{e.preventDefault();let t=Array.from(e.dataTransfer.files);t.length>0&&D(t)},[D]),L=(0,U.useCallback)(e=>{e.preventDefault()},[]),R=(0,U.useCallback)(()=>{C.current?.click()},[]),z=(0,U.useCallback)(e=>{let t=Array.from(e.target.files??[]);t.length>0&&D(t),e.target.value=``},[D]),B=_.trim().length>0||y.some(e=>e.status===`ready`),V=t&&!B;return(0,Z.jsxs)(`div`,{className:`p-2 md:p-3 bg-background`,children:[(0,Z.jsxs)(`div`,{className:`border border-border rounded-xl md:rounded-2xl bg-surface shadow-sm cursor-text`,onClick:()=>!r&&(S.current??x.current)?.focus(),children:[(0,Z.jsx)(Xe,{attachments:y,onRemove:O}),(0,Z.jsxs)(`div`,{className:`flex items-end gap-1 md:hidden px-2 py-2`,children:[(0,Z.jsx)(`button`,{type:`button`,onClick:e=>{e.stopPropagation(),R()},disabled:r,className:`flex items-center justify-center size-7 shrink-0 rounded-full text-text-subtle hover:text-text-primary transition-colors disabled:opacity-50`,"aria-label":`Attach file`,children:(0,Z.jsx)(ee,{className:`size-4`})}),(0,Z.jsx)(`textarea`,{ref:S,value:_,onChange:e=>{N(e.target.value),P(e)},onKeyDown:A,onPaste:F,onDrop:I,onDragOver:L,placeholder:t?`Follow-up...`:`Ask anything...`,disabled:r,rows:1,className:`flex-1 resize-none bg-transparent py-1.5 text-sm text-foreground placeholder:text-text-subtle focus:outline-none disabled:opacity-50 max-h-20`}),V?(0,Z.jsx)(`button`,{onClick:e=>{e.stopPropagation(),n?.()},className:`flex items-center justify-center size-7 shrink-0 rounded-full bg-red-600 text-white hover:bg-red-500 transition-colors`,"aria-label":`Stop`,children:(0,Z.jsx)(H,{className:`size-3`})}):(0,Z.jsx)(`button`,{onClick:e=>{e.stopPropagation(),k()},disabled:r||!B,className:`flex items-center justify-center size-7 shrink-0 rounded-full bg-primary text-primary-foreground hover:bg-primary/90 disabled:opacity-30 transition-colors`,"aria-label":`Send`,children:(0,Z.jsx)(j,{className:`size-3.5`})})]}),(0,Z.jsxs)(`div`,{className:`hidden md:block`,children:[(0,Z.jsx)(`textarea`,{ref:x,value:_,onChange:e=>{N(e.target.value),P(e)},onKeyDown:A,onPaste:F,onDrop:I,onDragOver:L,placeholder:t?`Follow-up or Stop...`:`Ask anything...`,disabled:r,rows:1,className:`w-full resize-none bg-transparent px-4 pt-3 pb-1 text-sm text-foreground placeholder:text-text-subtle focus:outline-none disabled:opacity-50 max-h-40`}),(0,Z.jsxs)(`div`,{className:`flex items-center justify-between px-3 pb-2`,children:[(0,Z.jsx)(`div`,{className:`flex items-center gap-1`,children:(0,Z.jsx)(`button`,{type:`button`,onClick:e=>{e.stopPropagation(),R()},disabled:r,className:`flex items-center justify-center size-8 rounded-full text-text-subtle hover:text-text-primary hover:bg-surface-elevated transition-colors disabled:opacity-50`,"aria-label":`Attach file`,children:(0,Z.jsx)(ee,{className:`size-4`})})}),(0,Z.jsx)(`div`,{className:`flex items-center gap-1`,children:V?(0,Z.jsx)(`button`,{onClick:e=>{e.stopPropagation(),n?.()},className:`flex items-center justify-center size-8 rounded-full bg-red-600 text-white hover:bg-red-500 transition-colors`,"aria-label":`Stop response`,children:(0,Z.jsx)(H,{className:`size-3.5`})}):(0,Z.jsx)(`button`,{onClick:e=>{e.stopPropagation(),k()},disabled:r||!B,className:`flex items-center justify-center size-8 rounded-full bg-primary text-primary-foreground hover:bg-primary/90 disabled:opacity-30 disabled:cursor-not-allowed transition-colors`,"aria-label":`Send message`,children:(0,Z.jsx)(j,{className:`size-4`})})})]})]})]}),(0,Z.jsx)(`input`,{ref:C,type:`file`,multiple:!0,className:`hidden`,onChange:z})]})}function et({items:e,filter:t,onSelect:n,onClose:r,visible:i}){let[a,o]=(0,U.useState)(0),s=(0,U.useRef)(null),c=e.filter(e=>{let n=t.toLowerCase();return e.name.toLowerCase().includes(n)||e.description.toLowerCase().includes(n)});(0,U.useEffect)(()=>{o(0)},[t]),(0,U.useEffect)(()=>{let e=s.current;e&&e.children[a]?.scrollIntoView({block:`nearest`})},[a]);let l=(0,U.useCallback)(e=>{if(!i||c.length===0)return!1;switch(e.key){case`ArrowUp`:return e.preventDefault(),o(e=>e>0?e-1:c.length-1),!0;case`ArrowDown`:return e.preventDefault(),o(e=>e<c.length-1?e+1:0),!0;case`Enter`:case`Tab`:return e.preventDefault(),c[a]&&n(c[a]),!0;case`Escape`:return e.preventDefault(),r(),!0}return!1},[i,c,a,n,r]);return(0,U.useEffect)(()=>{if(!i)return;let e=e=>{l(e)};return document.addEventListener(`keydown`,e,!0),()=>document.removeEventListener(`keydown`,e,!0)},[i,l]),!i||c.length===0?null:(0,Z.jsx)(`div`,{className:`max-h-52 overflow-y-auto border-b border-border bg-surface`,children:(0,Z.jsx)(`div`,{ref:s,className:`py-1`,children:c.map((e,t)=>(0,Z.jsxs)(`button`,{className:`flex items-start gap-3 w-full px-3 py-2 text-left transition-colors ${t===a?`bg-primary/10 text-primary`:`hover:bg-surface-hover text-text-primary`}`,onMouseEnter:()=>o(t),onClick:()=>n(e),children:[(0,Z.jsx)(`span`,{className:`shrink-0 mt-0.5`,children:e.type===`skill`?(0,Z.jsx)(te,{className:`size-4 text-amber-500`}):(0,Z.jsx)(x,{className:`size-4 text-blue-500`})}),(0,Z.jsxs)(`div`,{className:`min-w-0 flex-1`,children:[(0,Z.jsxs)(`div`,{className:`flex items-baseline gap-2`,children:[(0,Z.jsxs)(`span`,{className:`font-medium text-sm`,children:[`/`,e.name]}),e.argumentHint&&(0,Z.jsx)(`span`,{className:`text-xs text-text-subtle`,children:e.argumentHint}),(0,Z.jsx)(`span`,{className:`text-xs text-text-subtle capitalize ml-auto`,children:e.scope===`user`?`global`:e.type})]}),e.description&&(0,Z.jsx)(`p`,{className:`text-xs text-text-subtle mt-0.5 line-clamp-2`,children:e.description})]})]},`${e.type}-${e.name}`))})})}function tt(e){return e>=90?`text-red-500`:e>=70?`text-amber-500`:`text-green-500`}function nt(e){return e>=90?`bg-red-500`:e>=70?`bg-amber-500`:`bg-green-500`}function rt(e){if(!e)return null;let t=null;if(e.resetsInMinutes!=null)t=e.resetsInMinutes;else if(e.resetsInHours!=null)t=Math.round(e.resetsInHours*60);else if(e.resetsAt){let n=new Date(e.resetsAt).getTime()-Date.now();t=n>0?Math.ceil(n/6e4):0}if(t==null)return null;if(t<=0)return`now`;let n=Math.floor(t/1440),r=Math.floor(t%1440/60),i=t%60;return n>0?i>0?`${n}d ${r}h ${i}m`:r>0?`${n}d ${r}h`:`${n}d`:r>0?i>0?`${r}h ${i}m`:`${r}h`:`${i}m`}function $({label:e,bucket:t}){if(!t)return null;let n=Math.round(t.utilization*100),r=rt(t);return(0,Z.jsxs)(`div`,{className:`space-y-1`,children:[(0,Z.jsxs)(`div`,{className:`flex items-center justify-between`,children:[(0,Z.jsx)(`span`,{className:`text-xs font-medium text-text-primary`,children:e}),r&&(0,Z.jsxs)(`span`,{className:`text-[10px] text-text-subtle`,children:[`↻ `,r]})]}),(0,Z.jsxs)(`div`,{className:`flex items-center gap-2`,children:[(0,Z.jsx)(`div`,{className:`flex-1 h-2 rounded-full bg-border overflow-hidden`,children:(0,Z.jsx)(`div`,{className:`h-full rounded-full transition-all ${nt(n)}`,style:{width:`${Math.min(n,100)}%`}})}),(0,Z.jsxs)(`span`,{className:`text-xs font-medium tabular-nums w-10 text-right ${tt(n)}`,children:[n,`%`]})]})]})}function it(e){if(!e)return null;let t=Math.round((Date.now()-e)/1e3);return t<5?`just now`:t<60?`${t}s ago`:`${Math.floor(t/60)}m ago`}function at({usage:e,visible:t,onClose:n,onReload:r,loading:i,lastFetchedAt:a}){if(!t)return null;let o=e.queryCostUsd!=null||e.totalCostUsd!=null,s=e.session||e.weekly||e.weeklyOpus||e.weeklySonnet;return(0,Z.jsxs)(`div`,{className:`border-b border-border bg-surface px-3 py-2.5 space-y-2.5`,children:[(0,Z.jsxs)(`div`,{className:`flex items-center justify-between`,children:[(0,Z.jsxs)(`div`,{className:`flex items-center gap-2`,children:[(0,Z.jsx)(`span`,{className:`text-xs font-semibold text-text-primary`,children:`Usage Limits`}),a&&(0,Z.jsx)(`span`,{className:`text-[10px] text-text-subtle`,children:it(new Date(a).getTime())})]}),(0,Z.jsxs)(`div`,{className:`flex items-center gap-1`,children:[r&&(0,Z.jsx)(`button`,{onClick:r,disabled:i,className:`text-xs text-text-subtle hover:text-text-primary px-1 disabled:opacity-50`,title:`Refresh usage data`,children:(0,Z.jsx)(b,{className:`size-3 ${i?`animate-spin`:``}`})}),(0,Z.jsx)(`button`,{onClick:n,className:`text-xs text-text-subtle hover:text-text-primary px-1`,children:`✕`})]})]}),s?(0,Z.jsxs)(`div`,{className:`space-y-2.5`,children:[(0,Z.jsx)($,{label:`5-Hour Session`,bucket:e.session}),(0,Z.jsx)($,{label:`Weekly`,bucket:e.weekly}),(0,Z.jsx)($,{label:`Weekly (Opus)`,bucket:e.weeklyOpus}),(0,Z.jsx)($,{label:`Weekly (Sonnet)`,bucket:e.weeklySonnet})]}):(0,Z.jsxs)(`p`,{className:`text-xs text-text-subtle`,children:[`No data — run `,(0,Z.jsx)(`code`,{className:`bg-surface-elevated px-1 rounded`,children:`bun install`})]}),o&&(0,Z.jsxs)(`div`,{className:`border-t border-border pt-2 space-y-1`,children:[e.queryCostUsd!=null&&(0,Z.jsxs)(`div`,{className:`flex items-center justify-between text-xs`,children:[(0,Z.jsx)(`span`,{className:`text-text-subtle`,children:`Last query`}),(0,Z.jsxs)(`span`,{className:`text-text-primary font-medium tabular-nums`,children:[`$`,e.queryCostUsd.toFixed(4)]})]}),e.totalCostUsd!=null&&(0,Z.jsxs)(`div`,{className:`flex items-center justify-between text-xs`,children:[(0,Z.jsx)(`span`,{className:`text-text-subtle`,children:`Session total`}),(0,Z.jsxs)(`span`,{className:`text-text-primary font-medium tabular-nums`,children:[`$`,e.totalCostUsd.toFixed(4)]})]})]})]})}function ot(e){try{return new Date(e).toLocaleDateString(void 0,{month:`short`,day:`numeric`})}catch{return``}}function st(e){return e>=90?`text-red-500`:e>=70?`text-amber-500`:`text-green-500`}function ct({projectName:e,usageInfo:t,contextWindowPct:n,usageLoading:r,refreshUsage:i,lastFetchedAt:o,sessionId:s,onSelectSession:c,onBugReport:l,isConnected:d,onReconnect:m}){let[h,g]=(0,U.useState)(null),[_,v]=(0,U.useState)([]),[y,x]=(0,U.useState)(!1),[S,C]=(0,U.useState)(``),T=a(e=>e.openTab),E=e=>{g(t=>t===e?null:e)},k=(0,U.useCallback)(async()=>{if(e){x(!0);try{v(await f.get(`${u(e)}/chat/sessions`))}catch{}finally{x(!1)}}},[e]);(0,U.useEffect)(()=>{h===`history`&&_.length===0&&k()},[h]);function j(t){c?(c(t),g(null)):T({type:`chat`,title:t.title||`Chat`,projectId:e??null,metadata:{projectName:e,sessionId:t.id},closable:!0})}let M=S.trim()?_.filter(e=>(e.title||``).toLowerCase().includes(S.toLowerCase())):_,N=t.fiveHour==null?null:Math.round(t.fiveHour*100),P=t.sevenDay==null?null:Math.round(t.sevenDay*100),F=N!=null||P!=null?st(Math.max(N??0,P??0)):`text-text-subtle`;return(0,Z.jsxs)(`div`,{className:`border-b border-border/50`,children:[(0,Z.jsxs)(`div`,{className:`flex items-center gap-1 px-2 py-1`,children:[(0,Z.jsxs)(`button`,{onClick:()=>E(`history`),className:`flex items-center gap-1 px-1.5 py-0.5 rounded text-[11px] transition-colors ${h===`history`?`text-primary bg-primary/10`:`text-text-secondary hover:text-foreground hover:bg-surface-elevated`}`,children:[(0,Z.jsx)(L,{className:`size-3`}),(0,Z.jsx)(`span`,{children:`History`})]}),(0,Z.jsx)(`button`,{onClick:()=>E(`config`),className:`p-1 rounded transition-colors ${h===`config`?`text-primary bg-primary/10`:`text-text-subtle hover:text-text-secondary hover:bg-surface-elevated`}`,title:`AI Settings`,children:(0,Z.jsx)(B,{className:`size-3`})}),(0,Z.jsxs)(`button`,{onClick:()=>E(`usage`),className:`flex items-center gap-1 px-1.5 py-0.5 rounded text-[11px] font-medium tabular-nums transition-colors hover:bg-surface-elevated ${h===`usage`?`bg-primary/10`:``} ${F}`,title:`Usage limits`,children:[(0,Z.jsx)(A,{className:`size-3`}),(0,Z.jsxs)(`span`,{children:[`5h:`,N==null?`--%`:`${N}%`]}),(0,Z.jsx)(`span`,{className:`text-text-subtle`,children:`·`}),(0,Z.jsxs)(`span`,{children:[`Wk:`,P==null?`--%`:`${P}%`]}),n!=null&&(0,Z.jsxs)(Z.Fragment,{children:[(0,Z.jsx)(`span`,{className:`text-text-subtle`,children:`·`}),(0,Z.jsxs)(`span`,{className:st(n),children:[`Ctx:`,n,`%`]})]})]}),(0,Z.jsx)(`div`,{className:`flex-1`}),m&&(0,Z.jsx)(`button`,{onClick:m,className:`size-4 flex items-center justify-center`,title:d?`Connected`:`Disconnected — click to reconnect`,children:(0,Z.jsx)(`span`,{className:`size-2 rounded-full ${d?`bg-green-500`:`bg-red-500 animate-pulse`}`})})]}),h===`history`&&(0,Z.jsxs)(`div`,{className:`border-t border-border/30 bg-surface`,children:[(0,Z.jsxs)(`div`,{className:`flex items-center gap-1.5 px-2 py-1 border-b border-border/30`,children:[(0,Z.jsx)(O,{className:`size-3 text-text-subtle shrink-0`}),(0,Z.jsx)(`input`,{type:`text`,value:S,onChange:e=>C(e.target.value),placeholder:`Search sessions...`,className:`flex-1 bg-transparent text-[11px] text-text-primary outline-none placeholder:text-text-subtle`}),(0,Z.jsx)(`button`,{onClick:k,disabled:y,className:`p-0.5 rounded text-text-subtle hover:text-text-secondary transition-colors disabled:opacity-50`,title:`Refresh`,children:(0,Z.jsx)(b,{className:`size-3 ${y?`animate-spin`:``}`})})]}),(0,Z.jsx)(`div`,{className:`max-h-[200px] overflow-y-auto`,children:y&&_.length===0?(0,Z.jsx)(`div`,{className:`flex items-center justify-center py-3`,children:(0,Z.jsx)(D,{className:`size-3.5 animate-spin text-text-subtle`})}):M.length===0?(0,Z.jsx)(`div`,{className:`flex items-center justify-center py-3 text-[11px] text-text-subtle`,children:S?`No matching sessions`:`No sessions yet`}):M.map(e=>(0,Z.jsxs)(`button`,{onClick:()=>j(e),className:`flex items-center gap-2 w-full px-3 py-1.5 text-left hover:bg-surface-elevated transition-colors`,children:[(0,Z.jsx)(p,{className:`size-3 shrink-0 text-text-subtle`}),(0,Z.jsx)(`span`,{className:`text-[11px] truncate flex-1`,children:e.title||`Untitled`}),e.updatedAt&&(0,Z.jsx)(`span`,{className:`text-[10px] text-text-subtle shrink-0`,children:ot(e.updatedAt)})]},e.id))})]}),h===`config`&&(0,Z.jsx)(`div`,{className:`border-t border-border/30 bg-surface px-3 py-2 max-h-[280px] overflow-y-auto`,children:(0,Z.jsx)(w,{compact:!0})}),h===`usage`&&(0,Z.jsx)(at,{usage:t,visible:!0,onClose:()=>g(null),onReload:i,loading:r,lastFetchedAt:o})]})}function lt({metadata:e,tabId:t}){let[n,r]=(0,U.useState)(e?.sessionId??null),[i,o]=(0,U.useState)(e?.providerId??`claude`),[c,l]=(0,U.useState)([]),[d,p]=(0,U.useState)(!1),[m,h]=(0,U.useState)(``),[g,_]=(0,U.useState)(null),[v,y]=(0,U.useState)([]),[b,x]=(0,U.useState)(!1),[C,w]=(0,U.useState)(``),[E,D]=(0,U.useState)(null),[O,k]=(0,U.useState)(!1),[A,j]=(0,U.useState)(null),M=(0,U.useRef)(0),N=e?.projectName??``,P=a(e=>e.updateTab),F=s(e=>e.version),{usageInfo:I,usageLoading:L,lastFetchedAt:R,refreshUsage:z}=ce(N,i);(0,U.useEffect)(()=>{!t||!n||P(t,{metadata:{...e,sessionId:n,providerId:i}})},[n,i]);let{messages:ee,messagesLoading:B,isStreaming:V,streamingStatus:te,connectingElapsed:H,thinkingWarningThreshold:re,pendingApproval:ie,contextWindowPct:ae,sendMessage:W,respondToApproval:se,cancelStreaming:le,reconnect:ue,refetchMessages:de,isConnected:G}=oe(n,i,N),K=(0,U.useRef)(e?.pendingMessage);(0,U.useEffect)(()=>{if(K.current&&G&&n){let n=K.current;K.current=void 0,t&&P(t,{metadata:{...e,pendingMessage:void 0}}),setTimeout(()=>W(n),100)}},[G,n]),(0,U.useCallback)(()=>{a.getState().openTab({type:`chat`,title:`AI Chat`,metadata:{projectName:N},projectId:N||null,closable:!0})},[N]);let fe=(0,U.useCallback)(e=>{r(e.id),o(e.providerId)},[]),q=(0,U.useCallback)(async e=>{if(!(!n||!N))try{let{api:t,projectUrl:r}=await S(async()=>{let{api:e,projectUrl:t}=await import(`./api-client-BxCvlogn.js`).then(e=>e.n);return{api:e,projectUrl:t}},__vite__mapDeps([0,1])),o=await t.post(`${r(N)}/chat/sessions/${n}/fork?providerId=${i}`);a.getState().openTab({type:`chat`,title:`Fork: ${e.slice(0,30)}`,metadata:{projectName:N,sessionId:o.id,providerId:i,pendingMessage:e},projectId:N||null,closable:!0})}catch(e){console.error(`Fork failed:`,e)}},[n,N,i]),J=(0,U.useCallback)((e,t)=>{if(t.length===0)return e;let n=t.filter(e=>e.serverPath).map(e=>e.serverPath).join(`
|
|
7
|
-
`);return n?(t.length===1?`[Attached file: ${n}]\n\n`:`[Attached files:\n${n}\n]\n\n`)+e:e},[]),Y=(0,U.useCallback)(async(e,t=[])=>{let a=J(e,t);if(a.trim()){if(!n)try{let t=N,n=await f.post(`${u(t)}/chat/sessions`,{providerId:i,title:e.slice(0,50)});r(n.id),o(n.providerId),setTimeout(()=>{W(a)},500);return}catch(e){console.error(`Failed to create session:`,e);return}W(a)}},[n,i,N,W,J]),pe=(0,U.useCallback)((e,t)=>{p(e),h(t)},[]),me=(0,U.useCallback)(e=>{_(e),p(!1),h(``),setTimeout(()=>_(null),50)},[]),X=(0,U.useCallback)(()=>{p(!1),h(``)},[]),he=(0,U.useCallback)((e,t)=>{x(e),w(t)},[]),ge=(0,U.useCallback)(e=>{D(e),x(!1),w(``),setTimeout(()=>D(null),50)},[]),_e=(0,U.useCallback)(()=>{x(!1),w(``)},[]);return(0,Z.jsxs)(`div`,{className:`flex flex-col h-full relative`,onDragEnter:(0,U.useCallback)(e=>{e.preventDefault(),M.current++,e.dataTransfer.types.includes(`Files`)&&k(!0)},[]),onDragLeave:(0,U.useCallback)(e=>{e.preventDefault(),M.current--,M.current===0&&k(!1)},[]),onDragOver:(0,U.useCallback)(e=>{e.preventDefault()},[]),onDrop:(0,U.useCallback)(e=>{e.preventDefault(),M.current=0,k(!1);let t=Array.from(e.dataTransfer.files);t.length>0&&(j(t),setTimeout(()=>j(null),100))},[]),children:[O&&(0,Z.jsx)(`div`,{className:`absolute inset-0 z-50 flex items-center justify-center bg-background/80 backdrop-blur-sm border-2 border-dashed border-primary rounded-lg pointer-events-none`,children:(0,Z.jsxs)(`div`,{className:`flex flex-col items-center gap-2 text-primary`,children:[(0,Z.jsx)(ne,{className:`size-8`}),(0,Z.jsx)(`span`,{className:`text-sm font-medium`,children:`Drop files to attach`})]})}),(0,Z.jsx)(Te,{messages:ee,messagesLoading:B,pendingApproval:ie,onApprovalResponse:se,isStreaming:V,streamingStatus:te,connectingElapsed:H,thinkingWarningThreshold:re,projectName:N,onFork:V?void 0:q}),(0,Z.jsxs)(`div`,{className:`border-t border-border bg-background shrink-0`,children:[(0,Z.jsx)(ct,{projectName:N,usageInfo:I,contextWindowPct:ae,usageLoading:L,refreshUsage:z,lastFetchedAt:R,sessionId:n,onSelectSession:fe,onBugReport:n?()=>T(F,{sessionId:n,projectName:N}):void 0,isConnected:G,onReconnect:()=>{G||ue(),de()}}),(0,Z.jsx)(et,{items:c,filter:m,onSelect:me,onClose:X,visible:d}),(0,Z.jsx)(Qe,{items:v,filter:C,onSelect:ge,onClose:_e,visible:b}),(0,Z.jsx)($e,{onSend:Y,isStreaming:V,onCancel:le,projectName:N,onSlashStateChange:pe,onSlashItemsLoaded:l,slashSelected:g,onFileStateChange:he,onFileItemsLoaded:y,fileSelected:E,externalFiles:A})]})]})}export{lt as ChatTab};
|