@code-yeongyu/senpi 2026.5.21-2 → 2026.5.23-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.
- package/CHANGELOG.md +65 -0
- package/README.md +1 -1
- package/dist/cli/file-processor.d.ts.map +1 -1
- package/dist/cli/file-processor.js +2 -3
- package/dist/cli/file-processor.js.map +1 -1
- package/dist/config.d.ts.map +1 -1
- package/dist/config.js +3 -10
- package/dist/config.js.map +1 -1
- package/dist/core/agent-session-runtime.d.ts.map +1 -1
- package/dist/core/agent-session-runtime.js +2 -1
- package/dist/core/agent-session-runtime.js.map +1 -1
- package/dist/core/agent-session-services.d.ts.map +1 -1
- package/dist/core/agent-session-services.js +3 -2
- package/dist/core/agent-session-services.js.map +1 -1
- package/dist/core/agent-session.d.ts +2 -0
- package/dist/core/agent-session.d.ts.map +1 -1
- package/dist/core/agent-session.js +28 -4
- package/dist/core/agent-session.js.map +1 -1
- package/dist/core/auth-storage.d.ts.map +1 -1
- package/dist/core/auth-storage.js +2 -1
- package/dist/core/auth-storage.js.map +1 -1
- package/dist/core/export-html/index.d.ts.map +1 -1
- package/dist/core/export-html/index.js +8 -7
- package/dist/core/export-html/index.js.map +1 -1
- package/dist/core/export-html/template.js +6 -3
- package/dist/core/extensions/builtin/compaction/index.d.ts.map +1 -1
- package/dist/core/extensions/builtin/compaction/index.js +9 -0
- package/dist/core/extensions/builtin/compaction/index.js.map +1 -1
- package/dist/core/extensions/builtin/history-search/filter.d.ts +3 -0
- package/dist/core/extensions/builtin/history-search/filter.d.ts.map +1 -0
- package/dist/core/extensions/builtin/history-search/filter.js +22 -0
- package/dist/core/extensions/builtin/history-search/filter.js.map +1 -0
- package/dist/core/extensions/builtin/history-search/index.d.ts +7 -0
- package/dist/core/extensions/builtin/history-search/index.d.ts.map +1 -0
- package/dist/core/extensions/builtin/history-search/index.js +45 -0
- package/dist/core/extensions/builtin/history-search/index.js.map +1 -0
- package/dist/core/extensions/builtin/history-search/indexer.d.ts +3 -0
- package/dist/core/extensions/builtin/history-search/indexer.d.ts.map +1 -0
- package/dist/core/extensions/builtin/history-search/indexer.js +161 -0
- package/dist/core/extensions/builtin/history-search/indexer.js.map +1 -0
- package/dist/core/extensions/builtin/history-search/overlay.d.ts +30 -0
- package/dist/core/extensions/builtin/history-search/overlay.d.ts.map +1 -0
- package/dist/core/extensions/builtin/history-search/overlay.js +115 -0
- package/dist/core/extensions/builtin/history-search/overlay.js.map +1 -0
- package/dist/core/extensions/builtin/history-search/types.d.ts +8 -0
- package/dist/core/extensions/builtin/history-search/types.d.ts.map +1 -0
- package/dist/core/extensions/builtin/history-search/types.js +2 -0
- package/dist/core/extensions/builtin/history-search/types.js.map +1 -0
- package/dist/core/extensions/builtin/index.d.ts.map +1 -1
- package/dist/core/extensions/builtin/index.js +4 -0
- package/dist/core/extensions/builtin/index.js.map +1 -1
- package/dist/core/extensions/builtin/session-observer/index.d.ts +5 -0
- package/dist/core/extensions/builtin/session-observer/index.d.ts.map +1 -0
- package/dist/core/extensions/builtin/session-observer/index.js +36 -0
- package/dist/core/extensions/builtin/session-observer/index.js.map +1 -0
- package/dist/core/extensions/builtin/session-observer/loader.d.ts +3 -0
- package/dist/core/extensions/builtin/session-observer/loader.d.ts.map +1 -0
- package/dist/core/extensions/builtin/session-observer/loader.js +20 -0
- package/dist/core/extensions/builtin/session-observer/loader.js.map +1 -0
- package/dist/core/extensions/builtin/session-observer/overlay-format.d.ts +7 -0
- package/dist/core/extensions/builtin/session-observer/overlay-format.d.ts.map +1 -0
- package/dist/core/extensions/builtin/session-observer/overlay-format.js +30 -0
- package/dist/core/extensions/builtin/session-observer/overlay-format.js.map +1 -0
- package/dist/core/extensions/builtin/session-observer/overlay.d.ts +51 -0
- package/dist/core/extensions/builtin/session-observer/overlay.d.ts.map +1 -0
- package/dist/core/extensions/builtin/session-observer/overlay.js +239 -0
- package/dist/core/extensions/builtin/session-observer/overlay.js.map +1 -0
- package/dist/core/extensions/builtin/session-observer/scanner.d.ts +10 -0
- package/dist/core/extensions/builtin/session-observer/scanner.d.ts.map +1 -0
- package/dist/core/extensions/builtin/session-observer/scanner.js +140 -0
- package/dist/core/extensions/builtin/session-observer/scanner.js.map +1 -0
- package/dist/core/extensions/builtin/session-observer/text.d.ts +7 -0
- package/dist/core/extensions/builtin/session-observer/text.d.ts.map +1 -0
- package/dist/core/extensions/builtin/session-observer/text.js +37 -0
- package/dist/core/extensions/builtin/session-observer/text.js.map +1 -0
- package/dist/core/extensions/builtin/session-observer/transcript-entries.d.ts +7 -0
- package/dist/core/extensions/builtin/session-observer/transcript-entries.d.ts.map +1 -0
- package/dist/core/extensions/builtin/session-observer/transcript-entries.js +71 -0
- package/dist/core/extensions/builtin/session-observer/transcript-entries.js.map +1 -0
- package/dist/core/extensions/builtin/session-observer/transcript-format.d.ts +11 -0
- package/dist/core/extensions/builtin/session-observer/transcript-format.d.ts.map +1 -0
- package/dist/core/extensions/builtin/session-observer/transcript-format.js +65 -0
- package/dist/core/extensions/builtin/session-observer/transcript-format.js.map +1 -0
- package/dist/core/extensions/builtin/session-observer/transcript.d.ts +4 -0
- package/dist/core/extensions/builtin/session-observer/transcript.d.ts.map +1 -0
- package/dist/core/extensions/builtin/session-observer/transcript.js +81 -0
- package/dist/core/extensions/builtin/session-observer/transcript.js.map +1 -0
- package/dist/core/extensions/builtin/session-observer/types.d.ts +33 -0
- package/dist/core/extensions/builtin/session-observer/types.d.ts.map +1 -0
- package/dist/core/extensions/builtin/session-observer/types.js +2 -0
- package/dist/core/extensions/builtin/session-observer/types.js.map +1 -0
- package/dist/core/extensions/loader.d.ts.map +1 -1
- package/dist/core/extensions/loader.js +13 -30
- package/dist/core/extensions/loader.js.map +1 -1
- package/dist/core/extensions/runner.d.ts.map +1 -1
- package/dist/core/extensions/runner.js +1 -0
- package/dist/core/extensions/runner.js.map +1 -1
- package/dist/core/keybindings.d.ts +10 -0
- package/dist/core/keybindings.d.ts.map +1 -1
- package/dist/core/keybindings.js +3 -0
- package/dist/core/keybindings.js.map +1 -1
- package/dist/core/model-registry.d.ts.map +1 -1
- package/dist/core/model-registry.js +5 -1
- package/dist/core/model-registry.js.map +1 -1
- package/dist/core/package-manager.d.ts +1 -0
- package/dist/core/package-manager.d.ts.map +1 -1
- package/dist/core/package-manager.js +47 -32
- package/dist/core/package-manager.js.map +1 -1
- package/dist/core/prompt-templates.d.ts.map +1 -1
- package/dist/core/prompt-templates.js +6 -20
- package/dist/core/prompt-templates.js.map +1 -1
- package/dist/core/resource-loader.d.ts.map +1 -1
- package/dist/core/resource-loader.js +38 -31
- package/dist/core/resource-loader.js.map +1 -1
- package/dist/core/sdk.d.ts.map +1 -1
- package/dist/core/sdk.js +9 -4
- package/dist/core/sdk.js.map +1 -1
- package/dist/core/session-manager.d.ts.map +1 -1
- package/dist/core/session-manager.js +32 -24
- package/dist/core/session-manager.js.map +1 -1
- package/dist/core/settings-manager.d.ts.map +1 -1
- package/dist/core/settings-manager.js +6 -13
- package/dist/core/settings-manager.js.map +1 -1
- package/dist/core/skills.d.ts.map +1 -1
- package/dist/core/skills.js +8 -22
- package/dist/core/skills.js.map +1 -1
- package/dist/core/tools/bash.d.ts.map +1 -1
- package/dist/core/tools/bash.js +54 -53
- package/dist/core/tools/bash.js.map +1 -1
- package/dist/core/tools/edit-diff.d.ts +3 -1
- package/dist/core/tools/edit-diff.d.ts.map +1 -1
- package/dist/core/tools/edit-diff.js +8 -1
- package/dist/core/tools/edit-diff.js.map +1 -1
- package/dist/core/tools/edit.d.ts +3 -1
- package/dist/core/tools/edit.d.ts.map +1 -1
- package/dist/core/tools/edit.js +44 -81
- package/dist/core/tools/edit.js.map +1 -1
- package/dist/core/tools/file-mutation-queue.d.ts.map +1 -1
- package/dist/core/tools/file-mutation-queue.js +27 -12
- package/dist/core/tools/file-mutation-queue.js.map +1 -1
- package/dist/core/tools/find.d.ts.map +1 -1
- package/dist/core/tools/find.js +2 -3
- package/dist/core/tools/find.js.map +1 -1
- package/dist/core/tools/grep.d.ts.map +1 -1
- package/dist/core/tools/grep.js +3 -3
- package/dist/core/tools/grep.js.map +1 -1
- package/dist/core/tools/ls.d.ts.map +1 -1
- package/dist/core/tools/ls.js +5 -5
- package/dist/core/tools/ls.js.map +1 -1
- package/dist/core/tools/output-accumulator.d.ts +2 -0
- package/dist/core/tools/output-accumulator.d.ts.map +1 -1
- package/dist/core/tools/output-accumulator.js +9 -3
- package/dist/core/tools/output-accumulator.js.map +1 -1
- package/dist/core/tools/path-utils.d.ts +2 -0
- package/dist/core/tools/path-utils.d.ts.map +1 -1
- package/dist/core/tools/path-utils.js +39 -21
- package/dist/core/tools/path-utils.js.map +1 -1
- package/dist/core/tools/read.d.ts.map +1 -1
- package/dist/core/tools/read.js +9 -8
- package/dist/core/tools/read.js.map +1 -1
- package/dist/core/tools/truncate.d.ts.map +1 -1
- package/dist/core/tools/truncate.js +12 -2
- package/dist/core/tools/truncate.js.map +1 -1
- package/dist/core/tools/write.d.ts.map +1 -1
- package/dist/core/tools/write.js +20 -35
- package/dist/core/tools/write.js.map +1 -1
- package/dist/main.d.ts.map +1 -1
- package/dist/main.js +5 -6
- package/dist/main.js.map +1 -1
- package/dist/modes/interactive/components/config-selector.d.ts.map +1 -1
- package/dist/modes/interactive/components/config-selector.js +1 -1
- package/dist/modes/interactive/components/config-selector.js.map +1 -1
- package/dist/modes/interactive/components/footer.d.ts +1 -0
- package/dist/modes/interactive/components/footer.d.ts.map +1 -1
- package/dist/modes/interactive/components/footer.js +87 -67
- package/dist/modes/interactive/components/footer.js.map +1 -1
- package/dist/modes/interactive/components/login-dialog.d.ts +9 -1
- package/dist/modes/interactive/components/login-dialog.d.ts.map +1 -1
- package/dist/modes/interactive/components/login-dialog.js +29 -4
- package/dist/modes/interactive/components/login-dialog.js.map +1 -1
- package/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
- package/dist/modes/interactive/interactive-mode.js +22 -4
- package/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/dist/modes/interactive/theme/theme.d.ts.map +1 -1
- package/dist/modes/interactive/theme/theme.js +37 -28
- package/dist/modes/interactive/theme/theme.js.map +1 -1
- package/dist/utils/clipboard-native.d.ts +3 -1
- package/dist/utils/clipboard-native.d.ts.map +1 -1
- package/dist/utils/clipboard-native.js +14 -8
- package/dist/utils/clipboard-native.js.map +1 -1
- package/dist/utils/image-resize-core.d.ts +30 -0
- package/dist/utils/image-resize-core.d.ts.map +1 -0
- package/dist/utils/image-resize-core.js +124 -0
- package/dist/utils/image-resize-core.js.map +1 -0
- package/dist/utils/image-resize-worker.d.ts +2 -0
- package/dist/utils/image-resize-worker.d.ts.map +1 -0
- package/dist/utils/image-resize-worker.js +31 -0
- package/dist/utils/image-resize-worker.js.map +1 -0
- package/dist/utils/image-resize.d.ts +7 -27
- package/dist/utils/image-resize.d.ts.map +1 -1
- package/dist/utils/image-resize.js +75 -115
- package/dist/utils/image-resize.js.map +1 -1
- package/dist/utils/paths.d.ts +16 -1
- package/dist/utils/paths.d.ts.map +1 -1
- package/dist/utils/paths.js +41 -7
- package/dist/utils/paths.js.map +1 -1
- package/docs/custom-provider.md +44 -12
- package/docs/models.md +8 -2
- package/docs/packages.md +5 -4
- package/docs/sdk.md +2 -0
- package/docs/usage.md +2 -2
- package/examples/extensions/custom-provider-anthropic/package-lock.json +2 -2
- package/examples/extensions/custom-provider-anthropic/package.json +1 -1
- package/examples/extensions/custom-provider-gitlab-duo/package.json +1 -1
- package/examples/extensions/sandbox/package-lock.json +2 -2
- package/examples/extensions/sandbox/package.json +1 -1
- package/examples/extensions/with-deps/package-lock.json +2 -2
- package/examples/extensions/with-deps/package.json +1 -1
- package/node_modules/@earendil-works/pi-agent-core/package.json +2 -2
- package/node_modules/@earendil-works/pi-ai/dist/cli.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/cli.js +14 -0
- package/node_modules/@earendil-works/pi-ai/dist/cli.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/index.d.ts +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/index.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/index.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/models.generated.d.ts +145 -225
- package/node_modules/@earendil-works/pi-ai/dist/models.generated.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/models.generated.js +134 -225
- package/node_modules/@earendil-works/pi-ai/dist/models.generated.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/amazon-bedrock.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/amazon-bedrock.js +2 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/amazon-bedrock.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/anthropic.d.ts +27 -8
- package/node_modules/@earendil-works/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/anthropic.js +35 -22
- package/node_modules/@earendil-works/pi-ai/dist/providers/anthropic.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/types.d.ts +10 -0
- package/node_modules/@earendil-works/pi-ai/dist/types.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/types.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/device-code.d.ts +19 -0
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/device-code.d.ts.map +1 -0
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/device-code.js +55 -0
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/device-code.js.map +1 -0
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/github-copilot.d.ts +3 -3
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/github-copilot.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/github-copilot.js +45 -69
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/github-copilot.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/index.d.ts +1 -0
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/index.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/index.js +1 -0
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/index.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/types.d.ts +8 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/types.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/oauth/types.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/package.json +2 -2
- package/node_modules/@earendil-works/pi-tui/package.json +1 -1
- package/npm-shrinkwrap.json +13 -13
- package/package.json +5 -5
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
import { basename } from "node:path";
|
|
2
|
+
import { Container, getKeybindings, Input, SelectList, Text } from "@earendil-works/pi-tui";
|
|
3
|
+
import { DynamicBorder } from "../../../../modes/interactive/components/dynamic-border.js";
|
|
4
|
+
import { filterHistory } from "./filter.js";
|
|
5
|
+
const MAX_VISIBLE_ROWS = 15;
|
|
6
|
+
const MAX_RENDERED_MATCHES = 250;
|
|
7
|
+
const SELECT_LIST_ACTIONS = [
|
|
8
|
+
"tui.select.up",
|
|
9
|
+
"tui.select.down",
|
|
10
|
+
"tui.select.confirm",
|
|
11
|
+
"tui.select.cancel",
|
|
12
|
+
];
|
|
13
|
+
function relativeTime(timestamp, now = Date.now()) {
|
|
14
|
+
const seconds = Math.max(0, Math.floor((now - timestamp) / 1_000));
|
|
15
|
+
if (seconds < 60)
|
|
16
|
+
return "now";
|
|
17
|
+
const minutes = Math.floor(seconds / 60);
|
|
18
|
+
if (minutes < 60)
|
|
19
|
+
return `${minutes}m ago`;
|
|
20
|
+
const hours = Math.floor(minutes / 60);
|
|
21
|
+
if (hours < 24)
|
|
22
|
+
return `${hours}h ago`;
|
|
23
|
+
const days = Math.floor(hours / 24);
|
|
24
|
+
if (days < 30)
|
|
25
|
+
return `${days}d ago`;
|
|
26
|
+
const months = Math.floor(days / 30);
|
|
27
|
+
if (months < 12)
|
|
28
|
+
return `${months}mo ago`;
|
|
29
|
+
return `${Math.floor(months / 12)}y ago`;
|
|
30
|
+
}
|
|
31
|
+
function describeEntry(entry) {
|
|
32
|
+
const shortId = entry.sessionId.length <= 8 ? entry.sessionId : entry.sessionId.slice(0, 8);
|
|
33
|
+
const cwdName = basename(entry.cwd);
|
|
34
|
+
const sessionLabel = cwdName ? `${cwdName}/${shortId}` : shortId;
|
|
35
|
+
return `${sessionLabel} · ${relativeTime(entry.timestamp)}`;
|
|
36
|
+
}
|
|
37
|
+
export class HistorySearchOverlay extends Container {
|
|
38
|
+
searchInput;
|
|
39
|
+
entriesByValue = new Map();
|
|
40
|
+
options;
|
|
41
|
+
list;
|
|
42
|
+
filteredEntries = [];
|
|
43
|
+
_focused = false;
|
|
44
|
+
constructor(options) {
|
|
45
|
+
super();
|
|
46
|
+
this.options = options;
|
|
47
|
+
this.searchInput = new Input();
|
|
48
|
+
this.searchInput.onEscape = () => this.options.done(undefined);
|
|
49
|
+
this.rebuild();
|
|
50
|
+
}
|
|
51
|
+
get focused() {
|
|
52
|
+
return this._focused;
|
|
53
|
+
}
|
|
54
|
+
set focused(value) {
|
|
55
|
+
this._focused = value;
|
|
56
|
+
this.searchInput.focused = value;
|
|
57
|
+
}
|
|
58
|
+
handleInput(input) {
|
|
59
|
+
const keybindings = getKeybindings();
|
|
60
|
+
if (SELECT_LIST_ACTIONS.some((action) => keybindings.matches(input, action))) {
|
|
61
|
+
this.list?.handleInput(input);
|
|
62
|
+
return;
|
|
63
|
+
}
|
|
64
|
+
const before = this.searchInput.getValue();
|
|
65
|
+
this.searchInput.handleInput(input);
|
|
66
|
+
if (before !== this.searchInput.getValue()) {
|
|
67
|
+
this.rebuild();
|
|
68
|
+
this.options.tui.requestRender();
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
getSearchValue() {
|
|
72
|
+
return this.searchInput.getValue();
|
|
73
|
+
}
|
|
74
|
+
getFilteredEntries() {
|
|
75
|
+
return this.filteredEntries;
|
|
76
|
+
}
|
|
77
|
+
rebuild() {
|
|
78
|
+
this.entriesByValue.clear();
|
|
79
|
+
this.filteredEntries = filterHistory(this.options.entries, this.searchInput.getValue());
|
|
80
|
+
const renderedEntries = this.filteredEntries.slice(0, MAX_RENDERED_MATCHES);
|
|
81
|
+
const items = renderedEntries.map((entry, index) => this.toSelectItem(entry, index));
|
|
82
|
+
const list = new SelectList(items, Math.min(MAX_VISIBLE_ROWS, Math.max(1, items.length)), {
|
|
83
|
+
selectedPrefix: (text) => this.options.theme.fg("accent", text),
|
|
84
|
+
selectedText: (text) => text,
|
|
85
|
+
description: (text) => this.options.theme.fg("muted", text),
|
|
86
|
+
scrollInfo: (text) => this.options.theme.fg("dim", text),
|
|
87
|
+
noMatch: (text) => this.options.theme.fg("warning", text.replace("commands", "prompts")),
|
|
88
|
+
});
|
|
89
|
+
list.onSelect = (item) => this.options.done(this.entriesByValue.get(item.value));
|
|
90
|
+
list.onCancel = () => this.options.done(undefined);
|
|
91
|
+
this.list = list;
|
|
92
|
+
this.renderContainer(list, this.filteredEntries.length);
|
|
93
|
+
}
|
|
94
|
+
toSelectItem(entry, index) {
|
|
95
|
+
const value = String(index);
|
|
96
|
+
this.entriesByValue.set(value, entry);
|
|
97
|
+
return {
|
|
98
|
+
value,
|
|
99
|
+
label: entry.text.replace(/[\r\n]+/g, " ").trim(),
|
|
100
|
+
description: describeEntry(entry),
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
renderContainer(list, matchCount) {
|
|
104
|
+
const title = this.options.theme.fg("accent", this.options.theme.bold(" Search prompt history"));
|
|
105
|
+
const count = this.options.theme.fg("dim", ` ${matchCount}/${this.options.entries.length} prompts`);
|
|
106
|
+
this.clear();
|
|
107
|
+
this.addChild(new DynamicBorder((text) => this.options.theme.fg("accent", text)));
|
|
108
|
+
this.addChild(new Text(`${title}${count}`, 0, 0));
|
|
109
|
+
this.addChild(this.searchInput);
|
|
110
|
+
this.addChild(list);
|
|
111
|
+
this.addChild(new Text(this.options.theme.fg("dim", " Type to filter • ↑↓ navigate • enter select • esc close"), 0, 0));
|
|
112
|
+
this.addChild(new DynamicBorder((text) => this.options.theme.fg("accent", text)));
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
//# sourceMappingURL=overlay.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"overlay.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/history-search/overlay.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAC;AAErC,OAAO,EAAE,SAAS,EAAE,cAAc,EAAE,KAAK,EAAmB,UAAU,EAAE,IAAI,EAAE,MAAM,wBAAwB,CAAC;AAC7G,OAAO,EAAE,aAAa,EAAE,MAAM,4DAA4D,CAAC;AAE3F,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAG5C,MAAM,gBAAgB,GAAG,EAAE,CAAC;AAC5B,MAAM,oBAAoB,GAAG,GAAG,CAAC;AAIjC,MAAM,mBAAmB,GAAgC;IACxD,eAAe;IACf,iBAAiB;IACjB,oBAAoB;IACpB,mBAAmB;CACnB,CAAC;AAWF,SAAS,YAAY,CAAC,SAAiB,EAAE,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,EAAU;IAClE,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,GAAG,SAAS,CAAC,GAAG,KAAK,CAAC,CAAC,CAAC;IACnE,IAAI,OAAO,GAAG,EAAE;QAAE,OAAO,KAAK,CAAC;IAC/B,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,EAAE,CAAC,CAAC;IACzC,IAAI,OAAO,GAAG,EAAE;QAAE,OAAO,GAAG,OAAO,OAAO,CAAC;IAC3C,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,EAAE,CAAC,CAAC;IACvC,IAAI,KAAK,GAAG,EAAE;QAAE,OAAO,GAAG,KAAK,OAAO,CAAC;IACvC,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,EAAE,CAAC,CAAC;IACpC,IAAI,IAAI,GAAG,EAAE;QAAE,OAAO,GAAG,IAAI,OAAO,CAAC;IACrC,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,GAAG,EAAE,CAAC,CAAC;IACrC,IAAI,MAAM,GAAG,EAAE;QAAE,OAAO,GAAG,MAAM,QAAQ,CAAC;IAC1C,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,EAAE,CAAC,OAAO,CAAC;AAAA,CACzC;AAED,SAAS,aAAa,CAAC,KAAmB,EAAU;IACnD,MAAM,OAAO,GAAG,KAAK,CAAC,SAAS,CAAC,MAAM,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;IAC5F,MAAM,OAAO,GAAG,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACpC,MAAM,YAAY,GAAG,OAAO,CAAC,CAAC,CAAC,GAAG,OAAO,IAAI,OAAO,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC;IACjE,OAAO,GAAG,YAAY,OAAM,YAAY,CAAC,KAAK,CAAC,SAAS,CAAC,EAAE,CAAC;AAAA,CAC5D;AAED,MAAM,OAAO,oBAAqB,SAAQ,SAAS;IACjC,WAAW,CAAQ;IACnB,cAAc,GAAG,IAAI,GAAG,EAAwB,CAAC;IACjD,OAAO,CAA8B;IAC9C,IAAI,CAAyB;IAC7B,eAAe,GAA4B,EAAE,CAAC;IAC9C,QAAQ,GAAG,KAAK,CAAC;IAEzB,YAAY,OAAoC,EAAE;QACjD,KAAK,EAAE,CAAC;QAER,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;QACvB,IAAI,CAAC,WAAW,GAAG,IAAI,KAAK,EAAE,CAAC;QAC/B,IAAI,CAAC,WAAW,CAAC,QAAQ,GAAG,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAC/D,IAAI,CAAC,OAAO,EAAE,CAAC;IAAA,CACf;IAED,IAAI,OAAO,GAAY;QACtB,OAAO,IAAI,CAAC,QAAQ,CAAC;IAAA,CACrB;IAED,IAAI,OAAO,CAAC,KAAc,EAAE;QAC3B,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;QACtB,IAAI,CAAC,WAAW,CAAC,OAAO,GAAG,KAAK,CAAC;IAAA,CACjC;IAED,WAAW,CAAC,KAAa,EAAQ;QAChC,MAAM,WAAW,GAAG,cAAc,EAAE,CAAC;QACrC,IAAI,mBAAmB,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,WAAW,CAAC,OAAO,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC,EAAE,CAAC;YAC9E,IAAI,CAAC,IAAI,EAAE,WAAW,CAAC,KAAK,CAAC,CAAC;YAC9B,OAAO;QACR,CAAC;QAED,MAAM,MAAM,GAAG,IAAI,CAAC,WAAW,CAAC,QAAQ,EAAE,CAAC;QAC3C,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;QACpC,IAAI,MAAM,KAAK,IAAI,CAAC,WAAW,CAAC,QAAQ,EAAE,EAAE,CAAC;YAC5C,IAAI,CAAC,OAAO,EAAE,CAAC;YACf,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;QAClC,CAAC;IAAA,CACD;IAED,cAAc,GAAW;QACxB,OAAO,IAAI,CAAC,WAAW,CAAC,QAAQ,EAAE,CAAC;IAAA,CACnC;IAED,kBAAkB,GAA4B;QAC7C,OAAO,IAAI,CAAC,eAAe,CAAC;IAAA,CAC5B;IAEO,OAAO,GAAS;QACvB,IAAI,CAAC,cAAc,CAAC,KAAK,EAAE,CAAC;QAC5B,IAAI,CAAC,eAAe,GAAG,aAAa,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,IAAI,CAAC,WAAW,CAAC,QAAQ,EAAE,CAAC,CAAC;QACxF,MAAM,eAAe,GAAG,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,EAAE,oBAAoB,CAAC,CAAC;QAC5E,MAAM,KAAK,GAAG,eAAe,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,KAAK,EAAE,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC;QACrF,MAAM,IAAI,GAAG,IAAI,UAAU,CAAC,KAAK,EAAE,IAAI,CAAC,GAAG,CAAC,gBAAgB,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,KAAK,CAAC,MAAM,CAAC,CAAC,EAAE;YACzF,cAAc,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,QAAQ,EAAE,IAAI,CAAC;YAC/D,YAAY,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI;YAC5B,WAAW,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,OAAO,EAAE,IAAI,CAAC;YAC3D,UAAU,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,EAAE,IAAI,CAAC;YACxD,OAAO,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,SAAS,EAAE,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE,SAAS,CAAC,CAAC;SACxF,CAAC,CAAC;QACH,IAAI,CAAC,QAAQ,GAAG,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC;QACjF,IAAI,CAAC,QAAQ,GAAG,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACnD,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,eAAe,CAAC,IAAI,EAAE,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC;IAAA,CACxD;IAEO,YAAY,CAAC,KAAmB,EAAE,KAAa,EAAc;QACpE,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;QAC5B,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;QACtC,OAAO;YACN,KAAK;YACL,KAAK,EAAE,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE,GAAG,CAAC,CAAC,IAAI,EAAE;YACjD,WAAW,EAAE,aAAa,CAAC,KAAK,CAAC;SACjC,CAAC;IAAA,CACF;IAEO,eAAe,CAAC,IAAgB,EAAE,UAAkB,EAAQ;QACnE,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,QAAQ,EAAE,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,wBAAwB,CAAC,CAAC,CAAC;QACjG,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,EAAE,IAAI,UAAU,IAAI,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,MAAM,UAAU,CAAC,CAAC;QACpG,IAAI,CAAC,KAAK,EAAE,CAAC;QACb,IAAI,CAAC,QAAQ,CAAC,IAAI,aAAa,CAAC,CAAC,IAAY,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC;QAC1F,IAAI,CAAC,QAAQ,CAAC,IAAI,IAAI,CAAC,GAAG,KAAK,GAAG,KAAK,EAAE,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;QAClD,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAChC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QACpB,IAAI,CAAC,QAAQ,CACZ,IAAI,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,EAAE,oEAA0D,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CACxG,CAAC;QACF,IAAI,CAAC,QAAQ,CAAC,IAAI,aAAa,CAAC,CAAC,IAAY,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC;IAAA,CAC1F;CACD","sourcesContent":["import { basename } from \"node:path\";\nimport type { Focusable, TUI } from \"@earendil-works/pi-tui\";\nimport { Container, getKeybindings, Input, type SelectItem, SelectList, Text } from \"@earendil-works/pi-tui\";\nimport { DynamicBorder } from \"../../../../modes/interactive/components/dynamic-border.ts\";\nimport type { Theme } from \"../../../../modes/interactive/theme/theme.ts\";\nimport { filterHistory } from \"./filter.ts\";\nimport type { HistoryEntry } from \"./types.ts\";\n\nconst MAX_VISIBLE_ROWS = 15;\nconst MAX_RENDERED_MATCHES = 250;\n\ntype SelectListAction = \"tui.select.up\" | \"tui.select.down\" | \"tui.select.confirm\" | \"tui.select.cancel\";\n\nconst SELECT_LIST_ACTIONS: readonly SelectListAction[] = [\n\t\"tui.select.up\",\n\t\"tui.select.down\",\n\t\"tui.select.confirm\",\n\t\"tui.select.cancel\",\n];\n\ntype HistorySearchTui = Pick<TUI, \"requestRender\">;\n\ntype HistorySearchOverlayOptions = {\n\treadonly tui: HistorySearchTui;\n\treadonly entries: readonly HistoryEntry[];\n\treadonly theme: Theme;\n\treadonly done: (entry: HistoryEntry | undefined) => void;\n};\n\nfunction relativeTime(timestamp: number, now = Date.now()): string {\n\tconst seconds = Math.max(0, Math.floor((now - timestamp) / 1_000));\n\tif (seconds < 60) return \"now\";\n\tconst minutes = Math.floor(seconds / 60);\n\tif (minutes < 60) return `${minutes}m ago`;\n\tconst hours = Math.floor(minutes / 60);\n\tif (hours < 24) return `${hours}h ago`;\n\tconst days = Math.floor(hours / 24);\n\tif (days < 30) return `${days}d ago`;\n\tconst months = Math.floor(days / 30);\n\tif (months < 12) return `${months}mo ago`;\n\treturn `${Math.floor(months / 12)}y ago`;\n}\n\nfunction describeEntry(entry: HistoryEntry): string {\n\tconst shortId = entry.sessionId.length <= 8 ? entry.sessionId : entry.sessionId.slice(0, 8);\n\tconst cwdName = basename(entry.cwd);\n\tconst sessionLabel = cwdName ? `${cwdName}/${shortId}` : shortId;\n\treturn `${sessionLabel} · ${relativeTime(entry.timestamp)}`;\n}\n\nexport class HistorySearchOverlay extends Container implements Focusable {\n\tprivate readonly searchInput: Input;\n\tprivate readonly entriesByValue = new Map<string, HistoryEntry>();\n\tprivate readonly options: HistorySearchOverlayOptions;\n\tprivate list: SelectList | undefined;\n\tprivate filteredEntries: readonly HistoryEntry[] = [];\n\tprivate _focused = false;\n\n\tconstructor(options: HistorySearchOverlayOptions) {\n\t\tsuper();\n\n\t\tthis.options = options;\n\t\tthis.searchInput = new Input();\n\t\tthis.searchInput.onEscape = () => this.options.done(undefined);\n\t\tthis.rebuild();\n\t}\n\n\tget focused(): boolean {\n\t\treturn this._focused;\n\t}\n\n\tset focused(value: boolean) {\n\t\tthis._focused = value;\n\t\tthis.searchInput.focused = value;\n\t}\n\n\thandleInput(input: string): void {\n\t\tconst keybindings = getKeybindings();\n\t\tif (SELECT_LIST_ACTIONS.some((action) => keybindings.matches(input, action))) {\n\t\t\tthis.list?.handleInput(input);\n\t\t\treturn;\n\t\t}\n\n\t\tconst before = this.searchInput.getValue();\n\t\tthis.searchInput.handleInput(input);\n\t\tif (before !== this.searchInput.getValue()) {\n\t\t\tthis.rebuild();\n\t\t\tthis.options.tui.requestRender();\n\t\t}\n\t}\n\n\tgetSearchValue(): string {\n\t\treturn this.searchInput.getValue();\n\t}\n\n\tgetFilteredEntries(): readonly HistoryEntry[] {\n\t\treturn this.filteredEntries;\n\t}\n\n\tprivate rebuild(): void {\n\t\tthis.entriesByValue.clear();\n\t\tthis.filteredEntries = filterHistory(this.options.entries, this.searchInput.getValue());\n\t\tconst renderedEntries = this.filteredEntries.slice(0, MAX_RENDERED_MATCHES);\n\t\tconst items = renderedEntries.map((entry, index) => this.toSelectItem(entry, index));\n\t\tconst list = new SelectList(items, Math.min(MAX_VISIBLE_ROWS, Math.max(1, items.length)), {\n\t\t\tselectedPrefix: (text) => this.options.theme.fg(\"accent\", text),\n\t\t\tselectedText: (text) => text,\n\t\t\tdescription: (text) => this.options.theme.fg(\"muted\", text),\n\t\t\tscrollInfo: (text) => this.options.theme.fg(\"dim\", text),\n\t\t\tnoMatch: (text) => this.options.theme.fg(\"warning\", text.replace(\"commands\", \"prompts\")),\n\t\t});\n\t\tlist.onSelect = (item) => this.options.done(this.entriesByValue.get(item.value));\n\t\tlist.onCancel = () => this.options.done(undefined);\n\t\tthis.list = list;\n\t\tthis.renderContainer(list, this.filteredEntries.length);\n\t}\n\n\tprivate toSelectItem(entry: HistoryEntry, index: number): SelectItem {\n\t\tconst value = String(index);\n\t\tthis.entriesByValue.set(value, entry);\n\t\treturn {\n\t\t\tvalue,\n\t\t\tlabel: entry.text.replace(/[\\r\\n]+/g, \" \").trim(),\n\t\t\tdescription: describeEntry(entry),\n\t\t};\n\t}\n\n\tprivate renderContainer(list: SelectList, matchCount: number): void {\n\t\tconst title = this.options.theme.fg(\"accent\", this.options.theme.bold(\" Search prompt history\"));\n\t\tconst count = this.options.theme.fg(\"dim\", ` ${matchCount}/${this.options.entries.length} prompts`);\n\t\tthis.clear();\n\t\tthis.addChild(new DynamicBorder((text: string) => this.options.theme.fg(\"accent\", text)));\n\t\tthis.addChild(new Text(`${title}${count}`, 0, 0));\n\t\tthis.addChild(this.searchInput);\n\t\tthis.addChild(list);\n\t\tthis.addChild(\n\t\t\tnew Text(this.options.theme.fg(\"dim\", \" Type to filter • ↑↓ navigate • enter select • esc close\"), 0, 0),\n\t\t);\n\t\tthis.addChild(new DynamicBorder((text: string) => this.options.theme.fg(\"accent\", text)));\n\t}\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/history-search/types.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,YAAY,GAAG;IAC1B,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAC;IACtB,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAC;IAC3B,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC;IAC7B,QAAQ,CAAC,GAAG,EAAE,MAAM,CAAC;IACrB,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAC;CAC3B,CAAC","sourcesContent":["export type HistoryEntry = {\n\treadonly text: string;\n\treadonly sessionId: string;\n\treadonly sessionFile: string;\n\treadonly cwd: string;\n\treadonly timestamp: number;\n};\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/history-search/types.ts"],"names":[],"mappings":"","sourcesContent":["export type HistoryEntry = {\n\treadonly text: string;\n\treadonly sessionId: string;\n\treadonly sessionFile: string;\n\treadonly cwd: string;\n\treadonly timestamp: number;\n};\n"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/core/extensions/builtin/index.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,aAAa,CAAC;AAKpD,OAAO,aAAa,MAAM,WAAW,CAAC;AACtC,OAAO,cAAc,MAAM,YAAY,CAAC;
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/core/extensions/builtin/index.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,aAAa,CAAC;AAKpD,OAAO,aAAa,MAAM,WAAW,CAAC;AACtC,OAAO,cAAc,MAAM,YAAY,CAAC;AAOxC,OAAO,wBAAwB,MAAM,wBAAwB,CAAC;AAM9D,OAAO,YAAY,MAAM,UAAU,CAAC;AAEpC,MAAM,WAAW,uBAAuB;IACvC,EAAE,EAAE,MAAM,CAAC;IACX,OAAO,EAAE,gBAAgB,CAAC;CAC1B;AAED,eAAO,MAAM,yBAAyB,wDAAyD,CAAC;AAEhG,eAAO,MAAM,+BAA+B;;;;;CAKoC,CAAC;AAEjF,eAAO,MAAM,iBAAiB,EAAE,uBAAuB,EAgBtD,CAAC","sourcesContent":["import type { ExtensionFactory } from \"../types.ts\";\nimport anthropicBashExtension from \"./anthropic-bash/index.ts\";\nimport anthropicWebSearchExtension from \"./anthropic-web-search/index.ts\";\nimport bashTimeoutExtension from \"./bash-timeout/index.ts\";\nimport compactionExtension from \"./compaction/index.ts\";\nimport diffExtension from \"./diff.ts\";\nimport filesExtension from \"./files.ts\";\nimport gptApplyPatchExtension from \"./gpt-apply-patch/index.ts\";\nimport historySearchExtension from \"./history-search/index.ts\";\nimport kimiWebSearchExtension from \"./kimi-web-search/index.ts\";\nimport openaiWebSearchExtension from \"./openai-web-search/index.ts\";\nimport permissionSystemExtension from \"./permission-system/index.ts\";\nimport promptPresetExtension from \"./prompt-preset/index.ts\";\nimport promptUrlWidgetExtension from \"./prompt-url-widget.ts\";\nimport redrawsExtension from \"./redraws.ts\";\nimport serviceTierExtension from \"./service-tier.ts\";\nimport sessionObserverExtension from \"./session-observer/index.ts\";\nimport todowriteExtension from \"./todotools/index.ts\";\nimport toolPairGuardExtension from \"./tool-pair-guard/index.ts\";\nimport tpsExtension from \"./tps.ts\";\n\nexport interface BuiltinExtensionFactory {\n\tid: string;\n\tfactory: ExtensionFactory;\n}\n\nexport const globalDefaultExtensionIds = [\"diff\", \"files\", \"prompt-url-widget\", \"tps\"] as const;\n\nexport const globalDefaultExtensionFactories = {\n\tdiff: diffExtension,\n\tfiles: filesExtension,\n\t\"prompt-url-widget\": promptUrlWidgetExtension,\n\ttps: tpsExtension,\n} satisfies Record<(typeof globalDefaultExtensionIds)[number], ExtensionFactory>;\n\nexport const builtinExtensions: BuiltinExtensionFactory[] = [\n\t{ id: \"permission-system\", factory: permissionSystemExtension },\n\t{ id: \"gpt-apply-patch\", factory: gptApplyPatchExtension },\n\t{ id: \"prompt-preset\", factory: promptPresetExtension },\n\t{ id: \"todowrite\", factory: todowriteExtension },\n\t{ id: \"redraws\", factory: redrawsExtension },\n\t{ id: \"anthropic-web-search\", factory: anthropicWebSearchExtension },\n\t{ id: \"anthropic-bash\", factory: anthropicBashExtension },\n\t{ id: \"openai-web-search\", factory: openaiWebSearchExtension },\n\t{ id: \"service-tier\", factory: serviceTierExtension },\n\t{ id: \"bash-timeout\", factory: bashTimeoutExtension },\n\t{ id: \"tool-pair-guard\", factory: toolPairGuardExtension },\n\t{ id: \"compaction\", factory: compactionExtension },\n\t{ id: \"history-search\", factory: historySearchExtension },\n\t{ id: \"session-observer\", factory: sessionObserverExtension },\n\t{ id: \"kimi-web-search\", factory: kimiWebSearchExtension },\n];\n"]}
|
|
@@ -5,6 +5,7 @@ import compactionExtension from "./compaction/index.js";
|
|
|
5
5
|
import diffExtension from "./diff.js";
|
|
6
6
|
import filesExtension from "./files.js";
|
|
7
7
|
import gptApplyPatchExtension from "./gpt-apply-patch/index.js";
|
|
8
|
+
import historySearchExtension from "./history-search/index.js";
|
|
8
9
|
import kimiWebSearchExtension from "./kimi-web-search/index.js";
|
|
9
10
|
import openaiWebSearchExtension from "./openai-web-search/index.js";
|
|
10
11
|
import permissionSystemExtension from "./permission-system/index.js";
|
|
@@ -12,6 +13,7 @@ import promptPresetExtension from "./prompt-preset/index.js";
|
|
|
12
13
|
import promptUrlWidgetExtension from "./prompt-url-widget.js";
|
|
13
14
|
import redrawsExtension from "./redraws.js";
|
|
14
15
|
import serviceTierExtension from "./service-tier.js";
|
|
16
|
+
import sessionObserverExtension from "./session-observer/index.js";
|
|
15
17
|
import todowriteExtension from "./todotools/index.js";
|
|
16
18
|
import toolPairGuardExtension from "./tool-pair-guard/index.js";
|
|
17
19
|
import tpsExtension from "./tps.js";
|
|
@@ -35,6 +37,8 @@ export const builtinExtensions = [
|
|
|
35
37
|
{ id: "bash-timeout", factory: bashTimeoutExtension },
|
|
36
38
|
{ id: "tool-pair-guard", factory: toolPairGuardExtension },
|
|
37
39
|
{ id: "compaction", factory: compactionExtension },
|
|
40
|
+
{ id: "history-search", factory: historySearchExtension },
|
|
41
|
+
{ id: "session-observer", factory: sessionObserverExtension },
|
|
38
42
|
{ id: "kimi-web-search", factory: kimiWebSearchExtension },
|
|
39
43
|
];
|
|
40
44
|
//# sourceMappingURL=index.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/core/extensions/builtin/index.ts"],"names":[],"mappings":"AACA,OAAO,sBAAsB,MAAM,2BAA2B,CAAC;AAC/D,OAAO,2BAA2B,MAAM,iCAAiC,CAAC;AAC1E,OAAO,oBAAoB,MAAM,yBAAyB,CAAC;AAC3D,OAAO,mBAAmB,MAAM,uBAAuB,CAAC;AACxD,OAAO,aAAa,MAAM,WAAW,CAAC;AACtC,OAAO,cAAc,MAAM,YAAY,CAAC;AACxC,OAAO,sBAAsB,MAAM,4BAA4B,CAAC;AAChE,OAAO,sBAAsB,MAAM,4BAA4B,CAAC;AAChE,OAAO,wBAAwB,MAAM,8BAA8B,CAAC;AACpE,OAAO,yBAAyB,MAAM,8BAA8B,CAAC;AACrE,OAAO,qBAAqB,MAAM,0BAA0B,CAAC;AAC7D,OAAO,wBAAwB,MAAM,wBAAwB,CAAC;AAC9D,OAAO,gBAAgB,MAAM,cAAc,CAAC;AAC5C,OAAO,oBAAoB,MAAM,mBAAmB,CAAC;AACrD,OAAO,kBAAkB,MAAM,sBAAsB,CAAC;AACtD,OAAO,sBAAsB,MAAM,4BAA4B,CAAC;AAChE,OAAO,YAAY,MAAM,UAAU,CAAC;AAOpC,MAAM,CAAC,MAAM,yBAAyB,GAAG,CAAC,MAAM,EAAE,OAAO,EAAE,mBAAmB,EAAE,KAAK,CAAU,CAAC;AAEhG,MAAM,CAAC,MAAM,+BAA+B,GAAG;IAC9C,IAAI,EAAE,aAAa;IACnB,KAAK,EAAE,cAAc;IACrB,mBAAmB,EAAE,wBAAwB;IAC7C,GAAG,EAAE,YAAY;CAC8D,CAAC;AAEjF,MAAM,CAAC,MAAM,iBAAiB,GAA8B;IAC3D,EAAE,EAAE,EAAE,mBAAmB,EAAE,OAAO,EAAE,yBAAyB,EAAE;IAC/D,EAAE,EAAE,EAAE,iBAAiB,EAAE,OAAO,EAAE,sBAAsB,EAAE;IAC1D,EAAE,EAAE,EAAE,eAAe,EAAE,OAAO,EAAE,qBAAqB,EAAE;IACvD,EAAE,EAAE,EAAE,WAAW,EAAE,OAAO,EAAE,kBAAkB,EAAE;IAChD,EAAE,EAAE,EAAE,SAAS,EAAE,OAAO,EAAE,gBAAgB,EAAE;IAC5C,EAAE,EAAE,EAAE,sBAAsB,EAAE,OAAO,EAAE,2BAA2B,EAAE;IACpE,EAAE,EAAE,EAAE,gBAAgB,EAAE,OAAO,EAAE,sBAAsB,EAAE;IACzD,EAAE,EAAE,EAAE,mBAAmB,EAAE,OAAO,EAAE,wBAAwB,EAAE;IAC9D,EAAE,EAAE,EAAE,cAAc,EAAE,OAAO,EAAE,oBAAoB,EAAE;IACrD,EAAE,EAAE,EAAE,cAAc,EAAE,OAAO,EAAE,oBAAoB,EAAE;IACrD,EAAE,EAAE,EAAE,iBAAiB,EAAE,OAAO,EAAE,sBAAsB,EAAE;IAC1D,EAAE,EAAE,EAAE,YAAY,EAAE,OAAO,EAAE,mBAAmB,EAAE;IAClD,EAAE,EAAE,EAAE,iBAAiB,EAAE,OAAO,EAAE,sBAAsB,EAAE;CAC1D,CAAC","sourcesContent":["import type { ExtensionFactory } from \"../types.ts\";\nimport anthropicBashExtension from \"./anthropic-bash/index.ts\";\nimport anthropicWebSearchExtension from \"./anthropic-web-search/index.ts\";\nimport bashTimeoutExtension from \"./bash-timeout/index.ts\";\nimport compactionExtension from \"./compaction/index.ts\";\nimport diffExtension from \"./diff.ts\";\nimport filesExtension from \"./files.ts\";\nimport gptApplyPatchExtension from \"./gpt-apply-patch/index.ts\";\nimport kimiWebSearchExtension from \"./kimi-web-search/index.ts\";\nimport openaiWebSearchExtension from \"./openai-web-search/index.ts\";\nimport permissionSystemExtension from \"./permission-system/index.ts\";\nimport promptPresetExtension from \"./prompt-preset/index.ts\";\nimport promptUrlWidgetExtension from \"./prompt-url-widget.ts\";\nimport redrawsExtension from \"./redraws.ts\";\nimport serviceTierExtension from \"./service-tier.ts\";\nimport todowriteExtension from \"./todotools/index.ts\";\nimport toolPairGuardExtension from \"./tool-pair-guard/index.ts\";\nimport tpsExtension from \"./tps.ts\";\n\nexport interface BuiltinExtensionFactory {\n\tid: string;\n\tfactory: ExtensionFactory;\n}\n\nexport const globalDefaultExtensionIds = [\"diff\", \"files\", \"prompt-url-widget\", \"tps\"] as const;\n\nexport const globalDefaultExtensionFactories = {\n\tdiff: diffExtension,\n\tfiles: filesExtension,\n\t\"prompt-url-widget\": promptUrlWidgetExtension,\n\ttps: tpsExtension,\n} satisfies Record<(typeof globalDefaultExtensionIds)[number], ExtensionFactory>;\n\nexport const builtinExtensions: BuiltinExtensionFactory[] = [\n\t{ id: \"permission-system\", factory: permissionSystemExtension },\n\t{ id: \"gpt-apply-patch\", factory: gptApplyPatchExtension },\n\t{ id: \"prompt-preset\", factory: promptPresetExtension },\n\t{ id: \"todowrite\", factory: todowriteExtension },\n\t{ id: \"redraws\", factory: redrawsExtension },\n\t{ id: \"anthropic-web-search\", factory: anthropicWebSearchExtension },\n\t{ id: \"anthropic-bash\", factory: anthropicBashExtension },\n\t{ id: \"openai-web-search\", factory: openaiWebSearchExtension },\n\t{ id: \"service-tier\", factory: serviceTierExtension },\n\t{ id: \"bash-timeout\", factory: bashTimeoutExtension },\n\t{ id: \"tool-pair-guard\", factory: toolPairGuardExtension },\n\t{ id: \"compaction\", factory: compactionExtension },\n\t{ id: \"kimi-web-search\", factory: kimiWebSearchExtension },\n];\n"]}
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/core/extensions/builtin/index.ts"],"names":[],"mappings":"AACA,OAAO,sBAAsB,MAAM,2BAA2B,CAAC;AAC/D,OAAO,2BAA2B,MAAM,iCAAiC,CAAC;AAC1E,OAAO,oBAAoB,MAAM,yBAAyB,CAAC;AAC3D,OAAO,mBAAmB,MAAM,uBAAuB,CAAC;AACxD,OAAO,aAAa,MAAM,WAAW,CAAC;AACtC,OAAO,cAAc,MAAM,YAAY,CAAC;AACxC,OAAO,sBAAsB,MAAM,4BAA4B,CAAC;AAChE,OAAO,sBAAsB,MAAM,2BAA2B,CAAC;AAC/D,OAAO,sBAAsB,MAAM,4BAA4B,CAAC;AAChE,OAAO,wBAAwB,MAAM,8BAA8B,CAAC;AACpE,OAAO,yBAAyB,MAAM,8BAA8B,CAAC;AACrE,OAAO,qBAAqB,MAAM,0BAA0B,CAAC;AAC7D,OAAO,wBAAwB,MAAM,wBAAwB,CAAC;AAC9D,OAAO,gBAAgB,MAAM,cAAc,CAAC;AAC5C,OAAO,oBAAoB,MAAM,mBAAmB,CAAC;AACrD,OAAO,wBAAwB,MAAM,6BAA6B,CAAC;AACnE,OAAO,kBAAkB,MAAM,sBAAsB,CAAC;AACtD,OAAO,sBAAsB,MAAM,4BAA4B,CAAC;AAChE,OAAO,YAAY,MAAM,UAAU,CAAC;AAOpC,MAAM,CAAC,MAAM,yBAAyB,GAAG,CAAC,MAAM,EAAE,OAAO,EAAE,mBAAmB,EAAE,KAAK,CAAU,CAAC;AAEhG,MAAM,CAAC,MAAM,+BAA+B,GAAG;IAC9C,IAAI,EAAE,aAAa;IACnB,KAAK,EAAE,cAAc;IACrB,mBAAmB,EAAE,wBAAwB;IAC7C,GAAG,EAAE,YAAY;CAC8D,CAAC;AAEjF,MAAM,CAAC,MAAM,iBAAiB,GAA8B;IAC3D,EAAE,EAAE,EAAE,mBAAmB,EAAE,OAAO,EAAE,yBAAyB,EAAE;IAC/D,EAAE,EAAE,EAAE,iBAAiB,EAAE,OAAO,EAAE,sBAAsB,EAAE;IAC1D,EAAE,EAAE,EAAE,eAAe,EAAE,OAAO,EAAE,qBAAqB,EAAE;IACvD,EAAE,EAAE,EAAE,WAAW,EAAE,OAAO,EAAE,kBAAkB,EAAE;IAChD,EAAE,EAAE,EAAE,SAAS,EAAE,OAAO,EAAE,gBAAgB,EAAE;IAC5C,EAAE,EAAE,EAAE,sBAAsB,EAAE,OAAO,EAAE,2BAA2B,EAAE;IACpE,EAAE,EAAE,EAAE,gBAAgB,EAAE,OAAO,EAAE,sBAAsB,EAAE;IACzD,EAAE,EAAE,EAAE,mBAAmB,EAAE,OAAO,EAAE,wBAAwB,EAAE;IAC9D,EAAE,EAAE,EAAE,cAAc,EAAE,OAAO,EAAE,oBAAoB,EAAE;IACrD,EAAE,EAAE,EAAE,cAAc,EAAE,OAAO,EAAE,oBAAoB,EAAE;IACrD,EAAE,EAAE,EAAE,iBAAiB,EAAE,OAAO,EAAE,sBAAsB,EAAE;IAC1D,EAAE,EAAE,EAAE,YAAY,EAAE,OAAO,EAAE,mBAAmB,EAAE;IAClD,EAAE,EAAE,EAAE,gBAAgB,EAAE,OAAO,EAAE,sBAAsB,EAAE;IACzD,EAAE,EAAE,EAAE,kBAAkB,EAAE,OAAO,EAAE,wBAAwB,EAAE;IAC7D,EAAE,EAAE,EAAE,iBAAiB,EAAE,OAAO,EAAE,sBAAsB,EAAE;CAC1D,CAAC","sourcesContent":["import type { ExtensionFactory } from \"../types.ts\";\nimport anthropicBashExtension from \"./anthropic-bash/index.ts\";\nimport anthropicWebSearchExtension from \"./anthropic-web-search/index.ts\";\nimport bashTimeoutExtension from \"./bash-timeout/index.ts\";\nimport compactionExtension from \"./compaction/index.ts\";\nimport diffExtension from \"./diff.ts\";\nimport filesExtension from \"./files.ts\";\nimport gptApplyPatchExtension from \"./gpt-apply-patch/index.ts\";\nimport historySearchExtension from \"./history-search/index.ts\";\nimport kimiWebSearchExtension from \"./kimi-web-search/index.ts\";\nimport openaiWebSearchExtension from \"./openai-web-search/index.ts\";\nimport permissionSystemExtension from \"./permission-system/index.ts\";\nimport promptPresetExtension from \"./prompt-preset/index.ts\";\nimport promptUrlWidgetExtension from \"./prompt-url-widget.ts\";\nimport redrawsExtension from \"./redraws.ts\";\nimport serviceTierExtension from \"./service-tier.ts\";\nimport sessionObserverExtension from \"./session-observer/index.ts\";\nimport todowriteExtension from \"./todotools/index.ts\";\nimport toolPairGuardExtension from \"./tool-pair-guard/index.ts\";\nimport tpsExtension from \"./tps.ts\";\n\nexport interface BuiltinExtensionFactory {\n\tid: string;\n\tfactory: ExtensionFactory;\n}\n\nexport const globalDefaultExtensionIds = [\"diff\", \"files\", \"prompt-url-widget\", \"tps\"] as const;\n\nexport const globalDefaultExtensionFactories = {\n\tdiff: diffExtension,\n\tfiles: filesExtension,\n\t\"prompt-url-widget\": promptUrlWidgetExtension,\n\ttps: tpsExtension,\n} satisfies Record<(typeof globalDefaultExtensionIds)[number], ExtensionFactory>;\n\nexport const builtinExtensions: BuiltinExtensionFactory[] = [\n\t{ id: \"permission-system\", factory: permissionSystemExtension },\n\t{ id: \"gpt-apply-patch\", factory: gptApplyPatchExtension },\n\t{ id: \"prompt-preset\", factory: promptPresetExtension },\n\t{ id: \"todowrite\", factory: todowriteExtension },\n\t{ id: \"redraws\", factory: redrawsExtension },\n\t{ id: \"anthropic-web-search\", factory: anthropicWebSearchExtension },\n\t{ id: \"anthropic-bash\", factory: anthropicBashExtension },\n\t{ id: \"openai-web-search\", factory: openaiWebSearchExtension },\n\t{ id: \"service-tier\", factory: serviceTierExtension },\n\t{ id: \"bash-timeout\", factory: bashTimeoutExtension },\n\t{ id: \"tool-pair-guard\", factory: toolPairGuardExtension },\n\t{ id: \"compaction\", factory: compactionExtension },\n\t{ id: \"history-search\", factory: historySearchExtension },\n\t{ id: \"session-observer\", factory: sessionObserverExtension },\n\t{ id: \"kimi-web-search\", factory: kimiWebSearchExtension },\n];\n"]}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import type { ExtensionAPI } from "../../types.ts";
|
|
2
|
+
export { resolveSessionHudRoot, scanSessionHudEntries } from "./scanner.ts";
|
|
3
|
+
export { renderTranscript } from "./transcript.ts";
|
|
4
|
+
export default function sessionHudExtension(pi: ExtensionAPI): void;
|
|
5
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/index.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAKnD,OAAO,EAAE,qBAAqB,EAAE,qBAAqB,EAAE,MAAM,cAAc,CAAC;AAC5E,OAAO,EAAE,gBAAgB,EAAE,MAAM,iBAAiB,CAAC;AAEnD,MAAM,CAAC,OAAO,UAAU,mBAAmB,CAAC,EAAE,EAAE,YAAY,GAAG,IAAI,CAmClE","sourcesContent":["import { getSessionsDir } from \"../../../../config.ts\";\nimport type { ExtensionAPI } from \"../../types.ts\";\nimport { SessionHudOverlay } from \"./overlay.ts\";\nimport { resolveSessionHudRoot, scanSessionHudEntries } from \"./scanner.ts\";\nimport type { SessionHudEntry } from \"./types.ts\";\n\nexport { resolveSessionHudRoot, scanSessionHudEntries } from \"./scanner.ts\";\nexport { renderTranscript } from \"./transcript.ts\";\n\nexport default function sessionHudExtension(pi: ExtensionAPI): void {\n\tpi.registerCommand(\"sessions\", {\n\t\tdescription: \"Peek at previous session transcripts in a HUD\",\n\t\thandler: async (_args, ctx) => {\n\t\t\tif (!ctx.hasUI) {\n\t\t\t\tctx.ui.notify(\"No UI available\", \"info\");\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tlet sessions: readonly SessionHudEntry[];\n\t\t\ttry {\n\t\t\t\tconst root = resolveSessionHudRoot(ctx.sessionManager.getSessionDir(), getSessionsDir());\n\t\t\t\tsessions = await scanSessionHudEntries(root, ctx.sessionManager.getSessionFile());\n\t\t\t} catch (error) {\n\t\t\t\tconst message = error instanceof Error ? error.message : String(error);\n\t\t\t\tctx.ui.notify(`Failed to read sessions: ${message}`, \"error\");\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif (sessions.length === 0) {\n\t\t\t\tctx.ui.notify(\"No sessions found\", \"info\");\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tawait ctx.ui.custom<void>(\n\t\t\t\t(tui, _theme, _keybindings, done) =>\n\t\t\t\t\tnew SessionHudOverlay({\n\t\t\t\t\t\tsessions,\n\t\t\t\t\t\tdone,\n\t\t\t\t\t\trequestRender: () => tui.requestRender(),\n\t\t\t\t\t}),\n\t\t\t\t{ overlay: true, overlayOptions: { width: \"94%\", maxHeight: \"90%\", minWidth: 72, margin: 1 } },\n\t\t\t);\n\t\t},\n\t});\n}\n"]}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { getSessionsDir } from "../../../../config.js";
|
|
2
|
+
import { SessionHudOverlay } from "./overlay.js";
|
|
3
|
+
import { resolveSessionHudRoot, scanSessionHudEntries } from "./scanner.js";
|
|
4
|
+
export { resolveSessionHudRoot, scanSessionHudEntries } from "./scanner.js";
|
|
5
|
+
export { renderTranscript } from "./transcript.js";
|
|
6
|
+
export default function sessionHudExtension(pi) {
|
|
7
|
+
pi.registerCommand("sessions", {
|
|
8
|
+
description: "Peek at previous session transcripts in a HUD",
|
|
9
|
+
handler: async (_args, ctx) => {
|
|
10
|
+
if (!ctx.hasUI) {
|
|
11
|
+
ctx.ui.notify("No UI available", "info");
|
|
12
|
+
return;
|
|
13
|
+
}
|
|
14
|
+
let sessions;
|
|
15
|
+
try {
|
|
16
|
+
const root = resolveSessionHudRoot(ctx.sessionManager.getSessionDir(), getSessionsDir());
|
|
17
|
+
sessions = await scanSessionHudEntries(root, ctx.sessionManager.getSessionFile());
|
|
18
|
+
}
|
|
19
|
+
catch (error) {
|
|
20
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
21
|
+
ctx.ui.notify(`Failed to read sessions: ${message}`, "error");
|
|
22
|
+
return;
|
|
23
|
+
}
|
|
24
|
+
if (sessions.length === 0) {
|
|
25
|
+
ctx.ui.notify("No sessions found", "info");
|
|
26
|
+
return;
|
|
27
|
+
}
|
|
28
|
+
await ctx.ui.custom((tui, _theme, _keybindings, done) => new SessionHudOverlay({
|
|
29
|
+
sessions,
|
|
30
|
+
done,
|
|
31
|
+
requestRender: () => tui.requestRender(),
|
|
32
|
+
}), { overlay: true, overlayOptions: { width: "94%", maxHeight: "90%", minWidth: 72, margin: 1 } });
|
|
33
|
+
},
|
|
34
|
+
});
|
|
35
|
+
}
|
|
36
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAEvD,OAAO,EAAE,iBAAiB,EAAE,MAAM,cAAc,CAAC;AACjD,OAAO,EAAE,qBAAqB,EAAE,qBAAqB,EAAE,MAAM,cAAc,CAAC;AAG5E,OAAO,EAAE,qBAAqB,EAAE,qBAAqB,EAAE,MAAM,cAAc,CAAC;AAC5E,OAAO,EAAE,gBAAgB,EAAE,MAAM,iBAAiB,CAAC;AAEnD,MAAM,CAAC,OAAO,UAAU,mBAAmB,CAAC,EAAgB,EAAQ;IACnE,EAAE,CAAC,eAAe,CAAC,UAAU,EAAE;QAC9B,WAAW,EAAE,+CAA+C;QAC5D,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,GAAG,EAAE,EAAE,CAAC;YAC9B,IAAI,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC;gBAChB,GAAG,CAAC,EAAE,CAAC,MAAM,CAAC,iBAAiB,EAAE,MAAM,CAAC,CAAC;gBACzC,OAAO;YACR,CAAC;YAED,IAAI,QAAoC,CAAC;YACzC,IAAI,CAAC;gBACJ,MAAM,IAAI,GAAG,qBAAqB,CAAC,GAAG,CAAC,cAAc,CAAC,aAAa,EAAE,EAAE,cAAc,EAAE,CAAC,CAAC;gBACzF,QAAQ,GAAG,MAAM,qBAAqB,CAAC,IAAI,EAAE,GAAG,CAAC,cAAc,CAAC,cAAc,EAAE,CAAC,CAAC;YACnF,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBAChB,MAAM,OAAO,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;gBACvE,GAAG,CAAC,EAAE,CAAC,MAAM,CAAC,4BAA4B,OAAO,EAAE,EAAE,OAAO,CAAC,CAAC;gBAC9D,OAAO;YACR,CAAC;YAED,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC3B,GAAG,CAAC,EAAE,CAAC,MAAM,CAAC,mBAAmB,EAAE,MAAM,CAAC,CAAC;gBAC3C,OAAO;YACR,CAAC;YAED,MAAM,GAAG,CAAC,EAAE,CAAC,MAAM,CAClB,CAAC,GAAG,EAAE,MAAM,EAAE,YAAY,EAAE,IAAI,EAAE,EAAE,CACnC,IAAI,iBAAiB,CAAC;gBACrB,QAAQ;gBACR,IAAI;gBACJ,aAAa,EAAE,GAAG,EAAE,CAAC,GAAG,CAAC,aAAa,EAAE;aACxC,CAAC,EACH,EAAE,OAAO,EAAE,IAAI,EAAE,cAAc,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,SAAS,EAAE,KAAK,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,CAAC,EAAE,EAAE,CAC9F,CAAC;QAAA,CACF;KACD,CAAC,CAAC;AAAA,CACH","sourcesContent":["import { getSessionsDir } from \"../../../../config.ts\";\nimport type { ExtensionAPI } from \"../../types.ts\";\nimport { SessionHudOverlay } from \"./overlay.ts\";\nimport { resolveSessionHudRoot, scanSessionHudEntries } from \"./scanner.ts\";\nimport type { SessionHudEntry } from \"./types.ts\";\n\nexport { resolveSessionHudRoot, scanSessionHudEntries } from \"./scanner.ts\";\nexport { renderTranscript } from \"./transcript.ts\";\n\nexport default function sessionHudExtension(pi: ExtensionAPI): void {\n\tpi.registerCommand(\"sessions\", {\n\t\tdescription: \"Peek at previous session transcripts in a HUD\",\n\t\thandler: async (_args, ctx) => {\n\t\t\tif (!ctx.hasUI) {\n\t\t\t\tctx.ui.notify(\"No UI available\", \"info\");\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tlet sessions: readonly SessionHudEntry[];\n\t\t\ttry {\n\t\t\t\tconst root = resolveSessionHudRoot(ctx.sessionManager.getSessionDir(), getSessionsDir());\n\t\t\t\tsessions = await scanSessionHudEntries(root, ctx.sessionManager.getSessionFile());\n\t\t\t} catch (error) {\n\t\t\t\tconst message = error instanceof Error ? error.message : String(error);\n\t\t\t\tctx.ui.notify(`Failed to read sessions: ${message}`, \"error\");\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif (sessions.length === 0) {\n\t\t\t\tctx.ui.notify(\"No sessions found\", \"info\");\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tawait ctx.ui.custom<void>(\n\t\t\t\t(tui, _theme, _keybindings, done) =>\n\t\t\t\t\tnew SessionHudOverlay({\n\t\t\t\t\t\tsessions,\n\t\t\t\t\t\tdone,\n\t\t\t\t\t\trequestRender: () => tui.requestRender(),\n\t\t\t\t\t}),\n\t\t\t\t{ overlay: true, overlayOptions: { width: \"94%\", maxHeight: \"90%\", minWidth: 72, margin: 1 } },\n\t\t\t);\n\t\t},\n\t});\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"loader.d.ts","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/loader.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAC;AAErD,wBAAsB,sBAAsB,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,kBAAkB,CAAC,CAc1F","sourcesContent":["import { readFile } from \"node:fs/promises\";\nimport type { SessionMessageEntry } from \"../../../session-manager.ts\";\nimport { parseSessionEntries } from \"../../../session-manager.ts\";\nimport type { TranscriptSnapshot } from \"./types.ts\";\n\nexport async function loadTranscriptSnapshot(filePath: string): Promise<TranscriptSnapshot> {\n\tconst content = await readFile(filePath, \"utf-8\");\n\tconst entries = parseSessionEntries(content);\n\tconst messages: SessionMessageEntry[] = [];\n\tlet model: string | undefined;\n\tfor (const entry of entries) {\n\t\tif (entry.type === \"message\") {\n\t\t\tmessages.push(entry);\n\t\t\tif (!model && entry.message.role === \"assistant\") model = entry.message.responseModel ?? entry.message.model;\n\t\t} else if (entry.type === \"model_change\") {\n\t\t\tmodel = `${entry.provider}/${entry.modelId}`;\n\t\t}\n\t}\n\treturn { entries: messages, model };\n}\n"]}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { readFile } from "node:fs/promises";
|
|
2
|
+
import { parseSessionEntries } from "../../../session-manager.js";
|
|
3
|
+
export async function loadTranscriptSnapshot(filePath) {
|
|
4
|
+
const content = await readFile(filePath, "utf-8");
|
|
5
|
+
const entries = parseSessionEntries(content);
|
|
6
|
+
const messages = [];
|
|
7
|
+
let model;
|
|
8
|
+
for (const entry of entries) {
|
|
9
|
+
if (entry.type === "message") {
|
|
10
|
+
messages.push(entry);
|
|
11
|
+
if (!model && entry.message.role === "assistant")
|
|
12
|
+
model = entry.message.responseModel ?? entry.message.model;
|
|
13
|
+
}
|
|
14
|
+
else if (entry.type === "model_change") {
|
|
15
|
+
model = `${entry.provider}/${entry.modelId}`;
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
return { entries: messages, model };
|
|
19
|
+
}
|
|
20
|
+
//# sourceMappingURL=loader.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"loader.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/loader.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,kBAAkB,CAAC;AAE5C,OAAO,EAAE,mBAAmB,EAAE,MAAM,6BAA6B,CAAC;AAGlE,MAAM,CAAC,KAAK,UAAU,sBAAsB,CAAC,QAAgB,EAA+B;IAC3F,MAAM,OAAO,GAAG,MAAM,QAAQ,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;IAClD,MAAM,OAAO,GAAG,mBAAmB,CAAC,OAAO,CAAC,CAAC;IAC7C,MAAM,QAAQ,GAA0B,EAAE,CAAC;IAC3C,IAAI,KAAyB,CAAC;IAC9B,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;QAC7B,IAAI,KAAK,CAAC,IAAI,KAAK,SAAS,EAAE,CAAC;YAC9B,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACrB,IAAI,CAAC,KAAK,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,KAAK,WAAW;gBAAE,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,aAAa,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;QAC9G,CAAC;aAAM,IAAI,KAAK,CAAC,IAAI,KAAK,cAAc,EAAE,CAAC;YAC1C,KAAK,GAAG,GAAG,KAAK,CAAC,QAAQ,IAAI,KAAK,CAAC,OAAO,EAAE,CAAC;QAC9C,CAAC;IACF,CAAC;IACD,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC;AAAA,CACpC","sourcesContent":["import { readFile } from \"node:fs/promises\";\nimport type { SessionMessageEntry } from \"../../../session-manager.ts\";\nimport { parseSessionEntries } from \"../../../session-manager.ts\";\nimport type { TranscriptSnapshot } from \"./types.ts\";\n\nexport async function loadTranscriptSnapshot(filePath: string): Promise<TranscriptSnapshot> {\n\tconst content = await readFile(filePath, \"utf-8\");\n\tconst entries = parseSessionEntries(content);\n\tconst messages: SessionMessageEntry[] = [];\n\tlet model: string | undefined;\n\tfor (const entry of entries) {\n\t\tif (entry.type === \"message\") {\n\t\t\tmessages.push(entry);\n\t\t\tif (!model && entry.message.role === \"assistant\") model = entry.message.responseModel ?? entry.message.model;\n\t\t} else if (entry.type === \"model_change\") {\n\t\t\tmodel = `${entry.provider}/${entry.modelId}`;\n\t\t}\n\t}\n\treturn { entries: messages, model };\n}\n"]}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import type { SessionHudEntry } from "./types.ts";
|
|
2
|
+
export declare function sessionAge(session: SessionHudEntry): string;
|
|
3
|
+
export declare function describeSession(session: SessionHudEntry): string;
|
|
4
|
+
export declare function pickerLabel(session: SessionHudEntry): string;
|
|
5
|
+
export declare function renderLine(text: string, width: number): string;
|
|
6
|
+
export declare function viewerFooter(scroll: string): string;
|
|
7
|
+
//# sourceMappingURL=overlay-format.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"overlay-format.d.ts","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/overlay-format.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAElD,wBAAgB,UAAU,CAAC,OAAO,EAAE,eAAe,GAAG,MAAM,CAE3D;AAED,wBAAgB,eAAe,CAAC,OAAO,EAAE,eAAe,GAAG,MAAM,CAGhE;AAED,wBAAgB,WAAW,CAAC,OAAO,EAAE,eAAe,GAAG,MAAM,CAE5D;AAED,wBAAgB,UAAU,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,MAAM,CAE9D;AAED,wBAAgB,YAAY,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,CAUnD","sourcesContent":["import { TruncatedText } from \"@earendil-works/pi-tui\";\nimport { keyHint, keyText } from \"../../../../modes/interactive/components/keybinding-hints.ts\";\nimport { theme } from \"../../../../modes/interactive/theme/theme.ts\";\nimport { shortenPath } from \"../../../../utils/paths.ts\";\nimport type {} from \"../../../keybindings.ts\";\nimport { compactWhitespace, formatSessionDate } from \"./text.ts\";\nimport type { SessionHudEntry } from \"./types.ts\";\n\nexport function sessionAge(session: SessionHudEntry): string {\n\treturn session.isCurrent ? \"live\" : formatSessionDate(new Date(session.modifiedAt));\n}\n\nexport function describeSession(session: SessionHudEntry): string {\n\tconst cwd = shortenPath(session.cwd) || \"unknown\";\n\treturn `${cwd} · ${sessionAge(session)} · ${session.messageCount} msg`;\n}\n\nexport function pickerLabel(session: SessionHudEntry): string {\n\treturn compactWhitespace(session.lastUserText) || \"(no user prompt)\";\n}\n\nexport function renderLine(text: string, width: number): string {\n\treturn new TruncatedText(text, 0, 0).render(width)[0] ?? \"\";\n}\n\nexport function viewerFooter(scroll: string): string {\n\tconst scrollKeys = `${theme.fg(\"dim\", `${keyText(\"tui.select.up\")}/${keyText(\"tui.select.down\")}`)}${theme.fg(\"muted\", \" scroll\")}`;\n\treturn [\n\t\tscrollKeys,\n\t\tkeyHint(\"tui.select.confirm\", \"expand\"),\n\t\tkeyHint(\"tui.select.cancel\", \"sessions\"),\n\t\tkeyHint(\"app.sessions.observe\", \"close\"),\n\t]\n\t\t.join(theme.fg(\"muted\", \" · \"))\n\t\t.concat(scroll);\n}\n"]}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { TruncatedText } from "@earendil-works/pi-tui";
|
|
2
|
+
import { keyHint, keyText } from "../../../../modes/interactive/components/keybinding-hints.js";
|
|
3
|
+
import { theme } from "../../../../modes/interactive/theme/theme.js";
|
|
4
|
+
import { shortenPath } from "../../../../utils/paths.js";
|
|
5
|
+
import { compactWhitespace, formatSessionDate } from "./text.js";
|
|
6
|
+
export function sessionAge(session) {
|
|
7
|
+
return session.isCurrent ? "live" : formatSessionDate(new Date(session.modifiedAt));
|
|
8
|
+
}
|
|
9
|
+
export function describeSession(session) {
|
|
10
|
+
const cwd = shortenPath(session.cwd) || "unknown";
|
|
11
|
+
return `${cwd} · ${sessionAge(session)} · ${session.messageCount} msg`;
|
|
12
|
+
}
|
|
13
|
+
export function pickerLabel(session) {
|
|
14
|
+
return compactWhitespace(session.lastUserText) || "(no user prompt)";
|
|
15
|
+
}
|
|
16
|
+
export function renderLine(text, width) {
|
|
17
|
+
return new TruncatedText(text, 0, 0).render(width)[0] ?? "";
|
|
18
|
+
}
|
|
19
|
+
export function viewerFooter(scroll) {
|
|
20
|
+
const scrollKeys = `${theme.fg("dim", `${keyText("tui.select.up")}/${keyText("tui.select.down")}`)}${theme.fg("muted", " scroll")}`;
|
|
21
|
+
return [
|
|
22
|
+
scrollKeys,
|
|
23
|
+
keyHint("tui.select.confirm", "expand"),
|
|
24
|
+
keyHint("tui.select.cancel", "sessions"),
|
|
25
|
+
keyHint("app.sessions.observe", "close"),
|
|
26
|
+
]
|
|
27
|
+
.join(theme.fg("muted", " · "))
|
|
28
|
+
.concat(scroll);
|
|
29
|
+
}
|
|
30
|
+
//# sourceMappingURL=overlay-format.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"overlay-format.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/overlay-format.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,8DAA8D,CAAC;AAChG,OAAO,EAAE,KAAK,EAAE,MAAM,8CAA8C,CAAC;AACrE,OAAO,EAAE,WAAW,EAAE,MAAM,4BAA4B,CAAC;AAEzD,OAAO,EAAE,iBAAiB,EAAE,iBAAiB,EAAE,MAAM,WAAW,CAAC;AAGjE,MAAM,UAAU,UAAU,CAAC,OAAwB,EAAU;IAC5D,OAAO,OAAO,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,iBAAiB,CAAC,IAAI,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,CAAC;AAAA,CACpF;AAED,MAAM,UAAU,eAAe,CAAC,OAAwB,EAAU;IACjE,MAAM,GAAG,GAAG,WAAW,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,SAAS,CAAC;IAClD,OAAO,GAAG,GAAG,OAAM,UAAU,CAAC,OAAO,CAAC,OAAM,OAAO,CAAC,YAAY,MAAM,CAAC;AAAA,CACvE;AAED,MAAM,UAAU,WAAW,CAAC,OAAwB,EAAU;IAC7D,OAAO,iBAAiB,CAAC,OAAO,CAAC,YAAY,CAAC,IAAI,kBAAkB,CAAC;AAAA,CACrE;AAED,MAAM,UAAU,UAAU,CAAC,IAAY,EAAE,KAAa,EAAU;IAC/D,OAAO,IAAI,aAAa,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;AAAA,CAC5D;AAED,MAAM,UAAU,YAAY,CAAC,MAAc,EAAU;IACpD,MAAM,UAAU,GAAG,GAAG,KAAK,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,OAAO,CAAC,eAAe,CAAC,IAAI,OAAO,CAAC,iBAAiB,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,EAAE,CAAC,OAAO,EAAE,SAAS,CAAC,EAAE,CAAC;IACpI,OAAO;QACN,UAAU;QACV,OAAO,CAAC,oBAAoB,EAAE,QAAQ,CAAC;QACvC,OAAO,CAAC,mBAAmB,EAAE,UAAU,CAAC;QACxC,OAAO,CAAC,sBAAsB,EAAE,OAAO,CAAC;KACxC;SACC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,OAAO,EAAE,MAAK,CAAC,CAAC;SAC9B,MAAM,CAAC,MAAM,CAAC,CAAC;AAAA,CACjB","sourcesContent":["import { TruncatedText } from \"@earendil-works/pi-tui\";\nimport { keyHint, keyText } from \"../../../../modes/interactive/components/keybinding-hints.ts\";\nimport { theme } from \"../../../../modes/interactive/theme/theme.ts\";\nimport { shortenPath } from \"../../../../utils/paths.ts\";\nimport type {} from \"../../../keybindings.ts\";\nimport { compactWhitespace, formatSessionDate } from \"./text.ts\";\nimport type { SessionHudEntry } from \"./types.ts\";\n\nexport function sessionAge(session: SessionHudEntry): string {\n\treturn session.isCurrent ? \"live\" : formatSessionDate(new Date(session.modifiedAt));\n}\n\nexport function describeSession(session: SessionHudEntry): string {\n\tconst cwd = shortenPath(session.cwd) || \"unknown\";\n\treturn `${cwd} · ${sessionAge(session)} · ${session.messageCount} msg`;\n}\n\nexport function pickerLabel(session: SessionHudEntry): string {\n\treturn compactWhitespace(session.lastUserText) || \"(no user prompt)\";\n}\n\nexport function renderLine(text: string, width: number): string {\n\treturn new TruncatedText(text, 0, 0).render(width)[0] ?? \"\";\n}\n\nexport function viewerFooter(scroll: string): string {\n\tconst scrollKeys = `${theme.fg(\"dim\", `${keyText(\"tui.select.up\")}/${keyText(\"tui.select.down\")}`)}${theme.fg(\"muted\", \" scroll\")}`;\n\treturn [\n\t\tscrollKeys,\n\t\tkeyHint(\"tui.select.confirm\", \"expand\"),\n\t\tkeyHint(\"tui.select.cancel\", \"sessions\"),\n\t\tkeyHint(\"app.sessions.observe\", \"close\"),\n\t]\n\t\t.join(theme.fg(\"muted\", \" · \"))\n\t\t.concat(scroll);\n}\n"]}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { Container, type Focusable } from "@earendil-works/pi-tui";
|
|
2
|
+
import type { SessionHudEntry } from "./types.ts";
|
|
3
|
+
type Mode = "picker" | "viewer";
|
|
4
|
+
interface SessionHudOverlayOptions {
|
|
5
|
+
readonly sessions: readonly SessionHudEntry[];
|
|
6
|
+
readonly done: () => void;
|
|
7
|
+
readonly requestRender: () => void;
|
|
8
|
+
}
|
|
9
|
+
export declare class SessionHudOverlay extends Container implements Focusable {
|
|
10
|
+
private readonly options;
|
|
11
|
+
private readonly sessionsByValue;
|
|
12
|
+
private readonly topBorder;
|
|
13
|
+
private readonly middleBorder;
|
|
14
|
+
private readonly bottomBorder;
|
|
15
|
+
private list;
|
|
16
|
+
private mode;
|
|
17
|
+
private selectedSession;
|
|
18
|
+
private snapshot;
|
|
19
|
+
private renderedLines;
|
|
20
|
+
private ranges;
|
|
21
|
+
private selectedEntryIndex;
|
|
22
|
+
private shouldSelectLastOnLoad;
|
|
23
|
+
private expandedEntries;
|
|
24
|
+
private scrollOffset;
|
|
25
|
+
private viewportHeight;
|
|
26
|
+
private loadingText;
|
|
27
|
+
private _focused;
|
|
28
|
+
constructor(options: SessionHudOverlayOptions);
|
|
29
|
+
get focused(): boolean;
|
|
30
|
+
set focused(value: boolean);
|
|
31
|
+
handleInput(input: string): void;
|
|
32
|
+
render(width: number): string[];
|
|
33
|
+
getMode(): Mode;
|
|
34
|
+
getSelectedEntryIndex(): number;
|
|
35
|
+
getExpandedEntryCount(): number;
|
|
36
|
+
private handlePickerInput;
|
|
37
|
+
private rebuildPicker;
|
|
38
|
+
private toPickerItem;
|
|
39
|
+
private openSession;
|
|
40
|
+
private resetViewerState;
|
|
41
|
+
private rebuildTranscript;
|
|
42
|
+
private renderViewer;
|
|
43
|
+
private handleViewerInput;
|
|
44
|
+
private backToPicker;
|
|
45
|
+
private moveSelection;
|
|
46
|
+
private jumpTo;
|
|
47
|
+
private toggleExpanded;
|
|
48
|
+
private scrollToSelected;
|
|
49
|
+
}
|
|
50
|
+
export {};
|
|
51
|
+
//# sourceMappingURL=overlay.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"overlay.d.ts","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/overlay.ts"],"names":[],"mappings":"AAAA,OAAO,EACN,SAAS,EACT,KAAK,SAAS,EAOd,MAAM,wBAAwB,CAAC;AAUhC,OAAO,KAAK,EAAE,eAAe,EAAwC,MAAM,YAAY,CAAC;AAIxF,KAAK,IAAI,GAAG,QAAQ,GAAG,QAAQ,CAAC;AAUhC,UAAU,wBAAwB;IACjC,QAAQ,CAAC,QAAQ,EAAE,SAAS,eAAe,EAAE,CAAC;IAC9C,QAAQ,CAAC,IAAI,EAAE,MAAM,IAAI,CAAC;IAC1B,QAAQ,CAAC,aAAa,EAAE,MAAM,IAAI,CAAC;CACnC;AAED,qBAAa,iBAAkB,SAAQ,SAAU,YAAW,SAAS;IACpE,OAAO,CAAC,QAAQ,CAAC,OAAO,CAA2B;IACnD,OAAO,CAAC,QAAQ,CAAC,eAAe,CAAsC;IACtE,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAyD;IACnF,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAyD;IACtF,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAyD;IACtF,OAAO,CAAC,IAAI,CAAyB;IACrC,OAAO,CAAC,IAAI,CAAkB;IAC9B,OAAO,CAAC,eAAe,CAA8B;IACrD,OAAO,CAAC,QAAQ,CAAiC;IACjD,OAAO,CAAC,aAAa,CAAyB;IAC9C,OAAO,CAAC,MAAM,CAAmC;IACjD,OAAO,CAAC,kBAAkB,CAAM;IAChC,OAAO,CAAC,sBAAsB,CAAS;IACvC,OAAO,CAAC,eAAe,CAAqB;IAC5C,OAAO,CAAC,YAAY,CAAK;IACzB,OAAO,CAAC,cAAc,CAAM;IAC5B,OAAO,CAAC,WAAW,CAAqB;IACxC,OAAO,CAAC,QAAQ,CAAS;IAEzB,YAAY,OAAO,EAAE,wBAAwB,EAI5C;IAED,IAAI,OAAO,IAAI,OAAO,CAErB;IAED,IAAI,OAAO,CAAC,KAAK,EAAE,OAAO,EAEzB;IAED,WAAW,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI,CAM/B;IAEQ,MAAM,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,EAAE,CAGvC;IAED,OAAO,IAAI,IAAI,CAEd;IAED,qBAAqB,IAAI,MAAM,CAE9B;IAED,qBAAqB,IAAI,MAAM,CAE9B;IAED,OAAO,CAAC,iBAAiB;IAOzB,OAAO,CAAC,aAAa;IAqCrB,OAAO,CAAC,YAAY;YAMN,WAAW;IAiBzB,OAAO,CAAC,gBAAgB;IASxB,OAAO,CAAC,iBAAiB;IAkBzB,OAAO,CAAC,YAAY;IA4BpB,OAAO,CAAC,iBAAiB;IAgBzB,OAAO,CAAC,YAAY;IAKpB,OAAO,CAAC,aAAa;IAMrB,OAAO,CAAC,MAAM;IAMd,OAAO,CAAC,cAAc;IAMtB,OAAO,CAAC,gBAAgB;CAQxB","sourcesContent":["import {\n\tContainer,\n\ttype Focusable,\n\tgetKeybindings,\n\ttype SelectItem,\n\tSelectList,\n\tSpacer,\n\tText,\n\tTruncatedText,\n} from \"@earendil-works/pi-tui\";\nimport { DynamicBorder } from \"../../../../modes/interactive/components/dynamic-border.ts\";\nimport { keyHint } from \"../../../../modes/interactive/components/keybinding-hints.ts\";\nimport { getMarkdownTheme, theme } from \"../../../../modes/interactive/theme/theme.ts\";\nimport { shortenPath } from \"../../../../utils/paths.ts\";\nimport type {} from \"../../../keybindings.ts\";\nimport { loadTranscriptSnapshot } from \"./loader.ts\";\nimport { describeSession, pickerLabel, renderLine, sessionAge, viewerFooter } from \"./overlay-format.ts\";\nimport { sanitizeLine } from \"./text.ts\";\nimport { renderTranscript } from \"./transcript.ts\";\nimport type { SessionHudEntry, TranscriptSnapshot, ViewerEntryRange } from \"./types.ts\";\n\nconst MAX_VISIBLE_SESSIONS = 12;\n\ntype Mode = \"picker\" | \"viewer\";\ntype PickerAction = \"tui.select.up\" | \"tui.select.down\" | \"tui.select.confirm\" | \"tui.select.cancel\";\n\nconst PICKER_ACTIONS: readonly PickerAction[] = [\n\t\"tui.select.up\",\n\t\"tui.select.down\",\n\t\"tui.select.confirm\",\n\t\"tui.select.cancel\",\n];\n\ninterface SessionHudOverlayOptions {\n\treadonly sessions: readonly SessionHudEntry[];\n\treadonly done: () => void;\n\treadonly requestRender: () => void;\n}\n\nexport class SessionHudOverlay extends Container implements Focusable {\n\tprivate readonly options: SessionHudOverlayOptions;\n\tprivate readonly sessionsByValue = new Map<string, SessionHudEntry>();\n\tprivate readonly topBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate readonly middleBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate readonly bottomBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate list: SelectList | undefined;\n\tprivate mode: Mode = \"picker\";\n\tprivate selectedSession: SessionHudEntry | undefined;\n\tprivate snapshot: TranscriptSnapshot | undefined;\n\tprivate renderedLines: readonly string[] = [];\n\tprivate ranges: readonly ViewerEntryRange[] = [];\n\tprivate selectedEntryIndex = -1;\n\tprivate shouldSelectLastOnLoad = false;\n\tprivate expandedEntries = new Set<number>();\n\tprivate scrollOffset = 0;\n\tprivate viewportHeight = 12;\n\tprivate loadingText: string | undefined;\n\tprivate _focused = false;\n\n\tconstructor(options: SessionHudOverlayOptions) {\n\t\tsuper();\n\t\tthis.options = options;\n\t\tthis.rebuildPicker();\n\t}\n\n\tget focused(): boolean {\n\t\treturn this._focused;\n\t}\n\n\tset focused(value: boolean) {\n\t\tthis._focused = value;\n\t}\n\n\thandleInput(input: string): void {\n\t\tif (this.mode === \"picker\") {\n\t\t\tthis.handlePickerInput(input);\n\t\t\treturn;\n\t\t}\n\t\tthis.handleViewerInput(input);\n\t}\n\n\toverride render(width: number): string[] {\n\t\tif (this.mode === \"picker\") return super.render(width);\n\t\treturn this.renderViewer(width);\n\t}\n\n\tgetMode(): Mode {\n\t\treturn this.mode;\n\t}\n\n\tgetSelectedEntryIndex(): number {\n\t\treturn this.selectedEntryIndex;\n\t}\n\n\tgetExpandedEntryCount(): number {\n\t\treturn this.expandedEntries.size;\n\t}\n\n\tprivate handlePickerInput(input: string): void {\n\t\tconst keybindings = getKeybindings();\n\t\tif (PICKER_ACTIONS.some((action) => keybindings.matches(input, action))) {\n\t\t\tthis.list?.handleInput(input);\n\t\t}\n\t}\n\n\tprivate rebuildPicker(): void {\n\t\tthis.sessionsByValue.clear();\n\t\tconst items = this.options.sessions.map((session, index) => this.toPickerItem(session, index));\n\t\tconst list = new SelectList(items, Math.min(MAX_VISIBLE_SESSIONS, Math.max(1, items.length)), {\n\t\t\tselectedPrefix: (text) => theme.fg(\"accent\", text),\n\t\t\tselectedText: (text) => text,\n\t\t\tdescription: (text) => theme.fg(\"muted\", text),\n\t\t\tscrollInfo: (text) => theme.fg(\"dim\", text),\n\t\t\tnoMatch: (text) => theme.fg(\"warning\", text.replace(\"commands\", \"sessions\")),\n\t\t});\n\t\tlist.onSelect = (item) => {\n\t\t\tconst session = this.sessionsByValue.get(item.value);\n\t\t\tif (session) void this.openSession(session);\n\t\t};\n\t\tlist.onCancel = () => this.options.done();\n\t\tthis.list = list;\n\t\tthis.clear();\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(this.topBorder);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(\n\t\t\tnew Text(\n\t\t\t\t`${theme.bold(theme.fg(\"accent\", \" Sessions\"))}${theme.fg(\"dim\", ` ${this.options.sessions.length} sessions`)}`,\n\t\t\t\t0,\n\t\t\t\t0,\n\t\t\t),\n\t\t);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(list);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(\n\t\t\tnew TruncatedText(`${keyHint(\"tui.select.confirm\", \"view\")} ${keyHint(\"tui.select.cancel\", \"close\")}`, 0, 0),\n\t\t);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(this.bottomBorder);\n\t}\n\n\tprivate toPickerItem(session: SessionHudEntry, index: number): SelectItem {\n\t\tconst value = String(index);\n\t\tthis.sessionsByValue.set(value, session);\n\t\treturn { value, label: pickerLabel(session), description: describeSession(session) };\n\t}\n\n\tprivate async openSession(session: SessionHudEntry): Promise<void> {\n\t\tthis.mode = \"viewer\";\n\t\tthis.selectedSession = session;\n\t\tthis.snapshot = undefined;\n\t\tthis.loadingText = \"Loading session transcript...\";\n\t\tthis.resetViewerState();\n\t\tthis.options.requestRender();\n\t\ttry {\n\t\t\tthis.snapshot = await loadTranscriptSnapshot(session.path);\n\t\t\tthis.loadingText = undefined;\n\t\t} catch (error) {\n\t\t\tthis.loadingText = `Failed to read session: ${error instanceof Error ? error.message : String(error)}`;\n\t\t}\n\t\tthis.rebuildTranscript(process.stdout.columns || 80);\n\t\tthis.options.requestRender();\n\t}\n\n\tprivate resetViewerState(): void {\n\t\tthis.expandedEntries = new Set<number>();\n\t\tthis.scrollOffset = 0;\n\t\tthis.selectedEntryIndex = -1;\n\t\tthis.shouldSelectLastOnLoad = true;\n\t\tthis.renderedLines = [];\n\t\tthis.ranges = [];\n\t}\n\n\tprivate rebuildTranscript(width: number): void {\n\t\tif (!this.snapshot) return;\n\t\tconst rendered = renderTranscript(this.snapshot.entries, {\n\t\t\twidth,\n\t\t\tselectedIndex: this.selectedEntryIndex,\n\t\t\texpandedEntries: this.expandedEntries,\n\t\t\tmarkdownTheme: getMarkdownTheme(),\n\t\t});\n\t\tthis.renderedLines = rendered.lines;\n\t\tthis.ranges = rendered.ranges;\n\t\tif (this.ranges.length > 0 && this.shouldSelectLastOnLoad) {\n\t\t\tthis.shouldSelectLastOnLoad = false;\n\t\t\tthis.selectedEntryIndex = this.ranges.length - 1;\n\t\t\tthis.rebuildTranscript(width);\n\t\t}\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate renderViewer(width: number): string[] {\n\t\tthis.viewportHeight = Math.max(5, (process.stdout.rows || 32) - 8);\n\t\tthis.rebuildTranscript(width);\n\t\tconst session = this.selectedSession;\n\t\tconst title = session ? `Sessions > ${shortenPath(session.cwd) || \"unknown\"} · ${session.shortId}` : \"Sessions\";\n\t\tconst status = session\n\t\t\t? `${session.messageCount} messages · ${sessionAge(session)}${this.snapshot?.model ? ` · ${this.snapshot.model}` : \"\"}`\n\t\t\t: \"\";\n\t\tconst maxScroll = Math.max(0, this.renderedLines.length - this.viewportHeight);\n\t\tthis.scrollOffset = Math.max(0, Math.min(this.scrollOffset, maxScroll));\n\t\tconst content = this.loadingText ? [theme.fg(\"dim\", this.loadingText)] : this.renderedLines;\n\t\tconst visible = content.slice(this.scrollOffset, this.scrollOffset + this.viewportHeight);\n\t\tconst lines: string[] = [];\n\t\tlines.push(...this.topBorder.render(width));\n\t\tlines.push(renderLine(` ${theme.bold(theme.fg(\"accent\", title))}`, width));\n\t\tif (status) lines.push(renderLine(` ${theme.fg(\"dim\", status)}`, width));\n\t\tlines.push(...this.middleBorder.render(width));\n\t\tfor (const line of visible) lines.push(` ${sanitizeLine(line, width - 2)}`);\n\t\tfor (let index = visible.length; index < this.viewportHeight; index += 1) lines.push(\"\");\n\t\tconst scroll =\n\t\t\tcontent.length > this.viewportHeight\n\t\t\t\t? ` [${this.scrollOffset + 1}-${Math.min(this.scrollOffset + this.viewportHeight, content.length)}/${content.length}]`\n\t\t\t\t: \"\";\n\t\tlines.push(renderLine(` ${viewerFooter(scroll)}`, width));\n\t\tlines.push(...this.bottomBorder.render(width));\n\t\treturn lines;\n\t}\n\n\tprivate handleViewerInput(input: string): void {\n\t\tconst keybindings = getKeybindings();\n\t\tif (keybindings.matches(input, \"app.sessions.observe\")) {\n\t\t\tthis.options.done();\n\t\t\treturn;\n\t\t} else if (keybindings.matches(input, \"tui.select.cancel\")) this.backToPicker();\n\t\telse if (input === \"j\" || keybindings.matches(input, \"tui.select.down\")) this.moveSelection(1);\n\t\telse if (input === \"k\" || keybindings.matches(input, \"tui.select.up\")) this.moveSelection(-1);\n\t\telse if (keybindings.matches(input, \"tui.select.pageDown\")) this.moveSelection(5);\n\t\telse if (keybindings.matches(input, \"tui.select.pageUp\")) this.moveSelection(-5);\n\t\telse if (input === \"g\") this.jumpTo(0);\n\t\telse if (input === \"G\") this.jumpTo(this.ranges.length - 1);\n\t\telse if (keybindings.matches(input, \"tui.select.confirm\")) this.toggleExpanded();\n\t\tthis.options.requestRender();\n\t}\n\n\tprivate backToPicker(): void {\n\t\tthis.mode = \"picker\";\n\t\tthis.rebuildPicker();\n\t}\n\n\tprivate moveSelection(delta: number): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tthis.selectedEntryIndex = Math.max(0, Math.min(this.selectedEntryIndex + delta, this.ranges.length - 1));\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate jumpTo(index: number): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tthis.selectedEntryIndex = Math.max(0, Math.min(index, this.ranges.length - 1));\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate toggleExpanded(): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tif (this.expandedEntries.has(this.selectedEntryIndex)) this.expandedEntries.delete(this.selectedEntryIndex);\n\t\telse this.expandedEntries.add(this.selectedEntryIndex);\n\t}\n\n\tprivate scrollToSelected(): void {\n\t\tconst selected = this.ranges[this.selectedEntryIndex];\n\t\tif (!selected) return;\n\t\tconst bottom = selected.lineStart + selected.lineCount;\n\t\tif (selected.lineStart < this.scrollOffset) this.scrollOffset = Math.max(0, selected.lineStart - 1);\n\t\tif (bottom > this.scrollOffset + this.viewportHeight)\n\t\t\tthis.scrollOffset = Math.max(0, bottom - this.viewportHeight + 1);\n\t}\n}\n"]}
|