@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.
Files changed (117) hide show
  1. package/CHANGELOG.md +48 -0
  2. package/bin/gjc.js +4 -0
  3. package/dist/types/cli/plugin-cli.d.ts +2 -0
  4. package/dist/types/commands/plugin.d.ts +6 -0
  5. package/dist/types/commands/session.d.ts +6 -0
  6. package/dist/types/config/model-profile-activation.d.ts +8 -1
  7. package/dist/types/extensibility/gjc-plugins/compiler.d.ts +19 -0
  8. package/dist/types/extensibility/gjc-plugins/constrained-hooks.d.ts +29 -0
  9. package/dist/types/extensibility/gjc-plugins/index.d.ts +9 -0
  10. package/dist/types/extensibility/gjc-plugins/injection.d.ts +9 -0
  11. package/dist/types/extensibility/gjc-plugins/installer.d.ts +13 -0
  12. package/dist/types/extensibility/gjc-plugins/mcp-policy.d.ts +26 -0
  13. package/dist/types/extensibility/gjc-plugins/observability.d.ts +27 -0
  14. package/dist/types/extensibility/gjc-plugins/prompt-appendix.d.ts +16 -0
  15. package/dist/types/extensibility/gjc-plugins/registry.d.ts +32 -0
  16. package/dist/types/extensibility/gjc-plugins/runtime-adapters.d.ts +64 -0
  17. package/dist/types/extensibility/gjc-plugins/session-validation.d.ts +42 -0
  18. package/dist/types/extensibility/gjc-plugins/types.d.ts +158 -2
  19. package/dist/types/extensibility/gjc-plugins/validation.d.ts +8 -1
  20. package/dist/types/gjc-runtime/launch-tmux.d.ts +1 -0
  21. package/dist/types/gjc-runtime/psmux-detect.d.ts +78 -0
  22. package/dist/types/gjc-runtime/team-runtime.d.ts +2 -0
  23. package/dist/types/gjc-runtime/tmux-common.d.ts +20 -1
  24. package/dist/types/gjc-runtime/tmux-sessions.d.ts +18 -0
  25. package/dist/types/main.d.ts +2 -0
  26. package/dist/types/modes/components/model-selector.d.ts +6 -0
  27. package/dist/types/notifications/html-format.d.ts +11 -0
  28. package/dist/types/notifications/index.d.ts +149 -1
  29. package/dist/types/notifications/lifecycle-commands.d.ts +72 -0
  30. package/dist/types/notifications/lifecycle-control-runtime.d.ts +98 -0
  31. package/dist/types/notifications/lifecycle-orchestrator.d.ts +144 -0
  32. package/dist/types/notifications/rate-limit-pool.d.ts +2 -0
  33. package/dist/types/notifications/recent-activity.d.ts +35 -0
  34. package/dist/types/notifications/telegram-daemon.d.ts +60 -0
  35. package/dist/types/notifications/telegram-reference.d.ts +3 -1
  36. package/dist/types/notifications/topic-registry.d.ts +10 -9
  37. package/dist/types/runtime-mcp/types.d.ts +7 -0
  38. package/dist/types/sdk.d.ts +2 -0
  39. package/dist/types/session/agent-session.d.ts +14 -4
  40. package/dist/types/session/blob-store.d.ts +25 -0
  41. package/dist/types/session/session-manager.d.ts +57 -0
  42. package/dist/types/slash-commands/helpers/fast-status-report.d.ts +6 -0
  43. package/dist/types/system-prompt.d.ts +2 -0
  44. package/dist/types/task/executor.d.ts +9 -1
  45. package/dist/types/tools/index.d.ts +3 -1
  46. package/dist/types/utils/changelog.d.ts +1 -0
  47. package/package.json +11 -9
  48. package/scripts/g004-tmux-smoke.ts +100 -0
  49. package/scripts/g005-daemon-smoke.ts +181 -0
  50. package/scripts/g011-daemon-path-smoke.ts +153 -0
  51. package/src/cli/plugin-cli.ts +66 -3
  52. package/src/cli.ts +21 -4
  53. package/src/commands/plugin.ts +4 -0
  54. package/src/commands/session.ts +18 -0
  55. package/src/config/model-profile-activation.ts +55 -7
  56. package/src/defaults/gjc/extensions/grok-cli-vendor/biome.json +1 -1
  57. package/src/defaults/gjc/skills/deep-interview/SKILL.md +3 -3
  58. package/src/defaults/gjc/skills/team/SKILL.md +5 -4
  59. package/src/defaults/gjc/skills/ultragoal/SKILL.md +41 -13
  60. package/src/export/html/index.ts +2 -2
  61. package/src/extensibility/gjc-plugins/compiler.ts +351 -0
  62. package/src/extensibility/gjc-plugins/constrained-hooks.ts +170 -0
  63. package/src/extensibility/gjc-plugins/index.ts +9 -0
  64. package/src/extensibility/gjc-plugins/injection.ts +109 -0
  65. package/src/extensibility/gjc-plugins/installer.ts +434 -0
  66. package/src/extensibility/gjc-plugins/loader.ts +3 -1
  67. package/src/extensibility/gjc-plugins/mcp-policy.ts +239 -0
  68. package/src/extensibility/gjc-plugins/observability.ts +84 -0
  69. package/src/extensibility/gjc-plugins/paths.ts +1 -1
  70. package/src/extensibility/gjc-plugins/prompt-appendix.ts +109 -0
  71. package/src/extensibility/gjc-plugins/registry.ts +180 -0
  72. package/src/extensibility/gjc-plugins/runtime-adapters.ts +234 -0
  73. package/src/extensibility/gjc-plugins/schema.ts +250 -20
  74. package/src/extensibility/gjc-plugins/session-validation.ts +147 -0
  75. package/src/extensibility/gjc-plugins/types.ts +199 -3
  76. package/src/extensibility/gjc-plugins/validation.ts +80 -0
  77. package/src/extensibility/skills.ts +15 -0
  78. package/src/gjc-runtime/launch-tmux.ts +61 -7
  79. package/src/gjc-runtime/psmux-detect.ts +239 -0
  80. package/src/gjc-runtime/team-runtime.ts +56 -23
  81. package/src/gjc-runtime/tmux-common.ts +27 -2
  82. package/src/gjc-runtime/tmux-sessions.ts +51 -1
  83. package/src/gjc-runtime/ultragoal-runtime.ts +75 -15
  84. package/src/internal-urls/docs-index.generated.ts +5 -4
  85. package/src/main.ts +14 -3
  86. package/src/modes/components/hook-editor.ts +1 -1
  87. package/src/modes/components/hook-selector.ts +67 -43
  88. package/src/modes/components/model-selector.ts +44 -11
  89. package/src/modes/controllers/extension-ui-controller.ts +0 -27
  90. package/src/modes/controllers/selector-controller.ts +50 -11
  91. package/src/modes/interactive-mode.ts +2 -0
  92. package/src/modes/utils/hotkeys-markdown.ts +1 -1
  93. package/src/notifications/html-format.ts +38 -0
  94. package/src/notifications/index.ts +242 -12
  95. package/src/notifications/lifecycle-commands.ts +228 -0
  96. package/src/notifications/lifecycle-control-runtime.ts +400 -0
  97. package/src/notifications/lifecycle-orchestrator.ts +358 -0
  98. package/src/notifications/rate-limit-pool.ts +19 -0
  99. package/src/notifications/recent-activity.ts +132 -0
  100. package/src/notifications/telegram-daemon.ts +433 -8
  101. package/src/notifications/telegram-reference.ts +25 -7
  102. package/src/notifications/topic-registry.ts +18 -9
  103. package/src/prompts/agents/executor.md +2 -2
  104. package/src/runtime-mcp/transports/stdio.ts +38 -4
  105. package/src/runtime-mcp/types.ts +7 -0
  106. package/src/sdk.ts +157 -10
  107. package/src/session/agent-session.ts +166 -74
  108. package/src/session/blob-store.ts +196 -8
  109. package/src/session/session-manager.ts +678 -7
  110. package/src/slash-commands/builtin-registry.ts +23 -3
  111. package/src/slash-commands/helpers/fast-status-report.ts +13 -3
  112. package/src/system-prompt.ts +9 -0
  113. package/src/task/executor.ts +31 -7
  114. package/src/task/index.ts +2 -0
  115. package/src/tools/ask.ts +5 -1
  116. package/src/tools/index.ts +3 -1
  117. 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.map(e => e.content).join("\n\n");
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
- const maxScrollOffset = Math.max(0, lines.length - this.#maxRows);
136
- this.#lastMaxScrollOffset = maxScrollOffset;
137
- this.#scrollOffset = Math.max(0, Math.min(this.#scrollOffset, maxScrollOffset));
119
+ if (lines.length <= this.#maxRows) {
120
+ this.#lastMaxScrollOffset = 0;
121
+ this.#scrollOffset = 0;
122
+ return lines;
123
+ }
138
124
 
139
- const visibleLines = lines.slice(this.#scrollOffset, this.#scrollOffset + this.#maxRows);
140
- if (maxScrollOffset === 0 || visibleLines.length === 0) {
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
- const indicator =
145
- this.#scrollOffset === 0
146
- ? theme.fg("dim", " PgDn↓")
147
- : this.#scrollOffset >= maxScrollOffset
148
- ? theme.fg("dim", " PgUp↑")
149
- : theme.fg("dim", " PgUp/PgDn↕");
150
- const lastIndex = visibleLines.length - 1;
151
- const availableWidth = Math.max(1, width - visibleWidth(indicator));
152
- const fittedLine = truncateToWidth(visibleLines[lastIndex] ?? "", availableWidth);
153
- visibleLines[lastIndex] = `${fittedLine}${indicator}`;
154
- return visibleLines;
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 scrollHint = this.#scrollTitleRows === undefined ? "" : " wheel/PgUp/PgDn scroll question";
547
- this.#helpTextComponent.setText(
548
- theme.fg("dim", `enter submit esc back to options ctrl+g external editor${scrollHint}`),
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
- let roleMatched = false;
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
- // Subagent roles (task.agentModelOverrides) run under task.serviceTier, so
985
- // their must reflect the effective subagent tier, not the main session tier.
986
- const roleFast =
987
- roleInfo.settingsPath === "task.agentModelOverrides"
988
- ? this.#isFastForSubagentProvider(assigned.model.provider)
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 even when it carries no role badge. Skip when a role token for
996
- // this row already rendered the glyph (duplicate-glyph guard).
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
- !roleMatched &&
1031
+ !currentModelEffectiveGlyphRendered &&
999
1032
  this.#currentModel !== undefined &&
1000
1033
  modelsAreEqual(this.#currentModel, item.model) &&
1001
- this.#isFastForProvider(item.model.provider)
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
- const selector = new ModelSelectorComponent(
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: ${selector ?? model.id}`);
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: ${selector ?? model.id}`);
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 overrides = this.ctx.settings.get("task.agentModelOverrides");
730
- const value = selector ?? `${model.provider}/${model.id}`;
731
- this.ctx.settings.set("task.agentModelOverrides", { ...overrides, [role]: value });
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: selector, focus: selector };
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