@juliusbrussee/caveman-tui 0.65.2

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.
Files changed (162) hide show
  1. package/README.md +767 -0
  2. package/dist/autocomplete.d.ts +52 -0
  3. package/dist/autocomplete.d.ts.map +1 -0
  4. package/dist/autocomplete.js +623 -0
  5. package/dist/autocomplete.js.map +1 -0
  6. package/dist/chord.d.ts +57 -0
  7. package/dist/chord.d.ts.map +1 -0
  8. package/dist/chord.js +97 -0
  9. package/dist/chord.js.map +1 -0
  10. package/dist/color-depth.d.ts +17 -0
  11. package/dist/color-depth.d.ts.map +1 -0
  12. package/dist/color-depth.js +147 -0
  13. package/dist/color-depth.js.map +1 -0
  14. package/dist/components/Chapters.d.ts +41 -0
  15. package/dist/components/Chapters.d.ts.map +1 -0
  16. package/dist/components/Chapters.js +103 -0
  17. package/dist/components/Chapters.js.map +1 -0
  18. package/dist/components/DiffView.d.ts +75 -0
  19. package/dist/components/DiffView.d.ts.map +1 -0
  20. package/dist/components/DiffView.js +170 -0
  21. package/dist/components/DiffView.js.map +1 -0
  22. package/dist/components/StatusLine.d.ts +135 -0
  23. package/dist/components/StatusLine.d.ts.map +1 -0
  24. package/dist/components/StatusLine.js +133 -0
  25. package/dist/components/StatusLine.js.map +1 -0
  26. package/dist/components/SubagentOverlay.d.ts +63 -0
  27. package/dist/components/SubagentOverlay.d.ts.map +1 -0
  28. package/dist/components/SubagentOverlay.js +124 -0
  29. package/dist/components/SubagentOverlay.js.map +1 -0
  30. package/dist/components/box.d.ts +22 -0
  31. package/dist/components/box.d.ts.map +1 -0
  32. package/dist/components/box.js +104 -0
  33. package/dist/components/box.js.map +1 -0
  34. package/dist/components/cancellable-loader.d.ts +22 -0
  35. package/dist/components/cancellable-loader.d.ts.map +1 -0
  36. package/dist/components/cancellable-loader.js +35 -0
  37. package/dist/components/cancellable-loader.js.map +1 -0
  38. package/dist/components/editor.d.ts +244 -0
  39. package/dist/components/editor.d.ts.map +1 -0
  40. package/dist/components/editor.js +1861 -0
  41. package/dist/components/editor.js.map +1 -0
  42. package/dist/components/grouped-select-list.d.ts +60 -0
  43. package/dist/components/grouped-select-list.d.ts.map +1 -0
  44. package/dist/components/grouped-select-list.js +312 -0
  45. package/dist/components/grouped-select-list.js.map +1 -0
  46. package/dist/components/image.d.ts +28 -0
  47. package/dist/components/image.d.ts.map +1 -0
  48. package/dist/components/image.js +69 -0
  49. package/dist/components/image.js.map +1 -0
  50. package/dist/components/input.d.ts +37 -0
  51. package/dist/components/input.d.ts.map +1 -0
  52. package/dist/components/input.js +426 -0
  53. package/dist/components/input.js.map +1 -0
  54. package/dist/components/loader.d.ts +26 -0
  55. package/dist/components/loader.d.ts.map +1 -0
  56. package/dist/components/loader.js +67 -0
  57. package/dist/components/loader.js.map +1 -0
  58. package/dist/components/markdown.d.ts +95 -0
  59. package/dist/components/markdown.d.ts.map +1 -0
  60. package/dist/components/markdown.js +663 -0
  61. package/dist/components/markdown.js.map +1 -0
  62. package/dist/components/select-list.d.ts +50 -0
  63. package/dist/components/select-list.d.ts.map +1 -0
  64. package/dist/components/select-list.js +159 -0
  65. package/dist/components/select-list.js.map +1 -0
  66. package/dist/components/settings-list.d.ts +50 -0
  67. package/dist/components/settings-list.d.ts.map +1 -0
  68. package/dist/components/settings-list.js +185 -0
  69. package/dist/components/settings-list.js.map +1 -0
  70. package/dist/components/spacer.d.ts +12 -0
  71. package/dist/components/spacer.d.ts.map +1 -0
  72. package/dist/components/spacer.js +23 -0
  73. package/dist/components/spacer.js.map +1 -0
  74. package/dist/components/spinner.d.ts +35 -0
  75. package/dist/components/spinner.d.ts.map +1 -0
  76. package/dist/components/spinner.js +77 -0
  77. package/dist/components/spinner.js.map +1 -0
  78. package/dist/components/streaming-markdown.d.ts +39 -0
  79. package/dist/components/streaming-markdown.d.ts.map +1 -0
  80. package/dist/components/streaming-markdown.js +137 -0
  81. package/dist/components/streaming-markdown.js.map +1 -0
  82. package/dist/components/text.d.ts +19 -0
  83. package/dist/components/text.d.ts.map +1 -0
  84. package/dist/components/text.js +89 -0
  85. package/dist/components/text.js.map +1 -0
  86. package/dist/components/truncated-text.d.ts +13 -0
  87. package/dist/components/truncated-text.d.ts.map +1 -0
  88. package/dist/components/truncated-text.js +51 -0
  89. package/dist/components/truncated-text.js.map +1 -0
  90. package/dist/editor-component.d.ts +39 -0
  91. package/dist/editor-component.d.ts.map +1 -0
  92. package/dist/editor-component.js +2 -0
  93. package/dist/editor-component.js.map +1 -0
  94. package/dist/fuzzy.d.ts +16 -0
  95. package/dist/fuzzy.d.ts.map +1 -0
  96. package/dist/fuzzy.js +107 -0
  97. package/dist/fuzzy.js.map +1 -0
  98. package/dist/index.d.ts +38 -0
  99. package/dist/index.d.ts.map +1 -0
  100. package/dist/index.js +59 -0
  101. package/dist/index.js.map +1 -0
  102. package/dist/keybindings.d.ts +193 -0
  103. package/dist/keybindings.d.ts.map +1 -0
  104. package/dist/keybindings.js +174 -0
  105. package/dist/keybindings.js.map +1 -0
  106. package/dist/keys.d.ts +170 -0
  107. package/dist/keys.d.ts.map +1 -0
  108. package/dist/keys.js +1124 -0
  109. package/dist/keys.js.map +1 -0
  110. package/dist/kill-ring.d.ts +28 -0
  111. package/dist/kill-ring.d.ts.map +1 -0
  112. package/dist/kill-ring.js +44 -0
  113. package/dist/kill-ring.js.map +1 -0
  114. package/dist/notifications.d.ts +35 -0
  115. package/dist/notifications.d.ts.map +1 -0
  116. package/dist/notifications.js +62 -0
  117. package/dist/notifications.js.map +1 -0
  118. package/dist/osc52.d.ts +28 -0
  119. package/dist/osc52.d.ts.map +1 -0
  120. package/dist/osc52.js +53 -0
  121. package/dist/osc52.js.map +1 -0
  122. package/dist/scroll-buffer.d.ts +67 -0
  123. package/dist/scroll-buffer.d.ts.map +1 -0
  124. package/dist/scroll-buffer.js +222 -0
  125. package/dist/scroll-buffer.js.map +1 -0
  126. package/dist/spinners.d.ts +26 -0
  127. package/dist/spinners.d.ts.map +1 -0
  128. package/dist/spinners.js +136 -0
  129. package/dist/spinners.js.map +1 -0
  130. package/dist/stdin-buffer.d.ts +48 -0
  131. package/dist/stdin-buffer.d.ts.map +1 -0
  132. package/dist/stdin-buffer.js +317 -0
  133. package/dist/stdin-buffer.js.map +1 -0
  134. package/dist/sync-output.d.ts +58 -0
  135. package/dist/sync-output.d.ts.map +1 -0
  136. package/dist/sync-output.js +79 -0
  137. package/dist/sync-output.js.map +1 -0
  138. package/dist/terminal-detect.d.ts +66 -0
  139. package/dist/terminal-detect.d.ts.map +1 -0
  140. package/dist/terminal-detect.js +315 -0
  141. package/dist/terminal-detect.js.map +1 -0
  142. package/dist/terminal-image.d.ts +68 -0
  143. package/dist/terminal-image.d.ts.map +1 -0
  144. package/dist/terminal-image.js +288 -0
  145. package/dist/terminal-image.js.map +1 -0
  146. package/dist/terminal.d.ts +105 -0
  147. package/dist/terminal.d.ts.map +1 -0
  148. package/dist/terminal.js +427 -0
  149. package/dist/terminal.js.map +1 -0
  150. package/dist/tui.d.ts +268 -0
  151. package/dist/tui.d.ts.map +1 -0
  152. package/dist/tui.js +1161 -0
  153. package/dist/tui.js.map +1 -0
  154. package/dist/undo-stack.d.ts +17 -0
  155. package/dist/undo-stack.d.ts.map +1 -0
  156. package/dist/undo-stack.js +25 -0
  157. package/dist/undo-stack.js.map +1 -0
  158. package/dist/utils.d.ts +78 -0
  159. package/dist/utils.d.ts.map +1 -0
  160. package/dist/utils.js +960 -0
  161. package/dist/utils.js.map +1 -0
  162. package/package.json +59 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"autocomplete.js","sourceRoot":"","sources":["../src/autocomplete.ts"],"names":[],"mappings":"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","sourcesContent":["import { spawn } from \"child_process\";\nimport { readdirSync, statSync } from \"fs\";\nimport { homedir } from \"os\";\nimport { basename, dirname, join } from \"path\";\nimport { fuzzyFilter } from \"./fuzzy.js\";\n\nconst PATH_DELIMITERS = new Set([\" \", \"\\t\", '\"', \"'\", \"=\"]);\n\nfunction toDisplayPath(value: string): string {\n\treturn value.replace(/\\\\/g, \"/\");\n}\n\nfunction escapeRegex(value: string): string {\n\treturn value.replace(/[.*+?^${}()|[\\]\\\\]/g, \"\\\\$&\");\n}\n\nfunction buildFdPathQuery(query: string): string {\n\tconst normalized = toDisplayPath(query);\n\tif (!normalized.includes(\"/\")) {\n\t\treturn normalized;\n\t}\n\n\tconst hasTrailingSeparator = normalized.endsWith(\"/\");\n\tconst trimmed = normalized.replace(/^\\/+|\\/+$/g, \"\");\n\tif (!trimmed) {\n\t\treturn normalized;\n\t}\n\n\tconst separatorPattern = \"[\\\\\\\\/]\";\n\tconst segments = trimmed\n\t\t.split(\"/\")\n\t\t.filter(Boolean)\n\t\t.map((segment) => escapeRegex(segment));\n\tif (segments.length === 0) {\n\t\treturn normalized;\n\t}\n\n\tlet pattern = segments.join(separatorPattern);\n\tif (hasTrailingSeparator) {\n\t\tpattern += separatorPattern;\n\t}\n\treturn pattern;\n}\n\nfunction findLastDelimiter(text: string): number {\n\tfor (let i = text.length - 1; i >= 0; i -= 1) {\n\t\tif (PATH_DELIMITERS.has(text[i] ?? \"\")) {\n\t\t\treturn i;\n\t\t}\n\t}\n\treturn -1;\n}\n\nfunction findUnclosedQuoteStart(text: string): number | null {\n\tlet inQuotes = false;\n\tlet quoteStart = -1;\n\n\tfor (let i = 0; i < text.length; i += 1) {\n\t\tif (text[i] === '\"') {\n\t\t\tinQuotes = !inQuotes;\n\t\t\tif (inQuotes) {\n\t\t\t\tquoteStart = i;\n\t\t\t}\n\t\t}\n\t}\n\n\treturn inQuotes ? quoteStart : null;\n}\n\nfunction isTokenStart(text: string, index: number): boolean {\n\treturn index === 0 || PATH_DELIMITERS.has(text[index - 1] ?? \"\");\n}\n\nfunction extractQuotedPrefix(text: string): string | null {\n\tconst quoteStart = findUnclosedQuoteStart(text);\n\tif (quoteStart === null) {\n\t\treturn null;\n\t}\n\n\tif (quoteStart > 0 && text[quoteStart - 1] === \"@\") {\n\t\tif (!isTokenStart(text, quoteStart - 1)) {\n\t\t\treturn null;\n\t\t}\n\t\treturn text.slice(quoteStart - 1);\n\t}\n\n\tif (!isTokenStart(text, quoteStart)) {\n\t\treturn null;\n\t}\n\n\treturn text.slice(quoteStart);\n}\n\nfunction parsePathPrefix(prefix: string): { rawPrefix: string; isAtPrefix: boolean; isQuotedPrefix: boolean } {\n\tif (prefix.startsWith('@\"')) {\n\t\treturn { rawPrefix: prefix.slice(2), isAtPrefix: true, isQuotedPrefix: true };\n\t}\n\tif (prefix.startsWith('\"')) {\n\t\treturn { rawPrefix: prefix.slice(1), isAtPrefix: false, isQuotedPrefix: true };\n\t}\n\tif (prefix.startsWith(\"@\")) {\n\t\treturn { rawPrefix: prefix.slice(1), isAtPrefix: true, isQuotedPrefix: false };\n\t}\n\treturn { rawPrefix: prefix, isAtPrefix: false, isQuotedPrefix: false };\n}\n\nfunction buildCompletionValue(\n\tpath: string,\n\toptions: { isDirectory: boolean; isAtPrefix: boolean; isQuotedPrefix: boolean },\n): string {\n\tconst needsQuotes = options.isQuotedPrefix || path.includes(\" \");\n\tconst prefix = options.isAtPrefix ? \"@\" : \"\";\n\n\tif (!needsQuotes) {\n\t\treturn `${prefix}${path}`;\n\t}\n\n\tconst openQuote = `${prefix}\"`;\n\tconst closeQuote = '\"';\n\treturn `${openQuote}${path}${closeQuote}`;\n}\n\n// Use fd to walk directory tree (fast, respects .gitignore)\nasync function walkDirectoryWithFd(\n\tbaseDir: string,\n\tfdPath: string,\n\tquery: string,\n\tmaxResults: number,\n\tsignal: AbortSignal,\n): Promise<Array<{ path: string; isDirectory: boolean }>> {\n\tconst args = [\n\t\t\"--base-directory\",\n\t\tbaseDir,\n\t\t\"--max-results\",\n\t\tString(maxResults),\n\t\t\"--type\",\n\t\t\"f\",\n\t\t\"--type\",\n\t\t\"d\",\n\t\t\"--full-path\",\n\t\t\"--hidden\",\n\t\t\"--exclude\",\n\t\t\".git\",\n\t\t\"--exclude\",\n\t\t\".git/*\",\n\t\t\"--exclude\",\n\t\t\".git/**\",\n\t];\n\n\tif (query) {\n\t\targs.push(buildFdPathQuery(query));\n\t}\n\n\treturn await new Promise((resolve) => {\n\t\tif (signal.aborted) {\n\t\t\tresolve([]);\n\t\t\treturn;\n\t\t}\n\n\t\tconst child = spawn(fdPath, args, {\n\t\t\tstdio: [\"ignore\", \"pipe\", \"pipe\"],\n\t\t});\n\t\tlet stdout = \"\";\n\t\tlet resolved = false;\n\n\t\tconst finish = (results: Array<{ path: string; isDirectory: boolean }>) => {\n\t\t\tif (resolved) return;\n\t\t\tresolved = true;\n\t\t\tsignal.removeEventListener(\"abort\", onAbort);\n\t\t\tresolve(results);\n\t\t};\n\n\t\tconst onAbort = () => {\n\t\t\tif (child.exitCode === null) {\n\t\t\t\tchild.kill(\"SIGKILL\");\n\t\t\t}\n\t\t};\n\n\t\tsignal.addEventListener(\"abort\", onAbort, { once: true });\n\t\tchild.stdout.setEncoding(\"utf-8\");\n\t\tchild.stdout.on(\"data\", (chunk: string) => {\n\t\t\tstdout += chunk;\n\t\t});\n\t\tchild.on(\"error\", () => {\n\t\t\tfinish([]);\n\t\t});\n\t\tchild.on(\"close\", (code) => {\n\t\t\tif (signal.aborted || code !== 0 || !stdout) {\n\t\t\t\tfinish([]);\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst lines = stdout.trim().split(\"\\n\").filter(Boolean);\n\t\t\tconst results: Array<{ path: string; isDirectory: boolean }> = [];\n\n\t\t\tfor (const line of lines) {\n\t\t\t\tconst displayLine = toDisplayPath(line);\n\t\t\t\tconst hasTrailingSeparator = displayLine.endsWith(\"/\");\n\t\t\t\tconst normalizedPath = hasTrailingSeparator ? displayLine.slice(0, -1) : displayLine;\n\t\t\t\tif (normalizedPath === \".git\" || normalizedPath.startsWith(\".git/\") || normalizedPath.includes(\"/.git/\")) {\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\n\t\t\t\tresults.push({\n\t\t\t\t\tpath: displayLine,\n\t\t\t\t\tisDirectory: hasTrailingSeparator,\n\t\t\t\t});\n\t\t\t}\n\n\t\t\tfinish(results);\n\t\t});\n\t});\n}\n\nexport interface AutocompleteItem {\n\tvalue: string;\n\tlabel: string;\n\tdescription?: string;\n}\n\ntype Awaitable<T> = T | Promise<T>;\n\nexport interface SlashCommand {\n\tname: string;\n\tdescription?: string;\n\t// Function to get argument completions for this command\n\t// Returns null if no argument completion is available\n\tgetArgumentCompletions?(argumentPrefix: string): Awaitable<AutocompleteItem[] | null>;\n}\n\nexport interface AutocompleteSuggestions {\n\titems: AutocompleteItem[];\n\tprefix: string; // What we're matching against (e.g., \"/\" or \"src/\")\n}\n\nexport interface AutocompleteProvider {\n\t// Get autocomplete suggestions for current text/cursor position\n\t// Returns null if no suggestions available\n\tgetSuggestions(\n\t\tlines: string[],\n\t\tcursorLine: number,\n\t\tcursorCol: number,\n\t\toptions: { signal: AbortSignal; force?: boolean },\n\t): Promise<AutocompleteSuggestions | null>;\n\n\t// Apply the selected item\n\t// Returns the new text and cursor position\n\tapplyCompletion(\n\t\tlines: string[],\n\t\tcursorLine: number,\n\t\tcursorCol: number,\n\t\titem: AutocompleteItem,\n\t\tprefix: string,\n\t): {\n\t\tlines: string[];\n\t\tcursorLine: number;\n\t\tcursorCol: number;\n\t};\n}\n\n// Combined provider that handles both slash commands and file paths\nexport class CombinedAutocompleteProvider implements AutocompleteProvider {\n\tprivate commands: (SlashCommand | AutocompleteItem)[];\n\tprivate basePath: string;\n\tprivate fdPath: string | null;\n\n\tconstructor(\n\t\tcommands: (SlashCommand | AutocompleteItem)[] = [],\n\t\tbasePath: string = process.cwd(),\n\t\tfdPath: string | null = null,\n\t) {\n\t\tthis.commands = commands;\n\t\tthis.basePath = basePath;\n\t\tthis.fdPath = fdPath;\n\t}\n\n\tasync getSuggestions(\n\t\tlines: string[],\n\t\tcursorLine: number,\n\t\tcursorCol: number,\n\t\toptions: { signal: AbortSignal; force?: boolean },\n\t): Promise<AutocompleteSuggestions | null> {\n\t\tconst currentLine = lines[cursorLine] || \"\";\n\t\tconst textBeforeCursor = currentLine.slice(0, cursorCol);\n\n\t\tconst atPrefix = this.extractAtPrefix(textBeforeCursor);\n\t\tif (atPrefix) {\n\t\t\tconst { rawPrefix, isQuotedPrefix } = parsePathPrefix(atPrefix);\n\t\t\tconst suggestions = await this.getFuzzyFileSuggestions(rawPrefix, {\n\t\t\t\tisQuotedPrefix,\n\t\t\t\tsignal: options.signal,\n\t\t\t});\n\t\t\tif (suggestions.length === 0) return null;\n\n\t\t\treturn {\n\t\t\t\titems: suggestions,\n\t\t\t\tprefix: atPrefix,\n\t\t\t};\n\t\t}\n\n\t\tif (!options.force && textBeforeCursor.startsWith(\"/\")) {\n\t\t\tconst spaceIndex = textBeforeCursor.indexOf(\" \");\n\n\t\t\tif (spaceIndex === -1) {\n\t\t\t\tconst prefix = textBeforeCursor.slice(1);\n\t\t\t\tconst commandItems = this.commands.map((cmd) => ({\n\t\t\t\t\tname: \"name\" in cmd ? cmd.name : cmd.value,\n\t\t\t\t\tlabel: \"name\" in cmd ? cmd.name : cmd.label,\n\t\t\t\t\tdescription: cmd.description,\n\t\t\t\t}));\n\n\t\t\t\tconst filtered = fuzzyFilter(commandItems, prefix, (item) => item.name).map((item) => ({\n\t\t\t\t\tvalue: item.name,\n\t\t\t\t\tlabel: item.label,\n\t\t\t\t\t...(item.description && { description: item.description }),\n\t\t\t\t}));\n\n\t\t\t\tif (filtered.length === 0) return null;\n\n\t\t\t\treturn {\n\t\t\t\t\titems: filtered,\n\t\t\t\t\tprefix: textBeforeCursor,\n\t\t\t\t};\n\t\t\t}\n\n\t\t\tconst commandName = textBeforeCursor.slice(1, spaceIndex);\n\t\t\tconst argumentText = textBeforeCursor.slice(spaceIndex + 1);\n\n\t\t\tconst command = this.commands.find((cmd) => {\n\t\t\t\tconst name = \"name\" in cmd ? cmd.name : cmd.value;\n\t\t\t\treturn name === commandName;\n\t\t\t});\n\t\t\tif (!command || !(\"getArgumentCompletions\" in command) || !command.getArgumentCompletions) {\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\tconst argumentSuggestions = await command.getArgumentCompletions(argumentText);\n\t\t\tif (!Array.isArray(argumentSuggestions) || argumentSuggestions.length === 0) {\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\treturn {\n\t\t\t\titems: argumentSuggestions,\n\t\t\t\tprefix: argumentText,\n\t\t\t};\n\t\t}\n\n\t\tconst pathMatch = this.extractPathPrefix(textBeforeCursor, options.force ?? false);\n\t\tif (pathMatch === null) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst suggestions = this.getFileSuggestions(pathMatch);\n\t\tif (suggestions.length === 0) return null;\n\n\t\treturn {\n\t\t\titems: suggestions,\n\t\t\tprefix: pathMatch,\n\t\t};\n\t}\n\n\tapplyCompletion(\n\t\tlines: string[],\n\t\tcursorLine: number,\n\t\tcursorCol: number,\n\t\titem: AutocompleteItem,\n\t\tprefix: string,\n\t): { lines: string[]; cursorLine: number; cursorCol: number } {\n\t\tconst currentLine = lines[cursorLine] || \"\";\n\t\tconst beforePrefix = currentLine.slice(0, cursorCol - prefix.length);\n\t\tconst afterCursor = currentLine.slice(cursorCol);\n\t\tconst isQuotedPrefix = prefix.startsWith('\"') || prefix.startsWith('@\"');\n\t\tconst hasLeadingQuoteAfterCursor = afterCursor.startsWith('\"');\n\t\tconst hasTrailingQuoteInItem = item.value.endsWith('\"');\n\t\tconst adjustedAfterCursor =\n\t\t\tisQuotedPrefix && hasTrailingQuoteInItem && hasLeadingQuoteAfterCursor ? afterCursor.slice(1) : afterCursor;\n\n\t\t// Check if we're completing a slash command (prefix starts with \"/\" but NOT a file path)\n\t\t// Slash commands are at the start of the line and don't contain path separators after the first /\n\t\tconst isSlashCommand = prefix.startsWith(\"/\") && beforePrefix.trim() === \"\" && !prefix.slice(1).includes(\"/\");\n\t\tif (isSlashCommand) {\n\t\t\t// This is a command name completion\n\t\t\tconst newLine = `${beforePrefix}/${item.value} ${adjustedAfterCursor}`;\n\t\t\tconst newLines = [...lines];\n\t\t\tnewLines[cursorLine] = newLine;\n\n\t\t\treturn {\n\t\t\t\tlines: newLines,\n\t\t\t\tcursorLine,\n\t\t\t\tcursorCol: beforePrefix.length + item.value.length + 2, // +2 for \"/\" and space\n\t\t\t};\n\t\t}\n\n\t\t// Check if we're completing a file attachment (prefix starts with \"@\")\n\t\tif (prefix.startsWith(\"@\")) {\n\t\t\t// This is a file attachment completion\n\t\t\t// Don't add space after directories so user can continue autocompleting\n\t\t\tconst isDirectory = item.label.endsWith(\"/\");\n\t\t\tconst suffix = isDirectory ? \"\" : \" \";\n\t\t\tconst newLine = `${beforePrefix + item.value}${suffix}${adjustedAfterCursor}`;\n\t\t\tconst newLines = [...lines];\n\t\t\tnewLines[cursorLine] = newLine;\n\n\t\t\tconst hasTrailingQuote = item.value.endsWith('\"');\n\t\t\tconst cursorOffset = isDirectory && hasTrailingQuote ? item.value.length - 1 : item.value.length;\n\n\t\t\treturn {\n\t\t\t\tlines: newLines,\n\t\t\t\tcursorLine,\n\t\t\t\tcursorCol: beforePrefix.length + cursorOffset + suffix.length,\n\t\t\t};\n\t\t}\n\n\t\t// Check if we're in a slash command context (beforePrefix contains \"/command \")\n\t\tconst textBeforeCursor = currentLine.slice(0, cursorCol);\n\t\tif (textBeforeCursor.includes(\"/\") && textBeforeCursor.includes(\" \")) {\n\t\t\t// This is likely a command argument completion\n\t\t\tconst newLine = beforePrefix + item.value + adjustedAfterCursor;\n\t\t\tconst newLines = [...lines];\n\t\t\tnewLines[cursorLine] = newLine;\n\n\t\t\tconst isDirectory = item.label.endsWith(\"/\");\n\t\t\tconst hasTrailingQuote = item.value.endsWith('\"');\n\t\t\tconst cursorOffset = isDirectory && hasTrailingQuote ? item.value.length - 1 : item.value.length;\n\n\t\t\treturn {\n\t\t\t\tlines: newLines,\n\t\t\t\tcursorLine,\n\t\t\t\tcursorCol: beforePrefix.length + cursorOffset,\n\t\t\t};\n\t\t}\n\n\t\t// For file paths, complete the path\n\t\tconst newLine = beforePrefix + item.value + adjustedAfterCursor;\n\t\tconst newLines = [...lines];\n\t\tnewLines[cursorLine] = newLine;\n\n\t\tconst isDirectory = item.label.endsWith(\"/\");\n\t\tconst hasTrailingQuote = item.value.endsWith('\"');\n\t\tconst cursorOffset = isDirectory && hasTrailingQuote ? item.value.length - 1 : item.value.length;\n\n\t\treturn {\n\t\t\tlines: newLines,\n\t\t\tcursorLine,\n\t\t\tcursorCol: beforePrefix.length + cursorOffset,\n\t\t};\n\t}\n\n\t// Extract @ prefix for fuzzy file suggestions\n\tprivate extractAtPrefix(text: string): string | null {\n\t\tconst quotedPrefix = extractQuotedPrefix(text);\n\t\tif (quotedPrefix?.startsWith('@\"')) {\n\t\t\treturn quotedPrefix;\n\t\t}\n\n\t\tconst lastDelimiterIndex = findLastDelimiter(text);\n\t\tconst tokenStart = lastDelimiterIndex === -1 ? 0 : lastDelimiterIndex + 1;\n\n\t\tif (text[tokenStart] === \"@\") {\n\t\t\treturn text.slice(tokenStart);\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t// Extract a path-like prefix from the text before cursor\n\tprivate extractPathPrefix(text: string, forceExtract: boolean = false): string | null {\n\t\tconst quotedPrefix = extractQuotedPrefix(text);\n\t\tif (quotedPrefix) {\n\t\t\treturn quotedPrefix;\n\t\t}\n\n\t\tconst lastDelimiterIndex = findLastDelimiter(text);\n\t\tconst pathPrefix = lastDelimiterIndex === -1 ? text : text.slice(lastDelimiterIndex + 1);\n\n\t\t// For forced extraction (Tab key), always return something\n\t\tif (forceExtract) {\n\t\t\treturn pathPrefix;\n\t\t}\n\n\t\t// For natural triggers, return if it looks like a path, ends with /, starts with ~/, .\n\t\t// Only return empty string if the text looks like it's starting a path context\n\t\tif (pathPrefix.includes(\"/\") || pathPrefix.startsWith(\".\") || pathPrefix.startsWith(\"~/\")) {\n\t\t\treturn pathPrefix;\n\t\t}\n\n\t\t// Return empty string only after a space (not for completely empty text)\n\t\t// Empty text should not trigger file suggestions - that's for forced Tab completion\n\t\tif (pathPrefix === \"\" && text.endsWith(\" \")) {\n\t\t\treturn pathPrefix;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t// Expand home directory (~/) to actual home path\n\tprivate expandHomePath(path: string): string {\n\t\tif (path.startsWith(\"~/\")) {\n\t\t\tconst expandedPath = join(homedir(), path.slice(2));\n\t\t\t// Preserve trailing slash if original path had one\n\t\t\treturn path.endsWith(\"/\") && !expandedPath.endsWith(\"/\") ? `${expandedPath}/` : expandedPath;\n\t\t} else if (path === \"~\") {\n\t\t\treturn homedir();\n\t\t}\n\t\treturn path;\n\t}\n\n\tprivate resolveScopedFuzzyQuery(rawQuery: string): { baseDir: string; query: string; displayBase: string } | null {\n\t\tconst normalizedQuery = toDisplayPath(rawQuery);\n\t\tconst slashIndex = normalizedQuery.lastIndexOf(\"/\");\n\t\tif (slashIndex === -1) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst displayBase = normalizedQuery.slice(0, slashIndex + 1);\n\t\tconst query = normalizedQuery.slice(slashIndex + 1);\n\n\t\tlet baseDir: string;\n\t\tif (displayBase.startsWith(\"~/\")) {\n\t\t\tbaseDir = this.expandHomePath(displayBase);\n\t\t} else if (displayBase.startsWith(\"/\")) {\n\t\t\tbaseDir = displayBase;\n\t\t} else {\n\t\t\tbaseDir = join(this.basePath, displayBase);\n\t\t}\n\n\t\ttry {\n\t\t\tif (!statSync(baseDir).isDirectory()) {\n\t\t\t\treturn null;\n\t\t\t}\n\t\t} catch {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn { baseDir, query, displayBase };\n\t}\n\n\tprivate scopedPathForDisplay(displayBase: string, relativePath: string): string {\n\t\tconst normalizedRelativePath = toDisplayPath(relativePath);\n\t\tif (displayBase === \"/\") {\n\t\t\treturn `/${normalizedRelativePath}`;\n\t\t}\n\t\treturn `${toDisplayPath(displayBase)}${normalizedRelativePath}`;\n\t}\n\n\t// Get file/directory suggestions for a given path prefix\n\tprivate getFileSuggestions(prefix: string): AutocompleteItem[] {\n\t\ttry {\n\t\t\tlet searchDir: string;\n\t\t\tlet searchPrefix: string;\n\t\t\tconst { rawPrefix, isAtPrefix, isQuotedPrefix } = parsePathPrefix(prefix);\n\t\t\tlet expandedPrefix = rawPrefix;\n\n\t\t\t// Handle home directory expansion\n\t\t\tif (expandedPrefix.startsWith(\"~\")) {\n\t\t\t\texpandedPrefix = this.expandHomePath(expandedPrefix);\n\t\t\t}\n\n\t\t\tconst isRootPrefix =\n\t\t\t\trawPrefix === \"\" ||\n\t\t\t\trawPrefix === \"./\" ||\n\t\t\t\trawPrefix === \"../\" ||\n\t\t\t\trawPrefix === \"~\" ||\n\t\t\t\trawPrefix === \"~/\" ||\n\t\t\t\trawPrefix === \"/\" ||\n\t\t\t\t(isAtPrefix && rawPrefix === \"\");\n\n\t\t\tif (isRootPrefix) {\n\t\t\t\t// Complete from specified position\n\t\t\t\tif (rawPrefix.startsWith(\"~\") || expandedPrefix.startsWith(\"/\")) {\n\t\t\t\t\tsearchDir = expandedPrefix;\n\t\t\t\t} else {\n\t\t\t\t\tsearchDir = join(this.basePath, expandedPrefix);\n\t\t\t\t}\n\t\t\t\tsearchPrefix = \"\";\n\t\t\t} else if (rawPrefix.endsWith(\"/\")) {\n\t\t\t\t// If prefix ends with /, show contents of that directory\n\t\t\t\tif (rawPrefix.startsWith(\"~\") || expandedPrefix.startsWith(\"/\")) {\n\t\t\t\t\tsearchDir = expandedPrefix;\n\t\t\t\t} else {\n\t\t\t\t\tsearchDir = join(this.basePath, expandedPrefix);\n\t\t\t\t}\n\t\t\t\tsearchPrefix = \"\";\n\t\t\t} else {\n\t\t\t\t// Split into directory and file prefix\n\t\t\t\tconst dir = dirname(expandedPrefix);\n\t\t\t\tconst file = basename(expandedPrefix);\n\t\t\t\tif (rawPrefix.startsWith(\"~\") || expandedPrefix.startsWith(\"/\")) {\n\t\t\t\t\tsearchDir = dir;\n\t\t\t\t} else {\n\t\t\t\t\tsearchDir = join(this.basePath, dir);\n\t\t\t\t}\n\t\t\t\tsearchPrefix = file;\n\t\t\t}\n\n\t\t\tconst entries = readdirSync(searchDir, { withFileTypes: true });\n\t\t\tconst suggestions: AutocompleteItem[] = [];\n\n\t\t\tfor (const entry of entries) {\n\t\t\t\tif (!entry.name.toLowerCase().startsWith(searchPrefix.toLowerCase())) {\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\n\t\t\t\t// Check if entry is a directory (or a symlink pointing to a directory)\n\t\t\t\tlet isDirectory = entry.isDirectory();\n\t\t\t\tif (!isDirectory && entry.isSymbolicLink()) {\n\t\t\t\t\ttry {\n\t\t\t\t\t\tconst fullPath = join(searchDir, entry.name);\n\t\t\t\t\t\tisDirectory = statSync(fullPath).isDirectory();\n\t\t\t\t\t} catch {\n\t\t\t\t\t\t// Broken symlink or permission error - treat as file\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tlet relativePath: string;\n\t\t\t\tconst name = entry.name;\n\t\t\t\tconst displayPrefix = rawPrefix;\n\n\t\t\t\tif (displayPrefix.endsWith(\"/\")) {\n\t\t\t\t\t// If prefix ends with /, append entry to the prefix\n\t\t\t\t\trelativePath = displayPrefix + name;\n\t\t\t\t} else if (displayPrefix.includes(\"/\") || displayPrefix.includes(\"\\\\\")) {\n\t\t\t\t\t// Preserve ~/ format for home directory paths\n\t\t\t\t\tif (displayPrefix.startsWith(\"~/\")) {\n\t\t\t\t\t\tconst homeRelativeDir = displayPrefix.slice(2); // Remove ~/\n\t\t\t\t\t\tconst dir = dirname(homeRelativeDir);\n\t\t\t\t\t\trelativePath = `~/${dir === \".\" ? name : join(dir, name)}`;\n\t\t\t\t\t} else if (displayPrefix.startsWith(\"/\")) {\n\t\t\t\t\t\t// Absolute path - construct properly\n\t\t\t\t\t\tconst dir = dirname(displayPrefix);\n\t\t\t\t\t\tif (dir === \"/\") {\n\t\t\t\t\t\t\trelativePath = `/${name}`;\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\trelativePath = `${dir}/${name}`;\n\t\t\t\t\t\t}\n\t\t\t\t\t} else {\n\t\t\t\t\t\trelativePath = join(dirname(displayPrefix), name);\n\t\t\t\t\t\t// path.join normalizes away ./ prefix, preserve it\n\t\t\t\t\t\tif (displayPrefix.startsWith(\"./\") && !relativePath.startsWith(\"./\")) {\n\t\t\t\t\t\t\trelativePath = `./${relativePath}`;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t} else {\n\t\t\t\t\t// For standalone entries, preserve ~/ if original prefix was ~/\n\t\t\t\t\tif (displayPrefix.startsWith(\"~\")) {\n\t\t\t\t\t\trelativePath = `~/${name}`;\n\t\t\t\t\t} else {\n\t\t\t\t\t\trelativePath = name;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\trelativePath = toDisplayPath(relativePath);\n\t\t\t\tconst pathValue = isDirectory ? `${relativePath}/` : relativePath;\n\t\t\t\tconst value = buildCompletionValue(pathValue, {\n\t\t\t\t\tisDirectory,\n\t\t\t\t\tisAtPrefix,\n\t\t\t\t\tisQuotedPrefix,\n\t\t\t\t});\n\n\t\t\t\tsuggestions.push({\n\t\t\t\t\tvalue,\n\t\t\t\t\tlabel: name + (isDirectory ? \"/\" : \"\"),\n\t\t\t\t});\n\t\t\t}\n\n\t\t\t// Sort directories first, then alphabetically\n\t\t\tsuggestions.sort((a, b) => {\n\t\t\t\tconst aIsDir = a.value.endsWith(\"/\");\n\t\t\t\tconst bIsDir = b.value.endsWith(\"/\");\n\t\t\t\tif (aIsDir && !bIsDir) return -1;\n\t\t\t\tif (!aIsDir && bIsDir) return 1;\n\t\t\t\treturn a.label.localeCompare(b.label);\n\t\t\t});\n\n\t\t\treturn suggestions;\n\t\t} catch (_e) {\n\t\t\t// Directory doesn't exist or not accessible\n\t\t\treturn [];\n\t\t}\n\t}\n\n\t// Score an entry against the query (higher = better match)\n\t// isDirectory adds bonus to prioritize folders\n\tprivate scoreEntry(filePath: string, query: string, isDirectory: boolean): number {\n\t\tconst fileName = basename(filePath);\n\t\tconst lowerFileName = fileName.toLowerCase();\n\t\tconst lowerQuery = query.toLowerCase();\n\n\t\tlet score = 0;\n\n\t\t// Exact filename match (highest)\n\t\tif (lowerFileName === lowerQuery) score = 100;\n\t\t// Filename starts with query\n\t\telse if (lowerFileName.startsWith(lowerQuery)) score = 80;\n\t\t// Substring match in filename\n\t\telse if (lowerFileName.includes(lowerQuery)) score = 50;\n\t\t// Substring match in full path\n\t\telse if (filePath.toLowerCase().includes(lowerQuery)) score = 30;\n\n\t\t// Directories get a bonus to appear first\n\t\tif (isDirectory && score > 0) score += 10;\n\n\t\treturn score;\n\t}\n\n\t// Fuzzy file search using fd (fast, respects .gitignore)\n\tprivate async getFuzzyFileSuggestions(\n\t\tquery: string,\n\t\toptions: { isQuotedPrefix: boolean; signal: AbortSignal },\n\t): Promise<AutocompleteItem[]> {\n\t\tif (!this.fdPath || options.signal.aborted) {\n\t\t\treturn [];\n\t\t}\n\n\t\ttry {\n\t\t\tconst scopedQuery = this.resolveScopedFuzzyQuery(query);\n\t\t\tconst fdBaseDir = scopedQuery?.baseDir ?? this.basePath;\n\t\t\tconst fdQuery = scopedQuery?.query ?? query;\n\t\t\tconst entries = await walkDirectoryWithFd(fdBaseDir, this.fdPath, fdQuery, 100, options.signal);\n\t\t\tif (options.signal.aborted) {\n\t\t\t\treturn [];\n\t\t\t}\n\n\t\t\tconst scoredEntries = entries\n\t\t\t\t.map((entry) => ({\n\t\t\t\t\t...entry,\n\t\t\t\t\tscore: fdQuery ? this.scoreEntry(entry.path, fdQuery, entry.isDirectory) : 1,\n\t\t\t\t}))\n\t\t\t\t.filter((entry) => entry.score > 0);\n\n\t\t\tscoredEntries.sort((a, b) => b.score - a.score);\n\t\t\tconst topEntries = scoredEntries.slice(0, 20);\n\n\t\t\tconst suggestions: AutocompleteItem[] = [];\n\t\t\tfor (const { path: entryPath, isDirectory } of topEntries) {\n\t\t\t\tconst pathWithoutSlash = isDirectory ? entryPath.slice(0, -1) : entryPath;\n\t\t\t\tconst displayPath = scopedQuery\n\t\t\t\t\t? this.scopedPathForDisplay(scopedQuery.displayBase, pathWithoutSlash)\n\t\t\t\t\t: pathWithoutSlash;\n\t\t\t\tconst entryName = basename(pathWithoutSlash);\n\t\t\t\tconst completionPath = isDirectory ? `${displayPath}/` : displayPath;\n\t\t\t\tconst value = buildCompletionValue(completionPath, {\n\t\t\t\t\tisDirectory,\n\t\t\t\t\tisAtPrefix: true,\n\t\t\t\t\tisQuotedPrefix: options.isQuotedPrefix,\n\t\t\t\t});\n\n\t\t\t\tsuggestions.push({\n\t\t\t\t\tvalue,\n\t\t\t\t\tlabel: entryName + (isDirectory ? \"/\" : \"\"),\n\t\t\t\t\tdescription: displayPath,\n\t\t\t\t});\n\t\t\t}\n\n\t\t\treturn suggestions;\n\t\t} catch {\n\t\t\treturn [];\n\t\t}\n\t}\n\n\t// Check if we should trigger file completion (called on Tab key)\n\tshouldTriggerFileCompletion(lines: string[], cursorLine: number, cursorCol: number): boolean {\n\t\tconst currentLine = lines[cursorLine] || \"\";\n\t\tconst textBeforeCursor = currentLine.slice(0, cursorCol);\n\n\t\t// Don't trigger if we're typing a slash command at the start of the line\n\t\tif (textBeforeCursor.trim().startsWith(\"/\") && !textBeforeCursor.trim().includes(\" \")) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn true;\n\t}\n}\n"]}
@@ -0,0 +1,57 @@
1
+ /**
2
+ * Chord-aware keybinding matcher.
3
+ *
4
+ * A chord is a sequence of single key presses separated by ` ` in the KeyId
5
+ * (e.g. `"ctrl+x ctrl+s"` means: press Ctrl+X, then Ctrl+S). The plain
6
+ * single-key matcher in `keybindings.ts` does not understand chords; this
7
+ * module adds a stateful matcher that the consumer drives one input event at
8
+ * a time.
9
+ *
10
+ * Reference: claude-code keybindings/parser.ts:1 + match.ts state machine.
11
+ *
12
+ * Usage:
13
+ * ```
14
+ * const session = new ChordSession();
15
+ * for (const data of inputStream) {
16
+ * const result = session.feed(data, keys);
17
+ * if (result.match) handle(result.match);
18
+ * }
19
+ * ```
20
+ */
21
+ import type { Keybinding, KeybindingsManager } from "./keybindings.js";
22
+ export type ChordSegment = string;
23
+ /** Split a chord notation into its single-press segments. */
24
+ export declare function parseChord(keyId: string): ChordSegment[];
25
+ export interface ChordResult {
26
+ /** True when the just-fed input completes one of the supplied chords. */
27
+ match?: {
28
+ keybinding: Keybinding;
29
+ key: string;
30
+ };
31
+ /** True when at least one chord is still mid-sequence (consumer should suppress default handling). */
32
+ partial: boolean;
33
+ }
34
+ /**
35
+ * Stateful chord matcher. One instance per input focus context. Call
36
+ * `feed(data, keybindings)` once per input event; the session decides
37
+ * whether the event advances any in-flight chord, completes one, or resets.
38
+ */
39
+ export declare class ChordSession {
40
+ private pending;
41
+ private timeoutMs;
42
+ private lastFedAt;
43
+ constructor(opts?: {
44
+ timeoutMs?: number;
45
+ });
46
+ /** Drop any in-flight chord state. */
47
+ reset(): void;
48
+ /** True while at least one chord is mid-sequence. */
49
+ hasPartial(): boolean;
50
+ /**
51
+ * Advance the matcher with one terminal-input event. Returns
52
+ * `{match, partial}`. `match` is set only when the input completes a
53
+ * chord; partial=true while the consumer should hold its default action.
54
+ */
55
+ feed(data: string, manager: KeybindingsManager, candidates: Iterable<Keybinding>): ChordResult;
56
+ }
57
+ //# sourceMappingURL=chord.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"chord.d.ts","sourceRoot":"","sources":["../src/chord.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;GAmBG;AAEH,OAAO,KAAK,EAAE,UAAU,EAAE,kBAAkB,EAAE,MAAM,kBAAkB,CAAC;AAGvE,MAAM,MAAM,YAAY,GAAG,MAAM,CAAC;AAElC,6DAA6D;AAC7D,wBAAgB,UAAU,CAAC,KAAK,EAAE,MAAM,GAAG,YAAY,EAAE,CAKxD;AAED,MAAM,WAAW,WAAW;IAC3B,yEAAyE;IACzE,KAAK,CAAC,EAAE;QAAE,UAAU,EAAE,UAAU,CAAC;QAAC,GAAG,EAAE,MAAM,CAAA;KAAE,CAAC;IAChD,sGAAsG;IACtG,OAAO,EAAE,OAAO,CAAC;CACjB;AASD;;;;GAIG;AACH,qBAAa,YAAY;IACxB,OAAO,CAAC,OAAO,CAAsB;IACrC,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,SAAS,CAAK;IAEtB,YAAY,IAAI,GAAE;QAAE,SAAS,CAAC,EAAE,MAAM,CAAA;KAAO,EAE5C;IAED,sCAAsC;IACtC,KAAK,IAAI,IAAI,CAEZ;IAED,qDAAqD;IACrD,UAAU,IAAI,OAAO,CAEpB;IAED;;;;OAIG;IACH,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,kBAAkB,EAAE,UAAU,EAAE,QAAQ,CAAC,UAAU,CAAC,GAAG,WAAW,CAyC7F;CACD","sourcesContent":["/**\n * Chord-aware keybinding matcher.\n *\n * A chord is a sequence of single key presses separated by ` ` in the KeyId\n * (e.g. `\"ctrl+x ctrl+s\"` means: press Ctrl+X, then Ctrl+S). The plain\n * single-key matcher in `keybindings.ts` does not understand chords; this\n * module adds a stateful matcher that the consumer drives one input event at\n * a time.\n *\n * Reference: claude-code keybindings/parser.ts:1 + match.ts state machine.\n *\n * Usage:\n * ```\n * const session = new ChordSession();\n * for (const data of inputStream) {\n * const result = session.feed(data, keys);\n * if (result.match) handle(result.match);\n * }\n * ```\n */\n\nimport type { Keybinding, KeybindingsManager } from \"./keybindings.js\";\nimport { matchesKey } from \"./keys.js\";\n\nexport type ChordSegment = string; // a single key id like \"ctrl+x\"\n\n/** Split a chord notation into its single-press segments. */\nexport function parseChord(keyId: string): ChordSegment[] {\n\treturn keyId\n\t\t.trim()\n\t\t.split(/\\s+/)\n\t\t.filter((s) => s.length > 0);\n}\n\nexport interface ChordResult {\n\t/** True when the just-fed input completes one of the supplied chords. */\n\tmatch?: { keybinding: Keybinding; key: string };\n\t/** True when at least one chord is still mid-sequence (consumer should suppress default handling). */\n\tpartial: boolean;\n}\n\ninterface PendingChord {\n\tkeybinding: Keybinding;\n\tkey: string;\n\tsegments: ChordSegment[];\n\tcursor: number;\n}\n\n/**\n * Stateful chord matcher. One instance per input focus context. Call\n * `feed(data, keybindings)` once per input event; the session decides\n * whether the event advances any in-flight chord, completes one, or resets.\n */\nexport class ChordSession {\n\tprivate pending: PendingChord[] = [];\n\tprivate timeoutMs: number;\n\tprivate lastFedAt = 0;\n\n\tconstructor(opts: { timeoutMs?: number } = {}) {\n\t\tthis.timeoutMs = opts.timeoutMs ?? 1500;\n\t}\n\n\t/** Drop any in-flight chord state. */\n\treset(): void {\n\t\tthis.pending = [];\n\t}\n\n\t/** True while at least one chord is mid-sequence. */\n\thasPartial(): boolean {\n\t\treturn this.pending.length > 0;\n\t}\n\n\t/**\n\t * Advance the matcher with one terminal-input event. Returns\n\t * `{match, partial}`. `match` is set only when the input completes a\n\t * chord; partial=true while the consumer should hold its default action.\n\t */\n\tfeed(data: string, manager: KeybindingsManager, candidates: Iterable<Keybinding>): ChordResult {\n\t\tconst now = Date.now();\n\t\tif (this.pending.length > 0 && now - this.lastFedAt > this.timeoutMs) {\n\t\t\tthis.pending = [];\n\t\t}\n\t\tthis.lastFedAt = now;\n\n\t\t// First, advance any pending chords.\n\t\tconst advanced: PendingChord[] = [];\n\t\tlet match: ChordResult[\"match\"];\n\t\tfor (const p of this.pending) {\n\t\t\tconst segment = p.segments[p.cursor];\n\t\t\tif (segment && matchesKey(data, segment as Parameters<typeof matchesKey>[1])) {\n\t\t\t\tconst next = p.cursor + 1;\n\t\t\t\tif (next === p.segments.length) {\n\t\t\t\t\tmatch = { keybinding: p.keybinding, key: p.key };\n\t\t\t\t} else {\n\t\t\t\t\tadvanced.push({ ...p, cursor: next });\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t// Seed new chord candidates if no chord was already in flight.\n\t\tif (this.pending.length === 0) {\n\t\t\tfor (const kb of candidates) {\n\t\t\t\tfor (const key of manager.getKeys(kb)) {\n\t\t\t\t\tconst segments = parseChord(String(key));\n\t\t\t\t\tif (segments.length <= 1) continue; // single-key, handled elsewhere\n\t\t\t\t\tif (matchesKey(data, segments[0] as Parameters<typeof matchesKey>[1])) {\n\t\t\t\t\t\tif (segments.length === 1) {\n\t\t\t\t\t\t\tmatch = { keybinding: kb, key: String(key) };\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tadvanced.push({ keybinding: kb, key: String(key), segments, cursor: 1 });\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tthis.pending = match ? [] : advanced;\n\t\treturn { match, partial: this.pending.length > 0 };\n\t}\n}\n"]}
package/dist/chord.js ADDED
@@ -0,0 +1,97 @@
1
+ /**
2
+ * Chord-aware keybinding matcher.
3
+ *
4
+ * A chord is a sequence of single key presses separated by ` ` in the KeyId
5
+ * (e.g. `"ctrl+x ctrl+s"` means: press Ctrl+X, then Ctrl+S). The plain
6
+ * single-key matcher in `keybindings.ts` does not understand chords; this
7
+ * module adds a stateful matcher that the consumer drives one input event at
8
+ * a time.
9
+ *
10
+ * Reference: claude-code keybindings/parser.ts:1 + match.ts state machine.
11
+ *
12
+ * Usage:
13
+ * ```
14
+ * const session = new ChordSession();
15
+ * for (const data of inputStream) {
16
+ * const result = session.feed(data, keys);
17
+ * if (result.match) handle(result.match);
18
+ * }
19
+ * ```
20
+ */
21
+ import { matchesKey } from "./keys.js";
22
+ /** Split a chord notation into its single-press segments. */
23
+ export function parseChord(keyId) {
24
+ return keyId
25
+ .trim()
26
+ .split(/\s+/)
27
+ .filter((s) => s.length > 0);
28
+ }
29
+ /**
30
+ * Stateful chord matcher. One instance per input focus context. Call
31
+ * `feed(data, keybindings)` once per input event; the session decides
32
+ * whether the event advances any in-flight chord, completes one, or resets.
33
+ */
34
+ export class ChordSession {
35
+ pending = [];
36
+ timeoutMs;
37
+ lastFedAt = 0;
38
+ constructor(opts = {}) {
39
+ this.timeoutMs = opts.timeoutMs ?? 1500;
40
+ }
41
+ /** Drop any in-flight chord state. */
42
+ reset() {
43
+ this.pending = [];
44
+ }
45
+ /** True while at least one chord is mid-sequence. */
46
+ hasPartial() {
47
+ return this.pending.length > 0;
48
+ }
49
+ /**
50
+ * Advance the matcher with one terminal-input event. Returns
51
+ * `{match, partial}`. `match` is set only when the input completes a
52
+ * chord; partial=true while the consumer should hold its default action.
53
+ */
54
+ feed(data, manager, candidates) {
55
+ const now = Date.now();
56
+ if (this.pending.length > 0 && now - this.lastFedAt > this.timeoutMs) {
57
+ this.pending = [];
58
+ }
59
+ this.lastFedAt = now;
60
+ // First, advance any pending chords.
61
+ const advanced = [];
62
+ let match;
63
+ for (const p of this.pending) {
64
+ const segment = p.segments[p.cursor];
65
+ if (segment && matchesKey(data, segment)) {
66
+ const next = p.cursor + 1;
67
+ if (next === p.segments.length) {
68
+ match = { keybinding: p.keybinding, key: p.key };
69
+ }
70
+ else {
71
+ advanced.push({ ...p, cursor: next });
72
+ }
73
+ }
74
+ }
75
+ // Seed new chord candidates if no chord was already in flight.
76
+ if (this.pending.length === 0) {
77
+ for (const kb of candidates) {
78
+ for (const key of manager.getKeys(kb)) {
79
+ const segments = parseChord(String(key));
80
+ if (segments.length <= 1)
81
+ continue; // single-key, handled elsewhere
82
+ if (matchesKey(data, segments[0])) {
83
+ if (segments.length === 1) {
84
+ match = { keybinding: kb, key: String(key) };
85
+ }
86
+ else {
87
+ advanced.push({ keybinding: kb, key: String(key), segments, cursor: 1 });
88
+ }
89
+ }
90
+ }
91
+ }
92
+ }
93
+ this.pending = match ? [] : advanced;
94
+ return { match, partial: this.pending.length > 0 };
95
+ }
96
+ }
97
+ //# sourceMappingURL=chord.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"chord.js","sourceRoot":"","sources":["../src/chord.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Chord-aware keybinding matcher.\n *\n * A chord is a sequence of single key presses separated by ` ` in the KeyId\n * (e.g. `\"ctrl+x ctrl+s\"` means: press Ctrl+X, then Ctrl+S). The plain\n * single-key matcher in `keybindings.ts` does not understand chords; this\n * module adds a stateful matcher that the consumer drives one input event at\n * a time.\n *\n * Reference: claude-code keybindings/parser.ts:1 + match.ts state machine.\n *\n * Usage:\n * ```\n * const session = new ChordSession();\n * for (const data of inputStream) {\n * const result = session.feed(data, keys);\n * if (result.match) handle(result.match);\n * }\n * ```\n */\n\nimport type { Keybinding, KeybindingsManager } from \"./keybindings.js\";\nimport { matchesKey } from \"./keys.js\";\n\nexport type ChordSegment = string; // a single key id like \"ctrl+x\"\n\n/** Split a chord notation into its single-press segments. */\nexport function parseChord(keyId: string): ChordSegment[] {\n\treturn keyId\n\t\t.trim()\n\t\t.split(/\\s+/)\n\t\t.filter((s) => s.length > 0);\n}\n\nexport interface ChordResult {\n\t/** True when the just-fed input completes one of the supplied chords. */\n\tmatch?: { keybinding: Keybinding; key: string };\n\t/** True when at least one chord is still mid-sequence (consumer should suppress default handling). */\n\tpartial: boolean;\n}\n\ninterface PendingChord {\n\tkeybinding: Keybinding;\n\tkey: string;\n\tsegments: ChordSegment[];\n\tcursor: number;\n}\n\n/**\n * Stateful chord matcher. One instance per input focus context. Call\n * `feed(data, keybindings)` once per input event; the session decides\n * whether the event advances any in-flight chord, completes one, or resets.\n */\nexport class ChordSession {\n\tprivate pending: PendingChord[] = [];\n\tprivate timeoutMs: number;\n\tprivate lastFedAt = 0;\n\n\tconstructor(opts: { timeoutMs?: number } = {}) {\n\t\tthis.timeoutMs = opts.timeoutMs ?? 1500;\n\t}\n\n\t/** Drop any in-flight chord state. */\n\treset(): void {\n\t\tthis.pending = [];\n\t}\n\n\t/** True while at least one chord is mid-sequence. */\n\thasPartial(): boolean {\n\t\treturn this.pending.length > 0;\n\t}\n\n\t/**\n\t * Advance the matcher with one terminal-input event. Returns\n\t * `{match, partial}`. `match` is set only when the input completes a\n\t * chord; partial=true while the consumer should hold its default action.\n\t */\n\tfeed(data: string, manager: KeybindingsManager, candidates: Iterable<Keybinding>): ChordResult {\n\t\tconst now = Date.now();\n\t\tif (this.pending.length > 0 && now - this.lastFedAt > this.timeoutMs) {\n\t\t\tthis.pending = [];\n\t\t}\n\t\tthis.lastFedAt = now;\n\n\t\t// First, advance any pending chords.\n\t\tconst advanced: PendingChord[] = [];\n\t\tlet match: ChordResult[\"match\"];\n\t\tfor (const p of this.pending) {\n\t\t\tconst segment = p.segments[p.cursor];\n\t\t\tif (segment && matchesKey(data, segment as Parameters<typeof matchesKey>[1])) {\n\t\t\t\tconst next = p.cursor + 1;\n\t\t\t\tif (next === p.segments.length) {\n\t\t\t\t\tmatch = { keybinding: p.keybinding, key: p.key };\n\t\t\t\t} else {\n\t\t\t\t\tadvanced.push({ ...p, cursor: next });\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t// Seed new chord candidates if no chord was already in flight.\n\t\tif (this.pending.length === 0) {\n\t\t\tfor (const kb of candidates) {\n\t\t\t\tfor (const key of manager.getKeys(kb)) {\n\t\t\t\t\tconst segments = parseChord(String(key));\n\t\t\t\t\tif (segments.length <= 1) continue; // single-key, handled elsewhere\n\t\t\t\t\tif (matchesKey(data, segments[0] as Parameters<typeof matchesKey>[1])) {\n\t\t\t\t\t\tif (segments.length === 1) {\n\t\t\t\t\t\t\tmatch = { keybinding: kb, key: String(key) };\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tadvanced.push({ keybinding: kb, key: String(key), segments, cursor: 1 });\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tthis.pending = match ? [] : advanced;\n\t\treturn { match, partial: this.pending.length > 0 };\n\t}\n}\n"]}
@@ -0,0 +1,17 @@
1
+ /**
2
+ * Color depth detection and SGR emission.
3
+ *
4
+ * Covers cavekit-terminal-blend R7: emit truecolor when the terminal supports it,
5
+ * 256-color when it doesn't, and 16-color as a last resort so sequences degrade
6
+ * cleanly under tmux, screen, and linux-console.
7
+ */
8
+ export type ColorDepth = "truecolor" | "256" | "16";
9
+ export declare function detectColorDepth(env?: NodeJS.ProcessEnv): ColorDepth;
10
+ export declare function resetColorDepthCache(): void;
11
+ /**
12
+ * Emit the SGR escape sequence to set the given color as foreground or background
13
+ * at the given color depth. Returns "" for invalid input (empty/undefined hex).
14
+ */
15
+ export declare function hexToSgr(hex: string | undefined | null, depth: ColorDepth, kind: "fg" | "bg"): string;
16
+ export declare function sgrReset(): string;
17
+ //# sourceMappingURL=color-depth.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"color-depth.d.ts","sourceRoot":"","sources":["../src/color-depth.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,MAAM,MAAM,UAAU,GAAG,WAAW,GAAG,KAAK,GAAG,IAAI,CAAC;AAIpD,wBAAgB,gBAAgB,CAAC,GAAG,GAAE,MAAM,CAAC,UAAwB,GAAG,UAAU,CAyBjF;AAED,wBAAgB,oBAAoB,IAAI,IAAI,CAE3C;AA6ED;;;GAGG;AACH,wBAAgB,QAAQ,CAAC,GAAG,EAAE,MAAM,GAAG,SAAS,GAAG,IAAI,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,IAAI,GAAG,IAAI,GAAG,MAAM,CAwBrG;AAED,wBAAgB,QAAQ,IAAI,MAAM,CAEjC","sourcesContent":["/**\n * Color depth detection and SGR emission.\n *\n * Covers cavekit-terminal-blend R7: emit truecolor when the terminal supports it,\n * 256-color when it doesn't, and 16-color as a last resort so sequences degrade\n * cleanly under tmux, screen, and linux-console.\n */\n\nexport type ColorDepth = \"truecolor\" | \"256\" | \"16\";\n\nlet cachedDepth: ColorDepth | undefined;\n\nexport function detectColorDepth(env: NodeJS.ProcessEnv = process.env): ColorDepth {\n\tif (cachedDepth !== undefined && env === process.env) return cachedDepth;\n\n\tconst colorterm = (env.COLORTERM || \"\").toLowerCase();\n\tconst term = (env.TERM || \"\").toLowerCase();\n\tconst tmux = Boolean(env.TMUX);\n\n\tlet depth: ColorDepth;\n\n\tif (term === \"linux\") {\n\t\tdepth = \"16\";\n\t} else if (colorterm === \"truecolor\" || colorterm === \"24bit\") {\n\t\t// tmux without explicit truecolor indicator downgrades to 256.\n\t\tdepth = tmux && term.includes(\"screen\") && !colorterm ? \"256\" : \"truecolor\";\n\t} else if (tmux && !colorterm) {\n\t\t// tmux without truecolor indicator: 256 max per R7.\n\t\tdepth = term.includes(\"256color\") ? \"256\" : \"16\";\n\t} else if (term.includes(\"256color\") || term.includes(\"256colour\")) {\n\t\tdepth = \"256\";\n\t} else {\n\t\tdepth = \"16\";\n\t}\n\n\tif (env === process.env) cachedDepth = depth;\n\treturn depth;\n}\n\nexport function resetColorDepthCache(): void {\n\tcachedDepth = undefined;\n}\n\nfunction parseHex(hex: string): { r: number; g: number; b: number } | null {\n\tif (!hex) return null;\n\tconst v = hex.startsWith(\"#\") ? hex.slice(1) : hex;\n\tif (!/^[0-9a-fA-F]{6}$/.test(v)) return null;\n\treturn {\n\t\tr: parseInt(v.slice(0, 2), 16),\n\t\tg: parseInt(v.slice(2, 4), 16),\n\t\tb: parseInt(v.slice(4, 6), 16),\n\t};\n}\n\n/** Convert 0-255 channel to 0-5 cube index (6x6x6 cube in 256-color). */\nfunction channelTo6(v: number): number {\n\tif (v < 48) return 0;\n\tif (v < 115) return 1;\n\treturn Math.floor((v - 35) / 40);\n}\n\n/** Map (r,g,b) 0-255 to an xterm 256-color index (16-231 cube or 232-255 grayscale). */\nfunction rgbTo256(r: number, g: number, b: number): number {\n\t// Prefer grayscale ramp if channels are close together.\n\tconst max = Math.max(r, g, b);\n\tconst min = Math.min(r, g, b);\n\tif (max - min < 8) {\n\t\tconst avg = (r + g + b) / 3;\n\t\tif (avg < 8) return 16; // black in cube\n\t\tif (avg > 248) return 231; // white in cube\n\t\treturn 232 + Math.min(23, Math.floor((avg - 8) / 10));\n\t}\n\tconst ri = channelTo6(r);\n\tconst gi = channelTo6(g);\n\tconst bi = channelTo6(b);\n\treturn 16 + 36 * ri + 6 * gi + bi;\n}\n\n/**\n * Standard 16-color ANSI palette (approximate RGB values used for nearest-match).\n * Base 0-7 map to SGR 30-37 (fg) / 40-47 (bg); bright 8-15 map to 90-97 / 100-107.\n */\nconst ANSI16: { r: number; g: number; b: number }[] = [\n\t{ r: 0, g: 0, b: 0 }, // black\n\t{ r: 205, g: 0, b: 0 }, // red\n\t{ r: 0, g: 205, b: 0 }, // green\n\t{ r: 205, g: 205, b: 0 }, // yellow\n\t{ r: 0, g: 0, b: 238 }, // blue\n\t{ r: 205, g: 0, b: 205 }, // magenta\n\t{ r: 0, g: 205, b: 205 }, // cyan\n\t{ r: 229, g: 229, b: 229 }, // white\n\t{ r: 127, g: 127, b: 127 }, // bright black\n\t{ r: 255, g: 0, b: 0 }, // bright red\n\t{ r: 0, g: 255, b: 0 }, // bright green\n\t{ r: 255, g: 255, b: 0 }, // bright yellow\n\t{ r: 92, g: 92, b: 255 }, // bright blue\n\t{ r: 255, g: 0, b: 255 }, // bright magenta\n\t{ r: 0, g: 255, b: 255 }, // bright cyan\n\t{ r: 255, g: 255, b: 255 }, // bright white\n];\n\nfunction rgbTo16(r: number, g: number, b: number): number {\n\tlet best = 0;\n\tlet bestDist = Infinity;\n\tfor (let i = 0; i < ANSI16.length; i++) {\n\t\tconst p = ANSI16[i];\n\t\tconst dr = r - p.r;\n\t\tconst dg = g - p.g;\n\t\tconst db = b - p.b;\n\t\tconst dist = dr * dr + dg * dg + db * db;\n\t\tif (dist < bestDist) {\n\t\t\tbestDist = dist;\n\t\t\tbest = i;\n\t\t}\n\t}\n\treturn best;\n}\n\n/**\n * Emit the SGR escape sequence to set the given color as foreground or background\n * at the given color depth. Returns \"\" for invalid input (empty/undefined hex).\n */\nexport function hexToSgr(hex: string | undefined | null, depth: ColorDepth, kind: \"fg\" | \"bg\"): string {\n\tif (!hex) return \"\";\n\tconst rgb = parseHex(hex);\n\tif (!rgb) return \"\";\n\n\tif (depth === \"truecolor\") {\n\t\tconst code = kind === \"fg\" ? 38 : 48;\n\t\treturn `\\x1b[${code};2;${rgb.r};${rgb.g};${rgb.b}m`;\n\t}\n\n\tif (depth === \"256\") {\n\t\tconst idx = rgbTo256(rgb.r, rgb.g, rgb.b);\n\t\tconst code = kind === \"fg\" ? 38 : 48;\n\t\treturn `\\x1b[${code};5;${idx}m`;\n\t}\n\n\t// 16-color\n\tconst idx = rgbTo16(rgb.r, rgb.g, rgb.b);\n\tif (idx < 8) {\n\t\tconst base = kind === \"fg\" ? 30 : 40;\n\t\treturn `\\x1b[${base + idx}m`;\n\t}\n\tconst base = kind === \"fg\" ? 90 : 100;\n\treturn `\\x1b[${base + (idx - 8)}m`;\n}\n\nexport function sgrReset(): string {\n\treturn \"\\x1b[0m\";\n}\n"]}
@@ -0,0 +1,147 @@
1
+ /**
2
+ * Color depth detection and SGR emission.
3
+ *
4
+ * Covers cavekit-terminal-blend R7: emit truecolor when the terminal supports it,
5
+ * 256-color when it doesn't, and 16-color as a last resort so sequences degrade
6
+ * cleanly under tmux, screen, and linux-console.
7
+ */
8
+ let cachedDepth;
9
+ export function detectColorDepth(env = process.env) {
10
+ if (cachedDepth !== undefined && env === process.env)
11
+ return cachedDepth;
12
+ const colorterm = (env.COLORTERM || "").toLowerCase();
13
+ const term = (env.TERM || "").toLowerCase();
14
+ const tmux = Boolean(env.TMUX);
15
+ let depth;
16
+ if (term === "linux") {
17
+ depth = "16";
18
+ }
19
+ else if (colorterm === "truecolor" || colorterm === "24bit") {
20
+ // tmux without explicit truecolor indicator downgrades to 256.
21
+ depth = tmux && term.includes("screen") && !colorterm ? "256" : "truecolor";
22
+ }
23
+ else if (tmux && !colorterm) {
24
+ // tmux without truecolor indicator: 256 max per R7.
25
+ depth = term.includes("256color") ? "256" : "16";
26
+ }
27
+ else if (term.includes("256color") || term.includes("256colour")) {
28
+ depth = "256";
29
+ }
30
+ else {
31
+ depth = "16";
32
+ }
33
+ if (env === process.env)
34
+ cachedDepth = depth;
35
+ return depth;
36
+ }
37
+ export function resetColorDepthCache() {
38
+ cachedDepth = undefined;
39
+ }
40
+ function parseHex(hex) {
41
+ if (!hex)
42
+ return null;
43
+ const v = hex.startsWith("#") ? hex.slice(1) : hex;
44
+ if (!/^[0-9a-fA-F]{6}$/.test(v))
45
+ return null;
46
+ return {
47
+ r: parseInt(v.slice(0, 2), 16),
48
+ g: parseInt(v.slice(2, 4), 16),
49
+ b: parseInt(v.slice(4, 6), 16),
50
+ };
51
+ }
52
+ /** Convert 0-255 channel to 0-5 cube index (6x6x6 cube in 256-color). */
53
+ function channelTo6(v) {
54
+ if (v < 48)
55
+ return 0;
56
+ if (v < 115)
57
+ return 1;
58
+ return Math.floor((v - 35) / 40);
59
+ }
60
+ /** Map (r,g,b) 0-255 to an xterm 256-color index (16-231 cube or 232-255 grayscale). */
61
+ function rgbTo256(r, g, b) {
62
+ // Prefer grayscale ramp if channels are close together.
63
+ const max = Math.max(r, g, b);
64
+ const min = Math.min(r, g, b);
65
+ if (max - min < 8) {
66
+ const avg = (r + g + b) / 3;
67
+ if (avg < 8)
68
+ return 16; // black in cube
69
+ if (avg > 248)
70
+ return 231; // white in cube
71
+ return 232 + Math.min(23, Math.floor((avg - 8) / 10));
72
+ }
73
+ const ri = channelTo6(r);
74
+ const gi = channelTo6(g);
75
+ const bi = channelTo6(b);
76
+ return 16 + 36 * ri + 6 * gi + bi;
77
+ }
78
+ /**
79
+ * Standard 16-color ANSI palette (approximate RGB values used for nearest-match).
80
+ * Base 0-7 map to SGR 30-37 (fg) / 40-47 (bg); bright 8-15 map to 90-97 / 100-107.
81
+ */
82
+ const ANSI16 = [
83
+ { r: 0, g: 0, b: 0 }, // black
84
+ { r: 205, g: 0, b: 0 }, // red
85
+ { r: 0, g: 205, b: 0 }, // green
86
+ { r: 205, g: 205, b: 0 }, // yellow
87
+ { r: 0, g: 0, b: 238 }, // blue
88
+ { r: 205, g: 0, b: 205 }, // magenta
89
+ { r: 0, g: 205, b: 205 }, // cyan
90
+ { r: 229, g: 229, b: 229 }, // white
91
+ { r: 127, g: 127, b: 127 }, // bright black
92
+ { r: 255, g: 0, b: 0 }, // bright red
93
+ { r: 0, g: 255, b: 0 }, // bright green
94
+ { r: 255, g: 255, b: 0 }, // bright yellow
95
+ { r: 92, g: 92, b: 255 }, // bright blue
96
+ { r: 255, g: 0, b: 255 }, // bright magenta
97
+ { r: 0, g: 255, b: 255 }, // bright cyan
98
+ { r: 255, g: 255, b: 255 }, // bright white
99
+ ];
100
+ function rgbTo16(r, g, b) {
101
+ let best = 0;
102
+ let bestDist = Infinity;
103
+ for (let i = 0; i < ANSI16.length; i++) {
104
+ const p = ANSI16[i];
105
+ const dr = r - p.r;
106
+ const dg = g - p.g;
107
+ const db = b - p.b;
108
+ const dist = dr * dr + dg * dg + db * db;
109
+ if (dist < bestDist) {
110
+ bestDist = dist;
111
+ best = i;
112
+ }
113
+ }
114
+ return best;
115
+ }
116
+ /**
117
+ * Emit the SGR escape sequence to set the given color as foreground or background
118
+ * at the given color depth. Returns "" for invalid input (empty/undefined hex).
119
+ */
120
+ export function hexToSgr(hex, depth, kind) {
121
+ if (!hex)
122
+ return "";
123
+ const rgb = parseHex(hex);
124
+ if (!rgb)
125
+ return "";
126
+ if (depth === "truecolor") {
127
+ const code = kind === "fg" ? 38 : 48;
128
+ return `\x1b[${code};2;${rgb.r};${rgb.g};${rgb.b}m`;
129
+ }
130
+ if (depth === "256") {
131
+ const idx = rgbTo256(rgb.r, rgb.g, rgb.b);
132
+ const code = kind === "fg" ? 38 : 48;
133
+ return `\x1b[${code};5;${idx}m`;
134
+ }
135
+ // 16-color
136
+ const idx = rgbTo16(rgb.r, rgb.g, rgb.b);
137
+ if (idx < 8) {
138
+ const base = kind === "fg" ? 30 : 40;
139
+ return `\x1b[${base + idx}m`;
140
+ }
141
+ const base = kind === "fg" ? 90 : 100;
142
+ return `\x1b[${base + (idx - 8)}m`;
143
+ }
144
+ export function sgrReset() {
145
+ return "\x1b[0m";
146
+ }
147
+ //# sourceMappingURL=color-depth.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"color-depth.js","sourceRoot":"","sources":["../src/color-depth.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Color depth detection and SGR emission.\n *\n * Covers cavekit-terminal-blend R7: emit truecolor when the terminal supports it,\n * 256-color when it doesn't, and 16-color as a last resort so sequences degrade\n * cleanly under tmux, screen, and linux-console.\n */\n\nexport type ColorDepth = \"truecolor\" | \"256\" | \"16\";\n\nlet cachedDepth: ColorDepth | undefined;\n\nexport function detectColorDepth(env: NodeJS.ProcessEnv = process.env): ColorDepth {\n\tif (cachedDepth !== undefined && env === process.env) return cachedDepth;\n\n\tconst colorterm = (env.COLORTERM || \"\").toLowerCase();\n\tconst term = (env.TERM || \"\").toLowerCase();\n\tconst tmux = Boolean(env.TMUX);\n\n\tlet depth: ColorDepth;\n\n\tif (term === \"linux\") {\n\t\tdepth = \"16\";\n\t} else if (colorterm === \"truecolor\" || colorterm === \"24bit\") {\n\t\t// tmux without explicit truecolor indicator downgrades to 256.\n\t\tdepth = tmux && term.includes(\"screen\") && !colorterm ? \"256\" : \"truecolor\";\n\t} else if (tmux && !colorterm) {\n\t\t// tmux without truecolor indicator: 256 max per R7.\n\t\tdepth = term.includes(\"256color\") ? \"256\" : \"16\";\n\t} else if (term.includes(\"256color\") || term.includes(\"256colour\")) {\n\t\tdepth = \"256\";\n\t} else {\n\t\tdepth = \"16\";\n\t}\n\n\tif (env === process.env) cachedDepth = depth;\n\treturn depth;\n}\n\nexport function resetColorDepthCache(): void {\n\tcachedDepth = undefined;\n}\n\nfunction parseHex(hex: string): { r: number; g: number; b: number } | null {\n\tif (!hex) return null;\n\tconst v = hex.startsWith(\"#\") ? hex.slice(1) : hex;\n\tif (!/^[0-9a-fA-F]{6}$/.test(v)) return null;\n\treturn {\n\t\tr: parseInt(v.slice(0, 2), 16),\n\t\tg: parseInt(v.slice(2, 4), 16),\n\t\tb: parseInt(v.slice(4, 6), 16),\n\t};\n}\n\n/** Convert 0-255 channel to 0-5 cube index (6x6x6 cube in 256-color). */\nfunction channelTo6(v: number): number {\n\tif (v < 48) return 0;\n\tif (v < 115) return 1;\n\treturn Math.floor((v - 35) / 40);\n}\n\n/** Map (r,g,b) 0-255 to an xterm 256-color index (16-231 cube or 232-255 grayscale). */\nfunction rgbTo256(r: number, g: number, b: number): number {\n\t// Prefer grayscale ramp if channels are close together.\n\tconst max = Math.max(r, g, b);\n\tconst min = Math.min(r, g, b);\n\tif (max - min < 8) {\n\t\tconst avg = (r + g + b) / 3;\n\t\tif (avg < 8) return 16; // black in cube\n\t\tif (avg > 248) return 231; // white in cube\n\t\treturn 232 + Math.min(23, Math.floor((avg - 8) / 10));\n\t}\n\tconst ri = channelTo6(r);\n\tconst gi = channelTo6(g);\n\tconst bi = channelTo6(b);\n\treturn 16 + 36 * ri + 6 * gi + bi;\n}\n\n/**\n * Standard 16-color ANSI palette (approximate RGB values used for nearest-match).\n * Base 0-7 map to SGR 30-37 (fg) / 40-47 (bg); bright 8-15 map to 90-97 / 100-107.\n */\nconst ANSI16: { r: number; g: number; b: number }[] = [\n\t{ r: 0, g: 0, b: 0 }, // black\n\t{ r: 205, g: 0, b: 0 }, // red\n\t{ r: 0, g: 205, b: 0 }, // green\n\t{ r: 205, g: 205, b: 0 }, // yellow\n\t{ r: 0, g: 0, b: 238 }, // blue\n\t{ r: 205, g: 0, b: 205 }, // magenta\n\t{ r: 0, g: 205, b: 205 }, // cyan\n\t{ r: 229, g: 229, b: 229 }, // white\n\t{ r: 127, g: 127, b: 127 }, // bright black\n\t{ r: 255, g: 0, b: 0 }, // bright red\n\t{ r: 0, g: 255, b: 0 }, // bright green\n\t{ r: 255, g: 255, b: 0 }, // bright yellow\n\t{ r: 92, g: 92, b: 255 }, // bright blue\n\t{ r: 255, g: 0, b: 255 }, // bright magenta\n\t{ r: 0, g: 255, b: 255 }, // bright cyan\n\t{ r: 255, g: 255, b: 255 }, // bright white\n];\n\nfunction rgbTo16(r: number, g: number, b: number): number {\n\tlet best = 0;\n\tlet bestDist = Infinity;\n\tfor (let i = 0; i < ANSI16.length; i++) {\n\t\tconst p = ANSI16[i];\n\t\tconst dr = r - p.r;\n\t\tconst dg = g - p.g;\n\t\tconst db = b - p.b;\n\t\tconst dist = dr * dr + dg * dg + db * db;\n\t\tif (dist < bestDist) {\n\t\t\tbestDist = dist;\n\t\t\tbest = i;\n\t\t}\n\t}\n\treturn best;\n}\n\n/**\n * Emit the SGR escape sequence to set the given color as foreground or background\n * at the given color depth. Returns \"\" for invalid input (empty/undefined hex).\n */\nexport function hexToSgr(hex: string | undefined | null, depth: ColorDepth, kind: \"fg\" | \"bg\"): string {\n\tif (!hex) return \"\";\n\tconst rgb = parseHex(hex);\n\tif (!rgb) return \"\";\n\n\tif (depth === \"truecolor\") {\n\t\tconst code = kind === \"fg\" ? 38 : 48;\n\t\treturn `\\x1b[${code};2;${rgb.r};${rgb.g};${rgb.b}m`;\n\t}\n\n\tif (depth === \"256\") {\n\t\tconst idx = rgbTo256(rgb.r, rgb.g, rgb.b);\n\t\tconst code = kind === \"fg\" ? 38 : 48;\n\t\treturn `\\x1b[${code};5;${idx}m`;\n\t}\n\n\t// 16-color\n\tconst idx = rgbTo16(rgb.r, rgb.g, rgb.b);\n\tif (idx < 8) {\n\t\tconst base = kind === \"fg\" ? 30 : 40;\n\t\treturn `\\x1b[${base + idx}m`;\n\t}\n\tconst base = kind === \"fg\" ? 90 : 100;\n\treturn `\\x1b[${base + (idx - 8)}m`;\n}\n\nexport function sgrReset(): string {\n\treturn \"\\x1b[0m\";\n}\n"]}
@@ -0,0 +1,41 @@
1
+ /**
2
+ * Chapters — auto-fold transcript turns by detected intent.
3
+ *
4
+ * The TUI keeps a flat history; chapters group consecutive turns that share
5
+ * a detected intent ("debug", "refactor", "test", "explain", …) into a
6
+ * single foldable unit. Detection is heuristic (keyword + tool-pattern) so
7
+ * the UX stays predictable and predictable-feeling cheap.
8
+ */
9
+ export type Intent = "implement" | "debug" | "refactor" | "test" | "explain" | "review" | "plan" | "memory" | "other";
10
+ export interface Turn {
11
+ id: string;
12
+ role: "user" | "assistant" | "tool";
13
+ text: string;
14
+ tools?: string[];
15
+ timestamp: number;
16
+ }
17
+ export interface Chapter {
18
+ id: string;
19
+ intent: Intent;
20
+ title: string;
21
+ turns: Turn[];
22
+ startedAt: number;
23
+ endedAt: number;
24
+ folded: boolean;
25
+ }
26
+ /**
27
+ * Classify a single turn into an intent. The first matching rule wins; ties
28
+ * are broken by rule order, which mirrors how a user perceives priority
29
+ * (test/debug/refactor before generic implement).
30
+ */
31
+ export declare function detectIntent(turn: Turn): Intent;
32
+ /**
33
+ * Group a turn stream into chapters. Consecutive same-intent turns merge;
34
+ * tool-only turns inherit the previous user/assistant intent so they don't
35
+ * fragment a chapter.
36
+ */
37
+ export declare function groupTurnsIntoChapters(turns: Turn[]): Chapter[];
38
+ /** Toggle a chapter's folded state. Returns a new array (immutable). */
39
+ export declare function toggleChapter(chapters: Chapter[], id: string): Chapter[];
40
+ export declare const intentLabel: (intent: Intent) => string;
41
+ //# sourceMappingURL=Chapters.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Chapters.d.ts","sourceRoot":"","sources":["../../src/components/Chapters.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,MAAM,MAAM,MAAM,GAAG,WAAW,GAAG,OAAO,GAAG,UAAU,GAAG,MAAM,GAAG,SAAS,GAAG,QAAQ,GAAG,MAAM,GAAG,QAAQ,GAAG,OAAO,CAAC;AAEtH,MAAM,WAAW,IAAI;IACpB,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,GAAG,WAAW,GAAG,MAAM,CAAC;IACpC,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,CAAC,EAAE,MAAM,EAAE,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;CAClB;AAED,MAAM,WAAW,OAAO;IACvB,EAAE,EAAE,MAAM,CAAC;IACX,MAAM,EAAE,MAAM,CAAC;IACf,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,IAAI,EAAE,CAAC;IACd,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,OAAO,CAAC;CAChB;AA2BD;;;;GAIG;AACH,wBAAgB,YAAY,CAAC,IAAI,EAAE,IAAI,GAAG,MAAM,CAM/C;AAED;;;;GAIG;AACH,wBAAgB,sBAAsB,CAAC,KAAK,EAAE,IAAI,EAAE,GAAG,OAAO,EAAE,CAmC/D;AAoBD,wEAAwE;AACxE,wBAAgB,aAAa,CAAC,QAAQ,EAAE,OAAO,EAAE,EAAE,EAAE,EAAE,MAAM,GAAG,OAAO,EAAE,CAExE;AAED,eAAO,MAAM,WAAW,4BAAoD,CAAC","sourcesContent":["/**\n * Chapters — auto-fold transcript turns by detected intent.\n *\n * The TUI keeps a flat history; chapters group consecutive turns that share\n * a detected intent (\"debug\", \"refactor\", \"test\", \"explain\", …) into a\n * single foldable unit. Detection is heuristic (keyword + tool-pattern) so\n * the UX stays predictable and predictable-feeling cheap.\n */\n\nexport type Intent = \"implement\" | \"debug\" | \"refactor\" | \"test\" | \"explain\" | \"review\" | \"plan\" | \"memory\" | \"other\";\n\nexport interface Turn {\n\tid: string;\n\trole: \"user\" | \"assistant\" | \"tool\";\n\ttext: string;\n\ttools?: string[];\n\ttimestamp: number;\n}\n\nexport interface Chapter {\n\tid: string;\n\tintent: Intent;\n\ttitle: string;\n\tturns: Turn[];\n\tstartedAt: number;\n\tendedAt: number;\n\tfolded: boolean;\n}\n\ninterface IntentRule {\n\tintent: Intent;\n\tkeywords: RegExp;\n\ttools?: RegExp;\n}\n\nconst RULES: IntentRule[] = [\n\t{\n\t\tintent: \"debug\",\n\t\tkeywords:\n\t\t\t/\\b(bug|crash|stack trace|null|undefined|reproduce|fail(ed|ing)?|exception|TypeError|RangeError|why is|why does|broken)\\b/i,\n\t},\n\t{ intent: \"test\", keywords: /\\b(test|spec|vitest|jest|node:test|coverage|assertion)s?\\b/i, tools: /Bash|Test/ },\n\t{ intent: \"refactor\", keywords: /\\b(refactor|rename|extract|inline|deduplicat|cleanup|tidy|simplif|reorganiz)/i },\n\t{\n\t\tintent: \"implement\",\n\t\tkeywords: /\\b(implement|add|create|build|wire|hook up|introduce|scaffold)\\b/i,\n\t\ttools: /Edit|Write/,\n\t},\n\t{ intent: \"explain\", keywords: /\\b(explain|what does|how does|walk me through|tell me about|what is)\\b/i },\n\t{ intent: \"review\", keywords: /\\b(review|critique|feedback|audit|lgtm|nit:)/i },\n\t{ intent: \"plan\", keywords: /\\b(plan|outline|strategy|approach|design|architecture|sketch)\\b/i },\n\t{ intent: \"memory\", keywords: /\\b(remember|memory|memorize|recall|forget)\\b/i },\n];\n\n/**\n * Classify a single turn into an intent. The first matching rule wins; ties\n * are broken by rule order, which mirrors how a user perceives priority\n * (test/debug/refactor before generic implement).\n */\nexport function detectIntent(turn: Turn): Intent {\n\tfor (const rule of RULES) {\n\t\tif (rule.keywords.test(turn.text)) return rule.intent;\n\t\tif (rule.tools && turn.tools && turn.tools.some((t) => rule.tools?.test(t))) return rule.intent;\n\t}\n\treturn \"other\";\n}\n\n/**\n * Group a turn stream into chapters. Consecutive same-intent turns merge;\n * tool-only turns inherit the previous user/assistant intent so they don't\n * fragment a chapter.\n */\nexport function groupTurnsIntoChapters(turns: Turn[]): Chapter[] {\n\tconst out: Chapter[] = [];\n\tlet current: Chapter | undefined;\n\tlet lastNonToolIntent: Intent | undefined;\n\n\tfor (const turn of turns) {\n\t\tlet intent: Intent;\n\t\tif (turn.role === \"tool\" && lastNonToolIntent) {\n\t\t\tintent = lastNonToolIntent;\n\t\t} else {\n\t\t\tintent = detectIntent(turn);\n\t\t\tif (turn.role !== \"tool\") lastNonToolIntent = intent;\n\t\t}\n\n\t\tif (current && current.intent === intent) {\n\t\t\tcurrent.turns.push(turn);\n\t\t\tcurrent.endedAt = turn.timestamp;\n\t\t} else {\n\t\t\tcurrent = {\n\t\t\t\tid: `ch-${out.length + 1}`,\n\t\t\t\tintent,\n\t\t\t\ttitle: titleForIntent(intent, turn),\n\t\t\t\tturns: [turn],\n\t\t\t\tstartedAt: turn.timestamp,\n\t\t\t\tendedAt: turn.timestamp,\n\t\t\t\t// Fold every chapter except the last by default; the renderer\n\t\t\t\t// flips the most recent open after grouping.\n\t\t\t\tfolded: true,\n\t\t\t};\n\t\t\tout.push(current);\n\t\t}\n\t}\n\n\tif (out.length > 0) out[out.length - 1].folded = false;\n\treturn out;\n}\n\nfunction titleForIntent(intent: Intent, firstTurn: Turn): string {\n\tconst summary = firstTurn.text.replace(/\\s+/g, \" \").trim().slice(0, 60);\n\tconst prefix = INTENT_LABELS[intent];\n\treturn summary.length > 0 ? `${prefix}: ${summary}` : prefix;\n}\n\nconst INTENT_LABELS: Record<Intent, string> = {\n\timplement: \"Implement\",\n\tdebug: \"Debug\",\n\trefactor: \"Refactor\",\n\ttest: \"Test\",\n\texplain: \"Explain\",\n\treview: \"Review\",\n\tplan: \"Plan\",\n\tmemory: \"Memory\",\n\tother: \"Notes\",\n};\n\n/** Toggle a chapter's folded state. Returns a new array (immutable). */\nexport function toggleChapter(chapters: Chapter[], id: string): Chapter[] {\n\treturn chapters.map((c) => (c.id === id ? { ...c, folded: !c.folded } : c));\n}\n\nexport const intentLabel = (intent: Intent): string => INTENT_LABELS[intent];\n"]}