@pennyfarthing/cyclist 10.3.1 → 11.0.0-alpha.0

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 (271) hide show
  1. package/dist/api/agent-load.d.ts +1 -2
  2. package/dist/api/agent-load.d.ts.map +1 -1
  3. package/dist/api/agent-load.js +2 -123
  4. package/dist/api/agent-load.js.map +1 -1
  5. package/dist/api/audit-log.d.ts +1 -17
  6. package/dist/api/audit-log.d.ts.map +1 -1
  7. package/dist/api/audit-log.js +2 -162
  8. package/dist/api/audit-log.js.map +1 -1
  9. package/dist/api/background-tasks.d.ts +1 -26
  10. package/dist/api/background-tasks.d.ts.map +1 -1
  11. package/dist/api/background-tasks.js +2 -55
  12. package/dist/api/background-tasks.js.map +1 -1
  13. package/dist/api/bell.d.ts +1 -18
  14. package/dist/api/bell.d.ts.map +1 -1
  15. package/dist/api/bell.js +2 -33
  16. package/dist/api/bell.js.map +1 -1
  17. package/dist/api/code-markers.d.ts +1 -8
  18. package/dist/api/code-markers.d.ts.map +1 -1
  19. package/dist/api/code-markers.js +2 -61
  20. package/dist/api/code-markers.js.map +1 -1
  21. package/dist/api/complexity.d.ts +1 -2
  22. package/dist/api/complexity.d.ts.map +1 -1
  23. package/dist/api/complexity.js +2 -46
  24. package/dist/api/complexity.js.map +1 -1
  25. package/dist/api/context.d.ts +1 -37
  26. package/dist/api/context.d.ts.map +1 -1
  27. package/dist/api/context.js +2 -143
  28. package/dist/api/context.js.map +1 -1
  29. package/dist/api/dead-code.d.ts +1 -2
  30. package/dist/api/dead-code.d.ts.map +1 -1
  31. package/dist/api/dead-code.js +2 -69
  32. package/dist/api/dead-code.js.map +1 -1
  33. package/dist/api/dependencies.d.ts +1 -2
  34. package/dist/api/dependencies.d.ts.map +1 -1
  35. package/dist/api/dependencies.js +2 -42
  36. package/dist/api/dependencies.js.map +1 -1
  37. package/dist/api/evaluation.d.ts +1 -19
  38. package/dist/api/evaluation.d.ts.map +1 -1
  39. package/dist/api/evaluation.js +2 -127
  40. package/dist/api/evaluation.js.map +1 -1
  41. package/dist/api/file-browser.d.ts +1 -8
  42. package/dist/api/file-browser.d.ts.map +1 -1
  43. package/dist/api/file-browser.js +2 -114
  44. package/dist/api/file-browser.js.map +1 -1
  45. package/dist/api/git.d.ts +1 -46
  46. package/dist/api/git.d.ts.map +1 -1
  47. package/dist/api/git.js +2 -354
  48. package/dist/api/git.js.map +1 -1
  49. package/dist/api/health-score.d.ts +1 -2
  50. package/dist/api/health-score.d.ts.map +1 -1
  51. package/dist/api/health-score.js +2 -46
  52. package/dist/api/health-score.js.map +1 -1
  53. package/dist/api/hook-request.d.ts +1 -40
  54. package/dist/api/hook-request.d.ts.map +1 -1
  55. package/dist/api/hook-request.js +2 -277
  56. package/dist/api/hook-request.js.map +1 -1
  57. package/dist/api/hotspots.d.ts +1 -2
  58. package/dist/api/hotspots.d.ts.map +1 -1
  59. package/dist/api/hotspots.js +2 -61
  60. package/dist/api/hotspots.js.map +1 -1
  61. package/dist/api/identity.d.ts +1 -16
  62. package/dist/api/identity.d.ts.map +1 -1
  63. package/dist/api/identity.js +2 -78
  64. package/dist/api/identity.js.map +1 -1
  65. package/dist/api/index.d.ts +1 -34
  66. package/dist/api/index.d.ts.map +1 -1
  67. package/dist/api/index.js +2 -44
  68. package/dist/api/index.js.map +1 -1
  69. package/dist/api/mode.d.ts +1 -22
  70. package/dist/api/mode.d.ts.map +1 -1
  71. package/dist/api/mode.js +2 -37
  72. package/dist/api/mode.js.map +1 -1
  73. package/dist/api/otlp.d.ts +1 -2
  74. package/dist/api/otlp.d.ts.map +1 -1
  75. package/dist/api/otlp.js +2 -46
  76. package/dist/api/otlp.js.map +1 -1
  77. package/dist/api/permissions.d.ts +1 -15
  78. package/dist/api/permissions.d.ts.map +1 -1
  79. package/dist/api/permissions.js +2 -66
  80. package/dist/api/permissions.js.map +1 -1
  81. package/dist/api/persona.d.ts +1 -8
  82. package/dist/api/persona.d.ts.map +1 -1
  83. package/dist/api/persona.js +2 -67
  84. package/dist/api/persona.js.map +1 -1
  85. package/dist/api/portrait.d.ts +1 -5
  86. package/dist/api/portrait.d.ts.map +1 -1
  87. package/dist/api/portrait.js +2 -27
  88. package/dist/api/portrait.js.map +1 -1
  89. package/dist/api/settings.d.ts +1 -53
  90. package/dist/api/settings.d.ts.map +1 -1
  91. package/dist/api/settings.js +2 -464
  92. package/dist/api/settings.js.map +1 -1
  93. package/dist/api/spans.d.ts +1 -16
  94. package/dist/api/spans.d.ts.map +1 -1
  95. package/dist/api/spans.js +2 -244
  96. package/dist/api/spans.js.map +1 -1
  97. package/dist/api/stats.d.ts +1 -12
  98. package/dist/api/stats.d.ts.map +1 -1
  99. package/dist/api/stats.js +2 -84
  100. package/dist/api/stats.js.map +1 -1
  101. package/dist/api/story.d.ts +1 -2
  102. package/dist/api/story.d.ts.map +1 -1
  103. package/dist/api/story.js +2 -14
  104. package/dist/api/story.js.map +1 -1
  105. package/dist/api/telemetry.d.ts +1 -18
  106. package/dist/api/telemetry.d.ts.map +1 -1
  107. package/dist/api/telemetry.js +2 -164
  108. package/dist/api/telemetry.js.map +1 -1
  109. package/dist/api/theme-agents.d.ts +1 -60
  110. package/dist/api/theme-agents.d.ts.map +1 -1
  111. package/dist/api/theme-agents.js +2 -213
  112. package/dist/api/theme-agents.js.map +1 -1
  113. package/dist/api/todos.d.ts +1 -32
  114. package/dist/api/todos.d.ts.map +1 -1
  115. package/dist/api/todos.js +2 -43
  116. package/dist/api/todos.js.map +1 -1
  117. package/dist/api/token-stats.d.ts +1 -7
  118. package/dist/api/token-stats.d.ts.map +1 -1
  119. package/dist/api/token-stats.js +2 -35
  120. package/dist/api/token-stats.js.map +1 -1
  121. package/dist/api/welcome.d.ts +1 -21
  122. package/dist/api/welcome.d.ts.map +1 -1
  123. package/dist/api/welcome.js +2 -34
  124. package/dist/api/welcome.js.map +1 -1
  125. package/dist/env.d.ts +6 -0
  126. package/dist/env.d.ts.map +1 -0
  127. package/dist/env.js +10 -0
  128. package/dist/env.js.map +1 -0
  129. package/dist/focus.d.ts +53 -0
  130. package/dist/focus.d.ts.map +1 -0
  131. package/dist/focus.js +122 -0
  132. package/dist/focus.js.map +1 -0
  133. package/dist/menu-builder.d.ts.map +1 -1
  134. package/dist/menu-builder.js +0 -1
  135. package/dist/menu-builder.js.map +1 -1
  136. package/dist/public/css/react.css +1 -1
  137. package/dist/public/js/react/react.js +51 -59
  138. package/dist/server.d.ts +16 -85
  139. package/dist/server.d.ts.map +1 -1
  140. package/dist/server.js +38 -409
  141. package/dist/server.js.map +1 -1
  142. package/dist/sprint-data.d.ts +1 -1
  143. package/dist/sprint-data.d.ts.map +1 -1
  144. package/dist/sprint-data.js +29 -8
  145. package/dist/sprint-data.js.map +1 -1
  146. package/dist/websocket.d.ts +2 -0
  147. package/dist/websocket.d.ts.map +1 -1
  148. package/dist/websocket.js +45 -78
  149. package/dist/websocket.js.map +1 -1
  150. package/package.json +33 -35
  151. package/portraits/hogans-heroes/large/burkhalter-35312.png +0 -0
  152. package/portraits/hogans-heroes/large/carter-34352.png +0 -0
  153. package/portraits/hogans-heroes/large/hochstetter-45314.png +0 -0
  154. package/portraits/hogans-heroes/large/hogan-44541.png +0 -0
  155. package/portraits/hogans-heroes/large/kinch-35241.png +0 -0
  156. package/portraits/hogans-heroes/large/klink-23434.png +0 -0
  157. package/portraits/hogans-heroes/large/lebeau-45443.png +0 -0
  158. package/portraits/hogans-heroes/large/marya-53543.png +0 -0
  159. package/portraits/hogans-heroes/large/newkirk-54432.png +0 -0
  160. package/portraits/hogans-heroes/large/schultz-42453.png +0 -0
  161. package/portraits/hogans-heroes/large/underground-55131.png +0 -0
  162. package/portraits/hogans-heroes/medium/burkhalter-35312.png +0 -0
  163. package/portraits/hogans-heroes/medium/carter-34352.png +0 -0
  164. package/portraits/hogans-heroes/medium/hochstetter-45314.png +0 -0
  165. package/portraits/hogans-heroes/medium/hogan-44541.png +0 -0
  166. package/portraits/hogans-heroes/medium/kinch-35241.png +0 -0
  167. package/portraits/hogans-heroes/medium/klink-23434.png +0 -0
  168. package/portraits/hogans-heroes/medium/lebeau-45443.png +0 -0
  169. package/portraits/hogans-heroes/medium/marya-53543.png +0 -0
  170. package/portraits/hogans-heroes/medium/newkirk-54432.png +0 -0
  171. package/portraits/hogans-heroes/medium/schultz-42453.png +0 -0
  172. package/portraits/hogans-heroes/medium/underground-55131.png +0 -0
  173. package/portraits/monty-python/large/announcer-44441.png +0 -0
  174. package/portraits/monty-python/large/arguer-35412.png +0 -0
  175. package/portraits/monty-python/large/bicycle-repair-man-35241.png +0 -0
  176. package/portraits/monty-python/large/colonel-35423.png +0 -0
  177. package/portraits/monty-python/large/counsellor-45341.png +0 -0
  178. package/portraits/monty-python/large/gumbys-23524.png +0 -0
  179. package/portraits/monty-python/large/nudge-43533.png +0 -0
  180. package/portraits/monty-python/large/praline-45413.png +0 -0
  181. package/portraits/monty-python/large/silly-walks-55322.png +0 -0
  182. package/portraits/monty-python/large/wensleydale-54451.png +0 -0
  183. package/portraits/monty-python/large/xim-nez-43534.png +0 -0
  184. package/portraits/monty-python/medium/announcer-44441.png +0 -0
  185. package/portraits/monty-python/medium/arguer-35412.png +0 -0
  186. package/portraits/monty-python/medium/bicycle-repair-man-35241.png +0 -0
  187. package/portraits/monty-python/medium/colonel-35423.png +0 -0
  188. package/portraits/monty-python/medium/counsellor-45341.png +0 -0
  189. package/portraits/monty-python/medium/gumbys-23524.png +0 -0
  190. package/portraits/monty-python/medium/nudge-43533.png +0 -0
  191. package/portraits/monty-python/medium/praline-45413.png +0 -0
  192. package/portraits/monty-python/medium/silly-walks-55322.png +0 -0
  193. package/portraits/monty-python/medium/wensleydale-54451.png +0 -0
  194. package/portraits/monty-python/medium/xim-nez-43534.png +0 -0
  195. package/portraits/stephen-king/large/andy-55231.png +0 -0
  196. package/portraits/stephen-king/large/christine-25112.png +0 -0
  197. package/portraits/stephen-king/large/danny-53243.png +0 -0
  198. package/portraits/stephen-king/large/flagg-55311.png +0 -0
  199. package/portraits/stephen-king/large/gaunt-54421.png +0 -0
  200. package/portraits/stephen-king/large/jack-44224.png +0 -0
  201. package/portraits/stephen-king/large/johnny-44353.png +0 -0
  202. package/portraits/stephen-king/large/margaret-15415.png +0 -0
  203. package/portraits/stephen-king/large/paul-45233.png +0 -0
  204. package/portraits/stephen-king/large/pennywise-54411.png +0 -0
  205. package/portraits/stephen-king/large/roland-35121.png +0 -0
  206. package/portraits/stephen-king/medium/andy-55231.png +0 -0
  207. package/portraits/stephen-king/medium/christine-25112.png +0 -0
  208. package/portraits/stephen-king/medium/danny-53243.png +0 -0
  209. package/portraits/stephen-king/medium/flagg-55311.png +0 -0
  210. package/portraits/stephen-king/medium/gaunt-54421.png +0 -0
  211. package/portraits/stephen-king/medium/jack-44224.png +0 -0
  212. package/portraits/stephen-king/medium/johnny-44353.png +0 -0
  213. package/portraits/stephen-king/medium/margaret-15415.png +0 -0
  214. package/portraits/stephen-king/medium/paul-45233.png +0 -0
  215. package/portraits/stephen-king/medium/pennywise-54411.png +0 -0
  216. package/portraits/stephen-king/medium/roland-35121.png +0 -0
  217. package/src/public/App.tsx +21 -5
  218. package/src/public/components/BikeRackIndex.tsx +0 -1
  219. package/src/public/components/BikeRackWorkspace.tsx +86 -11
  220. package/src/public/components/DockviewWorkspace.tsx +19 -8
  221. package/src/public/components/StandalonePanel.tsx +1 -3
  222. package/src/public/components/panel-registry.ts +3 -1
  223. package/src/public/components/panels/AuditLogPanel.tsx +28 -4
  224. package/src/public/components/panels/GitPanel.tsx +1 -20
  225. package/src/public/components/panels/SettingsPanel.tsx +0 -1
  226. package/src/public/components/panels/SprintPanel.tsx +32 -1
  227. package/src/public/components/panels/index.ts +0 -2
  228. package/src/public/hooks/useFocusPanel.ts +137 -0
  229. package/src/public/hooks/useLayoutPersistence.ts +8 -5
  230. package/src/public/styles/dockview-theme.css +1 -84
  231. package/src/public/styles/tailwind.css +27 -32
  232. package/src/public/utils/slash-commands.ts +122 -98
  233. package/LICENSE +0 -14
  234. package/dist/hooks/cyclist-pretooluse-hook.d.ts +0 -60
  235. package/dist/hooks/cyclist-pretooluse-hook.d.ts.map +0 -1
  236. package/dist/hooks/cyclist-pretooluse-hook.js +0 -57
  237. package/dist/hooks/cyclist-pretooluse-hook.js.map +0 -1
  238. package/dist/hooks/pretooluse-hook.d.ts +0 -89
  239. package/dist/hooks/pretooluse-hook.d.ts.map +0 -1
  240. package/dist/hooks/pretooluse-hook.js +0 -235
  241. package/dist/hooks/pretooluse-hook.js.map +0 -1
  242. package/dist/notification-sound.d.ts +0 -59
  243. package/dist/notification-sound.d.ts.map +0 -1
  244. package/dist/notification-sound.js +0 -219
  245. package/dist/notification-sound.js.map +0 -1
  246. package/dist/plugin-loader.test.d.ts +0 -17
  247. package/dist/plugin-loader.test.d.ts.map +0 -1
  248. package/dist/plugin-loader.test.js +0 -407
  249. package/dist/plugin-loader.test.js.map +0 -1
  250. package/portraits/star-trek-tng/large/beverly-44352.png +0 -0
  251. package/portraits/star-trek-tng/large/data-55241.png +0 -0
  252. package/portraits/star-trek-tng/large/deanna-43353.png +0 -0
  253. package/portraits/star-trek-tng/large/geordi-54342.png +0 -0
  254. package/portraits/star-trek-tng/large/jean-luc-45342.png +0 -0
  255. package/portraits/star-trek-tng/large/kathryn-45332.png +0 -0
  256. package/portraits/star-trek-tng/large/miles-35342.png +0 -0
  257. package/portraits/star-trek-tng/large/q-53521.png +0 -0
  258. package/portraits/star-trek-tng/large/spock-45231.png +0 -0
  259. package/portraits/star-trek-tng/large/troi-44352.png +0 -0
  260. package/portraits/star-trek-tng/medium/beverly-44352.png +0 -0
  261. package/portraits/star-trek-tng/medium/data-55241.png +0 -0
  262. package/portraits/star-trek-tng/medium/deanna-43353.png +0 -0
  263. package/portraits/star-trek-tng/medium/geordi-54342.png +0 -0
  264. package/portraits/star-trek-tng/medium/jean-luc-45342.png +0 -0
  265. package/portraits/star-trek-tng/medium/kathryn-45332.png +0 -0
  266. package/portraits/star-trek-tng/medium/miles-35342.png +0 -0
  267. package/portraits/star-trek-tng/medium/q-53521.png +0 -0
  268. package/portraits/star-trek-tng/medium/spock-45231.png +0 -0
  269. package/portraits/star-trek-tng/medium/troi-44352.png +0 -0
  270. package/src/public/components/panels/TTYPanel.tsx +0 -299
  271. package/src/public/types/electron.d.ts +0 -18
@@ -1,299 +0,0 @@
1
- /**
2
- * TTYPanel - Terminal panel using xterm.js over WebSocket
3
- *
4
- * Story MSSCI-14211 - TTY Panel with xterm.js terminal emulator
5
- * Epic: Epic 76 - Dockview Panel Migration
6
- *
7
- * Features:
8
- * - Embeds xterm.js terminal in a Dockview panel
9
- * - Communicates with server via /ws/pty WebSocket (works in both Electron and web mode)
10
- * - Loads user's shell with environment (bash/zsh profile)
11
- * - Opens in project root directory
12
- * - Proper resize handling via FitAddon
13
- * - Standard terminal features (colors, cursor, scrollback)
14
- */
15
-
16
- import React, { useRef, useEffect, useState, useCallback } from 'react';
17
- import { Button } from '@/components/ui/button';
18
- import { Terminal } from 'xterm';
19
- import { FitAddon } from 'xterm-addon-fit';
20
- import 'xterm/css/xterm.css';
21
-
22
- /** Props for TTYPanel component */
23
- export interface TTYPanelProps {
24
- /** Project root directory for the terminal. Defaults to current project. */
25
- projectRoot?: string;
26
- }
27
-
28
- /** Terminal status states */
29
- type TerminalStatus = 'connecting' | 'connected' | 'error' | 'exited';
30
-
31
- /**
32
- * TTYPanel - Terminal emulator panel for Cyclist
33
- */
34
- export function TTYPanel({ projectRoot }: TTYPanelProps): React.ReactElement {
35
- const terminalRef = useRef<HTMLDivElement>(null);
36
- const xtermRef = useRef<Terminal | null>(null);
37
- const fitAddonRef = useRef<FitAddon | null>(null);
38
- const wsRef = useRef<WebSocket | null>(null);
39
- const resizeObserverRef = useRef<ResizeObserver | null>(null);
40
- const resizeTimeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
41
-
42
- const [status, setStatus] = useState<TerminalStatus>('connecting');
43
- const [errorMessage, setErrorMessage] = useState<string | null>(null);
44
-
45
- /**
46
- * Build the WebSocket URL for /ws/pty
47
- */
48
- const getWsUrl = useCallback(() => {
49
- const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
50
- return `${protocol}//${window.location.host}/ws/pty`;
51
- }, []);
52
-
53
- /**
54
- * Spawn a new PTY session via WebSocket
55
- */
56
- const spawnPty = useCallback(() => {
57
- const ws = wsRef.current;
58
- if (!ws || ws.readyState !== WebSocket.OPEN) return;
59
-
60
- setStatus('connecting');
61
- setErrorMessage(null);
62
-
63
- const fitAddon = fitAddonRef.current;
64
- const cols = fitAddon ? Math.max(fitAddon.proposeDimensions()?.cols ?? 80, 1) : 80;
65
- const rows = fitAddon ? Math.max(fitAddon.proposeDimensions()?.rows ?? 24, 1) : 24;
66
-
67
- ws.send(JSON.stringify({
68
- type: 'spawn',
69
- cwd: projectRoot,
70
- cols,
71
- rows,
72
- }));
73
- }, [projectRoot]);
74
-
75
- /**
76
- * Handle restart button click
77
- */
78
- const handleRestart = useCallback(() => {
79
- xtermRef.current?.clear();
80
- xtermRef.current?.reset();
81
- spawnPty();
82
- }, [spawnPty]);
83
-
84
- /**
85
- * Initialize terminal and WebSocket
86
- */
87
- useEffect(() => {
88
- if (!terminalRef.current) return;
89
-
90
- // Create xterm Terminal
91
- const terminal = new Terminal({
92
- allowProposedApi: true,
93
- allowTransparency: false,
94
- fontFamily: 'Menlo, Monaco, "Courier New", monospace',
95
- fontSize: 14,
96
- scrollback: 5000,
97
- cursorBlink: true,
98
- theme: {
99
- background: '#1e1e1e',
100
- foreground: '#d4d4d4',
101
- cursor: '#d4d4d4',
102
- cursorAccent: '#1e1e1e',
103
- },
104
- });
105
-
106
- const fitAddon = new FitAddon();
107
- terminal.loadAddon(fitAddon);
108
-
109
- xtermRef.current = terminal;
110
- fitAddonRef.current = fitAddon;
111
-
112
- terminal.open(terminalRef.current);
113
-
114
- // Initial fit after open
115
- setTimeout(() => {
116
- fitAddon.fit();
117
- terminal.focus();
118
- }, 0);
119
-
120
- // Connect WebSocket to /ws/pty
121
- const wsUrl = getWsUrl();
122
- const ws = new WebSocket(wsUrl);
123
- wsRef.current = ws;
124
-
125
- ws.onopen = () => {
126
- // Spawn PTY once connected
127
- const cols = Math.max(fitAddon.proposeDimensions()?.cols ?? 80, 1);
128
- const rows = Math.max(fitAddon.proposeDimensions()?.rows ?? 24, 1);
129
-
130
- ws.send(JSON.stringify({
131
- type: 'spawn',
132
- cwd: projectRoot,
133
- cols,
134
- rows,
135
- }));
136
- };
137
-
138
- ws.onmessage = (event) => {
139
- try {
140
- const msg = JSON.parse(event.data);
141
- if (msg.type === 'data') {
142
- terminal.write(msg.data);
143
- } else if (msg.type === 'spawn') {
144
- setStatus('connected');
145
- } else if (msg.type === 'exit') {
146
- setStatus('exited');
147
- } else if (msg.type === 'error') {
148
- setStatus('error');
149
- setErrorMessage(msg.error);
150
- }
151
- } catch {
152
- // Non-JSON data, write directly
153
- terminal.write(event.data);
154
- }
155
- };
156
-
157
- ws.onerror = () => {
158
- setStatus('error');
159
- setErrorMessage('WebSocket connection failed');
160
- };
161
-
162
- ws.onclose = () => {
163
- if (status === 'connected') {
164
- setStatus('exited');
165
- }
166
- };
167
-
168
- // Send terminal input to PTY via WebSocket
169
- const dataDisposable = terminal.onData((data) => {
170
- if (ws.readyState === WebSocket.OPEN) {
171
- ws.send(JSON.stringify({ type: 'data', data }));
172
- }
173
- });
174
-
175
- // Handle terminal resize - notify PTY via WebSocket
176
- const resizeDisposable = terminal.onResize(({ cols, rows }) => {
177
- if (ws.readyState === WebSocket.OPEN) {
178
- ws.send(JSON.stringify({ type: 'resize', cols, rows }));
179
- }
180
- });
181
-
182
- // ResizeObserver for container resize handling with debounce
183
- const resizeObserver = new ResizeObserver(() => {
184
- if (resizeTimeoutRef.current) {
185
- clearTimeout(resizeTimeoutRef.current);
186
- }
187
- resizeTimeoutRef.current = setTimeout(() => {
188
- fitAddonRef.current?.fit();
189
- }, 100);
190
- });
191
-
192
- resizeObserver.observe(terminalRef.current);
193
- resizeObserverRef.current = resizeObserver;
194
-
195
- // Cleanup on unmount
196
- return () => {
197
- if (resizeTimeoutRef.current) {
198
- clearTimeout(resizeTimeoutRef.current);
199
- }
200
- resizeObserverRef.current?.disconnect();
201
-
202
- // Kill PTY and close WebSocket
203
- if (ws.readyState === WebSocket.OPEN) {
204
- ws.send(JSON.stringify({ type: 'kill' }));
205
- ws.close();
206
- }
207
- wsRef.current = null;
208
-
209
- dataDisposable.dispose();
210
- resizeDisposable.dispose();
211
- terminal.dispose();
212
- };
213
- }, [getWsUrl, projectRoot]);
214
-
215
- // Render status message for error/exit states
216
- const renderStatusOverlay = () => {
217
- if (status === 'error') {
218
- return (
219
- <div className="tty-overlay" data-testid="tty-error">
220
- <div className="tty-overlay-content">
221
- <span className="tty-overlay-icon">⚠️</span>
222
- <span className="tty-overlay-message">Failed to spawn shell: {errorMessage}</span>
223
- <Button
224
- variant="outline"
225
- size="sm"
226
- className="tty-restart-button"
227
- onClick={handleRestart}
228
- aria-label="Restart terminal"
229
- >
230
- Restart
231
- </Button>
232
- </div>
233
- </div>
234
- );
235
- }
236
-
237
- if (status === 'exited') {
238
- return (
239
- <div className="tty-overlay" data-testid="tty-exited">
240
- <div className="tty-overlay-content">
241
- <span className="tty-overlay-icon">✓</span>
242
- <span className="tty-overlay-message">Process ended</span>
243
- <Button
244
- variant="outline"
245
- size="sm"
246
- className="tty-restart-button"
247
- onClick={handleRestart}
248
- aria-label="New session"
249
- >
250
- New Session
251
- </Button>
252
- </div>
253
- </div>
254
- );
255
- }
256
-
257
- return null;
258
- };
259
-
260
- return (
261
- <div
262
- className="tty-panel"
263
- data-testid="tty-panel"
264
- role="region"
265
- aria-label="Terminal"
266
- aria-describedby="tty-keyboard-help"
267
- >
268
- {/* Hidden keyboard help for screen readers */}
269
- <span id="tty-keyboard-help" className="sr-only">
270
- Terminal panel. Use keyboard to interact with the shell.
271
- </span>
272
-
273
- {/* Status announcer for screen readers */}
274
- <div
275
- data-testid="tty-status"
276
- aria-live="polite"
277
- className="sr-only"
278
- >
279
- {status === 'connecting' && 'Connecting to terminal...'}
280
- {status === 'connected' && 'Terminal connected'}
281
- {status === 'error' && `Terminal error: ${errorMessage}`}
282
- {status === 'exited' && 'Terminal session ended'}
283
- </div>
284
-
285
- {/* Terminal container */}
286
- <div
287
- ref={terminalRef}
288
- className="tty-terminal-container"
289
- data-testid="tty-terminal-container"
290
- style={{ height: '100%', width: '100%' }}
291
- />
292
-
293
- {/* Status overlay for error/exit states */}
294
- {renderStatusOverlay()}
295
- </div>
296
- );
297
- }
298
-
299
- export default TTYPanel;
@@ -1,18 +0,0 @@
1
- /**
2
- * Global type declarations for window.electronAPI
3
- *
4
- * Story MSSCI-12717 - React Migration
5
- *
6
- * This file provides TypeScript type definitions for the Electron API
7
- * exposed via preload.ts to the renderer process.
8
- */
9
-
10
- import type { ElectronAPI } from '../../preload';
11
-
12
- declare global {
13
- interface Window {
14
- electronAPI?: ElectronAPI;
15
- }
16
- }
17
-
18
- export {};