agentvibes 5.6.0 → 5.6.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (101) hide show
  1. package/.agentvibes/config.json +3 -38
  2. package/.claude/config/audio-effects.cfg +1 -1
  3. package/.claude/config/background-music-enabled.txt +1 -1
  4. package/.claude/config/background-music-position.txt +6 -6
  5. package/.claude/github-star-reminder.txt +1 -1
  6. package/.claude/hooks/play-tts-ssh-remote.sh +119 -42
  7. package/.claude/hooks/play-tts-windows-receiver.sh +31 -0
  8. package/.claude/hooks/stop.sh +2 -27
  9. package/.claude/hooks-windows/play-tts-windows-sapi.ps1 +108 -108
  10. package/.claude/hooks-windows/play-tts.ps1 +58 -8
  11. package/.claude/piper-voices-dir.txt +1 -1
  12. package/.clawdbot/skill/README.md +326 -0
  13. package/.mcp.json +17 -27
  14. package/README.md +15 -2
  15. package/RELEASE_NOTES.md +64 -0
  16. package/bin/agent-vibes +39 -39
  17. package/package.json +1 -1
  18. package/src/bmad-detector.js +71 -71
  19. package/src/cli/list-personalities.js +110 -110
  20. package/src/cli/list-voices.js +114 -114
  21. package/src/commands/bmad-voices.js +394 -394
  22. package/src/commands/install-mcp.js +476 -476
  23. package/src/console/brand-colors.js +13 -13
  24. package/src/console/constants/personalities.js +44 -44
  25. package/src/console/modals/modal-overlay.js +247 -247
  26. package/src/console/navigation.js +5 -1
  27. package/src/console/tabs/agents-tab.js +5 -5
  28. package/src/console/tabs/help-tab.js +314 -314
  29. package/src/console/tabs/readme-tab.js +272 -272
  30. package/src/console/tabs/setup-tab.js +32 -17
  31. package/src/console/tabs/voices-tab.js +2 -2
  32. package/src/console/widgets/destroy-list.js +25 -25
  33. package/src/console/widgets/notice.js +55 -55
  34. package/src/console/widgets/personality-picker.js +213 -213
  35. package/src/console/widgets/reverb-picker.js +97 -97
  36. package/src/console/widgets/track-picker.js +1 -1
  37. package/src/i18n/de.js +202 -202
  38. package/src/i18n/es.js +202 -202
  39. package/src/i18n/fr.js +202 -202
  40. package/src/i18n/hi.js +202 -202
  41. package/src/i18n/ja.js +202 -202
  42. package/src/i18n/ko.js +202 -202
  43. package/src/i18n/pt.js +202 -202
  44. package/src/i18n/strings.js +54 -54
  45. package/src/i18n/zh-CN.js +202 -202
  46. package/src/installer/language-screen.js +31 -31
  47. package/src/installer/music-file-input.js +304 -304
  48. package/src/services/agent-voice-store.js +420 -423
  49. package/src/services/config-service.js +264 -264
  50. package/src/services/language-service.js +47 -47
  51. package/src/services/llm-provider-service.js +11 -4
  52. package/src/services/navigation-service.js +34 -10
  53. package/src/services/provider-service.js +143 -143
  54. package/src/utils/audio-duration-validator.js +298 -298
  55. package/src/utils/audio-format-validator.js +277 -277
  56. package/src/utils/dependency-checker.js +469 -469
  57. package/src/utils/file-ownership-verifier.js +358 -358
  58. package/src/utils/list-formatter.js +194 -194
  59. package/src/utils/music-file-validator.js +285 -285
  60. package/src/utils/preview-list-prompt.js +136 -136
  61. package/src/utils/secure-music-storage.js +412 -412
  62. package/.agentvibes/LITE-MODE.md +0 -236
  63. package/.agentvibes/README.md +0 -136
  64. package/.agentvibes/backup/session-start-tts.sh.20251210_212814 +0 -141
  65. package/.agentvibes/backups/agents/analyst_20260204_144958.md +0 -78
  66. package/.agentvibes/backups/agents/architect_20260204_144958.md +0 -72
  67. package/.agentvibes/backups/agents/dev_20260204_144958.md +0 -74
  68. package/.agentvibes/backups/agents/pm_20260204_144958.md +0 -72
  69. package/.agentvibes/backups/agents/quick-flow-solo-dev_20260204_144958.md +0 -64
  70. package/.agentvibes/backups/agents/sm_20260204_144958.md +0 -87
  71. package/.agentvibes/backups/agents/tea_20260204_144958.md +0 -79
  72. package/.agentvibes/backups/agents/tech-writer_20260204_144958.md +0 -82
  73. package/.agentvibes/backups/agents/ux-designer_20260204_144958.md +0 -80
  74. package/.agentvibes/config/README-personality-defaults.md +0 -162
  75. package/.agentvibes/config/agentvibes.json +0 -1
  76. package/.agentvibes/config/mode.txt +0 -1
  77. package/.agentvibes/config/personality-voice-defaults.default.json +0 -21
  78. package/.agentvibes/config/save-audio.txt +0 -1
  79. package/.agentvibes/config/voice-metadata.json +0 -160
  80. package/.agentvibes/hooks/help.sh +0 -191
  81. package/.agentvibes/hooks/post-tool-use-lite.sh +0 -111
  82. package/.agentvibes/hooks/save-audio-manager.sh +0 -162
  83. package/.agentvibes/hooks/session-start-full-optimized.sh +0 -102
  84. package/.agentvibes/hooks/session-start-full.sh +0 -142
  85. package/.agentvibes/hooks/session-start-lite-v2.sh +0 -34
  86. package/.agentvibes/hooks/session-start-lite.sh +0 -29
  87. package/.agentvibes/hooks/stop-lite.sh +0 -115
  88. package/.agentvibes/hooks/switch-mode.sh +0 -215
  89. package/.agentvibes/output-styles/audio-summary.md +0 -30
  90. package/.claude/audio/voice-samples/piper/alan.wav +0 -0
  91. package/.claude/audio/voice-samples/piper/amy.wav +0 -0
  92. package/.claude/audio/voice-samples/piper/charlotte.wav +0 -0
  93. package/.claude/audio/voice-samples/piper/joe.wav +0 -0
  94. package/.claude/audio/voice-samples/piper/john.wav +0 -0
  95. package/.claude/audio/voice-samples/piper/katherine.wav +0 -0
  96. package/.claude/audio/voice-samples/piper/kristin.wav +0 -0
  97. package/.claude/audio/voice-samples/piper/linda.wav +0 -0
  98. package/.claude/audio/voice-samples/piper/marcus.wav +0 -0
  99. package/.claude/audio/voice-samples/piper/ryan.wav +0 -0
  100. package/.claude/hooks/post-response.sh +0 -41
  101. package/bin/ensure-soprano-running.sh +0 -43
@@ -1,314 +1,314 @@
1
- /**
2
- * AgentVibes TUI Console — Help Tab
3
- * Epic 13: Story 13.1
4
- *
5
- * Implements the Tab Component Contract:
6
- * createHelpTab(screen, services) → { box, show, hide, onFocus, onBlur, getFooterText, getFooterColor }
7
- *
8
- * Features: keyboard shortcuts reference, two sections, [/] search.
9
- */
10
-
11
- import { t } from '../../i18n/strings.js';
12
-
13
- const IS_TEST = process.env.AGENTVIBES_TEST_MODE === 'true';
14
-
15
- let blessed;
16
- if (!IS_TEST) {
17
- const { default: b } = await import('blessed');
18
- blessed = b;
19
- }
20
-
21
- // ---------------------------------------------------------------------------
22
-
23
- const COLORS = {
24
- contentBg: '#0a0e1a',
25
- sectionHdr: '#546e7a', // Blue-gray — Help tab
26
- labelFg: '#e3f2fd',
27
- keyFg: '#ffff00', // Yellow — keyboard shortcuts
28
- descFg: '#90a4ae', // Gray — descriptions
29
- borderFg: '#607d8b',
30
- footerBg: '#607d8b', // Gray — Help tab footer
31
- };
32
-
33
- // ---------------------------------------------------------------------------
34
-
35
- /**
36
- * Return all shortcut sections.
37
- * @returns {{ title: string, shortcuts: { key: string, desc: string }[] }[]}
38
- */
39
- export function getShortcutSections() {
40
- return [
41
- {
42
- title: 'Global Shortcuts',
43
- shortcuts: [
44
- { key: 'Q', desc: 'Quit the console' },
45
- { key: 'Ctrl+C', desc: 'Force quit' },
46
- { key: 'S', desc: 'Switch to Settings tab' },
47
- { key: 'V', desc: 'Switch to Voices tab' },
48
- { key: 'M', desc: 'Switch to Music tab' },
49
- { key: 'R', desc: 'Switch to Readme tab' },
50
- { key: 'H', desc: 'Switch to Help tab' },
51
- { key: 'I', desc: 'Switch to Install tab' },
52
- { key: 'Esc', desc: 'Close modal / go back' },
53
- ],
54
- },
55
- {
56
- title: 'Navigation Shortcuts',
57
- shortcuts: [
58
- { key: '↑↓ / j k', desc: 'Navigate lists' },
59
- { key: 'Enter', desc: 'Select / activate' },
60
- { key: 'Space', desc: 'Toggle / preview' },
61
- { key: 'Tab', desc: 'Next button' },
62
- { key: 'Shift+Tab', desc: 'Previous button' },
63
- { key: '/', desc: 'Open search/filter' },
64
- { key: 'F', desc: 'Toggle favorites filter (Voices/Music)' },
65
- { key: '*', desc: 'Toggle favorite (Music tab)' },
66
- { key: 'M', desc: 'Toggle music on/off (Music tab)' },
67
- ],
68
- },
69
- {
70
- title: 'Tab Color Guide',
71
- shortcuts: [
72
- { key: 'Blue (#2196f3)', desc: 'Settings tab footer' },
73
- { key: 'Teal (#00695c)', desc: 'Voices tab footer' },
74
- { key: 'Orange (#ff9800)', desc: 'Music tab footer' },
75
- { key: 'Dark (#455a64)', desc: 'Readme tab footer' },
76
- { key: 'Gray (#607d8b)', desc: 'Help tab footer' },
77
- { key: 'Indigo (#3f51b5)', desc: 'Install tab footer' },
78
- ],
79
- },
80
- ];
81
- }
82
-
83
- // ---------------------------------------------------------------------------
84
-
85
- const _FOOTER_TEXT_EN = '[↑↓/jk] Scroll [/] Search [PgUp/PgDn] Page [S/V/M/A/R] Tab [Q] Quit';
86
-
87
- function createTestStub() {
88
- return {
89
- box: {},
90
- show: () => {},
91
- hide: () => {},
92
- onFocus: () => {},
93
- onBlur: () => {},
94
- getFooterText: () => _FOOTER_TEXT_EN,
95
- getFooterColor: () => COLORS.footerBg,
96
- };
97
- }
98
-
99
- // ---------------------------------------------------------------------------
100
-
101
- /**
102
- * Create the Help tab component.
103
- *
104
- * @param {object} screen - Blessed screen instance
105
- * @param {object} services
106
- * @returns {{ box, show, hide, onFocus, onBlur, getFooterText, getFooterColor }}
107
- */
108
- export function createHelpTab(screen, services) {
109
- if (IS_TEST) return createTestStub();
110
-
111
- const { focusMainTabBar, languageService } = services;
112
- const _tl = (key) => languageService ? languageService.t(key) : t('en', key);
113
-
114
- function _buildSections() {
115
- return [
116
- {
117
- title: _tl('helpSectionGlobal'),
118
- shortcuts: [
119
- { key: 'Q', desc: _tl('helpQuit') },
120
- { key: 'Ctrl+C', desc: _tl('helpForceQuit') },
121
- { key: 'S', desc: _tl('helpSwitchSettings') },
122
- { key: 'V', desc: _tl('helpSwitchVoices') },
123
- { key: 'M', desc: _tl('helpSwitchMusic') },
124
- { key: 'R', desc: _tl('helpSwitchReadme') },
125
- { key: 'H', desc: _tl('helpSwitchHelp') },
126
- { key: 'I', desc: _tl('helpSwitchInstall') },
127
- { key: 'Esc', desc: _tl('helpCloseModal') },
128
- ],
129
- },
130
- {
131
- title: _tl('helpSectionNavigation'),
132
- shortcuts: [
133
- { key: '↑↓ / j k', desc: _tl('helpNavigateLists') },
134
- { key: 'Enter', desc: _tl('helpSelectActivate') },
135
- { key: 'Space', desc: _tl('helpTogglePreview') },
136
- { key: 'Tab', desc: _tl('helpNextButton') },
137
- { key: 'Shift+Tab', desc: _tl('helpPrevButton') },
138
- { key: '/', desc: _tl('helpOpenSearch') },
139
- { key: 'F', desc: _tl('helpToggleFavFilter') },
140
- { key: '*', desc: _tl('helpToggleFav') },
141
- { key: 'M', desc: _tl('helpToggleMusic') },
142
- ],
143
- },
144
- {
145
- title: _tl('helpSectionColors'),
146
- shortcuts: [
147
- { key: 'Blue (#2196f3)', desc: _tl('helpColorSettings') },
148
- { key: 'Teal (#00695c)', desc: _tl('helpColorVoices') },
149
- { key: 'Orange (#ff9800)', desc: _tl('helpColorMusic') },
150
- { key: 'Dark (#455a64)', desc: _tl('helpColorReadme') },
151
- { key: 'Gray (#607d8b)', desc: _tl('helpColorHelp') },
152
- { key: 'Indigo (#3f51b5)', desc: _tl('helpColorInstall') },
153
- ],
154
- },
155
- ];
156
- }
157
-
158
- // -------------------------------------------------------------------------
159
- // Container
160
-
161
- const box = blessed.box({
162
- parent: screen,
163
- top: 5,
164
- left: 0,
165
- width: '100%',
166
- bottom: 2,
167
- hidden: true,
168
- style: { fg: COLORS.labelFg, bg: COLORS.contentBg },
169
- border: { type: 'line' },
170
- borderStyle: { fg: COLORS.borderFg },
171
- });
172
-
173
- // -------------------------------------------------------------------------
174
- // Build content text
175
-
176
- function _buildContent(filterText) {
177
- const lines = [];
178
- for (const section of _buildSections()) {
179
- lines.push(`{bold}{#546e7a-fg}── ${section.title} ${'─'.repeat(Math.max(0, 60 - section.title.length))}{/#546e7a-fg}{/bold}`);
180
- for (const { key, desc } of section.shortcuts) {
181
- const displayKey = key.padEnd(20);
182
- const displayDesc = desc;
183
- if (filterText && !key.toLowerCase().includes(filterText) && !desc.toLowerCase().includes(filterText)) {
184
- continue;
185
- }
186
- lines.push(` {${COLORS.keyFg}-fg}${displayKey}{/${COLORS.keyFg}-fg} {${COLORS.descFg}-fg}${displayDesc}{/${COLORS.descFg}-fg}`);
187
- }
188
- lines.push('');
189
- }
190
- return lines.join('\n');
191
- }
192
-
193
- // -------------------------------------------------------------------------
194
- // Scrollable content
195
-
196
- const scrollBox = blessed.box({
197
- parent: box,
198
- top: 1,
199
- left: 2,
200
- width: '96%',
201
- bottom: 4,
202
- scrollable: true,
203
- alwaysScroll: true,
204
- tags: true,
205
- keys: true,
206
- vi: true,
207
- mouse: true,
208
- scrollbar: { ch: '│', style: { fg: COLORS.sectionHdr } },
209
- content: _buildContent(''),
210
- style: { fg: COLORS.labelFg, bg: COLORS.contentBg },
211
- });
212
-
213
- // -------------------------------------------------------------------------
214
- // Search
215
-
216
- const searchBox = blessed.textbox({
217
- parent: box,
218
- bottom: 2,
219
- left: 10,
220
- width: 30,
221
- height: 1,
222
- hidden: true,
223
- inputOnFocus: true,
224
- keys: true,
225
- style: { fg: COLORS.keyFg, bg: '#1a3a5c', focus: { bg: '#245a80' } },
226
- });
227
-
228
- const searchLabel = blessed.text({
229
- parent: box,
230
- bottom: 2,
231
- left: 2,
232
- content: _tl('helpSearchLabel'),
233
- style: { fg: COLORS.descFg, bg: COLORS.contentBg },
234
- });
235
-
236
- searchBox.on('keypress', () => {
237
- setTimeout(() => {
238
- const filter = searchBox.getValue().toLowerCase().trim();
239
- scrollBox.setContent(_buildContent(filter));
240
- screen.render();
241
- }, 0);
242
- });
243
-
244
- searchBox.key(['escape'], () => {
245
- searchBox.clearValue();
246
- searchBox.hide();
247
- scrollBox.setContent(_buildContent(''));
248
- scrollBox.focus();
249
- screen.render();
250
- });
251
-
252
- scrollBox.key(['/'], () => {
253
- searchBox.show();
254
- searchBox.clearValue();
255
- searchBox.focus();
256
- screen.render();
257
- });
258
-
259
- // [↑] at top of content → jump to main header tab bar
260
- scrollBox.key(['up'], () => {
261
- if (scrollBox.getScroll() === 0 && typeof focusMainTabBar === 'function') {
262
- focusMainTabBar();
263
- }
264
- });
265
-
266
- // Escape → return to header tab bar
267
- scrollBox.key(['escape'], () => {
268
- if (typeof focusMainTabBar === 'function') { focusMainTabBar(); screen.render(); }
269
- });
270
-
271
- // -------------------------------------------------------------------------
272
- // Language change handler
273
-
274
- if (languageService) {
275
- languageService.onChange(() => {
276
- scrollBox.setContent(_buildContent(''));
277
- searchLabel.setContent(_tl('helpSearchLabel'));
278
- screen.render();
279
- });
280
- }
281
-
282
- // -------------------------------------------------------------------------
283
- // Tab Component Contract
284
-
285
- return {
286
- box,
287
-
288
- show() {
289
- box.show();
290
- scrollBox.setContent(_buildContent(''));
291
- screen.render();
292
- },
293
-
294
- hide() {
295
- box.hide();
296
- screen.render();
297
- },
298
-
299
- onFocus() {
300
- scrollBox.focus();
301
- screen.render();
302
- },
303
-
304
- onBlur() {},
305
-
306
- getFooterText() {
307
- return _tl('helpFooter');
308
- },
309
-
310
- getFooterColor() {
311
- return COLORS.footerBg;
312
- },
313
- };
314
- }
1
+ /**
2
+ * AgentVibes TUI Console — Help Tab
3
+ * Epic 13: Story 13.1
4
+ *
5
+ * Implements the Tab Component Contract:
6
+ * createHelpTab(screen, services) → { box, show, hide, onFocus, onBlur, getFooterText, getFooterColor }
7
+ *
8
+ * Features: keyboard shortcuts reference, two sections, [/] search.
9
+ */
10
+
11
+ import { t } from '../../i18n/strings.js';
12
+
13
+ const IS_TEST = process.env.AGENTVIBES_TEST_MODE === 'true';
14
+
15
+ let blessed;
16
+ if (!IS_TEST) {
17
+ const { default: b } = await import('blessed');
18
+ blessed = b;
19
+ }
20
+
21
+ // ---------------------------------------------------------------------------
22
+
23
+ const COLORS = {
24
+ contentBg: '#0a0e1a',
25
+ sectionHdr: '#546e7a', // Blue-gray — Help tab
26
+ labelFg: '#e3f2fd',
27
+ keyFg: '#ffff00', // Yellow — keyboard shortcuts
28
+ descFg: '#90a4ae', // Gray — descriptions
29
+ borderFg: '#607d8b',
30
+ footerBg: '#607d8b', // Gray — Help tab footer
31
+ };
32
+
33
+ // ---------------------------------------------------------------------------
34
+
35
+ /**
36
+ * Return all shortcut sections.
37
+ * @returns {{ title: string, shortcuts: { key: string, desc: string }[] }[]}
38
+ */
39
+ export function getShortcutSections() {
40
+ return [
41
+ {
42
+ title: 'Global Shortcuts',
43
+ shortcuts: [
44
+ { key: 'Q', desc: 'Quit the console' },
45
+ { key: 'Ctrl+C', desc: 'Force quit' },
46
+ { key: 'S', desc: 'Switch to Settings tab' },
47
+ { key: 'V', desc: 'Switch to Voices tab' },
48
+ { key: 'M', desc: 'Switch to Music tab' },
49
+ { key: 'R', desc: 'Switch to Readme tab' },
50
+ { key: 'H', desc: 'Switch to Help tab' },
51
+ { key: 'I', desc: 'Switch to Install tab' },
52
+ { key: 'Esc', desc: 'Close modal / go back' },
53
+ ],
54
+ },
55
+ {
56
+ title: 'Navigation Shortcuts',
57
+ shortcuts: [
58
+ { key: '↑↓ / j k', desc: 'Navigate lists' },
59
+ { key: 'Enter', desc: 'Select / activate' },
60
+ { key: 'Space', desc: 'Toggle / preview' },
61
+ { key: 'Tab', desc: 'Next button' },
62
+ { key: 'Shift+Tab', desc: 'Previous button' },
63
+ { key: '/', desc: 'Open search/filter' },
64
+ { key: 'F', desc: 'Toggle favorites filter (Voices/Music)' },
65
+ { key: '*', desc: 'Toggle favorite (Music tab)' },
66
+ { key: 'M', desc: 'Toggle music on/off (Music tab)' },
67
+ ],
68
+ },
69
+ {
70
+ title: 'Tab Color Guide',
71
+ shortcuts: [
72
+ { key: 'Blue (#2196f3)', desc: 'Settings tab footer' },
73
+ { key: 'Teal (#00695c)', desc: 'Voices tab footer' },
74
+ { key: 'Orange (#ff9800)', desc: 'Music tab footer' },
75
+ { key: 'Dark (#455a64)', desc: 'Readme tab footer' },
76
+ { key: 'Gray (#607d8b)', desc: 'Help tab footer' },
77
+ { key: 'Indigo (#3f51b5)', desc: 'Install tab footer' },
78
+ ],
79
+ },
80
+ ];
81
+ }
82
+
83
+ // ---------------------------------------------------------------------------
84
+
85
+ const _FOOTER_TEXT_EN = '[↑↓/jk] Scroll [/] Search [PgUp/PgDn] Page [S/V/M/A/R] Tab [Q] Quit';
86
+
87
+ function createTestStub() {
88
+ return {
89
+ box: {},
90
+ show: () => {},
91
+ hide: () => {},
92
+ onFocus: () => {},
93
+ onBlur: () => {},
94
+ getFooterText: () => _FOOTER_TEXT_EN,
95
+ getFooterColor: () => COLORS.footerBg,
96
+ };
97
+ }
98
+
99
+ // ---------------------------------------------------------------------------
100
+
101
+ /**
102
+ * Create the Help tab component.
103
+ *
104
+ * @param {object} screen - Blessed screen instance
105
+ * @param {object} services
106
+ * @returns {{ box, show, hide, onFocus, onBlur, getFooterText, getFooterColor }}
107
+ */
108
+ export function createHelpTab(screen, services) {
109
+ if (IS_TEST) return createTestStub();
110
+
111
+ const { focusMainTabBar, languageService } = services;
112
+ const _tl = (key) => languageService ? languageService.t(key) : t('en', key);
113
+
114
+ function _buildSections() {
115
+ return [
116
+ {
117
+ title: _tl('helpSectionGlobal'),
118
+ shortcuts: [
119
+ { key: 'Q', desc: _tl('helpQuit') },
120
+ { key: 'Ctrl+C', desc: _tl('helpForceQuit') },
121
+ { key: 'S', desc: _tl('helpSwitchSettings') },
122
+ { key: 'V', desc: _tl('helpSwitchVoices') },
123
+ { key: 'M', desc: _tl('helpSwitchMusic') },
124
+ { key: 'R', desc: _tl('helpSwitchReadme') },
125
+ { key: 'H', desc: _tl('helpSwitchHelp') },
126
+ { key: 'I', desc: _tl('helpSwitchInstall') },
127
+ { key: 'Esc', desc: _tl('helpCloseModal') },
128
+ ],
129
+ },
130
+ {
131
+ title: _tl('helpSectionNavigation'),
132
+ shortcuts: [
133
+ { key: '↑↓ / j k', desc: _tl('helpNavigateLists') },
134
+ { key: 'Enter', desc: _tl('helpSelectActivate') },
135
+ { key: 'Space', desc: _tl('helpTogglePreview') },
136
+ { key: 'Tab', desc: _tl('helpNextButton') },
137
+ { key: 'Shift+Tab', desc: _tl('helpPrevButton') },
138
+ { key: '/', desc: _tl('helpOpenSearch') },
139
+ { key: 'F', desc: _tl('helpToggleFavFilter') },
140
+ { key: '*', desc: _tl('helpToggleFav') },
141
+ { key: 'M', desc: _tl('helpToggleMusic') },
142
+ ],
143
+ },
144
+ {
145
+ title: _tl('helpSectionColors'),
146
+ shortcuts: [
147
+ { key: 'Blue (#2196f3)', desc: _tl('helpColorSettings') },
148
+ { key: 'Teal (#00695c)', desc: _tl('helpColorVoices') },
149
+ { key: 'Orange (#ff9800)', desc: _tl('helpColorMusic') },
150
+ { key: 'Dark (#455a64)', desc: _tl('helpColorReadme') },
151
+ { key: 'Gray (#607d8b)', desc: _tl('helpColorHelp') },
152
+ { key: 'Indigo (#3f51b5)', desc: _tl('helpColorInstall') },
153
+ ],
154
+ },
155
+ ];
156
+ }
157
+
158
+ // -------------------------------------------------------------------------
159
+ // Container
160
+
161
+ const box = blessed.box({
162
+ parent: screen,
163
+ top: 5,
164
+ left: 0,
165
+ width: '100%',
166
+ bottom: 2,
167
+ hidden: true,
168
+ style: { fg: COLORS.labelFg, bg: COLORS.contentBg },
169
+ border: { type: 'line' },
170
+ borderStyle: { fg: COLORS.borderFg },
171
+ });
172
+
173
+ // -------------------------------------------------------------------------
174
+ // Build content text
175
+
176
+ function _buildContent(filterText) {
177
+ const lines = [];
178
+ for (const section of _buildSections()) {
179
+ lines.push(`{bold}{#546e7a-fg}── ${section.title} ${'─'.repeat(Math.max(0, 60 - section.title.length))}{/#546e7a-fg}{/bold}`);
180
+ for (const { key, desc } of section.shortcuts) {
181
+ const displayKey = key.padEnd(20);
182
+ const displayDesc = desc;
183
+ if (filterText && !key.toLowerCase().includes(filterText) && !desc.toLowerCase().includes(filterText)) {
184
+ continue;
185
+ }
186
+ lines.push(` {${COLORS.keyFg}-fg}${displayKey}{/${COLORS.keyFg}-fg} {${COLORS.descFg}-fg}${displayDesc}{/${COLORS.descFg}-fg}`);
187
+ }
188
+ lines.push('');
189
+ }
190
+ return lines.join('\n');
191
+ }
192
+
193
+ // -------------------------------------------------------------------------
194
+ // Scrollable content
195
+
196
+ const scrollBox = blessed.box({
197
+ parent: box,
198
+ top: 1,
199
+ left: 2,
200
+ width: '96%',
201
+ bottom: 4,
202
+ scrollable: true,
203
+ alwaysScroll: true,
204
+ tags: true,
205
+ keys: true,
206
+ vi: true,
207
+ mouse: true,
208
+ scrollbar: { ch: '│', style: { fg: COLORS.sectionHdr } },
209
+ content: _buildContent(''),
210
+ style: { fg: COLORS.labelFg, bg: COLORS.contentBg },
211
+ });
212
+
213
+ // -------------------------------------------------------------------------
214
+ // Search
215
+
216
+ const searchBox = blessed.textbox({
217
+ parent: box,
218
+ bottom: 2,
219
+ left: 10,
220
+ width: 30,
221
+ height: 1,
222
+ hidden: true,
223
+ inputOnFocus: true,
224
+ keys: true,
225
+ style: { fg: COLORS.keyFg, bg: '#1a3a5c', focus: { bg: '#245a80' } },
226
+ });
227
+
228
+ const searchLabel = blessed.text({
229
+ parent: box,
230
+ bottom: 2,
231
+ left: 2,
232
+ content: _tl('helpSearchLabel'),
233
+ style: { fg: COLORS.descFg, bg: COLORS.contentBg },
234
+ });
235
+
236
+ searchBox.on('keypress', () => {
237
+ setTimeout(() => {
238
+ const filter = searchBox.getValue().toLowerCase().trim();
239
+ scrollBox.setContent(_buildContent(filter));
240
+ screen.render();
241
+ }, 0);
242
+ });
243
+
244
+ searchBox.key(['escape'], () => {
245
+ searchBox.clearValue();
246
+ searchBox.hide();
247
+ scrollBox.setContent(_buildContent(''));
248
+ scrollBox.focus();
249
+ screen.render();
250
+ });
251
+
252
+ scrollBox.key(['/'], () => {
253
+ searchBox.show();
254
+ searchBox.clearValue();
255
+ searchBox.focus();
256
+ screen.render();
257
+ });
258
+
259
+ // [↑] at top of content → jump to main header tab bar
260
+ scrollBox.key(['up'], () => {
261
+ if (scrollBox.getScroll() === 0 && typeof focusMainTabBar === 'function') {
262
+ focusMainTabBar();
263
+ }
264
+ });
265
+
266
+ // Escape → return to header tab bar
267
+ scrollBox.key(['escape'], () => {
268
+ if (typeof focusMainTabBar === 'function') { focusMainTabBar(); screen.render(); }
269
+ });
270
+
271
+ // -------------------------------------------------------------------------
272
+ // Language change handler
273
+
274
+ if (languageService) {
275
+ languageService.onChange(() => {
276
+ scrollBox.setContent(_buildContent(''));
277
+ searchLabel.setContent(_tl('helpSearchLabel'));
278
+ screen.render();
279
+ });
280
+ }
281
+
282
+ // -------------------------------------------------------------------------
283
+ // Tab Component Contract
284
+
285
+ return {
286
+ box,
287
+
288
+ show() {
289
+ box.show();
290
+ scrollBox.setContent(_buildContent(''));
291
+ screen.render();
292
+ },
293
+
294
+ hide() {
295
+ box.hide();
296
+ screen.render();
297
+ },
298
+
299
+ onFocus() {
300
+ scrollBox.focus();
301
+ screen.render();
302
+ },
303
+
304
+ onBlur() {},
305
+
306
+ getFooterText() {
307
+ return _tl('helpFooter');
308
+ },
309
+
310
+ getFooterColor() {
311
+ return COLORS.footerBg;
312
+ },
313
+ };
314
+ }