@gajae-code/coding-agent 0.7.3 → 0.7.4
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 +48 -0
- package/bin/gjc.js +4 -0
- package/dist/types/cli/plugin-cli.d.ts +2 -0
- package/dist/types/commands/plugin.d.ts +6 -0
- package/dist/types/commands/session.d.ts +6 -0
- package/dist/types/config/model-profile-activation.d.ts +8 -1
- package/dist/types/extensibility/gjc-plugins/compiler.d.ts +19 -0
- package/dist/types/extensibility/gjc-plugins/constrained-hooks.d.ts +29 -0
- package/dist/types/extensibility/gjc-plugins/index.d.ts +9 -0
- package/dist/types/extensibility/gjc-plugins/injection.d.ts +9 -0
- package/dist/types/extensibility/gjc-plugins/installer.d.ts +13 -0
- package/dist/types/extensibility/gjc-plugins/mcp-policy.d.ts +26 -0
- package/dist/types/extensibility/gjc-plugins/observability.d.ts +27 -0
- package/dist/types/extensibility/gjc-plugins/prompt-appendix.d.ts +16 -0
- package/dist/types/extensibility/gjc-plugins/registry.d.ts +32 -0
- package/dist/types/extensibility/gjc-plugins/runtime-adapters.d.ts +64 -0
- package/dist/types/extensibility/gjc-plugins/session-validation.d.ts +42 -0
- package/dist/types/extensibility/gjc-plugins/types.d.ts +158 -2
- package/dist/types/extensibility/gjc-plugins/validation.d.ts +8 -1
- package/dist/types/gjc-runtime/launch-tmux.d.ts +1 -0
- package/dist/types/gjc-runtime/psmux-detect.d.ts +78 -0
- package/dist/types/gjc-runtime/team-runtime.d.ts +2 -0
- package/dist/types/gjc-runtime/tmux-common.d.ts +20 -1
- package/dist/types/gjc-runtime/tmux-sessions.d.ts +18 -0
- package/dist/types/main.d.ts +2 -0
- package/dist/types/modes/components/model-selector.d.ts +6 -0
- package/dist/types/notifications/html-format.d.ts +11 -0
- package/dist/types/notifications/index.d.ts +149 -1
- package/dist/types/notifications/lifecycle-commands.d.ts +72 -0
- package/dist/types/notifications/lifecycle-control-runtime.d.ts +98 -0
- package/dist/types/notifications/lifecycle-orchestrator.d.ts +144 -0
- package/dist/types/notifications/rate-limit-pool.d.ts +2 -0
- package/dist/types/notifications/recent-activity.d.ts +35 -0
- package/dist/types/notifications/telegram-daemon.d.ts +60 -0
- package/dist/types/notifications/telegram-reference.d.ts +3 -1
- package/dist/types/notifications/topic-registry.d.ts +10 -9
- package/dist/types/runtime-mcp/types.d.ts +7 -0
- package/dist/types/sdk.d.ts +2 -0
- package/dist/types/session/agent-session.d.ts +14 -4
- package/dist/types/session/blob-store.d.ts +25 -0
- package/dist/types/session/session-manager.d.ts +57 -0
- package/dist/types/slash-commands/helpers/fast-status-report.d.ts +6 -0
- package/dist/types/system-prompt.d.ts +2 -0
- package/dist/types/task/executor.d.ts +9 -1
- package/dist/types/tools/index.d.ts +3 -1
- package/dist/types/utils/changelog.d.ts +1 -0
- package/package.json +11 -9
- package/scripts/g004-tmux-smoke.ts +100 -0
- package/scripts/g005-daemon-smoke.ts +181 -0
- package/scripts/g011-daemon-path-smoke.ts +153 -0
- package/src/cli/plugin-cli.ts +66 -3
- package/src/cli.ts +21 -4
- package/src/commands/plugin.ts +4 -0
- package/src/commands/session.ts +18 -0
- package/src/config/model-profile-activation.ts +55 -7
- package/src/defaults/gjc/extensions/grok-cli-vendor/biome.json +1 -1
- package/src/defaults/gjc/skills/deep-interview/SKILL.md +3 -3
- package/src/defaults/gjc/skills/team/SKILL.md +5 -4
- package/src/defaults/gjc/skills/ultragoal/SKILL.md +41 -13
- package/src/export/html/index.ts +2 -2
- package/src/extensibility/gjc-plugins/compiler.ts +351 -0
- package/src/extensibility/gjc-plugins/constrained-hooks.ts +170 -0
- package/src/extensibility/gjc-plugins/index.ts +9 -0
- package/src/extensibility/gjc-plugins/injection.ts +109 -0
- package/src/extensibility/gjc-plugins/installer.ts +434 -0
- package/src/extensibility/gjc-plugins/loader.ts +3 -1
- package/src/extensibility/gjc-plugins/mcp-policy.ts +239 -0
- package/src/extensibility/gjc-plugins/observability.ts +84 -0
- package/src/extensibility/gjc-plugins/paths.ts +1 -1
- package/src/extensibility/gjc-plugins/prompt-appendix.ts +109 -0
- package/src/extensibility/gjc-plugins/registry.ts +180 -0
- package/src/extensibility/gjc-plugins/runtime-adapters.ts +234 -0
- package/src/extensibility/gjc-plugins/schema.ts +250 -20
- package/src/extensibility/gjc-plugins/session-validation.ts +147 -0
- package/src/extensibility/gjc-plugins/types.ts +199 -3
- package/src/extensibility/gjc-plugins/validation.ts +80 -0
- package/src/extensibility/skills.ts +15 -0
- package/src/gjc-runtime/launch-tmux.ts +61 -7
- package/src/gjc-runtime/psmux-detect.ts +239 -0
- package/src/gjc-runtime/team-runtime.ts +56 -23
- package/src/gjc-runtime/tmux-common.ts +27 -2
- package/src/gjc-runtime/tmux-sessions.ts +51 -1
- package/src/gjc-runtime/ultragoal-runtime.ts +75 -15
- package/src/internal-urls/docs-index.generated.ts +5 -4
- package/src/main.ts +14 -3
- package/src/modes/components/hook-editor.ts +1 -1
- package/src/modes/components/hook-selector.ts +67 -43
- package/src/modes/components/model-selector.ts +44 -11
- package/src/modes/controllers/extension-ui-controller.ts +0 -27
- package/src/modes/controllers/selector-controller.ts +50 -11
- package/src/modes/interactive-mode.ts +2 -0
- package/src/modes/utils/hotkeys-markdown.ts +1 -1
- package/src/notifications/html-format.ts +38 -0
- package/src/notifications/index.ts +242 -12
- package/src/notifications/lifecycle-commands.ts +228 -0
- package/src/notifications/lifecycle-control-runtime.ts +400 -0
- package/src/notifications/lifecycle-orchestrator.ts +358 -0
- package/src/notifications/rate-limit-pool.ts +19 -0
- package/src/notifications/recent-activity.ts +132 -0
- package/src/notifications/telegram-daemon.ts +433 -8
- package/src/notifications/telegram-reference.ts +25 -7
- package/src/notifications/topic-registry.ts +18 -9
- package/src/prompts/agents/executor.md +2 -2
- package/src/runtime-mcp/transports/stdio.ts +38 -4
- package/src/runtime-mcp/types.ts +7 -0
- package/src/sdk.ts +157 -10
- package/src/session/agent-session.ts +166 -74
- package/src/session/blob-store.ts +196 -8
- package/src/session/session-manager.ts +678 -7
- package/src/slash-commands/builtin-registry.ts +23 -3
- package/src/slash-commands/helpers/fast-status-report.ts +13 -3
- package/src/system-prompt.ts +9 -0
- package/src/task/executor.ts +31 -7
- package/src/task/index.ts +2 -0
- package/src/tools/ask.ts +5 -1
- package/src/tools/index.ts +3 -1
- package/src/utils/changelog.ts +8 -0
package/src/main.ts
CHANGED
|
@@ -52,7 +52,7 @@ import { formatModelOnboardingGuidance } from "./setup/model-onboarding-guidance
|
|
|
52
52
|
import { executeBuiltinSlashCommand } from "./slash-commands/builtin-registry";
|
|
53
53
|
import { resolvePromptInput } from "./system-prompt";
|
|
54
54
|
import type { LspStartupServerInfo } from "./tools";
|
|
55
|
-
import { getDisplayChangelogEntries, getNewEntries } from "./utils/changelog";
|
|
55
|
+
import { getDisplayChangelogEntries, getInstalledVersionChangelogEntry, getNewEntries } from "./utils/changelog";
|
|
56
56
|
import type { EventBus } from "./utils/event-bus";
|
|
57
57
|
|
|
58
58
|
async function checkForNewVersion(currentVersion: string): Promise<string | undefined> {
|
|
@@ -407,7 +407,7 @@ async function getChangelogForDisplay(parsed: Args): Promise<string | undefined>
|
|
|
407
407
|
if (entries.length > 0) {
|
|
408
408
|
settings.set("lastChangelogVersion", VERSION);
|
|
409
409
|
await flushChangelogVersion();
|
|
410
|
-
return entries
|
|
410
|
+
return getInstalledVersionChangelogEntry(entries, VERSION)?.content;
|
|
411
411
|
}
|
|
412
412
|
} else {
|
|
413
413
|
const newEntries = getNewEntries(entries, lastVersion);
|
|
@@ -429,7 +429,7 @@ async function flushChangelogVersion(): Promise<void> {
|
|
|
429
429
|
}
|
|
430
430
|
}
|
|
431
431
|
|
|
432
|
-
async function createSessionManager(
|
|
432
|
+
export async function createSessionManager(
|
|
433
433
|
parsed: Args,
|
|
434
434
|
cwd: string,
|
|
435
435
|
activeSettings: Settings = settings,
|
|
@@ -482,6 +482,17 @@ async function createSessionManager(
|
|
|
482
482
|
if (parsed.sessionDir) {
|
|
483
483
|
return SessionManager.create(cwd, parsed.sessionDir);
|
|
484
484
|
}
|
|
485
|
+
// A lifecycle `/session_create` child must start a FRESH session that adopts
|
|
486
|
+
// the pre-allocated id (GJC_SESSION_ID), never auto-resume existing history in
|
|
487
|
+
// the target cwd — otherwise the daemon/tmux id and the session header id
|
|
488
|
+
// diverge and close/resume-by-create-id break. Resume children are launched
|
|
489
|
+
// with `--resume <id>` (handled above) and carry no GJC_LIFECYCLE_REQUEST_ID.
|
|
490
|
+
if (
|
|
491
|
+
process.env.GJC_LIFECYCLE_REQUEST_ID &&
|
|
492
|
+
/^[A-Za-z0-9._-]{1,128}$/.test(process.env.GJC_SESSION_ID?.trim() ?? "")
|
|
493
|
+
) {
|
|
494
|
+
return undefined;
|
|
495
|
+
}
|
|
485
496
|
// Auto-resume: behave like --continue if the setting is enabled and a prior
|
|
486
497
|
// session exists. When a prior session is resumed, mark parsed.continue so
|
|
487
498
|
// buildSessionOptions restores the session's model/thinking instead of
|
|
@@ -66,7 +66,7 @@ export class HookEditorComponent extends Container {
|
|
|
66
66
|
|
|
67
67
|
// Hint
|
|
68
68
|
const hint = this.#promptStyle
|
|
69
|
-
? "enter submit esc cancel ctrl+g external editor"
|
|
69
|
+
? "enter submit shift+enter/ctrl+j newline esc cancel ctrl+g external editor"
|
|
70
70
|
: "ctrl+enter submit esc cancel ctrl+g external editor";
|
|
71
71
|
this.addChild(new Text(theme.fg("dim", hint), 1, 0));
|
|
72
72
|
|
|
@@ -28,22 +28,6 @@ import { getEditorCommand, openInEditor } from "../../utils/external-editor";
|
|
|
28
28
|
import { CountdownTimer } from "./countdown-timer";
|
|
29
29
|
import { DynamicBorder } from "./dynamic-border";
|
|
30
30
|
|
|
31
|
-
const SGR_MOUSE_PRESS_PATTERN = /^\x1b\[<(\d+);\d+;\d+M$/;
|
|
32
|
-
const MOUSE_WHEEL_TITLE_SCROLL_ROWS = 3;
|
|
33
|
-
|
|
34
|
-
function getMouseWheelTitleScrollRows(keyData: string): number {
|
|
35
|
-
const match = SGR_MOUSE_PRESS_PATTERN.exec(keyData);
|
|
36
|
-
if (!match) return 0;
|
|
37
|
-
|
|
38
|
-
const button = Number.parseInt(match[1] ?? "", 10);
|
|
39
|
-
if (!Number.isFinite(button) || (button & 64) === 0) return 0;
|
|
40
|
-
|
|
41
|
-
const wheelDirection = button & 3;
|
|
42
|
-
if (wheelDirection === 0) return -MOUSE_WHEEL_TITLE_SCROLL_ROWS;
|
|
43
|
-
if (wheelDirection === 1) return MOUSE_WHEEL_TITLE_SCROLL_ROWS;
|
|
44
|
-
return 0;
|
|
45
|
-
}
|
|
46
|
-
|
|
47
31
|
export interface HookSelectorOptions {
|
|
48
32
|
tui?: TUI;
|
|
49
33
|
timeout?: number;
|
|
@@ -132,26 +116,58 @@ class ScrollableTitle extends Container {
|
|
|
132
116
|
|
|
133
117
|
render(width: number): string[] {
|
|
134
118
|
const lines = this.#markdown.render(width);
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
119
|
+
if (lines.length <= this.#maxRows) {
|
|
120
|
+
this.#lastMaxScrollOffset = 0;
|
|
121
|
+
this.#scrollOffset = 0;
|
|
122
|
+
return lines;
|
|
123
|
+
}
|
|
138
124
|
|
|
139
|
-
|
|
140
|
-
|
|
125
|
+
if (this.#maxRows < 3) {
|
|
126
|
+
const maxScrollOffset = Math.max(0, lines.length - this.#maxRows);
|
|
127
|
+
this.#lastMaxScrollOffset = maxScrollOffset;
|
|
128
|
+
this.#scrollOffset = Math.max(0, Math.min(this.#scrollOffset, maxScrollOffset));
|
|
129
|
+
|
|
130
|
+
const visibleLines = lines.slice(this.#scrollOffset, this.#scrollOffset + this.#maxRows);
|
|
131
|
+
const indicator =
|
|
132
|
+
this.#scrollOffset === 0
|
|
133
|
+
? theme.fg("dim", " PgDn↓")
|
|
134
|
+
: this.#scrollOffset >= maxScrollOffset
|
|
135
|
+
? theme.fg("dim", " PgUp↑")
|
|
136
|
+
: theme.fg("dim", " PgUp/PgDn↕");
|
|
137
|
+
const lastIndex = visibleLines.length - 1;
|
|
138
|
+
const availableWidth = Math.max(1, width - visibleWidth(indicator));
|
|
139
|
+
const fittedLine = truncateToWidth(visibleLines[lastIndex] ?? "", availableWidth);
|
|
140
|
+
visibleLines[lastIndex] = `${fittedLine}${indicator}`;
|
|
141
141
|
return visibleLines;
|
|
142
142
|
}
|
|
143
143
|
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
144
|
+
let showTopIndicator = this.#scrollOffset > 0;
|
|
145
|
+
let showBottomIndicator = true;
|
|
146
|
+
let contentRows = 1;
|
|
147
|
+
let maxScrollOffset = 0;
|
|
148
|
+
|
|
149
|
+
for (let i = 0; i < 4; i++) {
|
|
150
|
+
contentRows = Math.max(1, this.#maxRows - (showTopIndicator ? 1 : 0) - (showBottomIndicator ? 1 : 0));
|
|
151
|
+
maxScrollOffset = Math.max(0, lines.length - contentRows);
|
|
152
|
+
this.#scrollOffset = Math.max(0, Math.min(this.#scrollOffset, maxScrollOffset));
|
|
153
|
+
|
|
154
|
+
const nextShowTopIndicator = this.#scrollOffset > 0;
|
|
155
|
+
const nextShowBottomIndicator = this.#scrollOffset + contentRows < lines.length;
|
|
156
|
+
if (nextShowTopIndicator === showTopIndicator && nextShowBottomIndicator === showBottomIndicator) {
|
|
157
|
+
break;
|
|
158
|
+
}
|
|
159
|
+
showTopIndicator = nextShowTopIndicator;
|
|
160
|
+
showBottomIndicator = nextShowBottomIndicator;
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
this.#lastMaxScrollOffset = maxScrollOffset;
|
|
164
|
+
|
|
165
|
+
const visibleLines = lines.slice(this.#scrollOffset, this.#scrollOffset + contentRows);
|
|
166
|
+
const result: string[] = [];
|
|
167
|
+
if (showTopIndicator) result.push(theme.fg("dim", truncateToWidth("▲ more", width)));
|
|
168
|
+
result.push(...visibleLines);
|
|
169
|
+
if (showBottomIndicator) result.push(theme.fg("dim", truncateToWidth("▼ more", width)));
|
|
170
|
+
return result.slice(0, this.#maxRows);
|
|
155
171
|
}
|
|
156
172
|
}
|
|
157
173
|
|
|
@@ -454,13 +470,6 @@ export class HookSelectorComponent extends Container {
|
|
|
454
470
|
// Reset countdown on any interaction
|
|
455
471
|
this.#countdown?.reset();
|
|
456
472
|
|
|
457
|
-
if (this.#scrollTitleRows !== undefined) {
|
|
458
|
-
const wheelRows = getMouseWheelTitleScrollRows(keyData);
|
|
459
|
-
if (wheelRows !== 0) {
|
|
460
|
-
this.#scrollableTitle?.scrollBy(wheelRows);
|
|
461
|
-
return;
|
|
462
|
-
}
|
|
463
|
-
}
|
|
464
473
|
if (this.#scrollTitleRows !== undefined && matchesKey(keyData, "pageUp")) {
|
|
465
474
|
this.#scrollableTitle?.scrollBy(-this.#scrollTitleRows);
|
|
466
475
|
return;
|
|
@@ -469,6 +478,14 @@ export class HookSelectorComponent extends Container {
|
|
|
469
478
|
this.#scrollableTitle?.scrollBy(this.#scrollTitleRows);
|
|
470
479
|
return;
|
|
471
480
|
}
|
|
481
|
+
if (!this.#inlineEditor && this.#scrollTitleRows !== undefined && matchesKey(keyData, "ctrl+u")) {
|
|
482
|
+
this.#scrollableTitle?.scrollBy(-this.#scrollTitleRows);
|
|
483
|
+
return;
|
|
484
|
+
}
|
|
485
|
+
if (!this.#inlineEditor && this.#scrollTitleRows !== undefined && matchesKey(keyData, "ctrl+d")) {
|
|
486
|
+
this.#scrollableTitle?.scrollBy(this.#scrollTitleRows);
|
|
487
|
+
return;
|
|
488
|
+
}
|
|
472
489
|
if (this.#inlineEditor) {
|
|
473
490
|
this.#handleInputModeKey(keyData, this.#inlineEditor);
|
|
474
491
|
return;
|
|
@@ -537,16 +554,23 @@ export class HookSelectorComponent extends Container {
|
|
|
537
554
|
editor.setBorderVisible(false);
|
|
538
555
|
editor.setPromptGutter("> ");
|
|
539
556
|
editor.disableSubmit = true;
|
|
557
|
+
// Mark the inline editor focused only when mirroring the app's hardware-cursor
|
|
558
|
+
// mode, so it emits CURSOR_MARKER at the input caret for IME preedit anchoring
|
|
559
|
+
// without changing legacy non-hardware-cursor layout.
|
|
560
|
+
const useTerminalCursor = this.#tui?.getShowHardwareCursor() ?? false;
|
|
561
|
+
editor.focused = useTerminalCursor;
|
|
562
|
+
editor.setUseTerminalCursor(useTerminalCursor);
|
|
540
563
|
if (this.#autocompleteProvider) {
|
|
541
564
|
editor.setAutocompleteProvider(this.#autocompleteProvider);
|
|
542
565
|
}
|
|
543
566
|
this.#inlineEditor = editor;
|
|
544
567
|
this.#inputArea.addChild(new Spacer(1));
|
|
545
568
|
this.#inputArea.addChild(editor);
|
|
546
|
-
const
|
|
547
|
-
|
|
548
|
-
|
|
549
|
-
|
|
569
|
+
const helpText =
|
|
570
|
+
this.#scrollTitleRows === undefined
|
|
571
|
+
? "enter submit esc back to options ctrl+g external editor"
|
|
572
|
+
: "enter submit esc back to options PgUp/PgDn: question · Wheel: transcript";
|
|
573
|
+
this.#helpTextComponent.setText(theme.fg("dim", helpText));
|
|
550
574
|
this.invalidate();
|
|
551
575
|
}
|
|
552
576
|
|
|
@@ -225,6 +225,7 @@ export class ModelSelectorComponent extends Container {
|
|
|
225
225
|
#activeModelProfile?: string;
|
|
226
226
|
#isFastForProvider: (provider?: string) => boolean = () => false;
|
|
227
227
|
#isFastForSubagentProvider: (provider?: string) => boolean = () => false;
|
|
228
|
+
#isCurrentModelFastModeActive: () => boolean = () => false;
|
|
228
229
|
#pendingActionItem?: ModelItem | CanonicalModelItem;
|
|
229
230
|
#selectedActionIndex: number = 0;
|
|
230
231
|
#pendingThinkingChoice?: PendingThinkingChoice;
|
|
@@ -260,6 +261,7 @@ export class ModelSelectorComponent extends Container {
|
|
|
260
261
|
sessionId?: string;
|
|
261
262
|
isFastForProvider?: (provider?: string) => boolean;
|
|
262
263
|
isFastForSubagentProvider?: (provider?: string) => boolean;
|
|
264
|
+
isCurrentModelFastModeActive?: () => boolean;
|
|
263
265
|
currentThinkingLevel?: ThinkingLevel;
|
|
264
266
|
activeModelProfile?: string;
|
|
265
267
|
},
|
|
@@ -279,6 +281,12 @@ export class ModelSelectorComponent extends Container {
|
|
|
279
281
|
this.#activeModelProfile = options?.activeModelProfile;
|
|
280
282
|
this.#isFastForProvider = options?.isFastForProvider ?? (() => false);
|
|
281
283
|
this.#isFastForSubagentProvider = options?.isFastForSubagentProvider ?? (() => false);
|
|
284
|
+
// Current-model EFFECTIVE fast state. Defaults to intent for the current
|
|
285
|
+
// model so existing callers/tests keep prior behavior; production wires the
|
|
286
|
+
// session's effective predicate so an auto-disabled provider shows no glyph.
|
|
287
|
+
this.#isCurrentModelFastModeActive =
|
|
288
|
+
options?.isCurrentModelFastModeActive ??
|
|
289
|
+
(() => (this.#currentModel ? this.#isFastForProvider(this.#currentModel.provider) : false));
|
|
282
290
|
const initialSearchInput = options?.initialSearchInput;
|
|
283
291
|
this.#viewMode = this.#temporaryOnly || initialSearchInput || scopedModels.length > 0 ? "models" : "presets";
|
|
284
292
|
|
|
@@ -381,6 +389,17 @@ export class ModelSelectorComponent extends Container {
|
|
|
381
389
|
}
|
|
382
390
|
}
|
|
383
391
|
|
|
392
|
+
refreshRoleAssignments(
|
|
393
|
+
options: { currentModel?: Model; currentThinkingLevel?: ThinkingLevel; activeModelProfile?: string } = {},
|
|
394
|
+
): void {
|
|
395
|
+
if ("currentModel" in options) this.#currentModel = options.currentModel;
|
|
396
|
+
if ("currentThinkingLevel" in options) this.#currentThinkingLevel = options.currentThinkingLevel;
|
|
397
|
+
if ("activeModelProfile" in options) this.#activeModelProfile = options.activeModelProfile;
|
|
398
|
+
this.#roles = {};
|
|
399
|
+
this.#loadRoleModels();
|
|
400
|
+
this.#applyTabFilter();
|
|
401
|
+
}
|
|
402
|
+
|
|
384
403
|
#sortModels(models: ModelItem[]): void {
|
|
385
404
|
// Sort: default-tagged model first, then MRU, then alphabetical
|
|
386
405
|
const mruOrder = this.#settings.getStorage()?.getModelUsageOrder() ?? [];
|
|
@@ -973,32 +992,46 @@ export class ModelSelectorComponent extends Container {
|
|
|
973
992
|
|
|
974
993
|
// Build role badges (inverted: color as background, black text)
|
|
975
994
|
const roleBadgeTokens: string[] = [];
|
|
976
|
-
|
|
995
|
+
// Whether a non-subagent (modelRoles) badge on the CURRENT model row already
|
|
996
|
+
// rendered the current-model EFFECTIVE glyph. Only that case should suppress
|
|
997
|
+
// the standalone current glyph below — a subagent-only match must NOT, since
|
|
998
|
+
// subagent badges reflect the subagent tier, not the current model.
|
|
999
|
+
let currentModelEffectiveGlyphRendered = false;
|
|
977
1000
|
for (const role of GJC_MODEL_ASSIGNMENT_TARGET_IDS) {
|
|
978
1001
|
const roleInfo = GJC_MODEL_ASSIGNMENT_TARGETS[role];
|
|
979
1002
|
const assigned = this.#roles[role];
|
|
980
1003
|
if (roleInfo.tag && assigned && modelsAreEqual(assigned.model, item.model)) {
|
|
981
|
-
roleMatched = true;
|
|
982
1004
|
const badge = makeInvertedBadge(roleInfo.tag, roleInfo.color ?? "muted");
|
|
983
1005
|
const thinkingLabel = getThinkingLevelMetadata(assigned.thinkingLevel).label;
|
|
984
|
-
|
|
985
|
-
//
|
|
986
|
-
|
|
987
|
-
|
|
988
|
-
|
|
1006
|
+
|
|
1007
|
+
// Subagent roles (task.agentModelOverrides) run under task.serviceTier,
|
|
1008
|
+
// so their ⚡ uses the effective subagent tier. A non-subagent
|
|
1009
|
+
// (modelRoles) badge on the CURRENT model row uses the current-model
|
|
1010
|
+
// effective predicate so a provider auto-disable hides the glyph;
|
|
1011
|
+
// other modelRoles rows show pure intent.
|
|
1012
|
+
const isSubagentRole = roleInfo.settingsPath === "task.agentModelOverrides";
|
|
1013
|
+
const isCurrentRow = this.#currentModel !== undefined && modelsAreEqual(this.#currentModel, item.model);
|
|
1014
|
+
const roleFast = isSubagentRole
|
|
1015
|
+
? this.#isFastForSubagentProvider(assigned.model.provider)
|
|
1016
|
+
: isCurrentRow
|
|
1017
|
+
? this.#isCurrentModelFastModeActive()
|
|
989
1018
|
: this.#isFastForProvider(assigned.model.provider);
|
|
1019
|
+
if (roleFast && isCurrentRow && !isSubagentRole) {
|
|
1020
|
+
currentModelEffectiveGlyphRendered = true;
|
|
1021
|
+
}
|
|
990
1022
|
const fastSuffix = roleFast ? ` ${theme.icon.fast}` : "";
|
|
991
1023
|
roleBadgeTokens.push(`${badge} ${theme.fg("dim", `(${thinkingLabel})`)}${fastSuffix}`);
|
|
992
1024
|
}
|
|
993
1025
|
}
|
|
994
1026
|
// Active/current non-role row: show the fast glyph on the session's current
|
|
995
|
-
// model row
|
|
996
|
-
//
|
|
1027
|
+
// model row. Suppress only when a non-subagent current-row badge already
|
|
1028
|
+
// rendered the current-model effective glyph (duplicate-glyph guard) — a
|
|
1029
|
+
// subagent-only match must not hide the current model's own indicator.
|
|
997
1030
|
if (
|
|
998
|
-
!
|
|
1031
|
+
!currentModelEffectiveGlyphRendered &&
|
|
999
1032
|
this.#currentModel !== undefined &&
|
|
1000
1033
|
modelsAreEqual(this.#currentModel, item.model) &&
|
|
1001
|
-
this.#
|
|
1034
|
+
this.#isCurrentModelFastModeActive()
|
|
1002
1035
|
) {
|
|
1003
1036
|
roleBadgeTokens.push(theme.icon.fast);
|
|
1004
1037
|
}
|
|
@@ -25,8 +25,6 @@ import type { InteractiveModeContext } from "../../modes/types";
|
|
|
25
25
|
import { setSessionTerminalTitle, setTerminalTitle } from "../../utils/title-generator";
|
|
26
26
|
|
|
27
27
|
const MAX_WIDGET_LINES = 10;
|
|
28
|
-
const HOOK_SELECTOR_MOUSE_REPORTING_ENABLE = "\x1b[?1006h\x1b[?1000h";
|
|
29
|
-
const HOOK_SELECTOR_MOUSE_REPORTING_DISABLE = "\x1b[?1000l\x1b[?1006l";
|
|
30
28
|
const HOOK_SELECTOR_CHROME_ROWS = 7;
|
|
31
29
|
const HOOK_SELECTOR_OUTLINE_ROWS = 2;
|
|
32
30
|
const HOOK_SELECTOR_INLINE_INPUT_ROWS = 2;
|
|
@@ -35,7 +33,6 @@ export class ExtensionUiController {
|
|
|
35
33
|
#extensionTerminalInputUnsubscribers = new Set<() => void>();
|
|
36
34
|
#hookWidgetsAbove = new Map<string, ExtensionUiComponent>();
|
|
37
35
|
#hookWidgetsBelow = new Map<string, ExtensionUiComponent>();
|
|
38
|
-
#hookSelectorMouseReportingEnabled = false;
|
|
39
36
|
#activeHookCustomComponent?: Component & { dispose?(): void };
|
|
40
37
|
#activeHookCustomOverlay?: OverlayHandle;
|
|
41
38
|
|
|
@@ -624,9 +621,6 @@ export class ExtensionUiController {
|
|
|
624
621
|
this.ctx.ui.terminal.rows - scrollOptionRows - listChromeRows - inlineInputRows - HOOK_SELECTOR_CHROME_ROWS;
|
|
625
622
|
const scrollTitleRows =
|
|
626
623
|
requestedTitleRows === undefined ? undefined : Math.max(1, Math.min(requestedTitleRows, availableTitleRows));
|
|
627
|
-
if (scrollTitleRows !== undefined) {
|
|
628
|
-
this.#enableHookSelectorMouseReporting();
|
|
629
|
-
}
|
|
630
624
|
|
|
631
625
|
this.ctx.hookSelector = new HookSelectorComponent(
|
|
632
626
|
title,
|
|
@@ -691,31 +685,10 @@ export class ExtensionUiController {
|
|
|
691
685
|
return promise;
|
|
692
686
|
}
|
|
693
687
|
|
|
694
|
-
#enableHookSelectorMouseReporting(): void {
|
|
695
|
-
if (this.#hookSelectorMouseReportingEnabled) return;
|
|
696
|
-
this.#hookSelectorMouseReportingEnabled = true;
|
|
697
|
-
this.#writeTerminalControl(HOOK_SELECTOR_MOUSE_REPORTING_ENABLE);
|
|
698
|
-
}
|
|
699
|
-
|
|
700
|
-
#disableHookSelectorMouseReporting(): void {
|
|
701
|
-
if (!this.#hookSelectorMouseReportingEnabled) return;
|
|
702
|
-
this.#hookSelectorMouseReportingEnabled = false;
|
|
703
|
-
this.#writeTerminalControl(HOOK_SELECTOR_MOUSE_REPORTING_DISABLE);
|
|
704
|
-
}
|
|
705
|
-
|
|
706
|
-
#writeTerminalControl(sequence: string): void {
|
|
707
|
-
try {
|
|
708
|
-
this.ctx.ui.terminal.write(sequence);
|
|
709
|
-
} catch {
|
|
710
|
-
// Terminal teardown can race selector cleanup; normal shutdown restores modes.
|
|
711
|
-
}
|
|
712
|
-
}
|
|
713
|
-
|
|
714
688
|
/**
|
|
715
689
|
* Hide the hook selector.
|
|
716
690
|
*/
|
|
717
691
|
hideHookSelector(): void {
|
|
718
|
-
this.#disableHookSelectorMouseReporting();
|
|
719
692
|
this.ctx.hookSelector?.dispose();
|
|
720
693
|
this.ctx.editorContainer.clear();
|
|
721
694
|
this.ctx.editorContainer.addChild(this.ctx.editor);
|
|
@@ -4,8 +4,10 @@ import type { OAuthProvider } from "@gajae-code/ai/utils/oauth/types";
|
|
|
4
4
|
import type { Component, OverlayHandle } from "@gajae-code/tui";
|
|
5
5
|
import { Input, Loader, Spacer, Text } from "@gajae-code/tui";
|
|
6
6
|
import { getAgentDbPath, getProjectDir } from "@gajae-code/utils";
|
|
7
|
-
import { activateModelProfile } from "../../config/model-profile-activation";
|
|
7
|
+
import { activateModelProfile, materializeActiveModelProfileAssignment } from "../../config/model-profile-activation";
|
|
8
8
|
import { recommendModelProfileForProvider } from "../../config/model-profiles";
|
|
9
|
+
import { GJC_MODEL_ASSIGNMENT_TARGETS } from "../../config/model-registry";
|
|
10
|
+
import { formatModelSelectorValue } from "../../config/model-resolver";
|
|
9
11
|
import { settings } from "../../config/settings";
|
|
10
12
|
import { DebugSelectorComponent } from "../../debug";
|
|
11
13
|
import { disableProvider, enableProvider } from "../../discovery";
|
|
@@ -663,7 +665,8 @@ export class SelectorController {
|
|
|
663
665
|
|
|
664
666
|
showModelSelector(options?: { temporaryOnly?: boolean }): void {
|
|
665
667
|
this.showSelector(done => {
|
|
666
|
-
|
|
668
|
+
let modelSelector: ModelSelectorComponent;
|
|
669
|
+
modelSelector = new ModelSelectorComponent(
|
|
667
670
|
this.ctx.ui,
|
|
668
671
|
this.ctx.session.model,
|
|
669
672
|
this.ctx.settings,
|
|
@@ -697,38 +700,73 @@ export class SelectorController {
|
|
|
697
700
|
this.ctx.ui.requestRender();
|
|
698
701
|
return;
|
|
699
702
|
}
|
|
700
|
-
const { model, role, thinkingLevel, selector } = selection;
|
|
703
|
+
const { model, role, thinkingLevel, selector: selectedSelector } = selection;
|
|
701
704
|
if (role === null) {
|
|
702
705
|
// Temporary: update agent state but don't persist to settings
|
|
703
706
|
await this.ctx.session.setModelTemporary(model, thinkingLevel);
|
|
704
707
|
this.ctx.statusLine.invalidate();
|
|
705
708
|
this.ctx.updateEditorBorderColor();
|
|
706
|
-
this.ctx.showStatus(`Temporary model: ${
|
|
709
|
+
this.ctx.showStatus(`Temporary model: ${selectedSelector ?? model.id}`);
|
|
707
710
|
done();
|
|
708
711
|
this.ctx.ui.requestRender();
|
|
709
712
|
} else if (role === "default") {
|
|
710
713
|
// Default: update agent state and persist as the active default model.
|
|
711
714
|
await this.ctx.session.setModel(model, role, {
|
|
712
|
-
selector,
|
|
715
|
+
selector: selectedSelector,
|
|
713
716
|
thinkingLevel,
|
|
714
717
|
});
|
|
718
|
+
const value = formatModelSelectorValue(
|
|
719
|
+
selectedSelector ?? `${model.provider}/${model.id}`,
|
|
720
|
+
thinkingLevel,
|
|
721
|
+
);
|
|
722
|
+
materializeActiveModelProfileAssignment({
|
|
723
|
+
session: this.ctx.session,
|
|
724
|
+
settings: this.ctx.settings,
|
|
725
|
+
role,
|
|
726
|
+
selector: value,
|
|
727
|
+
});
|
|
715
728
|
if (thinkingLevel && thinkingLevel !== ThinkingLevel.Inherit) {
|
|
716
729
|
this.ctx.session.setThinkingLevel(thinkingLevel);
|
|
717
730
|
}
|
|
731
|
+
modelSelector.refreshRoleAssignments({
|
|
732
|
+
currentModel: this.ctx.session.model,
|
|
733
|
+
currentThinkingLevel: this.ctx.session.thinkingLevel,
|
|
734
|
+
activeModelProfile:
|
|
735
|
+
this.ctx.session.getActiveModelProfile?.() ?? this.ctx.settings.get("modelProfile.default"),
|
|
736
|
+
});
|
|
718
737
|
this.ctx.statusLine.invalidate();
|
|
719
738
|
this.ctx.updateEditorBorderColor();
|
|
720
|
-
this.ctx.showStatus(`Default model: ${
|
|
739
|
+
this.ctx.showStatus(`Default model: ${selectedSelector ?? model.id}`);
|
|
721
740
|
done();
|
|
722
741
|
this.ctx.ui.requestRender();
|
|
723
742
|
} else {
|
|
724
|
-
// Role-agent assignments configure Task dispatch and must not switch the active chat model.
|
|
725
743
|
const apiKey = await this.ctx.session.modelRegistry.getApiKey(model, this.ctx.session.sessionId);
|
|
726
744
|
if (!apiKey) {
|
|
727
745
|
throw new Error(`No API key for ${model.provider}/${model.id}`);
|
|
728
746
|
}
|
|
729
|
-
const
|
|
730
|
-
|
|
731
|
-
|
|
747
|
+
const value =
|
|
748
|
+
selectedSelector ?? formatModelSelectorValue(`${model.provider}/${model.id}`, thinkingLevel);
|
|
749
|
+
const materializedProfile = materializeActiveModelProfileAssignment({
|
|
750
|
+
session: this.ctx.session,
|
|
751
|
+
settings: this.ctx.settings,
|
|
752
|
+
role,
|
|
753
|
+
selector: value,
|
|
754
|
+
});
|
|
755
|
+
if (!materializedProfile) {
|
|
756
|
+
const target = GJC_MODEL_ASSIGNMENT_TARGETS[role];
|
|
757
|
+
if (target.settingsPath === "modelRoles") {
|
|
758
|
+
this.ctx.settings.setModelRole(role, value);
|
|
759
|
+
} else {
|
|
760
|
+
const overrides = this.ctx.settings.get("task.agentModelOverrides");
|
|
761
|
+
this.ctx.settings.set("task.agentModelOverrides", { ...overrides, [role]: value });
|
|
762
|
+
}
|
|
763
|
+
}
|
|
764
|
+
modelSelector.refreshRoleAssignments({
|
|
765
|
+
currentModel: this.ctx.session.model,
|
|
766
|
+
currentThinkingLevel: this.ctx.session.thinkingLevel,
|
|
767
|
+
activeModelProfile:
|
|
768
|
+
this.ctx.session.getActiveModelProfile?.() ?? this.ctx.settings.get("modelProfile.default"),
|
|
769
|
+
});
|
|
732
770
|
this.ctx.settings.getStorage()?.recordModelUsage(`${model.provider}/${model.id}`);
|
|
733
771
|
this.ctx.showStatus(`${role} agent model: ${value}`);
|
|
734
772
|
done();
|
|
@@ -750,9 +788,10 @@ export class SelectorController {
|
|
|
750
788
|
this.ctx.session.getActiveModelProfile?.() ?? this.ctx.settings.get("modelProfile.default"),
|
|
751
789
|
isFastForProvider: provider => this.ctx.session.isFastForProvider(provider),
|
|
752
790
|
isFastForSubagentProvider: provider => this.ctx.session.isFastForSubagentProvider(provider),
|
|
791
|
+
isCurrentModelFastModeActive: () => this.ctx.session.isFastModeActive(),
|
|
753
792
|
},
|
|
754
793
|
);
|
|
755
|
-
return { component:
|
|
794
|
+
return { component: modelSelector, focus: modelSelector };
|
|
756
795
|
});
|
|
757
796
|
}
|
|
758
797
|
|
|
@@ -431,6 +431,8 @@ export class InteractiveMode implements InteractiveModeContext {
|
|
|
431
431
|
this.#resizeHandler = () => {
|
|
432
432
|
this.#syncEditorMaxHeight();
|
|
433
433
|
this.updateEditorChrome();
|
|
434
|
+
this.editor.invalidate();
|
|
435
|
+
this.ui.requestRender(true, "resize");
|
|
434
436
|
};
|
|
435
437
|
process.stdout.on("resize", this.#resizeHandler);
|
|
436
438
|
try {
|
|
@@ -23,7 +23,7 @@ export function buildHotkeysMarkdown(bindings: HotkeysMarkdownBindings): string
|
|
|
23
23
|
"|-----|--------|",
|
|
24
24
|
"| `Enter` | Send / queue while busy |",
|
|
25
25
|
`| \`${appKey(bindings, "app.message.queue")}\` | Queue message for next turn |`,
|
|
26
|
-
"| `Shift+Enter` | New line |",
|
|
26
|
+
"| `Shift+Enter` / `Ctrl+J` | New line |",
|
|
27
27
|
"| `Ctrl+W` / `Option+Backspace` | Delete word backwards |",
|
|
28
28
|
"| `Ctrl+U` | Delete to start of line |",
|
|
29
29
|
"| `Ctrl+K` | Delete to end of line |",
|
|
@@ -349,16 +349,54 @@ export function buttonLabel(label: string, index: number): string {
|
|
|
349
349
|
return `${index + 1}. ${stripped}`;
|
|
350
350
|
}
|
|
351
351
|
|
|
352
|
+
/** Numbered, escaped option list for the Telegram message body. */
|
|
353
|
+
export function numberedOptionList(labels: string[]): string {
|
|
354
|
+
return labels.map((label, i) => `${i + 1}. ${escapeHtml(label.replace(/^\s*\d+[.)]\s+/, ""))}`).join("\n");
|
|
355
|
+
}
|
|
356
|
+
|
|
357
|
+
/** Compact numeric button label; full option text belongs in the message body. */
|
|
358
|
+
export function choiceButtonLabel(index: number): string {
|
|
359
|
+
return String(index + 1);
|
|
360
|
+
}
|
|
361
|
+
|
|
352
362
|
export interface InlineButton {
|
|
353
363
|
text: string;
|
|
354
364
|
callback_data: string;
|
|
355
365
|
}
|
|
356
366
|
|
|
367
|
+
const COMPACT_BUTTONS_PER_ROW = 5;
|
|
368
|
+
|
|
357
369
|
/** A prefixed button label is "long" when it is wide or contains a newline. */
|
|
358
370
|
function isLongLabel(label: string): boolean {
|
|
359
371
|
return label.length > 18 || /[\r\n]/.test(label);
|
|
360
372
|
}
|
|
361
373
|
|
|
374
|
+
/**
|
|
375
|
+
* Lay out option callbacks as compact numeric buttons. Telegram mobile clients
|
|
376
|
+
* ellipsize long inline-keyboard labels and tall keyboards can be obscured by
|
|
377
|
+
* the composer, so the full choice text is rendered in the message body while
|
|
378
|
+
* the keyboard keeps only stable one-based tap targets.
|
|
379
|
+
*/
|
|
380
|
+
export function buildCompactChoiceGrid(
|
|
381
|
+
labels: string[],
|
|
382
|
+
callbackForIndex: (index: number) => string,
|
|
383
|
+
): InlineButton[][] {
|
|
384
|
+
const rows: InlineButton[][] = [];
|
|
385
|
+
let run: InlineButton[] = [];
|
|
386
|
+
const flush = () => {
|
|
387
|
+
if (run.length) {
|
|
388
|
+
rows.push(run);
|
|
389
|
+
run = [];
|
|
390
|
+
}
|
|
391
|
+
};
|
|
392
|
+
labels.forEach((_label, i) => {
|
|
393
|
+
run.push({ text: choiceButtonLabel(i), callback_data: callbackForIndex(i) });
|
|
394
|
+
if (run.length === COMPACT_BUTTONS_PER_ROW) flush();
|
|
395
|
+
});
|
|
396
|
+
flush();
|
|
397
|
+
return rows;
|
|
398
|
+
}
|
|
399
|
+
|
|
362
400
|
/**
|
|
363
401
|
* Lay out option labels as a numbered button grid. Long buttons take a
|
|
364
402
|
* full-width row; runs of short buttons are packed into rows of up to 3. The
|