centy 0.0.5 → 0.0.7

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 (81) hide show
  1. package/bin/run.js +14 -2
  2. package/dist/commands/start.d.ts +13 -0
  3. package/dist/commands/start.d.ts.map +1 -0
  4. package/dist/commands/start.js +79 -0
  5. package/dist/commands/start.js.map +1 -0
  6. package/dist/hooks/prerun.js +1 -1
  7. package/dist/hooks/prerun.js.map +1 -1
  8. package/dist/lib/start/daemon-binary-exists.d.ts +2 -0
  9. package/dist/lib/start/daemon-binary-exists.d.ts.map +1 -0
  10. package/dist/lib/start/daemon-binary-exists.js +10 -0
  11. package/dist/lib/start/daemon-binary-exists.js.map +1 -0
  12. package/dist/lib/start/find-daemon-binary.d.ts +2 -0
  13. package/dist/lib/start/find-daemon-binary.d.ts.map +1 -0
  14. package/dist/lib/start/find-daemon-binary.js +25 -0
  15. package/dist/lib/start/find-daemon-binary.js.map +1 -0
  16. package/dist/lib/start/wait-for-daemon.d.ts +7 -0
  17. package/dist/lib/start/wait-for-daemon.d.ts.map +1 -0
  18. package/dist/lib/start/wait-for-daemon.js +20 -0
  19. package/dist/lib/start/wait-for-daemon.js.map +1 -0
  20. package/dist/tui/App.d.ts +6 -0
  21. package/dist/tui/App.d.ts.map +1 -0
  22. package/dist/tui/App.js +63 -0
  23. package/dist/tui/App.js.map +1 -0
  24. package/dist/tui/components/domain/IssueList.d.ts +2 -0
  25. package/dist/tui/components/domain/IssueList.d.ts.map +1 -0
  26. package/dist/tui/components/domain/IssueList.js +52 -0
  27. package/dist/tui/components/domain/IssueList.js.map +1 -0
  28. package/dist/tui/components/domain/ProjectList.d.ts +2 -0
  29. package/dist/tui/components/domain/ProjectList.d.ts.map +1 -0
  30. package/dist/tui/components/domain/ProjectList.js +54 -0
  31. package/dist/tui/components/domain/ProjectList.js.map +1 -0
  32. package/dist/tui/components/layout/Header.d.ts +7 -0
  33. package/dist/tui/components/layout/Header.d.ts.map +1 -0
  34. package/dist/tui/components/layout/Header.js +7 -0
  35. package/dist/tui/components/layout/Header.js.map +1 -0
  36. package/dist/tui/components/layout/MainPanel.d.ts +8 -0
  37. package/dist/tui/components/layout/MainPanel.d.ts.map +1 -0
  38. package/dist/tui/components/layout/MainPanel.js +5 -0
  39. package/dist/tui/components/layout/MainPanel.js.map +1 -0
  40. package/dist/tui/components/layout/Sidebar.d.ts +9 -0
  41. package/dist/tui/components/layout/Sidebar.d.ts.map +1 -0
  42. package/dist/tui/components/layout/Sidebar.js +11 -0
  43. package/dist/tui/components/layout/Sidebar.js.map +1 -0
  44. package/dist/tui/components/layout/StatusBar.d.ts +11 -0
  45. package/dist/tui/components/layout/StatusBar.d.ts.map +1 -0
  46. package/dist/tui/components/layout/StatusBar.js +8 -0
  47. package/dist/tui/components/layout/StatusBar.js.map +1 -0
  48. package/dist/tui/hooks/useDaemonConnection.d.ts +5 -0
  49. package/dist/tui/hooks/useDaemonConnection.d.ts.map +1 -0
  50. package/dist/tui/hooks/useDaemonConnection.js +24 -0
  51. package/dist/tui/hooks/useDaemonConnection.js.map +1 -0
  52. package/dist/tui/hooks/useIssues.d.ts +8 -0
  53. package/dist/tui/hooks/useIssues.d.ts.map +1 -0
  54. package/dist/tui/hooks/useIssues.js +36 -0
  55. package/dist/tui/hooks/useIssues.js.map +1 -0
  56. package/dist/tui/hooks/useNavigation.d.ts +11 -0
  57. package/dist/tui/hooks/useNavigation.d.ts.map +1 -0
  58. package/dist/tui/hooks/useNavigation.js +24 -0
  59. package/dist/tui/hooks/useNavigation.js.map +1 -0
  60. package/dist/tui/hooks/useProjects.d.ts +8 -0
  61. package/dist/tui/hooks/useProjects.d.ts.map +1 -0
  62. package/dist/tui/hooks/useProjects.js +36 -0
  63. package/dist/tui/hooks/useProjects.js.map +1 -0
  64. package/dist/tui/index.d.ts +2 -0
  65. package/dist/tui/index.d.ts.map +1 -0
  66. package/dist/tui/index.js +39 -0
  67. package/dist/tui/index.js.map +1 -0
  68. package/dist/tui/services/daemon-service.d.ts +26 -0
  69. package/dist/tui/services/daemon-service.d.ts.map +1 -0
  70. package/dist/tui/services/daemon-service.js +81 -0
  71. package/dist/tui/services/daemon-service.js.map +1 -0
  72. package/dist/tui/state/app-state.d.ts +78 -0
  73. package/dist/tui/state/app-state.d.ts.map +1 -0
  74. package/dist/tui/state/app-state.js +93 -0
  75. package/dist/tui/state/app-state.js.map +1 -0
  76. package/dist/tui/types/views.d.ts +11 -0
  77. package/dist/tui/types/views.d.ts.map +1 -0
  78. package/dist/tui/types/views.js +25 -0
  79. package/dist/tui/types/views.js.map +1 -0
  80. package/oclif.manifest.json +34 -1
  81. package/package.json +9 -3
package/bin/run.js CHANGED
@@ -1,4 +1,16 @@
1
- #!/usr/bin/env node
1
+ #!/usr/bin/env bun
2
2
  import { execute } from '@oclif/core'
3
3
 
4
- await execute({ dir: import.meta.url })
4
+ // Check if running with no arguments (or just --interactive flag)
5
+ const args = process.argv.slice(2)
6
+ const hasCommand = args.length > 0 && !args[0].startsWith('-')
7
+ const isHelp = args.includes('--help') || args.includes('-h')
8
+ const isInteractive = args.includes('--interactive') || args.includes('-i')
9
+
10
+ // Launch TUI mode if no command provided (or explicit --interactive flag)
11
+ if ((!hasCommand && !isHelp) || isInteractive) {
12
+ const { startTUI } = await import('../dist/tui/index.js')
13
+ await startTUI()
14
+ } else {
15
+ await execute({ dir: import.meta.url })
16
+ }
@@ -0,0 +1,13 @@
1
+ import { Command } from '@oclif/core';
2
+ export default class Start extends Command {
3
+ static description: string;
4
+ static examples: string[];
5
+ static flags: {
6
+ foreground: import("@oclif/core/interfaces").BooleanFlag<boolean>;
7
+ };
8
+ run(): Promise<void>;
9
+ private startForeground;
10
+ private startBackground;
11
+ private handleSpawnError;
12
+ }
13
+ //# sourceMappingURL=start.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"start.d.ts","sourceRoot":"","sources":["../../src/commands/start.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,OAAO,EAAS,MAAM,aAAa,CAAA;AAO5C,MAAM,CAAC,OAAO,OAAO,KAAM,SAAQ,OAAO;IACxC,OAAgB,WAAW,SAA2B;IAEtD,OAAgB,QAAQ,WAIvB;IAED,OAAgB,KAAK;;MAMpB;IAEY,GAAG,IAAI,OAAO,CAAC,IAAI,CAAC;YAqBnB,eAAe;YAoBf,eAAe;IAqB7B,OAAO,CAAC,gBAAgB;CASzB"}
@@ -0,0 +1,79 @@
1
+ import { spawn } from 'node:child_process';
2
+ import { Command, Flags } from '@oclif/core';
3
+ import { checkDaemonConnection } from '../daemon/check-daemon-connection.js';
4
+ import { daemonBinaryExists } from '../lib/start/daemon-binary-exists.js';
5
+ import { findDaemonBinary } from '../lib/start/find-daemon-binary.js';
6
+ import { waitForDaemon } from '../lib/start/wait-for-daemon.js';
7
+ export default class Start extends Command {
8
+ static description = 'Start the centy daemon';
9
+ static examples = [
10
+ '<%= config.bin %> start',
11
+ '<%= config.bin %> start --foreground',
12
+ '<%= config.bin %> start -f',
13
+ ];
14
+ static flags = {
15
+ foreground: Flags.boolean({
16
+ char: 'f',
17
+ description: 'Run daemon in foreground (blocks terminal)',
18
+ default: false,
19
+ }),
20
+ };
21
+ async run() {
22
+ const { flags } = await this.parse(Start);
23
+ const status = await checkDaemonConnection();
24
+ if (status.connected) {
25
+ this.log('Daemon is already running');
26
+ return;
27
+ }
28
+ const daemonPath = findDaemonBinary();
29
+ if (!daemonBinaryExists(daemonPath)) {
30
+ this.error(`Daemon binary not found at: ${daemonPath}`);
31
+ }
32
+ if (flags.foreground) {
33
+ await this.startForeground(daemonPath);
34
+ }
35
+ else {
36
+ await this.startBackground(daemonPath);
37
+ }
38
+ }
39
+ async startForeground(daemonPath) {
40
+ this.log('Starting daemon in foreground mode...');
41
+ const child = spawn(daemonPath, [], { stdio: 'inherit' });
42
+ child.on('error', error => {
43
+ this.handleSpawnError(error, daemonPath);
44
+ });
45
+ await new Promise((resolve, reject) => {
46
+ child.on('exit', code => {
47
+ if (code === 0) {
48
+ resolve();
49
+ }
50
+ else {
51
+ reject(new Error(`Daemon exited with code ${code}`));
52
+ }
53
+ });
54
+ });
55
+ }
56
+ async startBackground(daemonPath) {
57
+ this.log('Starting daemon in background...');
58
+ const child = spawn(daemonPath, [], { detached: true, stdio: 'ignore' });
59
+ child.on('error', error => {
60
+ this.handleSpawnError(error, daemonPath);
61
+ });
62
+ child.unref();
63
+ const started = await waitForDaemon();
64
+ if (started) {
65
+ this.log('Daemon started successfully');
66
+ }
67
+ else {
68
+ this.error('Daemon process started but is not responding. Check daemon logs for errors.');
69
+ }
70
+ }
71
+ handleSpawnError(error, daemonPath) {
72
+ if (error.code === 'ENOENT') {
73
+ this.error(`Could not find centy-daemon binary at: ${daemonPath}. ` +
74
+ 'Make sure the daemon is built and accessible.');
75
+ }
76
+ this.error(`Failed to start daemon: ${error.message}`);
77
+ }
78
+ }
79
+ //# sourceMappingURL=start.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"start.js","sourceRoot":"","sources":["../../src/commands/start.ts"],"names":[],"mappings":"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"}
@@ -1,5 +1,5 @@
1
1
  import { checkDaemonConnection } from '../daemon/check-daemon-connection.js';
2
- const EXCLUDED_COMMANDS = ['info', 'shutdown', 'restart'];
2
+ const EXCLUDED_COMMANDS = ['info', 'shutdown', 'restart', 'start'];
3
3
  const hook = async function (options) {
4
4
  if (EXCLUDED_COMMANDS.includes(options.Command.id)) {
5
5
  return;
@@ -1 +1 @@
1
- {"version":3,"file":"prerun.js","sourceRoot":"","sources":["../../src/hooks/prerun.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,qBAAqB,EAAE,MAAM,sCAAsC,CAAA;AAE5E,MAAM,iBAAiB,GAAG,CAAC,MAAM,EAAE,UAAU,EAAE,SAAS,CAAC,CAAA;AAEzD,MAAM,IAAI,GAAmB,KAAK,WAAW,OAAO;IAClD,IAAI,iBAAiB,CAAC,QAAQ,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,CAAC;QACnD,OAAM;IACR,CAAC;IAED,MAAM,gBAAgB,GAAG,MAAM,qBAAqB,EAAE,CAAA;IACtD,IAAI,CAAC,gBAAgB,CAAC,SAAS,EAAE,CAAC;QAChC,MAAM,YAAY,GAChB,gBAAgB,CAAC,KAAK,KAAK,IAAI,IAAI,gBAAgB,CAAC,KAAK,KAAK,SAAS;YACrE,CAAC,CAAC,gBAAgB,CAAC,KAAK;YACxB,CAAC,CAAC,6DAA6D,CAAA;QACnE,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,CAAA;IAC1B,CAAC;AACH,CAAC,CAAA;AAED,eAAe,IAAI,CAAA"}
1
+ {"version":3,"file":"prerun.js","sourceRoot":"","sources":["../../src/hooks/prerun.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,qBAAqB,EAAE,MAAM,sCAAsC,CAAA;AAE5E,MAAM,iBAAiB,GAAG,CAAC,MAAM,EAAE,UAAU,EAAE,SAAS,EAAE,OAAO,CAAC,CAAA;AAElE,MAAM,IAAI,GAAmB,KAAK,WAAW,OAAO;IAClD,IAAI,iBAAiB,CAAC,QAAQ,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,CAAC;QACnD,OAAM;IACR,CAAC;IAED,MAAM,gBAAgB,GAAG,MAAM,qBAAqB,EAAE,CAAA;IACtD,IAAI,CAAC,gBAAgB,CAAC,SAAS,EAAE,CAAC;QAChC,MAAM,YAAY,GAChB,gBAAgB,CAAC,KAAK,KAAK,IAAI,IAAI,gBAAgB,CAAC,KAAK,KAAK,SAAS;YACrE,CAAC,CAAC,gBAAgB,CAAC,KAAK;YACxB,CAAC,CAAC,6DAA6D,CAAA;QACnE,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,CAAA;IAC1B,CAAC;AACH,CAAC,CAAA;AAED,eAAe,IAAI,CAAA"}
@@ -0,0 +1,2 @@
1
+ export declare function daemonBinaryExists(path: string): boolean;
2
+ //# sourceMappingURL=daemon-binary-exists.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"daemon-binary-exists.d.ts","sourceRoot":"","sources":["../../../src/lib/start/daemon-binary-exists.ts"],"names":[],"mappings":"AAIA,wBAAgB,kBAAkB,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAMxD"}
@@ -0,0 +1,10 @@
1
+ import { existsSync } from 'node:fs';
2
+ const DAEMON_BINARY_NAME = 'centy-daemon';
3
+ export function daemonBinaryExists(path) {
4
+ // PATH lookup case - assume binary is accessible
5
+ if (path === DAEMON_BINARY_NAME) {
6
+ return true;
7
+ }
8
+ return existsSync(path);
9
+ }
10
+ //# sourceMappingURL=daemon-binary-exists.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"daemon-binary-exists.js","sourceRoot":"","sources":["../../../src/lib/start/daemon-binary-exists.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAA;AAEpC,MAAM,kBAAkB,GAAG,cAAc,CAAA;AAEzC,MAAM,UAAU,kBAAkB,CAAC,IAAY;IAC7C,iDAAiD;IACjD,IAAI,IAAI,KAAK,kBAAkB,EAAE,CAAC;QAChC,OAAO,IAAI,CAAA;IACb,CAAC;IACD,OAAO,UAAU,CAAC,IAAI,CAAC,CAAA;AACzB,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare function findDaemonBinary(): string;
2
+ //# sourceMappingURL=find-daemon-binary.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"find-daemon-binary.d.ts","sourceRoot":"","sources":["../../../src/lib/start/find-daemon-binary.ts"],"names":[],"mappings":"AAMA,wBAAgB,gBAAgB,IAAI,MAAM,CAgCzC"}
@@ -0,0 +1,25 @@
1
+ import { existsSync } from 'node:fs';
2
+ import { dirname, join } from 'node:path';
3
+ import { fileURLToPath } from 'node:url';
4
+ const DAEMON_BINARY_NAME = 'centy-daemon';
5
+ export function findDaemonBinary() {
6
+ // 1. Check CENTY_DAEMON_PATH environment variable
7
+ const envPath = process.env['CENTY_DAEMON_PATH'];
8
+ if (envPath !== undefined && existsSync(envPath)) {
9
+ return envPath;
10
+ }
11
+ // 2. Check same directory as CLI binary
12
+ const __dirname = dirname(fileURLToPath(import.meta.url));
13
+ const sameDirPath = join(__dirname, '..', '..', '..', DAEMON_BINARY_NAME);
14
+ if (existsSync(sameDirPath)) {
15
+ return sameDirPath;
16
+ }
17
+ // 3. Check development path (sibling repo)
18
+ const devPath = join(__dirname, '..', '..', '..', '..', 'centy-daemon', 'target', 'release', DAEMON_BINARY_NAME);
19
+ if (existsSync(devPath)) {
20
+ return devPath;
21
+ }
22
+ // 4. Fallback to PATH lookup (will be resolved by spawn)
23
+ return DAEMON_BINARY_NAME;
24
+ }
25
+ //# sourceMappingURL=find-daemon-binary.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"find-daemon-binary.js","sourceRoot":"","sources":["../../../src/lib/start/find-daemon-binary.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAA;AACpC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAA;AACzC,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAA;AAExC,MAAM,kBAAkB,GAAG,cAAc,CAAA;AAEzC,MAAM,UAAU,gBAAgB;IAC9B,kDAAkD;IAClD,MAAM,OAAO,GAAG,OAAO,CAAC,GAAG,CAAC,mBAAmB,CAAC,CAAA;IAChD,IAAI,OAAO,KAAK,SAAS,IAAI,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC;QACjD,OAAO,OAAO,CAAA;IAChB,CAAC;IAED,wCAAwC;IACxC,MAAM,SAAS,GAAG,OAAO,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAA;IACzD,MAAM,WAAW,GAAG,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,kBAAkB,CAAC,CAAA;IACzE,IAAI,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;QAC5B,OAAO,WAAW,CAAA;IACpB,CAAC;IAED,2CAA2C;IAC3C,MAAM,OAAO,GAAG,IAAI,CAClB,SAAS,EACT,IAAI,EACJ,IAAI,EACJ,IAAI,EACJ,IAAI,EACJ,cAAc,EACd,QAAQ,EACR,SAAS,EACT,kBAAkB,CACnB,CAAA;IACD,IAAI,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC;QACxB,OAAO,OAAO,CAAA;IAChB,CAAC;IAED,yDAAyD;IACzD,OAAO,kBAAkB,CAAA;AAC3B,CAAC"}
@@ -0,0 +1,7 @@
1
+ interface WaitOptions {
2
+ maxAttempts?: number;
3
+ delayMs?: number;
4
+ }
5
+ export declare function waitForDaemon(options?: WaitOptions): Promise<boolean>;
6
+ export {};
7
+ //# sourceMappingURL=wait-for-daemon.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"wait-for-daemon.d.ts","sourceRoot":"","sources":["../../../src/lib/start/wait-for-daemon.ts"],"names":[],"mappings":"AAKA,UAAU,WAAW;IACnB,WAAW,CAAC,EAAE,MAAM,CAAA;IACpB,OAAO,CAAC,EAAE,MAAM,CAAA;CACjB;AAED,wBAAsB,aAAa,CAAC,OAAO,CAAC,EAAE,WAAW,GAAG,OAAO,CAAC,OAAO,CAAC,CAkB3E"}
@@ -0,0 +1,20 @@
1
+ import { checkDaemonConnection } from '../../daemon/check-daemon-connection.js';
2
+ const DEFAULT_MAX_ATTEMPTS = 5;
3
+ const DEFAULT_DELAY_MS = 500;
4
+ export async function waitForDaemon(options) {
5
+ const maxAttempts = options !== undefined && options.maxAttempts !== undefined
6
+ ? options.maxAttempts
7
+ : DEFAULT_MAX_ATTEMPTS;
8
+ const delayMs = options !== undefined && options.delayMs !== undefined
9
+ ? options.delayMs
10
+ : DEFAULT_DELAY_MS;
11
+ for (let attempt = 0; attempt < maxAttempts; attempt++) {
12
+ await new Promise(resolve => setTimeout(resolve, delayMs));
13
+ const status = await checkDaemonConnection();
14
+ if (status.connected) {
15
+ return true;
16
+ }
17
+ }
18
+ return false;
19
+ }
20
+ //# sourceMappingURL=wait-for-daemon.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"wait-for-daemon.js","sourceRoot":"","sources":["../../../src/lib/start/wait-for-daemon.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,qBAAqB,EAAE,MAAM,yCAAyC,CAAA;AAE/E,MAAM,oBAAoB,GAAG,CAAC,CAAA;AAC9B,MAAM,gBAAgB,GAAG,GAAG,CAAA;AAO5B,MAAM,CAAC,KAAK,UAAU,aAAa,CAAC,OAAqB;IACvD,MAAM,WAAW,GACf,OAAO,KAAK,SAAS,IAAI,OAAO,CAAC,WAAW,KAAK,SAAS;QACxD,CAAC,CAAC,OAAO,CAAC,WAAW;QACrB,CAAC,CAAC,oBAAoB,CAAA;IAC1B,MAAM,OAAO,GACX,OAAO,KAAK,SAAS,IAAI,OAAO,CAAC,OAAO,KAAK,SAAS;QACpD,CAAC,CAAC,OAAO,CAAC,OAAO;QACjB,CAAC,CAAC,gBAAgB,CAAA;IAEtB,KAAK,IAAI,OAAO,GAAG,CAAC,EAAE,OAAO,GAAG,WAAW,EAAE,OAAO,EAAE,EAAE,CAAC;QACvD,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC,CAAA;QAC1D,MAAM,MAAM,GAAG,MAAM,qBAAqB,EAAE,CAAA;QAC5C,IAAI,MAAM,CAAC,SAAS,EAAE,CAAC;YACrB,OAAO,IAAI,CAAA;QACb,CAAC;IACH,CAAC;IACD,OAAO,KAAK,CAAA;AACd,CAAC"}
@@ -0,0 +1,6 @@
1
+ interface AppProps {
2
+ onExit: () => void;
3
+ }
4
+ export declare function App({ onExit }: AppProps): import("react").ReactNode;
5
+ export {};
6
+ //# sourceMappingURL=App.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"App.d.ts","sourceRoot":"","sources":["../../src/tui/App.tsx"],"names":[],"mappings":"AAcA,UAAU,QAAQ;IAChB,MAAM,EAAE,MAAM,IAAI,CAAA;CACnB;AAED,wBAAgB,GAAG,CAAC,EAAE,MAAM,EAAE,EAAE,QAAQ,6BA0DvC"}
@@ -0,0 +1,63 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "@opentui/react/jsx-runtime";
2
+ import { useKeyboard } from '@opentui/react';
3
+ import { Header } from './components/layout/Header.js';
4
+ import { Sidebar } from './components/layout/Sidebar.js';
5
+ import { StatusBar } from './components/layout/StatusBar.js';
6
+ import { ProjectList } from './components/domain/ProjectList.js';
7
+ import { IssueList } from './components/domain/IssueList.js';
8
+ import { MainPanel } from './components/layout/MainPanel.js';
9
+ import { useNavigation } from './hooks/useNavigation.js';
10
+ import { useDaemonConnection } from './hooks/useDaemonConnection.js';
11
+ import { useAppState } from './state/app-state.js';
12
+ import { SIDEBAR_VIEWS } from './types/views.js';
13
+ export function App({ onExit }) {
14
+ const { connected } = useDaemonConnection();
15
+ const { currentView, sidebarIndex, navigate, selectSidebarItem } = useNavigation();
16
+ const { state } = useAppState();
17
+ // Global keyboard shortcuts
18
+ useKeyboard((event) => {
19
+ // Quit
20
+ if (event.name === 'q') {
21
+ onExit();
22
+ return;
23
+ }
24
+ // Navigate sidebar with Tab or number keys
25
+ if (event.name === 'tab') {
26
+ const nextIndex = (sidebarIndex + 1) % SIDEBAR_VIEWS.length;
27
+ selectSidebarItem(nextIndex);
28
+ navigate(SIDEBAR_VIEWS[nextIndex]);
29
+ }
30
+ // Number keys for quick navigation
31
+ const numKey = parseInt(event.name);
32
+ if (numKey >= 1 && numKey <= SIDEBAR_VIEWS.length) {
33
+ selectSidebarItem(numKey - 1);
34
+ navigate(SIDEBAR_VIEWS[numKey - 1]);
35
+ }
36
+ });
37
+ const shortcuts = [
38
+ { key: 'j/k', label: 'navigate' },
39
+ { key: 'Enter', label: 'select' },
40
+ { key: 'Tab', label: 'switch view' },
41
+ { key: 'q', label: 'quit' },
42
+ ];
43
+ return (_jsxs("box", { flexDirection: "column", width: "100%", height: "100%", children: [_jsx(Header, { title: "Centy", daemonConnected: connected }), _jsxs("box", { flexGrow: 1, flexDirection: "row", children: [_jsx(Sidebar, { currentView: currentView, selectedIndex: sidebarIndex, onNavigate: navigate }), renderView(currentView)] }), _jsx(StatusBar, { shortcuts: shortcuts, error: state.error })] }));
44
+ }
45
+ function renderView(view) {
46
+ switch (view) {
47
+ case 'projects':
48
+ return _jsx(ProjectList, {});
49
+ case 'issues':
50
+ return _jsx(IssueList, {});
51
+ case 'docs':
52
+ return (_jsx(MainPanel, { title: "Docs", children: _jsx("text", { fg: "gray", children: "Docs view coming soon..." }) }));
53
+ case 'assets':
54
+ return (_jsx(MainPanel, { title: "Assets", children: _jsx("text", { fg: "gray", children: "Assets view coming soon..." }) }));
55
+ case 'config':
56
+ return (_jsx(MainPanel, { title: "Config", children: _jsx("text", { fg: "gray", children: "Config view coming soon..." }) }));
57
+ case 'daemon':
58
+ return (_jsx(MainPanel, { title: "Daemon", children: _jsx("text", { fg: "gray", children: "Daemon control coming soon..." }) }));
59
+ default:
60
+ return _jsx(ProjectList, {});
61
+ }
62
+ }
63
+ //# sourceMappingURL=App.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"App.js","sourceRoot":"","sources":["../../src/tui/App.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export declare function IssueList(): import("react").ReactNode;
2
+ //# sourceMappingURL=IssueList.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"IssueList.d.ts","sourceRoot":"","sources":["../../../../src/tui/components/domain/IssueList.tsx"],"names":[],"mappings":"AAqBA,wBAAgB,SAAS,8BAmExB"}
@@ -0,0 +1,52 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "@opentui/react/jsx-runtime";
2
+ import { useState } from 'react';
3
+ import { useKeyboard } from '@opentui/react';
4
+ import { MainPanel } from '../layout/MainPanel.js';
5
+ import { useIssues } from '../../hooks/useIssues.js';
6
+ import { useAppState } from '../../state/app-state.js';
7
+ function getPriorityColor(priority) {
8
+ if (priority === 1)
9
+ return 'red';
10
+ if (priority === 2)
11
+ return 'yellow';
12
+ return 'green';
13
+ }
14
+ function getPriorityLabel(priority, label) {
15
+ if (label)
16
+ return label;
17
+ if (priority === 1)
18
+ return 'high';
19
+ if (priority === 2)
20
+ return 'med';
21
+ return 'low';
22
+ }
23
+ export function IssueList() {
24
+ const { issues, isLoading } = useIssues();
25
+ const { state } = useAppState();
26
+ const [selectedIndex, setSelectedIndex] = useState(0);
27
+ useKeyboard((event) => {
28
+ if (event.name === 'j' || event.name === 'down') {
29
+ setSelectedIndex((prev) => Math.min(prev + 1, issues.length - 1));
30
+ }
31
+ else if (event.name === 'k' || event.name === 'up') {
32
+ setSelectedIndex((prev) => Math.max(prev - 1, 0));
33
+ }
34
+ });
35
+ const projectName = state.selectedProjectPath?.split('/').pop() || 'Project';
36
+ if (!state.selectedProjectPath) {
37
+ return (_jsx(MainPanel, { title: "Issues", children: _jsx("text", { fg: "gray", children: "Select a project first." }) }));
38
+ }
39
+ if (isLoading) {
40
+ return (_jsx(MainPanel, { title: `Issues - ${projectName}`, children: _jsx("text", { fg: "gray", children: "Loading issues..." }) }));
41
+ }
42
+ if (issues.length === 0) {
43
+ return (_jsxs(MainPanel, { title: `Issues - ${projectName}`, children: [_jsx("text", { fg: "gray", children: "No issues found." }), _jsx("text", { fg: "gray", children: "Press `n` to create a new issue." })] }));
44
+ }
45
+ return (_jsx(MainPanel, { title: `Issues - ${projectName}`, children: issues.map((issue, index) => {
46
+ const isSelected = index === selectedIndex;
47
+ const priorityColor = getPriorityColor(issue.metadata.priority);
48
+ const priorityLabel = getPriorityLabel(issue.metadata.priority, issue.metadata.priorityLabel);
49
+ return (_jsxs("box", { height: 1, flexDirection: "row", children: [_jsx("text", { bg: isSelected ? 'gray' : undefined, children: isSelected ? _jsx("b", { children: '>' }) : ' ' }), _jsxs("text", { fg: "cyan", children: ["#", issue.displayNumber] }), _jsx("text", { children: " " }), _jsxs("text", { fg: priorityColor, children: ["[", priorityLabel, "]"] }), _jsx("text", { children: " " }), _jsxs("text", { fg: "gray", children: ["[", issue.metadata.status, "]"] }), _jsx("text", { children: " " }), _jsx("text", { bg: isSelected ? 'gray' : undefined, children: issue.title })] }, issue.id));
50
+ }) }));
51
+ }
52
+ //# sourceMappingURL=IssueList.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"IssueList.js","sourceRoot":"","sources":["../../../../src/tui/components/domain/IssueList.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export declare function ProjectList(): import("react").ReactNode;
2
+ //# sourceMappingURL=ProjectList.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ProjectList.d.ts","sourceRoot":"","sources":["../../../../src/tui/components/domain/ProjectList.tsx"],"names":[],"mappings":"AAyCA,wBAAgB,WAAW,8BAuE1B"}
@@ -0,0 +1,54 @@
1
+ import { jsxs as _jsxs, jsx as _jsx } from "@opentui/react/jsx-runtime";
2
+ import { useState, useRef, useEffect } from 'react';
3
+ import { useKeyboard } from '@opentui/react';
4
+ import { MainPanel } from '../layout/MainPanel.js';
5
+ import { useProjects } from '../../hooks/useProjects.js';
6
+ import { useNavigation } from '../../hooks/useNavigation.js';
7
+ const ITEM_HEIGHT = 2;
8
+ function ProjectItem({ project, isSelected }) {
9
+ const name = project.name || project.path.split('/').pop() || project.path;
10
+ return (_jsxs("box", { height: ITEM_HEIGHT, flexDirection: "column", children: [_jsx("box", { flexDirection: "row", children: _jsx("text", { bg: isSelected ? 'gray' : undefined, children: isSelected ? (_jsxs("b", { children: ['>', " ", name] })) : (` ${name}`) }) }), _jsxs("box", { flexDirection: "row", paddingLeft: 3, children: [_jsxs("text", { fg: "gray", children: [project.issueCount, " issues, ", project.docCount, " docs"] }), !project.initialized && _jsx("text", { fg: "yellow", children: " (not initialized)" })] })] }, project.path));
11
+ }
12
+ export function ProjectList() {
13
+ const { projects, isLoading, selectProject } = useProjects();
14
+ const { navigate } = useNavigation();
15
+ const [selectedIndex, setSelectedIndex] = useState(0);
16
+ const scrollBoxRef = useRef(null);
17
+ useEffect(() => {
18
+ if (scrollBoxRef.current) {
19
+ const targetScrollTop = selectedIndex * ITEM_HEIGHT;
20
+ const viewportHeight = scrollBoxRef.current.viewport.height;
21
+ const currentScrollTop = scrollBoxRef.current.scrollTop;
22
+ if (targetScrollTop < currentScrollTop) {
23
+ scrollBoxRef.current.scrollTo(targetScrollTop);
24
+ }
25
+ else if (targetScrollTop + ITEM_HEIGHT >
26
+ currentScrollTop + viewportHeight) {
27
+ scrollBoxRef.current.scrollTo(targetScrollTop - viewportHeight + ITEM_HEIGHT);
28
+ }
29
+ }
30
+ }, [selectedIndex]);
31
+ useKeyboard((event) => {
32
+ if (event.name === 'j' || event.name === 'down') {
33
+ setSelectedIndex((prev) => Math.min(prev + 1, projects.length - 1));
34
+ }
35
+ else if (event.name === 'k' || event.name === 'up') {
36
+ setSelectedIndex((prev) => Math.max(prev - 1, 0));
37
+ }
38
+ else if (event.name === 'return') {
39
+ const project = projects[selectedIndex];
40
+ if (project) {
41
+ selectProject(project.path);
42
+ navigate('issues');
43
+ }
44
+ }
45
+ });
46
+ if (isLoading) {
47
+ return (_jsx(MainPanel, { title: "Projects", children: _jsx("text", { fg: "gray", children: "Loading projects..." }) }));
48
+ }
49
+ if (projects.length === 0) {
50
+ return (_jsxs(MainPanel, { title: "Projects", children: [_jsx("text", { fg: "gray", children: "No projects found." }), _jsx("text", { fg: "gray", children: "Use `centy register project` to add a project." })] }));
51
+ }
52
+ return (_jsx(MainPanel, { title: "Projects", children: _jsx("scrollbox", { ref: scrollBoxRef, flexGrow: 1, scrollY: true, children: projects.map((project, index) => (_jsx(ProjectItem, { project: project, isSelected: index === selectedIndex }, project.path))) }) }));
53
+ }
54
+ //# sourceMappingURL=ProjectList.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ProjectList.js","sourceRoot":"","sources":["../../../../src/tui/components/domain/ProjectList.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ interface HeaderProps {
2
+ title: string;
3
+ daemonConnected: boolean;
4
+ }
5
+ export declare function Header({ title, daemonConnected }: HeaderProps): import("react").ReactNode;
6
+ export {};
7
+ //# sourceMappingURL=Header.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Header.d.ts","sourceRoot":"","sources":["../../../../src/tui/components/layout/Header.tsx"],"names":[],"mappings":"AAAA,UAAU,WAAW;IACnB,KAAK,EAAE,MAAM,CAAA;IACb,eAAe,EAAE,OAAO,CAAA;CACzB;AAED,wBAAgB,MAAM,CAAC,EAAE,KAAK,EAAE,eAAe,EAAE,EAAE,WAAW,6BAqB7D"}
@@ -0,0 +1,7 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "@opentui/react/jsx-runtime";
2
+ export function Header({ title, daemonConnected }) {
3
+ const statusColor = daemonConnected ? 'green' : 'red';
4
+ const statusText = daemonConnected ? 'Connected' : 'Disconnected';
5
+ return (_jsxs("box", { height: 1, width: "100%", flexDirection: "row", justifyContent: "space-between", children: [_jsx("text", { fg: "cyan", children: _jsx("b", { children: title }) }), _jsxs("box", { flexDirection: "row", children: [_jsx("text", { fg: statusColor, children: "\u25CF" }), _jsx("text", { children: " Daemon: " }), _jsx("text", { fg: statusColor, children: statusText })] })] }));
6
+ }
7
+ //# sourceMappingURL=Header.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Header.js","sourceRoot":"","sources":["../../../../src/tui/components/layout/Header.tsx"],"names":[],"mappings":";AAKA,MAAM,UAAU,MAAM,CAAC,EAAE,KAAK,EAAE,eAAe,EAAe;IAC5D,MAAM,WAAW,GAAG,eAAe,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,KAAK,CAAA;IACrD,MAAM,UAAU,GAAG,eAAe,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,cAAc,CAAA;IAEjE,OAAO,CACL,eACE,MAAM,EAAE,CAAC,EACT,KAAK,EAAC,MAAM,EACZ,aAAa,EAAC,KAAK,EACnB,cAAc,EAAC,eAAe,aAE9B,eAAM,EAAE,EAAC,MAAM,YACb,sBAAI,KAAK,GAAK,GACT,EACP,eAAK,aAAa,EAAC,KAAK,aACtB,eAAM,EAAE,EAAE,WAAW,uBAAU,EAC/B,uCAAsB,EACtB,eAAM,EAAE,EAAE,WAAW,YAAG,UAAU,GAAQ,IACtC,IACF,CACP,CAAA;AACH,CAAC"}
@@ -0,0 +1,8 @@
1
+ import type { ReactNode } from 'react';
2
+ interface MainPanelProps {
3
+ title: string;
4
+ children: ReactNode;
5
+ }
6
+ export declare function MainPanel({ title, children }: MainPanelProps): ReactNode;
7
+ export {};
8
+ //# sourceMappingURL=MainPanel.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"MainPanel.d.ts","sourceRoot":"","sources":["../../../../src/tui/components/layout/MainPanel.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,OAAO,CAAA;AAEtC,UAAU,cAAc;IACtB,KAAK,EAAE,MAAM,CAAA;IACb,QAAQ,EAAE,SAAS,CAAA;CACpB;AAED,wBAAgB,SAAS,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,EAAE,cAAc,aAmB5D"}
@@ -0,0 +1,5 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "@opentui/react/jsx-runtime";
2
+ export function MainPanel({ title, children }) {
3
+ return (_jsxs("box", { flexGrow: 1, flexDirection: "column", borderStyle: "single", children: [_jsx("box", { height: 1, justifyContent: "center", children: _jsx("text", { children: _jsx("b", { children: title }) }) }), _jsx("box", { flexGrow: 1, flexDirection: "column", paddingTop: 1, paddingLeft: 1, paddingRight: 1, children: children })] }));
4
+ }
5
+ //# sourceMappingURL=MainPanel.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"MainPanel.js","sourceRoot":"","sources":["../../../../src/tui/components/layout/MainPanel.tsx"],"names":[],"mappings":";AAOA,MAAM,UAAU,SAAS,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAkB;IAC3D,OAAO,CACL,eAAK,QAAQ,EAAE,CAAC,EAAE,aAAa,EAAC,QAAQ,EAAC,WAAW,EAAC,QAAQ,aAC3D,cAAK,MAAM,EAAE,CAAC,EAAE,cAAc,EAAC,QAAQ,YACrC,yBACE,sBAAI,KAAK,GAAK,GACT,GACH,EACN,cACE,QAAQ,EAAE,CAAC,EACX,aAAa,EAAC,QAAQ,EACtB,UAAU,EAAE,CAAC,EACb,WAAW,EAAE,CAAC,EACd,YAAY,EAAE,CAAC,YAEd,QAAQ,GACL,IACF,CACP,CAAA;AACH,CAAC"}
@@ -0,0 +1,9 @@
1
+ import type { ViewId } from '../../types/views.js';
2
+ interface SidebarProps {
3
+ currentView: ViewId;
4
+ selectedIndex: number;
5
+ onNavigate: (view: ViewId) => void;
6
+ }
7
+ export declare function Sidebar({ currentView, selectedIndex }: SidebarProps): import("react").ReactNode;
8
+ export {};
9
+ //# sourceMappingURL=Sidebar.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Sidebar.d.ts","sourceRoot":"","sources":["../../../../src/tui/components/layout/Sidebar.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,sBAAsB,CAAA;AAGlD,UAAU,YAAY;IACpB,WAAW,EAAE,MAAM,CAAA;IACnB,aAAa,EAAE,MAAM,CAAA;IACrB,UAAU,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,CAAA;CACnC;AAED,wBAAgB,OAAO,CAAC,EAAE,WAAW,EAAE,aAAa,EAAE,EAAE,YAAY,6BA4BnE"}
@@ -0,0 +1,11 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "@opentui/react/jsx-runtime";
2
+ import { SIDEBAR_VIEWS, VIEW_LABELS } from '../../types/views.js';
3
+ export function Sidebar({ currentView, selectedIndex }) {
4
+ return (_jsxs("box", { width: 20, flexDirection: "column", borderStyle: "single", children: [_jsx("box", { height: 1, justifyContent: "center", children: _jsx("text", { children: _jsx("b", { children: "Navigation" }) }) }), _jsx("box", { flexDirection: "column", paddingTop: 1, children: SIDEBAR_VIEWS.map((view, index) => {
5
+ const isSelected = index === selectedIndex;
6
+ const isCurrent = view === currentView;
7
+ const label = `${isSelected ? '>' : ' '} ${VIEW_LABELS[view]}`;
8
+ return (_jsx("box", { height: 1, children: _jsx("text", { fg: isCurrent ? 'cyan' : isSelected ? 'white' : 'gray', bg: isSelected ? 'gray' : undefined, children: isCurrent ? _jsx("b", { children: label }) : label }) }, view));
9
+ }) })] }));
10
+ }
11
+ //# sourceMappingURL=Sidebar.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Sidebar.js","sourceRoot":"","sources":["../../../../src/tui/components/layout/Sidebar.tsx"],"names":[],"mappings":";AACA,OAAO,EAAE,aAAa,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAA;AAQjE,MAAM,UAAU,OAAO,CAAC,EAAE,WAAW,EAAE,aAAa,EAAgB;IAClE,OAAO,CACL,eAAK,KAAK,EAAE,EAAE,EAAE,aAAa,EAAC,QAAQ,EAAC,WAAW,EAAC,QAAQ,aACzD,cAAK,MAAM,EAAE,CAAC,EAAE,cAAc,EAAC,QAAQ,YACrC,yBACE,qCAAiB,GACZ,GACH,EACN,cAAK,aAAa,EAAC,QAAQ,EAAC,UAAU,EAAE,CAAC,YACtC,aAAa,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE;oBACjC,MAAM,UAAU,GAAG,KAAK,KAAK,aAAa,CAAA;oBAC1C,MAAM,SAAS,GAAG,IAAI,KAAK,WAAW,CAAA;oBACtC,MAAM,KAAK,GAAG,GAAG,UAAU,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,IAAI,WAAW,CAAC,IAAI,CAAC,EAAE,CAAA;oBAE9D,OAAO,CACL,cAAgB,MAAM,EAAE,CAAC,YACvB,eACE,EAAE,EAAE,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,EACtD,EAAE,EAAE,UAAU,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,SAAS,YAElC,SAAS,CAAC,CAAC,CAAC,sBAAI,KAAK,GAAK,CAAC,CAAC,CAAC,KAAK,GAC9B,IANC,IAAI,CAOR,CACP,CAAA;gBACH,CAAC,CAAC,GACE,IACF,CACP,CAAA;AACH,CAAC"}
@@ -0,0 +1,11 @@
1
+ interface Shortcut {
2
+ key: string;
3
+ label: string;
4
+ }
5
+ interface StatusBarProps {
6
+ shortcuts: Shortcut[];
7
+ error?: string | null;
8
+ }
9
+ export declare function StatusBar({ shortcuts, error }: StatusBarProps): import("react").ReactNode;
10
+ export {};
11
+ //# sourceMappingURL=StatusBar.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"StatusBar.d.ts","sourceRoot":"","sources":["../../../../src/tui/components/layout/StatusBar.tsx"],"names":[],"mappings":"AAAA,UAAU,QAAQ;IAChB,GAAG,EAAE,MAAM,CAAA;IACX,KAAK,EAAE,MAAM,CAAA;CACd;AAED,UAAU,cAAc;IACtB,SAAS,EAAE,QAAQ,EAAE,CAAA;IACrB,KAAK,CAAC,EAAE,MAAM,GAAG,IAAI,CAAA;CACtB;AAED,wBAAgB,SAAS,CAAC,EAAE,SAAS,EAAE,KAAK,EAAE,EAAE,cAAc,6BAqB7D"}
@@ -0,0 +1,8 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "@opentui/react/jsx-runtime";
2
+ export function StatusBar({ shortcuts, error }) {
3
+ if (error) {
4
+ return (_jsx("box", { height: 1, width: "100%", children: _jsx("text", { fg: "red", children: error }) }));
5
+ }
6
+ return (_jsx("box", { height: 1, width: "100%", flexDirection: "row", gap: 2, children: shortcuts.map((shortcut, index) => (_jsxs("box", { flexDirection: "row", children: [_jsx("text", { fg: "yellow", children: _jsx("b", { children: shortcut.key }) }), _jsxs("text", { children: [": ", shortcut.label] })] }, index))) }));
7
+ }
8
+ //# sourceMappingURL=StatusBar.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"StatusBar.js","sourceRoot":"","sources":["../../../../src/tui/components/layout/StatusBar.tsx"],"names":[],"mappings":";AAUA,MAAM,UAAU,SAAS,CAAC,EAAE,SAAS,EAAE,KAAK,EAAkB;IAC5D,IAAI,KAAK,EAAE,CAAC;QACV,OAAO,CACL,cAAK,MAAM,EAAE,CAAC,EAAE,KAAK,EAAC,MAAM,YAC1B,eAAM,EAAE,EAAC,KAAK,YAAE,KAAK,GAAQ,GACzB,CACP,CAAA;IACH,CAAC;IAED,OAAO,CACL,cAAK,MAAM,EAAE,CAAC,EAAE,KAAK,EAAC,MAAM,EAAC,aAAa,EAAC,KAAK,EAAC,GAAG,EAAE,CAAC,YACpD,SAAS,CAAC,GAAG,CAAC,CAAC,QAAQ,EAAE,KAAK,EAAE,EAAE,CAAC,CAClC,eAAiB,aAAa,EAAC,KAAK,aAClC,eAAM,EAAE,EAAC,QAAQ,YACf,sBAAI,QAAQ,CAAC,GAAG,GAAK,GAChB,EACP,iCAAS,QAAQ,CAAC,KAAK,IAAQ,KAJvB,KAAK,CAKT,CACP,CAAC,GACE,CACP,CAAA;AACH,CAAC"}
@@ -0,0 +1,5 @@
1
+ export declare function useDaemonConnection(): {
2
+ connected: boolean;
3
+ retry: () => Promise<void>;
4
+ };
5
+ //# sourceMappingURL=useDaemonConnection.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useDaemonConnection.d.ts","sourceRoot":"","sources":["../../../src/tui/hooks/useDaemonConnection.ts"],"names":[],"mappings":"AAIA,wBAAgB,mBAAmB;;;EAsBlC"}