reasonix 0.40.0 → 0.43.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/README.md +47 -16
- package/README.zh-CN.md +19 -13
- package/dashboard/app.css +8 -4
- package/dashboard/dist/app.js +377 -227
- package/dashboard/dist/app.js.map +1 -1
- package/dist/cli/acp-DAGPCVFZ.js +713 -0
- package/dist/cli/acp-DAGPCVFZ.js.map +1 -0
- package/dist/cli/chat-7ES4IBNH.js +50 -0
- package/dist/cli/{chunk-E46ECXJD.js → chunk-2425HK6U.js} +2 -1
- package/dist/cli/{chunk-E46ECXJD.js.map → chunk-2425HK6U.js.map} +1 -1
- package/dist/cli/chunk-25T6CVUP.js +172 -0
- package/dist/cli/chunk-25T6CVUP.js.map +1 -0
- package/dist/cli/{chunk-7DLHHBGN.js → chunk-2K65GZBT.js} +16 -5
- package/dist/cli/chunk-2K65GZBT.js.map +1 -0
- package/dist/cli/{chunk-KMWKGPFZ.js → chunk-2KDUS647.js} +14 -4
- package/dist/cli/chunk-2KDUS647.js.map +1 -0
- package/dist/cli/chunk-2R4QCDOZ.js +11392 -0
- package/dist/cli/chunk-2R4QCDOZ.js.map +1 -0
- package/dist/cli/{chunk-3Q3C4W66.js → chunk-2UQP6H6T.js} +2 -1
- package/dist/cli/{chunk-3Q3C4W66.js.map → chunk-2UQP6H6T.js.map} +1 -1
- package/dist/cli/chunk-2Z35JOA4.js +96 -0
- package/dist/cli/chunk-2Z35JOA4.js.map +1 -0
- package/dist/cli/chunk-32TIKD5U.js +54 -0
- package/dist/cli/{chunk-JWCTX5S4.js.map → chunk-32TIKD5U.js.map} +1 -1
- package/dist/cli/{chunk-UVRXTSK3.js → chunk-3BXRZFWS.js} +65 -3
- package/dist/cli/chunk-3BXRZFWS.js.map +1 -0
- package/dist/cli/chunk-3Z6IBU3D.js +249 -0
- package/dist/cli/chunk-3Z6IBU3D.js.map +1 -0
- package/dist/cli/{chunk-VLNRQMCI.js → chunk-45U62RI3.js} +12 -5
- package/dist/cli/chunk-45U62RI3.js.map +1 -0
- package/dist/cli/{chunk-5GKJLNP2.js → chunk-4QUNBQQ2.js} +3 -2
- package/dist/cli/{chunk-5GKJLNP2.js.map → chunk-4QUNBQQ2.js.map} +1 -1
- package/dist/cli/{chunk-R4YTW7PR.js → chunk-5JJRUIPA.js} +57 -12
- package/dist/cli/chunk-5JJRUIPA.js.map +1 -0
- package/dist/cli/{chunk-HCC42PEI.js → chunk-6AK4EY3D.js} +12 -6
- package/dist/cli/chunk-6AK4EY3D.js.map +1 -0
- package/dist/cli/chunk-6G3CUUFG.js +34320 -0
- package/dist/cli/chunk-6G3CUUFG.js.map +1 -0
- package/dist/cli/{chunk-XST7BSZJ.js → chunk-6PBZN4VI.js} +21 -3
- package/dist/cli/chunk-6PBZN4VI.js.map +1 -0
- package/dist/cli/{chunk-A5LSGEEK.js → chunk-6PZ3CXBP.js} +88 -66
- package/dist/cli/chunk-6PZ3CXBP.js.map +1 -0
- package/dist/cli/chunk-74EX7SUH.js +25293 -0
- package/dist/cli/chunk-74EX7SUH.js.map +1 -0
- package/dist/cli/{chunk-FFNOMR32.js → chunk-7O5ALB4C.js} +3 -2
- package/dist/cli/{chunk-FFNOMR32.js.map → chunk-7O5ALB4C.js.map} +1 -1
- package/dist/cli/{chunk-UCMTWZKU.js → chunk-DOYHN4KB.js} +3 -2
- package/dist/cli/{chunk-UCMTWZKU.js.map → chunk-DOYHN4KB.js.map} +1 -1
- package/dist/cli/{chunk-XJLZ4HKU.js → chunk-F3PXYSNN.js} +3 -2
- package/dist/cli/{chunk-XJLZ4HKU.js.map → chunk-F3PXYSNN.js.map} +1 -1
- package/dist/cli/{chunk-XHQIK7B6.js → chunk-FHOGSSCH.js} +4 -3
- package/dist/cli/{chunk-XHQIK7B6.js.map → chunk-FHOGSSCH.js.map} +1 -1
- package/dist/cli/{chunk-IYF36OCJ.js → chunk-H6PS7IUE.js} +3 -2
- package/dist/cli/{chunk-IYF36OCJ.js.map → chunk-H6PS7IUE.js.map} +1 -1
- package/dist/cli/{chunk-ZTLZO42A.js → chunk-HFEAY5DT.js} +3 -2
- package/dist/cli/{chunk-ZTLZO42A.js.map → chunk-HFEAY5DT.js.map} +1 -1
- package/dist/cli/{chunk-FWGEHRB7.js → chunk-J5XJHLWM.js} +2 -1
- package/dist/cli/{chunk-FWGEHRB7.js.map → chunk-J5XJHLWM.js.map} +1 -1
- package/dist/cli/chunk-JMBMLOBP.js +26 -0
- package/dist/cli/chunk-JMBMLOBP.js.map +1 -0
- package/dist/cli/{chunk-SZH34P45.js → chunk-O52OLQL3.js} +52 -18
- package/dist/cli/chunk-O52OLQL3.js.map +1 -0
- package/dist/cli/{chunk-4DCHFFEY.js → chunk-OSZC7C6F.js} +3 -2
- package/dist/cli/{chunk-4DCHFFEY.js.map → chunk-OSZC7C6F.js.map} +1 -1
- package/dist/cli/chunk-P7EKE5ZQ.js +60641 -0
- package/dist/cli/chunk-P7EKE5ZQ.js.map +1 -0
- package/dist/cli/{chunk-FM57FNPJ.js → chunk-PLHAZOLZ.js} +2 -1
- package/dist/cli/{chunk-FM57FNPJ.js.map → chunk-PLHAZOLZ.js.map} +1 -1
- package/dist/cli/{chunk-RFX7TYVV.js → chunk-PQXPXJBJ.js} +16 -2
- package/dist/cli/chunk-PQXPXJBJ.js.map +1 -0
- package/dist/cli/{chunk-DAEAAVDF.js → chunk-PV55UMTO.js} +2 -1
- package/dist/cli/{chunk-DAEAAVDF.js.map → chunk-PV55UMTO.js.map} +1 -1
- package/dist/cli/{chunk-H7PHYVPM.js → chunk-RE4RAVFF.js} +85 -14
- package/dist/cli/chunk-RE4RAVFF.js.map +1 -0
- package/dist/cli/chunk-S4XVGLRW.js +499 -0
- package/dist/cli/chunk-S4XVGLRW.js.map +1 -0
- package/dist/cli/{chunk-WJ3YX4PZ.js → chunk-SZ5XES2N.js} +3 -2
- package/dist/cli/{chunk-WJ3YX4PZ.js.map → chunk-SZ5XES2N.js.map} +1 -1
- package/dist/cli/{chunk-4X3NY5ZM.js → chunk-TJX6BFZZ.js} +16 -9
- package/dist/cli/{chunk-4X3NY5ZM.js.map → chunk-TJX6BFZZ.js.map} +1 -1
- package/dist/cli/chunk-TUK7OWJA.js +51 -0
- package/dist/cli/{chunk-WKOMCPXP.js → chunk-VK5HG73G.js} +26 -17
- package/dist/cli/chunk-VK5HG73G.js.map +1 -0
- package/dist/cli/{chunk-CLAN6PVH.js → chunk-XCGGEJTI.js} +21 -8
- package/dist/cli/chunk-XCGGEJTI.js.map +1 -0
- package/dist/cli/{chunk-SOZE7V7V.js → chunk-XJXDHAES.js} +3 -2
- package/dist/cli/{chunk-SOZE7V7V.js.map → chunk-XJXDHAES.js.map} +1 -1
- package/dist/cli/chunk-XPDVG52A.js +2648 -0
- package/dist/cli/chunk-XPDVG52A.js.map +1 -0
- package/dist/cli/{chunk-CRPQUBP6.js → chunk-XXC2BYTV.js} +2 -1
- package/dist/cli/{chunk-CRPQUBP6.js.map → chunk-XXC2BYTV.js.map} +1 -1
- package/dist/cli/{chunk-AVB3WZWU.js → chunk-YFGF5NKA.js} +17 -14
- package/dist/cli/{chunk-AVB3WZWU.js.map → chunk-YFGF5NKA.js.map} +1 -1
- package/dist/cli/{chunk-ORM6PK57.js → chunk-YQ6NTIIE.js} +2 -1
- package/dist/cli/{chunk-ORM6PK57.js.map → chunk-YQ6NTIIE.js.map} +1 -1
- package/dist/cli/{chunk-ULBW7DYL.js → chunk-YYQAUTTN.js} +3 -2
- package/dist/cli/{chunk-ULBW7DYL.js.map → chunk-YYQAUTTN.js.map} +1 -1
- package/dist/cli/chunk-ZZM6QJ4W.js +109 -0
- package/dist/cli/chunk-ZZM6QJ4W.js.map +1 -0
- package/dist/cli/code-SMKEW6CD.js +154 -0
- package/dist/cli/code-SMKEW6CD.js.map +1 -0
- package/dist/cli/{commands-FQZOBLLZ.js → commands-FVVB5FZF.js} +7 -5
- package/dist/cli/{commands-FQZOBLLZ.js.map → commands-FVVB5FZF.js.map} +1 -1
- package/dist/cli/{commit-ZS24SHPG.js → commit-HE4VSPZ7.js} +7 -4
- package/dist/cli/{commit-ZS24SHPG.js.map → commit-HE4VSPZ7.js.map} +1 -1
- package/dist/cli/{desktop-6OLENOOO.js → desktop-Q7NDXCON.js} +379 -72
- package/dist/cli/desktop-Q7NDXCON.js.map +1 -0
- package/dist/cli/devtools-YECO25QO.js +3719 -0
- package/dist/cli/devtools-YECO25QO.js.map +1 -0
- package/dist/cli/diff-435UTPC5.js +165 -0
- package/dist/cli/{diff-2VUKNGEI.js.map → diff-435UTPC5.js.map} +1 -1
- package/dist/cli/doctor-OT7KH75K.js +27 -0
- package/dist/cli/{events-APSVNROZ.js → events-XEFAD5VX.js} +6 -4
- package/dist/cli/{events-APSVNROZ.js.map → events-XEFAD5VX.js.map} +1 -1
- package/dist/cli/index.js +3233 -123
- package/dist/cli/index.js.map +1 -1
- package/dist/cli/{mcp-DCKOE5RF.js → mcp-WUL2WO75.js} +6 -4
- package/dist/cli/{mcp-DCKOE5RF.js.map → mcp-WUL2WO75.js.map} +1 -1
- package/dist/cli/{mcp-browse-D6GBP5RQ.js → mcp-browse-RR7R4XET.js} +34 -19
- package/dist/cli/mcp-browse-RR7R4XET.js.map +1 -0
- package/dist/cli/{mcp-inspect-KFGFPJ3E.js → mcp-inspect-REGLYBWT.js} +9 -8
- package/dist/cli/{mcp-inspect-KFGFPJ3E.js.map → mcp-inspect-REGLYBWT.js.map} +1 -1
- package/dist/cli/package.json +3 -0
- package/dist/cli/prompt-UW6EFLVR.js +16 -0
- package/dist/cli/{prune-sessions-LV33R47N.js → prune-sessions-3RWUBYRS.js} +4 -2
- package/dist/cli/{prune-sessions-LV33R47N.js.map → prune-sessions-3RWUBYRS.js.map} +1 -1
- package/dist/cli/{replay-WFCYX7XF.js → replay-YOURXV4C.js} +42 -30
- package/dist/cli/{replay-WFCYX7XF.js.map → replay-YOURXV4C.js.map} +1 -1
- package/dist/cli/{run-IUJYEPMT.js → run-Q6BUXV66.js} +28 -27
- package/dist/cli/{run-IUJYEPMT.js.map → run-Q6BUXV66.js.map} +1 -1
- package/dist/cli/{server-CN4QPPVJ.js → server-XGDBRWMB.js} +44 -43
- package/dist/cli/server-XGDBRWMB.js.map +1 -0
- package/dist/cli/{sessions-F5GPGTJN.js → sessions-FH7QVYSY.js} +22 -19
- package/dist/cli/{sessions-F5GPGTJN.js.map → sessions-FH7QVYSY.js.map} +1 -1
- package/dist/cli/setup-VDS6SVEP.js +618 -0
- package/dist/cli/setup-VDS6SVEP.js.map +1 -0
- package/dist/cli/stats-MQVI2XQH.js +14 -0
- package/dist/cli/update-6ITLPRDV.js +15 -0
- package/dist/cli/update-6ITLPRDV.js.map +1 -0
- package/dist/cli/version-DAHGZY5N.js +33 -0
- package/dist/cli/{version-KQUPV6T5.js.map → version-DAHGZY5N.js.map} +1 -1
- package/dist/index.d.ts +157 -103
- package/dist/index.js +597 -178
- package/dist/index.js.map +1 -1
- package/package.json +2 -1
- package/dist/cli/chat-G7CUW4ZI.js +0 -45
- package/dist/cli/chunk-26UDIXLD.js +0 -16481
- package/dist/cli/chunk-26UDIXLD.js.map +0 -1
- package/dist/cli/chunk-4YV2GBYG.js +0 -5237
- package/dist/cli/chunk-4YV2GBYG.js.map +0 -1
- package/dist/cli/chunk-5X7LZJDE.js +0 -36
- package/dist/cli/chunk-5X7LZJDE.js.map +0 -1
- package/dist/cli/chunk-7DLHHBGN.js.map +0 -1
- package/dist/cli/chunk-A5LSGEEK.js.map +0 -1
- package/dist/cli/chunk-AFFZF3MW.js +0 -36
- package/dist/cli/chunk-AFFZF3MW.js.map +0 -1
- package/dist/cli/chunk-CLAN6PVH.js.map +0 -1
- package/dist/cli/chunk-CPOV2O73.js +0 -39
- package/dist/cli/chunk-CPOV2O73.js.map +0 -1
- package/dist/cli/chunk-CPTZ5OHX.js +0 -18
- package/dist/cli/chunk-CPTZ5OHX.js.map +0 -1
- package/dist/cli/chunk-CZSJILQP.js +0 -854
- package/dist/cli/chunk-CZSJILQP.js.map +0 -1
- package/dist/cli/chunk-H7PHYVPM.js.map +0 -1
- package/dist/cli/chunk-HCC42PEI.js.map +0 -1
- package/dist/cli/chunk-JWCTX5S4.js +0 -46
- package/dist/cli/chunk-KMWKGPFZ.js.map +0 -1
- package/dist/cli/chunk-MRLXEMZ7.js +0 -26
- package/dist/cli/chunk-MRLXEMZ7.js.map +0 -1
- package/dist/cli/chunk-R4YTW7PR.js.map +0 -1
- package/dist/cli/chunk-RFX7TYVV.js.map +0 -1
- package/dist/cli/chunk-SZH34P45.js.map +0 -1
- package/dist/cli/chunk-UVRXTSK3.js.map +0 -1
- package/dist/cli/chunk-VLNRQMCI.js.map +0 -1
- package/dist/cli/chunk-WKOMCPXP.js.map +0 -1
- package/dist/cli/chunk-XST7BSZJ.js.map +0 -1
- package/dist/cli/code-YQGVLIT2.js +0 -147
- package/dist/cli/code-YQGVLIT2.js.map +0 -1
- package/dist/cli/desktop-6OLENOOO.js.map +0 -1
- package/dist/cli/diff-2VUKNGEI.js +0 -153
- package/dist/cli/doctor-JO2WNN6C.js +0 -24
- package/dist/cli/mcp-browse-D6GBP5RQ.js.map +0 -1
- package/dist/cli/prompt-PKCCLLAD.js +0 -13
- package/dist/cli/server-CN4QPPVJ.js.map +0 -1
- package/dist/cli/setup-WWMDBPSB.js +0 -516
- package/dist/cli/setup-WWMDBPSB.js.map +0 -1
- package/dist/cli/stats-5RJCATCE.js +0 -12
- package/dist/cli/update-GUCWB4UN.js +0 -13
- package/dist/cli/version-KQUPV6T5.js +0 -30
- /package/dist/cli/{chat-G7CUW4ZI.js.map → chat-7ES4IBNH.js.map} +0 -0
- /package/dist/cli/{doctor-JO2WNN6C.js.map → chunk-TUK7OWJA.js.map} +0 -0
- /package/dist/cli/{prompt-PKCCLLAD.js.map → doctor-OT7KH75K.js.map} +0 -0
- /package/dist/cli/{stats-5RJCATCE.js.map → prompt-UW6EFLVR.js.map} +0 -0
- /package/dist/cli/{update-GUCWB4UN.js.map → stats-MQVI2XQH.js.map} +0 -0
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
+
import { createRequire as __cr } from 'node:module'; if (typeof globalThis.require === 'undefined') { globalThis.require = __cr(import.meta.url); }
|
|
2
3
|
|
|
3
4
|
// src/memory/session.ts
|
|
4
5
|
import { execFileSync } from "child_process";
|
|
@@ -15,7 +16,7 @@ import {
|
|
|
15
16
|
writeFileSync
|
|
16
17
|
} from "fs";
|
|
17
18
|
import { homedir } from "os";
|
|
18
|
-
import { dirname, join } from "path";
|
|
19
|
+
import { dirname, join, posix as posixPath, win32 as win32Path } from "path";
|
|
19
20
|
function detectGitBranch(cwd) {
|
|
20
21
|
try {
|
|
21
22
|
const out = execFileSync("git", ["branch", "--show-current"], {
|
|
@@ -42,6 +43,11 @@ function sanitizeName(name) {
|
|
|
42
43
|
function timestampSuffix() {
|
|
43
44
|
return (/* @__PURE__ */ new Date()).toISOString().replace(/[^\d]/g, "").slice(0, 12);
|
|
44
45
|
}
|
|
46
|
+
function freshSessionName(currentName) {
|
|
47
|
+
const base = currentName ? currentName.replace(/-\d{12,14}$/, "") : "default";
|
|
48
|
+
const stamp = (/* @__PURE__ */ new Date()).toISOString().replace(/[^\d]/g, "").slice(0, 14);
|
|
49
|
+
return `${base || "default"}-${stamp}`;
|
|
50
|
+
}
|
|
45
51
|
function findSessionsByPrefix(prefix) {
|
|
46
52
|
const dir = sessionsDir();
|
|
47
53
|
if (!existsSync(dir)) return [];
|
|
@@ -133,8 +139,19 @@ function listSessions() {
|
|
|
133
139
|
return [];
|
|
134
140
|
}
|
|
135
141
|
}
|
|
142
|
+
function normalizeWorkspace(p, platform = process.platform) {
|
|
143
|
+
if (typeof p !== "string" || p.length === 0) return "";
|
|
144
|
+
if (platform === "win32") {
|
|
145
|
+
const resolved = win32Path.resolve(p);
|
|
146
|
+
return resolved.replace(/\\/g, "/").replace(/^([A-Z]):/i, (_, d) => `${d.toLowerCase()}:`);
|
|
147
|
+
}
|
|
148
|
+
return posixPath.resolve(p);
|
|
149
|
+
}
|
|
136
150
|
function listSessionsForWorkspace(workspace) {
|
|
137
|
-
|
|
151
|
+
const want = normalizeWorkspace(workspace);
|
|
152
|
+
return listSessions().filter(
|
|
153
|
+
(s) => typeof s.meta.workspace === "string" && normalizeWorkspace(s.meta.workspace) === want
|
|
154
|
+
);
|
|
138
155
|
}
|
|
139
156
|
function metaPath(name) {
|
|
140
157
|
return join(sessionsDir(), `${sanitizeName(name)}.meta.json`);
|
|
@@ -247,6 +264,7 @@ export {
|
|
|
247
264
|
sessionPath,
|
|
248
265
|
sanitizeName,
|
|
249
266
|
timestampSuffix,
|
|
267
|
+
freshSessionName,
|
|
250
268
|
resolveSession,
|
|
251
269
|
loadSessionMessages,
|
|
252
270
|
appendSessionMessage,
|
|
@@ -260,4 +278,4 @@ export {
|
|
|
260
278
|
rewriteSession,
|
|
261
279
|
archiveSession
|
|
262
280
|
};
|
|
263
|
-
//# sourceMappingURL=chunk-
|
|
281
|
+
//# sourceMappingURL=chunk-6PBZN4VI.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/memory/session.ts"],"sourcesContent":["/** JSONL append-only message log under `~/.reasonix/sessions/`; concurrent-write safe. */\n\nimport { execFileSync } from \"node:child_process\";\nimport {\n appendFileSync,\n chmodSync,\n existsSync,\n mkdirSync,\n readFileSync,\n readdirSync,\n renameSync,\n statSync,\n unlinkSync,\n writeFileSync,\n} from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport { dirname, join, posix as posixPath, win32 as win32Path } from \"node:path\";\nimport type { ChatMessage } from \"../types.js\";\n\n/** Best-effort git branch sniff; returns undefined if not a git repo or git missing. */\nexport function detectGitBranch(cwd: string): string | undefined {\n try {\n const out = execFileSync(\"git\", [\"branch\", \"--show-current\"], {\n cwd,\n stdio: [\"ignore\", \"pipe\", \"ignore\"],\n timeout: 800,\n encoding: \"utf8\",\n }).trim();\n return out || undefined;\n } catch {\n return undefined;\n }\n}\n\nexport interface SessionInfo {\n name: string;\n path: string;\n size: number;\n messageCount: number;\n mtime: Date;\n meta: SessionMeta;\n}\n\nexport interface SessionMeta {\n branch?: string;\n summary?: string;\n totalCostUsd?: number;\n turnCount?: number;\n /** Absolute path of the workspace root the session was created/used in. */\n workspace?: string;\n /** Wallet currency at last save — used to format `totalCostUsd` in the picker without re-fetching balance. */\n balanceCurrency?: string;\n /** Cumulative cache hit / miss tokens across the session — survives resume so /status cache% isn't 0 on a fresh boot. */\n cacheHitTokens?: number;\n cacheMissTokens?: number;\n /** Last turn's promptTokens — lets /status render the context bar before the next turn fires. */\n lastPromptTokens?: number;\n}\n\nexport function sessionsDir(): string {\n return join(homedir(), \".reasonix\", \"sessions\");\n}\n\nexport function sessionPath(name: string): string {\n return join(sessionsDir(), `${sanitizeName(name)}.jsonl`);\n}\n\nexport function sanitizeName(name: string): string {\n const cleaned = name.replace(/[^\\w\\-\\u4e00-\\u9fa5]/g, \"_\").slice(0, 64);\n return cleaned || \"default\";\n}\n\n/** Sortable timestamp `YYYYMMDDHHmm` — used as a session-name suffix. */\nexport function timestampSuffix(): string {\n return new Date().toISOString().replace(/[^\\d]/g, \"\").slice(0, 12);\n}\n\n/** Unique name for an in-app \"new session\" — strips a trailing 12/14-digit timestamp from the current name and re-stamps with seconds precision so back-to-back clicks don't collide. */\nexport function freshSessionName(currentName: string | undefined): string {\n const base = currentName ? currentName.replace(/-\\d{12,14}$/, \"\") : \"default\";\n const stamp = new Date().toISOString().replace(/[^\\d]/g, \"\").slice(0, 14);\n return `${base || \"default\"}-${stamp}`;\n}\n\n/** Names of `.jsonl` sessions starting with `prefix`, newest-first by filename. */\nexport function findSessionsByPrefix(prefix: string): string[] {\n const dir = sessionsDir();\n if (!existsSync(dir)) return [];\n try {\n const files = readdirSync(dir)\n .filter((f) => f.endsWith(\".jsonl\") && !f.endsWith(\".events.jsonl\") && f.startsWith(prefix))\n .sort()\n .reverse();\n return files.map((f) => f.replace(/\\.jsonl$/, \"\"));\n } catch {\n return [];\n }\n}\n\nexport interface SessionPreview {\n messageCount: number;\n lastActive: Date;\n}\n\n/** Resolve launch-time session: forceNew → timestamped suffix; else latest `${name}-*` if any, else base. Preview returned only on the default branch when messages exist. */\nexport function resolveSession(\n sessionName: string | undefined,\n forceNew?: boolean,\n forceResume?: boolean,\n): { resolved: string | undefined; preview: SessionPreview | undefined } {\n let resolved = sessionName;\n let preview: SessionPreview | undefined;\n\n if (sessionName && forceNew) {\n resolved = `${sessionName}-${timestampSuffix()}`;\n } else if (sessionName && !forceResume) {\n let sessionToCheck = sessionName;\n const prefixed = findSessionsByPrefix(`${sessionName}-`);\n if (prefixed.length > 0) {\n sessionToCheck = prefixed[0]!;\n }\n const prior = loadSessionMessages(sessionToCheck);\n if (prior.length > 0) {\n resolved = sessionToCheck;\n const p = sessionPath(sessionToCheck);\n const mtime = existsSync(p) ? statSync(p).mtime : new Date();\n preview = { messageCount: prior.length, lastActive: mtime };\n }\n } else if (sessionName && forceResume) {\n const prefixed = findSessionsByPrefix(`${sessionName}-`);\n if (prefixed.length > 0) {\n resolved = prefixed[0]!;\n }\n }\n\n return { resolved, preview };\n}\n\nexport function loadSessionMessages(name: string): ChatMessage[] {\n const path = sessionPath(name);\n if (!existsSync(path)) return [];\n try {\n const raw = readFileSync(path, \"utf8\");\n const out: ChatMessage[] = [];\n for (const line of raw.split(/\\r?\\n/)) {\n const trimmed = line.trim();\n if (!trimmed) continue;\n try {\n const msg = JSON.parse(trimmed) as ChatMessage;\n if (msg && typeof msg === \"object\" && \"role\" in msg) out.push(msg);\n } catch {\n /* skip malformed line */\n }\n }\n return out;\n } catch {\n return [];\n }\n}\n\nexport function appendSessionMessage(name: string, message: ChatMessage): void {\n const path = sessionPath(name);\n mkdirSync(dirname(path), { recursive: true });\n appendFileSync(path, `${JSON.stringify(message)}\\n`, \"utf8\");\n try {\n chmodSync(path, 0o600);\n } catch {\n /* chmod not supported on this platform */\n }\n}\n\nexport function listSessions(): SessionInfo[] {\n const dir = sessionsDir();\n if (!existsSync(dir)) return [];\n try {\n // Exclude `.events.jsonl` sidecars — they share the .jsonl suffix.\n const files = readdirSync(dir).filter(\n (f) => f.endsWith(\".jsonl\") && !f.endsWith(\".events.jsonl\"),\n );\n return files\n .map((file) => {\n const path = join(dir, file);\n const stat = statSync(path);\n const name = file.replace(/\\.jsonl$/, \"\");\n const messageCount = countLines(path);\n return {\n name,\n path,\n size: stat.size,\n messageCount,\n mtime: stat.mtime,\n meta: loadSessionMeta(name),\n };\n })\n .sort((a, b) => b.mtime.getTime() - a.mtime.getTime());\n } catch {\n return [];\n }\n}\n\n/** Canonical form for workspace path comparisons — Windows drive-case + separator drift between session writes (yesterday) and reads (today) used to hide sessions from the sidebar. Issue #878. */\nexport function normalizeWorkspace(\n p: string | undefined,\n platform: NodeJS.Platform = process.platform,\n): string {\n if (typeof p !== \"string\" || p.length === 0) return \"\";\n if (platform === \"win32\") {\n const resolved = win32Path.resolve(p);\n return resolved\n .replace(/\\\\/g, \"/\")\n .replace(/^([A-Z]):/i, (_, d: string) => `${d.toLowerCase()}:`);\n }\n return posixPath.resolve(p);\n}\n\n/** Sessions without `meta.workspace` are still hidden — resume by name still works. */\nexport function listSessionsForWorkspace(workspace: string): SessionInfo[] {\n const want = normalizeWorkspace(workspace);\n return listSessions().filter(\n (s) => typeof s.meta.workspace === \"string\" && normalizeWorkspace(s.meta.workspace) === want,\n );\n}\n\nfunction metaPath(name: string): string {\n return join(sessionsDir(), `${sanitizeName(name)}.meta.json`);\n}\n\nexport function loadSessionMeta(name: string): SessionMeta {\n const p = metaPath(name);\n if (!existsSync(p)) return {};\n try {\n const raw = JSON.parse(readFileSync(p, \"utf8\")) as SessionMeta;\n return raw && typeof raw === \"object\" ? raw : {};\n } catch {\n return {};\n }\n}\n\nexport function patchSessionMeta(name: string, patch: Partial<SessionMeta>): SessionMeta {\n const cur = loadSessionMeta(name);\n const next: SessionMeta = { ...cur, ...patch };\n const p = metaPath(name);\n mkdirSync(dirname(p), { recursive: true });\n writeFileSync(p, JSON.stringify(next), \"utf8\");\n try {\n chmodSync(p, 0o600);\n } catch {\n /* chmod not supported */\n }\n return next;\n}\n\n/** Renames the JSONL plus all known sidecars together; returns false if target already exists. */\nexport function renameSession(oldName: string, newName: string): boolean {\n const safeOld = sanitizeName(oldName);\n const safeNew = sanitizeName(newName);\n if (safeOld === safeNew) return false;\n const oldJsonl = sessionPath(oldName);\n const newJsonl = sessionPath(newName);\n if (!existsSync(oldJsonl) || existsSync(newJsonl)) return false;\n renameSync(oldJsonl, newJsonl);\n for (const ext of [\".events.jsonl\", \".meta.json\", \".pending.json\", \".plan.json\"]) {\n const oldP = oldJsonl.replace(/\\.jsonl$/, ext);\n const newP = newJsonl.replace(/\\.jsonl$/, ext);\n if (existsSync(oldP)) {\n try {\n renameSync(oldP, newP);\n } catch {\n /* sidecar rename failed — leave the jsonl rename in place */\n }\n }\n }\n return true;\n}\n\n/** Best-effort: per-file delete errors are swallowed so partial pruning still finishes. */\nexport function pruneStaleSessions(daysOld = 90): string[] {\n const cutoff = Date.now() - daysOld * 24 * 60 * 60 * 1000;\n const deleted: string[] = [];\n for (const s of listSessions()) {\n if (s.mtime.getTime() < cutoff) {\n if (deleteSession(s.name)) deleted.push(s.name);\n }\n }\n return deleted;\n}\n\nexport function deleteSession(name: string): boolean {\n const path = sessionPath(name);\n try {\n unlinkSync(path);\n for (const ext of [\".events.jsonl\", \".pending.json\", \".meta.json\", \".plan.json\"]) {\n const sidecar = path.replace(/\\.jsonl$/, ext);\n try {\n unlinkSync(sidecar);\n } catch {\n /* expected when the sidecar doesn't exist */\n }\n }\n return true;\n } catch {\n return false;\n }\n}\n\n/** Non-atomic truncate+write window is acceptable — a concurrent crash leaves the session file empty, same end state as the user deleting it. */\nexport function rewriteSession(name: string, messages: ChatMessage[]): void {\n const path = sessionPath(name);\n mkdirSync(dirname(path), { recursive: true });\n const body = messages.map((m) => JSON.stringify(m)).join(\"\\n\");\n writeFileSync(path, body ? `${body}\\n` : \"\", \"utf8\");\n try {\n chmodSync(path, 0o600);\n } catch {\n /* chmod not supported */\n }\n}\n\n/** Rotate the live jsonl + sidecars to `<name>__archive_<ts>` so /new doesn't destroy history. Returns the archive name, or null if there was nothing to archive. */\nexport function archiveSession(name: string): string | null {\n const path = sessionPath(name);\n if (!existsSync(path)) return null;\n try {\n if (statSync(path).size === 0) return null;\n } catch {\n return null;\n }\n for (let attempt = 0; attempt < 5; attempt++) {\n const target = `${name}__archive_${timestampSuffix()}${attempt > 0 ? `_${attempt}` : \"\"}`;\n if (renameSession(name, target)) return target;\n }\n return null;\n}\n\nfunction countLines(path: string): number {\n try {\n const raw = readFileSync(path, \"utf8\");\n return raw.split(/\\r?\\n/).filter((l) => l.trim()).length;\n } catch {\n return 0;\n }\n}\n"],"mappings":"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","names":[]}
|
|
@@ -1,11 +1,20 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
+
import { createRequire as __cr } from 'node:module'; if (typeof globalThis.require === 'undefined') { globalThis.require = __cr(import.meta.url); }
|
|
2
3
|
import {
|
|
3
4
|
useColor
|
|
4
|
-
} from "./chunk-
|
|
5
|
+
} from "./chunk-TJX6BFZZ.js";
|
|
6
|
+
import {
|
|
7
|
+
Box_default,
|
|
8
|
+
Text,
|
|
9
|
+
require_react,
|
|
10
|
+
use_input_default
|
|
11
|
+
} from "./chunk-6G3CUUFG.js";
|
|
12
|
+
import {
|
|
13
|
+
__toESM
|
|
14
|
+
} from "./chunk-TUK7OWJA.js";
|
|
5
15
|
|
|
6
16
|
// src/cli/ui/keystroke-context.tsx
|
|
7
|
-
|
|
8
|
-
import React, { createContext, useContext, useEffect, useRef } from "react";
|
|
17
|
+
var import_react = __toESM(require_react(), 1);
|
|
9
18
|
|
|
10
19
|
// src/cli/ui/stdin-reader.ts
|
|
11
20
|
import { stdin } from "process";
|
|
@@ -66,6 +75,29 @@ function tryEscapelessCsi(chunk, i) {
|
|
|
66
75
|
}
|
|
67
76
|
return null;
|
|
68
77
|
}
|
|
78
|
+
var SGR_MOUSE_ESCAPELESS_RE = /^\[<\d+;\d+;\d+[Mm]/;
|
|
79
|
+
function decodeSgrMouseBody(body) {
|
|
80
|
+
const m = /^<(\d+);(\d+);(\d+)([Mm])$/.exec(body);
|
|
81
|
+
if (!m) return null;
|
|
82
|
+
const btn = Number.parseInt(m[1], 10);
|
|
83
|
+
const col = Number.parseInt(m[2], 10);
|
|
84
|
+
const row = Number.parseInt(m[3], 10);
|
|
85
|
+
if (!Number.isFinite(btn) || !Number.isFinite(col) || !Number.isFinite(row)) return null;
|
|
86
|
+
const tail = m[4];
|
|
87
|
+
if (tail === "m") return { input: "", mouseRelease: true, mouseRow: row, mouseCol: col };
|
|
88
|
+
if (btn === 64) return { input: "", mouseScrollUp: true, mouseRow: row, mouseCol: col };
|
|
89
|
+
if (btn === 65) return { input: "", mouseScrollDown: true, mouseRow: row, mouseCol: col };
|
|
90
|
+
if (btn === 0) return { input: "", mouseClick: true, mouseRow: row, mouseCol: col };
|
|
91
|
+
if (btn === 32) return { input: "", mouseDrag: true, mouseRow: row, mouseCol: col };
|
|
92
|
+
return null;
|
|
93
|
+
}
|
|
94
|
+
function tryEscapelessSgrMouse(chunk, i) {
|
|
95
|
+
if (chunk[i] !== "[") return null;
|
|
96
|
+
const m = SGR_MOUSE_ESCAPELESS_RE.exec(chunk.slice(i));
|
|
97
|
+
if (!m) return null;
|
|
98
|
+
const body = m[0].slice(1);
|
|
99
|
+
return { advance: m[0].length, ev: decodeSgrMouseBody(body) };
|
|
100
|
+
}
|
|
69
101
|
function isCsiFinal(ch) {
|
|
70
102
|
const code = ch.charCodeAt(0);
|
|
71
103
|
return code >= 64 && code <= 126;
|
|
@@ -108,6 +140,10 @@ function tryDecodeGenericCsi(seq) {
|
|
|
108
140
|
if (m) return decodeModifiedKey(Number.parseInt(m[1], 10), 1);
|
|
109
141
|
return null;
|
|
110
142
|
}
|
|
143
|
+
var PASTE_INVISIBLE_RE = /[\u200B\u200E\u200F\u202A-\u202E\u2060\u2066-\u2069\u00AD\uFEFF]/g;
|
|
144
|
+
function sanitizePasteText(s) {
|
|
145
|
+
return s.replace(PASTE_INVISIBLE_RE, "");
|
|
146
|
+
}
|
|
111
147
|
function looksLikeUnbracketedPaste(chunk) {
|
|
112
148
|
if (chunk.length < 2) return false;
|
|
113
149
|
if (chunk.includes(PASTE_START) || chunk.includes(PASTE_START_BARE)) return false;
|
|
@@ -239,7 +275,7 @@ var StdinReader = class {
|
|
|
239
275
|
break;
|
|
240
276
|
}
|
|
241
277
|
this.pasteBuf += chunk.slice(i, endIdx);
|
|
242
|
-
this.dispatch({ input: this.pasteBuf, paste: true });
|
|
278
|
+
this.dispatch({ input: sanitizePasteText(this.pasteBuf), paste: true });
|
|
243
279
|
this.pasteBuf = "";
|
|
244
280
|
this.state = "idle";
|
|
245
281
|
i = endIdx + endLen;
|
|
@@ -305,6 +341,12 @@ var StdinReader = class {
|
|
|
305
341
|
i += escapeless.advance;
|
|
306
342
|
continue;
|
|
307
343
|
}
|
|
344
|
+
const mouseEscapeless = tryEscapelessSgrMouse(chunk, i);
|
|
345
|
+
if (mouseEscapeless) {
|
|
346
|
+
if (mouseEscapeless.ev) this.dispatch(mouseEscapeless.ev);
|
|
347
|
+
i += mouseEscapeless.advance;
|
|
348
|
+
continue;
|
|
349
|
+
}
|
|
308
350
|
if (ch === "\r") {
|
|
309
351
|
this.dispatch({ input: "", return: true });
|
|
310
352
|
i++;
|
|
@@ -344,7 +386,7 @@ var StdinReader = class {
|
|
|
344
386
|
if (c === "\x7F" || c === "\b" || c === "") break;
|
|
345
387
|
const cc = c.charCodeAt(0);
|
|
346
388
|
if (cc >= 1 && cc <= 26) break;
|
|
347
|
-
if (c === "[" && tryEscapelessCsi(chunk, end)) break;
|
|
389
|
+
if (c === "[" && (tryEscapelessCsi(chunk, end) || tryEscapelessSgrMouse(chunk, end))) break;
|
|
348
390
|
if (chunk.slice(end, end + PASTE_START_BARE.length) === PASTE_START_BARE) break;
|
|
349
391
|
end++;
|
|
350
392
|
}
|
|
@@ -365,39 +407,9 @@ var StdinReader = class {
|
|
|
365
407
|
return;
|
|
366
408
|
}
|
|
367
409
|
if (seq.length > 1 && seq.charCodeAt(0) === 60) {
|
|
368
|
-
const
|
|
369
|
-
if (
|
|
370
|
-
|
|
371
|
-
const parts = body.split(";");
|
|
372
|
-
if (parts.length === 3) {
|
|
373
|
-
const btn = Number.parseInt(parts[0], 10);
|
|
374
|
-
const col = Number.parseInt(parts[1], 10);
|
|
375
|
-
const row = Number.parseInt(parts[2], 10);
|
|
376
|
-
if (Number.isFinite(btn) && Number.isFinite(col) && Number.isFinite(row)) {
|
|
377
|
-
if (tail === "M" && btn === 64) {
|
|
378
|
-
this.dispatch({ input: "", mouseScrollUp: true, mouseRow: row, mouseCol: col });
|
|
379
|
-
return;
|
|
380
|
-
}
|
|
381
|
-
if (tail === "M" && btn === 65) {
|
|
382
|
-
this.dispatch({ input: "", mouseScrollDown: true, mouseRow: row, mouseCol: col });
|
|
383
|
-
return;
|
|
384
|
-
}
|
|
385
|
-
if (tail === "M" && btn === 0) {
|
|
386
|
-
this.dispatch({ input: "", mouseClick: true, mouseRow: row, mouseCol: col });
|
|
387
|
-
return;
|
|
388
|
-
}
|
|
389
|
-
if (tail === "M" && btn === 32) {
|
|
390
|
-
this.dispatch({ input: "", mouseDrag: true, mouseRow: row, mouseCol: col });
|
|
391
|
-
return;
|
|
392
|
-
}
|
|
393
|
-
if (tail === "m") {
|
|
394
|
-
this.dispatch({ input: "", mouseRelease: true, mouseRow: row, mouseCol: col });
|
|
395
|
-
return;
|
|
396
|
-
}
|
|
397
|
-
return;
|
|
398
|
-
}
|
|
399
|
-
}
|
|
400
|
-
}
|
|
410
|
+
const ev2 = decodeSgrMouseBody(seq);
|
|
411
|
+
if (ev2) this.dispatch(ev2);
|
|
412
|
+
return;
|
|
401
413
|
}
|
|
402
414
|
const ev = lookupCsi(seq);
|
|
403
415
|
if (ev) {
|
|
@@ -418,13 +430,13 @@ function getStdinReader() {
|
|
|
418
430
|
}
|
|
419
431
|
|
|
420
432
|
// src/cli/ui/keystroke-context.tsx
|
|
421
|
-
var KeystrokeContext = createContext(null);
|
|
433
|
+
var KeystrokeContext = (0, import_react.createContext)(null);
|
|
422
434
|
function KeystrokeProvider({
|
|
423
435
|
children,
|
|
424
436
|
reader: providedReader
|
|
425
437
|
}) {
|
|
426
|
-
const handlersRef = useRef(/* @__PURE__ */ new Set());
|
|
427
|
-
const busRef = useRef(null);
|
|
438
|
+
const handlersRef = (0, import_react.useRef)(/* @__PURE__ */ new Set());
|
|
439
|
+
const busRef = (0, import_react.useRef)(null);
|
|
428
440
|
if (busRef.current === null) {
|
|
429
441
|
busRef.current = {
|
|
430
442
|
subscribe(handler) {
|
|
@@ -435,7 +447,7 @@ function KeystrokeProvider({
|
|
|
435
447
|
}
|
|
436
448
|
};
|
|
437
449
|
}
|
|
438
|
-
useEffect(() => {
|
|
450
|
+
(0, import_react.useEffect)(() => {
|
|
439
451
|
const reader = providedReader ?? getStdinReader();
|
|
440
452
|
reader.start();
|
|
441
453
|
const unsubscribe = reader.subscribe((ev) => {
|
|
@@ -445,17 +457,17 @@ function KeystrokeProvider({
|
|
|
445
457
|
unsubscribe();
|
|
446
458
|
};
|
|
447
459
|
}, [providedReader]);
|
|
448
|
-
return /* @__PURE__ */
|
|
460
|
+
return /* @__PURE__ */ import_react.default.createElement(KeystrokeContext.Provider, { value: busRef.current }, children);
|
|
449
461
|
}
|
|
450
462
|
function useKeystroke(handler, isActive = true) {
|
|
451
|
-
const bus = useContext(KeystrokeContext);
|
|
452
|
-
const handlerRef = useRef(handler);
|
|
463
|
+
const bus = (0, import_react.useContext)(KeystrokeContext);
|
|
464
|
+
const handlerRef = (0, import_react.useRef)(handler);
|
|
453
465
|
handlerRef.current = handler;
|
|
454
|
-
useEffect(() => {
|
|
466
|
+
(0, import_react.useEffect)(() => {
|
|
455
467
|
if (!bus || !isActive) return void 0;
|
|
456
468
|
return bus.subscribe((ev) => handlerRef.current(ev));
|
|
457
469
|
}, [bus, isActive]);
|
|
458
|
-
|
|
470
|
+
use_input_default(
|
|
459
471
|
(input, key) => {
|
|
460
472
|
if (bus) return;
|
|
461
473
|
handlerRef.current({
|
|
@@ -481,24 +493,25 @@ function useKeystroke(handler, isActive = true) {
|
|
|
481
493
|
}
|
|
482
494
|
|
|
483
495
|
// src/cli/ui/Select.tsx
|
|
484
|
-
|
|
485
|
-
import React2, { useState } from "react";
|
|
496
|
+
var import_react2 = __toESM(require_react(), 1);
|
|
486
497
|
function SingleSelect({
|
|
487
498
|
items,
|
|
488
499
|
initialValue,
|
|
489
500
|
onSubmit,
|
|
490
501
|
onTab,
|
|
491
502
|
onCancel,
|
|
492
|
-
footer
|
|
503
|
+
footer,
|
|
504
|
+
inlineHints = false,
|
|
505
|
+
ignoreKey
|
|
493
506
|
}) {
|
|
494
507
|
const color = useColor();
|
|
495
508
|
const initialIndex = Math.max(
|
|
496
509
|
0,
|
|
497
510
|
items.findIndex((i) => i.value === initialValue && !i.disabled)
|
|
498
511
|
);
|
|
499
|
-
const [index, setIndex] = useState(initialIndex === -1 ? 0 : initialIndex);
|
|
512
|
+
const [index, setIndex] = (0, import_react2.useState)(initialIndex === -1 ? 0 : initialIndex);
|
|
500
513
|
useKeystroke((ev) => {
|
|
501
|
-
if (ev.paste) return;
|
|
514
|
+
if (ev.paste || ignoreKey?.(ev)) return;
|
|
502
515
|
if (ev.upArrow) {
|
|
503
516
|
setIndex((i) => findNextEnabled(items, i, -1));
|
|
504
517
|
} else if (ev.downArrow) {
|
|
@@ -513,32 +526,35 @@ function SingleSelect({
|
|
|
513
526
|
onCancel();
|
|
514
527
|
}
|
|
515
528
|
});
|
|
516
|
-
return /* @__PURE__ */
|
|
529
|
+
return /* @__PURE__ */ import_react2.default.createElement(Box_default, { flexDirection: "column" }, items.map((item, i) => /* @__PURE__ */ import_react2.default.createElement(
|
|
517
530
|
SelectRow,
|
|
518
531
|
{
|
|
519
532
|
key: item.value,
|
|
520
533
|
item,
|
|
521
534
|
active: i === index,
|
|
522
535
|
marker: i === index ? "\u25B8" : " ",
|
|
523
|
-
color
|
|
536
|
+
color,
|
|
537
|
+
inlineHint: inlineHints
|
|
524
538
|
}
|
|
525
|
-
)), footer ? /* @__PURE__ */
|
|
539
|
+
)), footer ? /* @__PURE__ */ import_react2.default.createElement(Box_default, { marginTop: 1 }, /* @__PURE__ */ import_react2.default.createElement(Text, { dimColor: true }, footer)) : null);
|
|
526
540
|
}
|
|
527
541
|
function MultiSelect({
|
|
528
542
|
items,
|
|
529
543
|
initialSelected = [],
|
|
530
544
|
onSubmit,
|
|
531
545
|
onCancel,
|
|
532
|
-
footer
|
|
546
|
+
footer,
|
|
547
|
+
inlineHints = false,
|
|
548
|
+
ignoreKey
|
|
533
549
|
}) {
|
|
534
550
|
const color = useColor();
|
|
535
|
-
const [index, setIndex] = useState(() => {
|
|
551
|
+
const [index, setIndex] = (0, import_react2.useState)(() => {
|
|
536
552
|
const first = items.findIndex((i) => !i.disabled);
|
|
537
553
|
return first === -1 ? 0 : first;
|
|
538
554
|
});
|
|
539
|
-
const [selected, setSelected] = useState(new Set(initialSelected));
|
|
555
|
+
const [selected, setSelected] = (0, import_react2.useState)(new Set(initialSelected));
|
|
540
556
|
useKeystroke((ev) => {
|
|
541
|
-
if (ev.paste) return;
|
|
557
|
+
if (ev.paste || ignoreKey?.(ev)) return;
|
|
542
558
|
if (ev.upArrow) {
|
|
543
559
|
setIndex((i) => findNextEnabled(items, i, -1));
|
|
544
560
|
} else if (ev.downArrow) {
|
|
@@ -559,29 +575,35 @@ function MultiSelect({
|
|
|
559
575
|
onCancel();
|
|
560
576
|
}
|
|
561
577
|
});
|
|
562
|
-
return /* @__PURE__ */
|
|
578
|
+
return /* @__PURE__ */ import_react2.default.createElement(Box_default, { flexDirection: "column" }, items.map((item, i) => {
|
|
563
579
|
const checked = selected.has(item.value);
|
|
564
580
|
const marker = checked ? "[x]" : "[ ]";
|
|
565
|
-
return /* @__PURE__ */
|
|
581
|
+
return /* @__PURE__ */ import_react2.default.createElement(
|
|
566
582
|
SelectRow,
|
|
567
583
|
{
|
|
568
584
|
key: item.value,
|
|
569
585
|
item,
|
|
570
586
|
active: i === index,
|
|
571
587
|
marker: `${i === index ? "\u25B8" : " "} ${marker}`,
|
|
572
|
-
color
|
|
588
|
+
color,
|
|
589
|
+
inlineHint: inlineHints
|
|
573
590
|
}
|
|
574
591
|
);
|
|
575
|
-
}), footer ? /* @__PURE__ */
|
|
592
|
+
}), footer ? /* @__PURE__ */ import_react2.default.createElement(Box_default, { marginTop: 1 }, /* @__PURE__ */ import_react2.default.createElement(Text, { dimColor: true }, footer)) : null);
|
|
576
593
|
}
|
|
577
594
|
function SelectRow({
|
|
578
595
|
item,
|
|
579
596
|
active,
|
|
580
597
|
marker,
|
|
581
|
-
color
|
|
598
|
+
color,
|
|
599
|
+
inlineHint = false
|
|
582
600
|
}) {
|
|
583
601
|
const rowColor = item.disabled ? color.info : active ? color.primary : void 0;
|
|
584
|
-
|
|
602
|
+
const labelText = `${marker} ${item.label}`;
|
|
603
|
+
if (inlineHint) {
|
|
604
|
+
return /* @__PURE__ */ import_react2.default.createElement(Box_default, { flexDirection: "row", flexWrap: "nowrap", minHeight: 1 }, /* @__PURE__ */ import_react2.default.createElement(Text, { color: rowColor, bold: active, dimColor: item.disabled, wrap: "truncate" }, labelText), item.hint ? /* @__PURE__ */ import_react2.default.createElement(Text, { dimColor: true, wrap: "truncate" }, ` ${item.hint}`) : null);
|
|
605
|
+
}
|
|
606
|
+
return /* @__PURE__ */ import_react2.default.createElement(Box_default, { flexDirection: "column" }, /* @__PURE__ */ import_react2.default.createElement(Box_default, null, /* @__PURE__ */ import_react2.default.createElement(Text, { color: rowColor, bold: active, dimColor: item.disabled }, labelText)), item.hint ? /* @__PURE__ */ import_react2.default.createElement(Box_default, { paddingLeft: marker.length + 1 }, /* @__PURE__ */ import_react2.default.createElement(Text, { dimColor: true }, item.hint)) : null);
|
|
585
607
|
}
|
|
586
608
|
function findNextEnabled(items, from, step) {
|
|
587
609
|
if (items.length === 0) return 0;
|
|
@@ -599,4 +621,4 @@ export {
|
|
|
599
621
|
SingleSelect,
|
|
600
622
|
MultiSelect
|
|
601
623
|
};
|
|
602
|
-
//# sourceMappingURL=chunk-
|
|
624
|
+
//# sourceMappingURL=chunk-6PZ3CXBP.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/cli/ui/keystroke-context.tsx","../../src/cli/ui/stdin-reader.ts","../../src/cli/ui/Select.tsx"],"sourcesContent":["/**\n * KeystrokeContext — React surface in front of the raw stdin reader.\n *\n * Replaces Ink's `useInput` chain. Reasonix's components no longer\n * import `useInput` from \"ink\"; they call `useKeystroke(handler,\n * isActive)` from this module. The provider mounted once at App\n * level owns a `StdinReader`, subscribes a single fan-out function\n * to it, and dispatches each parsed `KeyEvent` to every active\n * consumer.\n *\n * Why a Context instead of a singleton import: the provider can be\n * disabled in tests / replay mode without touching the components,\n * and the lifecycle (start/stop on mount/unmount) is tied to the\n * React tree rather than a global side effect.\n *\n * Why not just keep Ink's useInput: Ink's parse-keypress uses a\n * 100 ms intra-CSI timeout that's too short for Windows ConPTY,\n * leaking arrow-key bytes / paste markers into the buffer. Our\n * reader uses 250 ms and recognises the ESC-stripped variants too\n * — see `stdin-reader.ts`.\n */\n\nimport { useInput } from \"ink\";\n// biome-ignore lint/style/useImportType: tsconfig jsx=react needs React as a runtime value\nimport React, { createContext, useContext, useEffect, useRef } from \"react\";\nimport { type KeyEvent, type StdinReader, getStdinReader } from \"./stdin-reader.js\";\n\ninterface KeystrokeBus {\n /** Subscribe — returns an unsubscribe function. */\n subscribe(handler: KeystrokeHandler): () => void;\n}\n\nexport type KeystrokeHandler = (ev: KeyEvent) => void;\n\nconst KeystrokeContext = createContext<KeystrokeBus | null>(null);\n\nexport interface KeystrokeProviderProps {\n children: React.ReactNode;\n /**\n * Optional reader override. Tests inject a synthetic reader so\n * they can `feed()` chunks instead of touching real stdin. Production\n * callers leave this unset and get the singleton.\n */\n reader?: StdinReader;\n}\n\nexport function KeystrokeProvider({\n children,\n reader: providedReader,\n}: KeystrokeProviderProps): React.ReactElement {\n const handlersRef = useRef<Set<KeystrokeHandler>>(new Set());\n // Ref so the bus value's identity is stable across re-renders —\n // consumers don't accidentally re-subscribe every render.\n const busRef = useRef<KeystrokeBus | null>(null);\n if (busRef.current === null) {\n busRef.current = {\n subscribe(handler) {\n handlersRef.current.add(handler);\n return () => {\n handlersRef.current.delete(handler);\n };\n },\n };\n }\n\n useEffect(() => {\n const reader = providedReader ?? getStdinReader();\n reader.start();\n const unsubscribe = reader.subscribe((ev) => {\n // Snapshot the handler set so handlers added/removed during\n // dispatch don't perturb iteration. Cheap — typical N=1-3.\n for (const fn of [...handlersRef.current]) fn(ev);\n });\n return () => {\n unsubscribe();\n // Don't `stop()` the singleton on every unmount — multiple\n // mounts (test reruns, hot-reload) must not tear down stdin.\n // The singleton's own start() is idempotent; stop() is the\n // process-exit handler's job.\n };\n }, [providedReader]);\n\n return <KeystrokeContext.Provider value={busRef.current}>{children}</KeystrokeContext.Provider>;\n}\n\n/** Subscribe to keystroke events; falls back to Ink's useInput when no KeystrokeProvider is mounted. */\nexport function useKeystroke(handler: KeystrokeHandler, isActive = true): void {\n const bus = useContext(KeystrokeContext);\n const handlerRef = useRef(handler);\n handlerRef.current = handler;\n\n useEffect(() => {\n if (!bus || !isActive) return undefined;\n return bus.subscribe((ev) => handlerRef.current(ev));\n }, [bus, isActive]);\n\n useInput(\n (input, key) => {\n if (bus) return;\n handlerRef.current({\n input,\n upArrow: key.upArrow,\n downArrow: key.downArrow,\n leftArrow: key.leftArrow,\n rightArrow: key.rightArrow,\n return: key.return,\n escape: key.escape,\n backspace: key.backspace,\n delete: key.delete,\n tab: key.tab,\n shift: key.shift,\n ctrl: key.ctrl,\n meta: key.meta,\n pageUp: key.pageUp,\n pageDown: key.pageDown,\n });\n },\n { isActive: !bus && isActive },\n );\n}\n\n/**\n * Lower-level hook for components that need a stable subscription\n * across the lifetime of the consumer (typically StdinReader-aware\n * unit tests).\n */\nexport function useKeystrokeBus(): KeystrokeBus | null {\n return useContext(KeystrokeContext);\n}\n\n/** Test helper — assemble a KeyEvent with sensible defaults. */\nexport function makeKeyEvent(overrides: Partial<KeyEvent> = {}): KeyEvent {\n return { input: \"\", ...overrides };\n}\n","/** Sole stdin owner; 250 ms ESC-ambiguity timer (ConPTY splits sequences past parse-keypress's 100 ms). */\n\nimport { stdin } from \"node:process\";\n\nexport interface KeyEvent {\n /** Empty for control keys (arrows / Enter / Esc); holds the letter for Ctrl+/Alt+. */\n input: string;\n upArrow?: boolean;\n downArrow?: boolean;\n leftArrow?: boolean;\n rightArrow?: boolean;\n pageUp?: boolean;\n pageDown?: boolean;\n home?: boolean;\n end?: boolean;\n delete?: boolean;\n backspace?: boolean;\n tab?: boolean;\n return?: boolean;\n escape?: boolean;\n shift?: boolean;\n ctrl?: boolean;\n meta?: boolean;\n /** Bracketed-paste content; consumers MUST NOT re-interpret as keystrokes (e.g. `\\n` ≠ submit). */\n paste?: boolean;\n /** xterm SGR mode 1006 wheel-up. */\n mouseScrollUp?: boolean;\n /** Mouse wheel down — symmetric to `mouseScrollUp`. */\n mouseScrollDown?: boolean;\n /** Left-button press; row/col are 1-based. */\n mouseClick?: boolean;\n /** Left-button motion (button held during drag). Mode 1002 only. */\n mouseDrag?: boolean;\n /** Any-button release. Mode 1002 only. */\n mouseRelease?: boolean;\n mouseRow?: number;\n mouseCol?: number;\n}\n\ntype Subscriber = (ev: KeyEvent) => void;\n\n/** ESC ambiguity timeout. Long enough for ConPTY-split sequences. */\nconst ESC_TIMEOUT_MS = 250;\n\n/** Bracketed-paste markers (DECSET 2004). */\nconst PASTE_START = \"\\x1b[200~\";\nconst PASTE_END = \"\\x1b[201~\";\n/** ESC-stripped variants — ConPTY occasionally eats the leading ESC. */\nconst PASTE_START_BARE = \"[200~\";\nconst PASTE_END_BARE = \"[201~\";\n\nconst CSI_TAIL_MAP: ReadonlyArray<{ tail: string; ev: KeyEvent }> = [\n { tail: \"A\", ev: { input: \"\", upArrow: true } },\n { tail: \"B\", ev: { input: \"\", downArrow: true } },\n { tail: \"C\", ev: { input: \"\", rightArrow: true } },\n { tail: \"D\", ev: { input: \"\", leftArrow: true } },\n { tail: \"H\", ev: { input: \"\", home: true } },\n { tail: \"F\", ev: { input: \"\", end: true } },\n { tail: \"1~\", ev: { input: \"\", home: true } },\n { tail: \"4~\", ev: { input: \"\", end: true } },\n { tail: \"5~\", ev: { input: \"\", pageUp: true } },\n { tail: \"6~\", ev: { input: \"\", pageDown: true } },\n { tail: \"3~\", ev: { input: \"\", delete: true } },\n { tail: \"Z\", ev: { input: \"\", shift: true, tab: true } },\n // Some Windows hosts (PowerShell 7.x conhost path) emit the\n // modifier-encoded back-tab `\\x1b[1;2Z` instead of bare `\\x1b[Z`.\n // Issue #373 — without this entry Shift+Tab is silently dropped.\n { tail: \"1;2Z\", ev: { input: \"\", shift: true, tab: true } },\n // modifyOtherKeys (xterm CSI > 4 ; 2 m) sequences for Enter / Tab\n // with modifiers. Only fired when App.tsx has enabled the mode at\n // startup; otherwise Shift+Enter stays indistinguishable from Enter.\n // Modifier encoding: 2=shift, 3=alt, 4=alt+shift, 5=ctrl,\n // 6=ctrl+shift, 7=ctrl+alt, 8=ctrl+alt+shift. Keycodes: 9=Tab, 13=Enter.\n { tail: \"27;2;9~\", ev: { input: \"\", tab: true, shift: true } },\n { tail: \"27;2;13~\", ev: { input: \"\", return: true, shift: true } },\n { tail: \"27;5;13~\", ev: { input: \"\", return: true, ctrl: true } },\n { tail: \"27;6;13~\", ev: { input: \"\", return: true, ctrl: true, shift: true } },\n // Kitty keyboard protocol — same idea, different envelope:\n // `\\x1b[<keycode>;<mod>u`. Some terminals (kitty, recent Windows\n // Terminal previews) prefer this shape. Harmless to map here too.\n { tail: \"9;2u\", ev: { input: \"\", tab: true, shift: true } },\n { tail: \"13;2u\", ev: { input: \"\", return: true, shift: true } },\n { tail: \"13;5u\", ev: { input: \"\", return: true, ctrl: true } },\n { tail: \"13;6u\", ev: { input: \"\", return: true, ctrl: true, shift: true } },\n];\n\n/** SS3 sequences (`\\x1bO<letter>`) — some terminals send these for arrows. */\nconst SS3_MAP: Record<string, KeyEvent> = {\n A: { input: \"\", upArrow: true },\n B: { input: \"\", downArrow: true },\n C: { input: \"\", rightArrow: true },\n D: { input: \"\", leftArrow: true },\n H: { input: \"\", home: true },\n F: { input: \"\", end: true },\n};\n\n/** ESC-stripped CSI lookahead — ConPTY occasionally drops the leading ESC. */\nfunction tryEscapelessCsi(chunk: string, i: number): { advance: number; ev: KeyEvent } | null {\n if (chunk[i] !== \"[\") return null;\n // Paste start as a special case (handled by caller).\n // Try each known tail.\n for (const entry of CSI_TAIL_MAP) {\n const candidate = `[${entry.tail}`;\n if (chunk.slice(i, i + candidate.length) === candidate) {\n return { advance: candidate.length, ev: entry.ev };\n }\n }\n return null;\n}\n\n/** `[<btn;col;row[Mm]` — SGR mouse report body (without leading ESC). */\nconst SGR_MOUSE_ESCAPELESS_RE = /^\\[<\\d+;\\d+;\\d+[Mm]/;\n\nfunction decodeSgrMouseBody(body: string): KeyEvent | null {\n const m = /^<(\\d+);(\\d+);(\\d+)([Mm])$/.exec(body);\n if (!m) return null;\n const btn = Number.parseInt(m[1]!, 10);\n const col = Number.parseInt(m[2]!, 10);\n const row = Number.parseInt(m[3]!, 10);\n if (!Number.isFinite(btn) || !Number.isFinite(col) || !Number.isFinite(row)) return null;\n const tail = m[4]!;\n if (tail === \"m\") return { input: \"\", mouseRelease: true, mouseRow: row, mouseCol: col };\n if (btn === 64) return { input: \"\", mouseScrollUp: true, mouseRow: row, mouseCol: col };\n if (btn === 65) return { input: \"\", mouseScrollDown: true, mouseRow: row, mouseCol: col };\n if (btn === 0) return { input: \"\", mouseClick: true, mouseRow: row, mouseCol: col };\n if (btn === 32) return { input: \"\", mouseDrag: true, mouseRow: row, mouseCol: col };\n return null;\n}\n\n/** ConPTY can strip the ESC off SGR mouse reports — match the bare shape and drop, issue #867. */\nfunction tryEscapelessSgrMouse(\n chunk: string,\n i: number,\n): { advance: number; ev: KeyEvent | null } | null {\n if (chunk[i] !== \"[\") return null;\n const m = SGR_MOUSE_ESCAPELESS_RE.exec(chunk.slice(i));\n if (!m) return null;\n const body = m[0].slice(1);\n return { advance: m[0].length, ev: decodeSgrMouseBody(body) };\n}\n\nfunction isCsiFinal(ch: string): boolean {\n const code = ch.charCodeAt(0);\n return code >= 0x40 && code <= 0x7e;\n}\n\n/** Unknown sequence → null → caller drops bytes silently (don't insert as text). */\nfunction lookupCsi(tail: string): KeyEvent | null {\n for (const entry of CSI_TAIL_MAP) {\n if (entry.tail === tail) return entry.ev;\n }\n return null;\n}\n\n/** modifyOtherKeys / Kitty: reconstruct the keystroke from `<codepoint>` + `<mod>`. */\nfunction decodeModifiedKey(cp: number, mod: number): KeyEvent | null {\n if (mod < 1 || mod > 8) return null;\n const bits = mod - 1;\n const shift = (bits & 1) !== 0;\n const alt = (bits & 2) !== 0;\n const ctrl = (bits & 4) !== 0;\n if (cp >= 0x20 && cp <= 0x7e && !ctrl && !alt) {\n const ev: KeyEvent = { input: String.fromCharCode(cp) };\n if (shift) ev.shift = true;\n return ev;\n }\n if (cp >= 0x20 && cp <= 0x7e && alt && !ctrl) {\n const ev: KeyEvent = { input: String.fromCharCode(cp), meta: true };\n if (shift) ev.shift = true;\n return ev;\n }\n if (cp >= 0x41 && cp <= 0x7a && ctrl && !alt) {\n const ev: KeyEvent = { input: String.fromCharCode(cp).toLowerCase(), ctrl: true };\n if (shift) ev.shift = true;\n return ev;\n }\n return null;\n}\n\n/** Generic modifyOtherKeys / Kitty envelope — picks up the keys lookupCsi misses (`@`, `_`, `[`, `\\`, `]`, `^` under `>4;2m`). */\nfunction tryDecodeGenericCsi(seq: string): KeyEvent | null {\n let m = /^27;(\\d+);(\\d+)~$/.exec(seq);\n if (m) return decodeModifiedKey(Number.parseInt(m[2]!, 10), Number.parseInt(m[1]!, 10));\n m = /^(\\d+);(\\d+)u$/.exec(seq);\n if (m) return decodeModifiedKey(Number.parseInt(m[1]!, 10), Number.parseInt(m[2]!, 10));\n m = /^(\\d+)u$/.exec(seq);\n if (m) return decodeModifiedKey(Number.parseInt(m[1]!, 10), 1);\n return null;\n}\n\n// Bidi controls + zero-width invisibles that browsers smuggle into the clipboard (e.g. a B-site tab title with RLE/PDF wrappers). They render as 0 cells but still occupy buffer offsets, so cursor + line-split math drifts. ZWJ / ZWNJ / variation selectors / combining marks are NOT in the class — emoji sequences and accented letters keep their semantics. Issue #849.\nconst PASTE_INVISIBLE_RE = /[\\u200B\\u200E\\u200F\\u202A-\\u202E\\u2060\\u2066-\\u2069\\u00AD\\uFEFF]/g;\n\nexport function sanitizePasteText(s: string): string {\n return s.replace(PASTE_INVISIBLE_RE, \"\");\n}\n\n/** Heuristic paste-burst detector — wraps raw multi-line chunks when the terminal didn't (#522). */\nexport function looksLikeUnbracketedPaste(chunk: string): boolean {\n if (chunk.length < 2) return false;\n if (chunk.includes(PASTE_START) || chunk.includes(PASTE_START_BARE)) return false;\n if (chunk.includes(PASTE_END) || chunk.includes(PASTE_END_BARE)) return false;\n // ESC anywhere = real keypress / control sequence, not a paste burst.\n if (chunk.includes(\"\\x1b\")) return false;\n // \\r\\n is one terminal-converted Enter, not two breaks — fold first.\n const norm = chunk.replace(/\\r\\n/g, \"\\n\");\n if (norm === \"\\r\" || norm === \"\\n\") return false;\n let breaks = 0;\n let firstBreakIdx = -1;\n for (let i = 0; i < norm.length; i++) {\n const c = norm[i];\n if (c === \"\\r\" || c === \"\\n\") {\n if (firstBreakIdx < 0) firstBreakIdx = i;\n breaks++;\n }\n }\n if (breaks >= 2) return true;\n // 1 break with non-empty text on BOTH sides — paste burst. (\"abc\\r\"\n // alone stays as type-then-Enter so a fast typist still submits.)\n if (breaks === 1) return firstBreakIdx > 0 && firstBreakIdx < norm.length - 1;\n return false;\n}\n\nexport class StdinReader {\n private subscribers = new Set<Subscriber>();\n private state: \"idle\" | \"esc\" | \"csi\" | \"ss3\" | \"paste\" = \"idle\";\n /** Buffer for partial sequences across chunks. */\n private csiBuf = \"\";\n /** Buffer for paste content. */\n private pasteBuf = \"\";\n private escTimer: NodeJS.Timeout | null = null;\n // Deferred-dispatch handle paired with `escTimer`. The timer\n // queues an Immediate that runs in the event loop's CHECK phase —\n // i.e. AFTER the POLL phase where stdin 'data' events fire — so\n // a multi-byte sequence whose chunks queued up while the loop was\n // blocked (heavy render, etc.) gets a chance to be processed\n // BEFORE we emit a bogus standalone-Esc. Fixes the \"I didn't press\n // Esc but it aborted the turn\" class of bug: previously the timer's\n // setTimeout callback ran in the timers phase ahead of poll, so a\n // split sequence like `\\x1b` + `[A` would dispatch escape+upArrow\n // even though the user only pressed Up.\n private escImmediate: NodeJS.Immediate | null = null;\n private started = false;\n /** The actual `data` listener — kept as a field so `stop()` can detach it. */\n private listener: ((chunk: Buffer | string) => void) | null = null;\n\n start(): void {\n if (this.started) return;\n // bun leaves `isTTY` undefined in a real terminal, so probe setRawMode directly.\n try {\n stdin.setRawMode(true);\n } catch {\n return;\n }\n stdin.setEncoding(\"utf8\");\n stdin.resume();\n this.listener = (chunk) =>\n this.handleChunk(typeof chunk === \"string\" ? chunk : chunk.toString(\"utf8\"));\n stdin.on(\"data\", this.listener);\n this.started = true;\n }\n\n stop(): void {\n if (!this.started) return;\n if (this.listener) {\n stdin.off(\"data\", this.listener);\n this.listener = null;\n }\n try {\n stdin.setRawMode(false);\n } catch {\n // setRawMode may throw if stdin is already closed; ignore.\n }\n stdin.pause();\n this.cancelEscTimer();\n this.state = \"idle\";\n this.csiBuf = \"\";\n this.pasteBuf = \"\";\n this.started = false;\n }\n\n subscribe(fn: Subscriber): () => void {\n this.subscribers.add(fn);\n return () => {\n this.subscribers.delete(fn);\n };\n }\n\n /** Test seam — drives the parser without a real TTY. */\n feed(chunk: string): void {\n this.handleChunk(chunk);\n }\n\n private dispatch(ev: KeyEvent): void {\n for (const sub of this.subscribers) sub(ev);\n }\n\n private cancelEscTimer(): void {\n if (this.escTimer) {\n clearTimeout(this.escTimer);\n this.escTimer = null;\n }\n if (this.escImmediate) {\n clearImmediate(this.escImmediate);\n this.escImmediate = null;\n }\n }\n\n private scheduleEscTimer(): void {\n this.cancelEscTimer();\n this.escTimer = setTimeout(() => {\n this.escTimer = null;\n // Defer the actual dispatch to the CHECK phase so any pending\n // stdin 'data' events that queued up during a long render still\n // get a chance to consume the rest of a split sequence. The\n // chunk handler cancels this Immediate at its start, so a\n // sequence completing first wins; only a truly-orphaned `\\x1b`\n // reaches the dispatch below.\n this.escImmediate = setImmediate(() => {\n this.escImmediate = null;\n if (this.state === \"esc\") {\n this.state = \"idle\";\n this.dispatch({ input: \"\", escape: true });\n }\n });\n }, ESC_TIMEOUT_MS);\n }\n\n private handleChunk(rawChunk: string): void {\n this.cancelEscTimer();\n // Paste rescue when DECSET 2004 markers don't arrive (multiplexers\n // strip them, some Windows pipes too) — otherwise each \\r in a\n // multi-line paste fires Enter and the loop submits N prompts (#522).\n const chunk =\n this.state === \"idle\" && looksLikeUnbracketedPaste(rawChunk)\n ? PASTE_START + rawChunk + PASTE_END\n : rawChunk;\n let i = 0;\n while (i < chunk.length) {\n // ── paste accumulator ──\n if (this.state === \"paste\") {\n // Look for end marker (with or without ESC).\n const endA = chunk.indexOf(PASTE_END, i);\n const endB = chunk.indexOf(PASTE_END_BARE, i);\n let endIdx = -1;\n let endLen = 0;\n if (endA !== -1 && (endB === -1 || endA <= endB)) {\n endIdx = endA;\n endLen = PASTE_END.length;\n } else if (endB !== -1) {\n endIdx = endB;\n endLen = PASTE_END_BARE.length;\n }\n if (endIdx === -1) {\n this.pasteBuf += chunk.slice(i);\n i = chunk.length;\n break;\n }\n this.pasteBuf += chunk.slice(i, endIdx);\n this.dispatch({ input: sanitizePasteText(this.pasteBuf), paste: true });\n this.pasteBuf = \"\";\n this.state = \"idle\";\n i = endIdx + endLen;\n continue;\n }\n\n // ── CSI accumulator ──\n if (this.state === \"csi\") {\n const ch = chunk[i]!;\n this.csiBuf += ch;\n if (isCsiFinal(ch)) {\n this.dispatchCsi(this.csiBuf);\n this.csiBuf = \"\";\n // Only reset state if `dispatchCsi` didn't already mutate it\n // (it transitions to `paste` for the `200~` start marker —\n // resetting here would clobber that and the paste content\n // would be parsed as keystrokes).\n if (this.state === \"csi\") this.state = \"idle\";\n }\n i++;\n continue;\n }\n\n // ── SS3 single-byte tail ──\n if (this.state === \"ss3\") {\n const ev = SS3_MAP[chunk[i]!];\n if (ev) this.dispatch(ev);\n this.state = \"idle\";\n i++;\n continue;\n }\n\n // ── ESC pending ──\n if (this.state === \"esc\") {\n const ch = chunk[i]!;\n if (ch === \"[\") {\n this.state = \"csi\";\n this.csiBuf = \"\";\n i++;\n continue;\n }\n if (ch === \"O\") {\n this.state = \"ss3\";\n i++;\n continue;\n }\n // Alt+Enter: ESC + CR (or ESC + LF). Universal newline shortcut on terminals\n // that don't support modifyOtherKeys (Shift+Enter falls through to plain Enter there).\n if (ch === \"\\r\" || ch === \"\\n\") {\n this.dispatch({ input: \"\", return: true, meta: true });\n this.state = \"idle\";\n i++;\n continue;\n }\n // ESC + any other char = Alt+key (rare; we still dispatch).\n this.dispatch({ input: ch, meta: true });\n this.state = \"idle\";\n i++;\n continue;\n }\n\n // ── idle ──\n const ch = chunk[i]!;\n\n if (ch === \"\\x1b\") {\n this.state = \"esc\";\n i++;\n continue;\n }\n\n // ESC-stripped paste-start (ConPTY): bare `[200~` at idle.\n if (chunk.slice(i, i + PASTE_START_BARE.length) === PASTE_START_BARE) {\n this.state = \"paste\";\n this.pasteBuf = \"\";\n i += PASTE_START_BARE.length;\n continue;\n }\n // ESC-stripped CSI tails — recover before treating `[` as text.\n const escapeless = tryEscapelessCsi(chunk, i);\n if (escapeless) {\n this.dispatch(escapeless.ev);\n i += escapeless.advance;\n continue;\n }\n const mouseEscapeless = tryEscapelessSgrMouse(chunk, i);\n if (mouseEscapeless) {\n if (mouseEscapeless.ev) this.dispatch(mouseEscapeless.ev);\n i += mouseEscapeless.advance;\n continue;\n }\n\n // Single-byte control keys.\n // \\r (CR, 0x0D) is Enter on every terminal in raw mode.\n // \\n (LF, 0x0A) is what Ctrl+J emits — keep it distinct so the\n // multiline reducer can map it to \"insert newline\" instead of\n // \"submit\". Pastes containing \\n still arrive via either the\n // bracketed-paste accumulator or a multi-byte printable chunk\n // that includes the newline; neither hits this single-byte\n // branch, so this split is safe.\n if (ch === \"\\r\") {\n this.dispatch({ input: \"\", return: true });\n i++;\n continue;\n }\n if (ch === \"\\n\") {\n this.dispatch({ input: \"j\", ctrl: true });\n i++;\n continue;\n }\n if (ch === \"\\t\") {\n this.dispatch({ input: \"\", tab: true });\n i++;\n continue;\n }\n if (ch === \"\\x7f\" || ch === \"\\b\") {\n this.dispatch({ input: \"\", backspace: true });\n i++;\n continue;\n }\n if (ch === \"\\x03\") {\n // Ctrl+C — terminate the process. Raw mode disables the\n // default SIGINT, so we have to handle it ourselves.\n this.dispatch({ input: \"c\", ctrl: true });\n i++;\n continue;\n }\n\n const code = ch.charCodeAt(0);\n // Other Ctrl+letter (0x01-0x1A → A-Z, except already-handled).\n if (code >= 1 && code <= 26) {\n const letter = String.fromCharCode(0x60 + code); // a..z\n this.dispatch({ input: letter, ctrl: true });\n i++;\n continue;\n }\n\n // Regular printable input. Coalesce a run of printable chars\n // into one event so a multi-byte UTF-8 paste-burst arrives as\n // one `input` rather than N adjacent events.\n let end = i + 1;\n while (end < chunk.length) {\n const c = chunk[end]!;\n if (c === \"\\x1b\" || c === \"\\r\" || c === \"\\n\" || c === \"\\t\") break;\n if (c === \"\\x7f\" || c === \"\\b\" || c === \"\\x03\") break;\n const cc = c.charCodeAt(0);\n if (cc >= 1 && cc <= 26) break;\n // Don't swallow into a printable run if a CSI / paste prefix\n // starts at this position.\n if (c === \"[\" && (tryEscapelessCsi(chunk, end) || tryEscapelessSgrMouse(chunk, end))) break;\n if (chunk.slice(end, end + PASTE_START_BARE.length) === PASTE_START_BARE) break;\n end++;\n }\n this.dispatch({ input: chunk.slice(i, end) });\n i = end;\n }\n\n // After processing, if we're still in `esc` state, schedule the\n // ambiguity timer. The next chunk may carry the rest of the CSI;\n // if not, the timer fires and dispatches a standalone Esc.\n if (this.state === \"esc\") {\n this.scheduleEscTimer();\n }\n }\n\n private dispatchCsi(seq: string): void {\n // seq is the bytes after `\\x1b[`, e.g. \"A\", \"5~\", \"200~\", \"Z\".\n if (seq === \"200~\") {\n this.state = \"paste\";\n this.pasteBuf = \"\";\n return;\n }\n if (seq === \"201~\") {\n // Stray paste-end — we shouldn't reach here outside paste mode,\n // but if we do, drop it silently.\n return;\n }\n // SGR mouse report — surface wheel/click/drag/release, drop the rest. Always consumes the bytes even when the button isn't one we map (issue #867).\n if (seq.length > 1 && seq.charCodeAt(0) === 60 /* '<' */) {\n const ev = decodeSgrMouseBody(seq);\n if (ev) this.dispatch(ev);\n return;\n }\n const ev = lookupCsi(seq);\n if (ev) {\n this.dispatch(ev);\n return;\n }\n const generic = tryDecodeGenericCsi(seq);\n if (generic) {\n this.dispatch(generic);\n return;\n }\n // Unknown CSI → drop. Do NOT insert raw bytes as text.\n }\n}\n\n/** Singleton — one reader per process. */\nlet singleton: StdinReader | null = null;\n\nexport function getStdinReader(): StdinReader {\n if (!singleton) singleton = new StdinReader();\n return singleton;\n}\n","/** Arrow-key list components for Ink — single-select and multi-select. */\n\nimport { Box, Text } from \"ink\";\nimport React, { useState } from \"react\";\nimport { useKeystroke } from \"./keystroke-context.js\";\nimport type { KeyEvent } from \"./stdin-reader.js\";\nimport { type UiColor, useColor } from \"./theme.js\";\n\nexport interface SelectItem<V extends string = string> {\n value: V;\n label: string;\n /** Optional descriptive text rendered dimmed. */\n hint?: string;\n /** Disabled rows render dimmed and are skipped on nav. */\n disabled?: boolean;\n}\n\nexport interface SingleSelectProps<V extends string> {\n items: SelectItem<V>[];\n initialValue?: V;\n onSubmit: (value: V) => void;\n onCancel?: () => void;\n /** Fired when Tab is pressed on the currently highlighted item. */\n onTab?: (value: V) => void;\n /** Optional dim footer beneath the list. */\n footer?: string;\n /** Render item hints on the same row as the label instead of a second row. */\n inlineHints?: boolean;\n /** Ignore matching keystrokes so an enclosing component can own them. */\n ignoreKey?: (ev: KeyEvent) => boolean;\n}\n\nexport function SingleSelect<V extends string>({\n items,\n initialValue,\n onSubmit,\n onTab,\n onCancel,\n footer,\n inlineHints = false,\n ignoreKey,\n}: SingleSelectProps<V>) {\n const color = useColor();\n const initialIndex = Math.max(\n 0,\n items.findIndex((i) => i.value === initialValue && !i.disabled),\n );\n const [index, setIndex] = useState(initialIndex === -1 ? 0 : initialIndex);\n\n useKeystroke((ev) => {\n if (ev.paste || ignoreKey?.(ev)) return;\n if (ev.upArrow) {\n setIndex((i) => findNextEnabled(items, i, -1));\n } else if (ev.downArrow) {\n setIndex((i) => findNextEnabled(items, i, +1));\n } else if (ev.return) {\n const chosen = items[index];\n if (chosen && !chosen.disabled) onSubmit(chosen.value);\n } else if (ev.tab) {\n const chosen = items[index];\n if (chosen && !chosen.disabled) onTab?.(chosen.value);\n } else if (ev.escape && onCancel) {\n onCancel();\n }\n });\n\n return (\n <Box flexDirection=\"column\">\n {items.map((item, i) => (\n <SelectRow\n key={item.value}\n item={item}\n active={i === index}\n marker={i === index ? \"▸\" : \" \"}\n color={color}\n inlineHint={inlineHints}\n />\n ))}\n {footer ? (\n <Box marginTop={1}>\n <Text dimColor>{footer}</Text>\n </Box>\n ) : null}\n </Box>\n );\n}\n\nexport interface MultiSelectProps<V extends string> {\n items: SelectItem<V>[];\n initialSelected?: V[];\n onSubmit: (values: V[]) => void;\n onCancel?: () => void;\n /** Footer hint under the list — e.g. \"[Space] toggle · [Enter] confirm\". */\n footer?: string;\n /** Render item hints on the same row as the label instead of a second row. */\n inlineHints?: boolean;\n /** Ignore matching keystrokes so an enclosing component can own them. */\n ignoreKey?: (ev: KeyEvent) => boolean;\n}\n\nexport function MultiSelect<V extends string>({\n items,\n initialSelected = [],\n onSubmit,\n onCancel,\n footer,\n inlineHints = false,\n ignoreKey,\n}: MultiSelectProps<V>) {\n const color = useColor();\n const [index, setIndex] = useState(() => {\n const first = items.findIndex((i) => !i.disabled);\n return first === -1 ? 0 : first;\n });\n const [selected, setSelected] = useState<Set<V>>(new Set(initialSelected));\n\n useKeystroke((ev) => {\n if (ev.paste || ignoreKey?.(ev)) return;\n if (ev.upArrow) {\n setIndex((i) => findNextEnabled(items, i, -1));\n } else if (ev.downArrow) {\n setIndex((i) => findNextEnabled(items, i, +1));\n } else if (ev.input === \" \") {\n const item = items[index];\n if (!item || item.disabled) return;\n setSelected((prev) => {\n const next = new Set(prev);\n if (next.has(item.value)) next.delete(item.value);\n else next.add(item.value);\n return next;\n });\n } else if (ev.return) {\n const ordered = items.filter((i) => selected.has(i.value)).map((i) => i.value);\n onSubmit(ordered);\n } else if (ev.escape && onCancel) {\n onCancel();\n }\n });\n\n return (\n <Box flexDirection=\"column\">\n {items.map((item, i) => {\n const checked = selected.has(item.value);\n const marker = checked ? \"[x]\" : \"[ ]\";\n return (\n <SelectRow\n key={item.value}\n item={item}\n active={i === index}\n marker={`${i === index ? \"▸\" : \" \"} ${marker}`}\n color={color}\n inlineHint={inlineHints}\n />\n );\n })}\n {footer ? (\n <Box marginTop={1}>\n <Text dimColor>{footer}</Text>\n </Box>\n ) : null}\n </Box>\n );\n}\n\nfunction SelectRow<V extends string>({\n item,\n active,\n marker,\n color,\n inlineHint = false,\n}: {\n item: SelectItem<V>;\n active: boolean;\n marker: string;\n color: UiColor;\n inlineHint?: boolean;\n}) {\n const rowColor = item.disabled ? color.info : active ? color.primary : undefined;\n const labelText = `${marker} ${item.label}`;\n if (inlineHint) {\n return (\n <Box flexDirection=\"row\" flexWrap=\"nowrap\" minHeight={1}>\n <Text color={rowColor} bold={active} dimColor={item.disabled} wrap=\"truncate\">\n {labelText}\n </Text>\n {item.hint ? <Text dimColor wrap=\"truncate\">{` ${item.hint}`}</Text> : null}\n </Box>\n );\n }\n return (\n <Box flexDirection=\"column\">\n <Box>\n <Text color={rowColor} bold={active} dimColor={item.disabled}>\n {labelText}\n </Text>\n </Box>\n {item.hint ? (\n <Box paddingLeft={marker.length + 1}>\n <Text dimColor>{item.hint}</Text>\n </Box>\n ) : null}\n </Box>\n );\n}\n\nfunction findNextEnabled<V extends string>(\n items: SelectItem<V>[],\n from: number,\n step: -1 | 1,\n): number {\n if (items.length === 0) return 0;\n let i = from;\n for (let tries = 0; tries < items.length; tries++) {\n i = (i + step + items.length) % items.length;\n if (!items[i]?.disabled) return i;\n }\n return from;\n}\n"],"mappings":"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","names":["ch","ev","React","import_react","React"]}
|