workos 0.0.24 → 0.1.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 (305) hide show
  1. package/.claude-plugin/plugin.json +13 -0
  2. package/LICENSE +47 -0
  3. package/README.md +154 -1
  4. package/dist/bin.d.ts +2 -0
  5. package/dist/bin.js +163 -0
  6. package/dist/bin.js.map +1 -0
  7. package/dist/cli.config.d.ts +52 -0
  8. package/dist/cli.config.js +70 -0
  9. package/dist/cli.config.js.map +1 -0
  10. package/dist/package.json +87 -0
  11. package/dist/src/commands/install-skill.d.ts +20 -0
  12. package/dist/src/commands/install-skill.js +130 -0
  13. package/dist/src/commands/install-skill.js.map +1 -0
  14. package/dist/src/commands/install.d.ts +22 -0
  15. package/dist/src/commands/install.js +57 -0
  16. package/dist/src/commands/install.js.map +1 -0
  17. package/dist/src/commands/login.d.ts +1 -0
  18. package/dist/src/commands/login.js +141 -0
  19. package/dist/src/commands/login.js.map +1 -0
  20. package/dist/src/commands/logout.d.ts +1 -0
  21. package/dist/src/commands/logout.js +17 -0
  22. package/dist/src/commands/logout.js.map +1 -0
  23. package/dist/src/dashboard/components/AnimatedLogo.d.ts +8 -0
  24. package/dist/src/dashboard/components/AnimatedLogo.js +16 -0
  25. package/dist/src/dashboard/components/AnimatedLogo.js.map +1 -0
  26. package/dist/src/dashboard/components/CompletionView.d.ts +13 -0
  27. package/dist/src/dashboard/components/CompletionView.js +21 -0
  28. package/dist/src/dashboard/components/CompletionView.js.map +1 -0
  29. package/dist/src/dashboard/components/ConfirmPrompt.d.ts +9 -0
  30. package/dist/src/dashboard/components/ConfirmPrompt.js +25 -0
  31. package/dist/src/dashboard/components/ConfirmPrompt.js.map +1 -0
  32. package/dist/src/dashboard/components/CredentialsForm.d.ts +10 -0
  33. package/dist/src/dashboard/components/CredentialsForm.js +47 -0
  34. package/dist/src/dashboard/components/CredentialsForm.js.map +1 -0
  35. package/dist/src/dashboard/components/Dashboard.d.ts +3 -0
  36. package/dist/src/dashboard/components/Dashboard.js +100 -0
  37. package/dist/src/dashboard/components/Dashboard.js.map +1 -0
  38. package/dist/src/dashboard/components/DashboardLayout.d.ts +24 -0
  39. package/dist/src/dashboard/components/DashboardLayout.js +25 -0
  40. package/dist/src/dashboard/components/DashboardLayout.js.map +1 -0
  41. package/dist/src/dashboard/components/DiffPanel.d.ts +9 -0
  42. package/dist/src/dashboard/components/DiffPanel.js +136 -0
  43. package/dist/src/dashboard/components/DiffPanel.js.map +1 -0
  44. package/dist/src/dashboard/components/InlinePrompt.d.ts +8 -0
  45. package/dist/src/dashboard/components/InlinePrompt.js +19 -0
  46. package/dist/src/dashboard/components/InlinePrompt.js.map +1 -0
  47. package/dist/src/dashboard/components/OutputPanel.d.ts +10 -0
  48. package/dist/src/dashboard/components/OutputPanel.js +100 -0
  49. package/dist/src/dashboard/components/OutputPanel.js.map +1 -0
  50. package/dist/src/dashboard/components/Panel.d.ts +12 -0
  51. package/dist/src/dashboard/components/Panel.js +6 -0
  52. package/dist/src/dashboard/components/Panel.js.map +1 -0
  53. package/dist/src/dashboard/components/TextInput.d.ts +13 -0
  54. package/dist/src/dashboard/components/TextInput.js +57 -0
  55. package/dist/src/dashboard/components/TextInput.js.map +1 -0
  56. package/dist/src/dashboard/components/WelcomeArt.d.ts +2 -0
  57. package/dist/src/dashboard/components/WelcomeArt.js +9 -0
  58. package/dist/src/dashboard/components/WelcomeArt.js.map +1 -0
  59. package/dist/src/dashboard/hooks/useAnimation.d.ts +7 -0
  60. package/dist/src/dashboard/hooks/useAnimation.js +24 -0
  61. package/dist/src/dashboard/hooks/useAnimation.js.map +1 -0
  62. package/dist/src/dashboard/hooks/useKeyboard.d.ts +8 -0
  63. package/dist/src/dashboard/hooks/useKeyboard.js +18 -0
  64. package/dist/src/dashboard/hooks/useKeyboard.js.map +1 -0
  65. package/dist/src/dashboard/hooks/useTerminalSize.d.ts +8 -0
  66. package/dist/src/dashboard/hooks/useTerminalSize.js +23 -0
  67. package/dist/src/dashboard/hooks/useTerminalSize.js.map +1 -0
  68. package/dist/src/dashboard/index.d.ts +6 -0
  69. package/dist/src/dashboard/index.js +36 -0
  70. package/dist/src/dashboard/index.js.map +1 -0
  71. package/dist/src/dashboard/lib/diff-utils.d.ts +21 -0
  72. package/dist/src/dashboard/lib/diff-utils.js +271 -0
  73. package/dist/src/dashboard/lib/diff-utils.js.map +1 -0
  74. package/dist/src/dashboard/lib/logo-frames.d.ts +20 -0
  75. package/dist/src/dashboard/lib/logo-frames.js +109 -0
  76. package/dist/src/dashboard/lib/logo-frames.js.map +1 -0
  77. package/dist/src/dashboard/lib/welcome-art.d.ts +1 -0
  78. package/dist/src/dashboard/lib/welcome-art.js +5 -0
  79. package/dist/src/dashboard/lib/welcome-art.js.map +1 -0
  80. package/dist/src/dashboard/types.d.ts +5 -0
  81. package/dist/src/dashboard/types.js +2 -0
  82. package/dist/src/dashboard/types.js.map +1 -0
  83. package/dist/src/lib/__tests__/test-utils.d.ts +40 -0
  84. package/dist/src/lib/__tests__/test-utils.js +108 -0
  85. package/dist/src/lib/__tests__/test-utils.js.map +1 -0
  86. package/dist/src/lib/adapters/cli-adapter.d.ts +56 -0
  87. package/dist/src/lib/adapters/cli-adapter.js +318 -0
  88. package/dist/src/lib/adapters/cli-adapter.js.map +1 -0
  89. package/dist/src/lib/adapters/dashboard-adapter.d.ts +30 -0
  90. package/dist/src/lib/adapters/dashboard-adapter.js +97 -0
  91. package/dist/src/lib/adapters/dashboard-adapter.js.map +1 -0
  92. package/dist/src/lib/adapters/index.d.ts +3 -0
  93. package/dist/src/lib/adapters/index.js +3 -0
  94. package/dist/src/lib/adapters/index.js.map +1 -0
  95. package/dist/src/lib/adapters/types.d.ts +41 -0
  96. package/dist/src/lib/adapters/types.js +2 -0
  97. package/dist/src/lib/adapters/types.js.map +1 -0
  98. package/dist/src/lib/agent-interface.d.ts +75 -0
  99. package/dist/src/lib/agent-interface.js +563 -0
  100. package/dist/src/lib/agent-interface.js.map +1 -0
  101. package/dist/src/lib/agent-runner.d.ts +9 -0
  102. package/dist/src/lib/agent-runner.js +213 -0
  103. package/dist/src/lib/agent-runner.js.map +1 -0
  104. package/dist/src/lib/api.d.ts +25 -0
  105. package/dist/src/lib/api.js +120 -0
  106. package/dist/src/lib/api.js.map +1 -0
  107. package/dist/src/lib/config.d.ts +60 -0
  108. package/dist/src/lib/config.js +88 -0
  109. package/dist/src/lib/config.js.map +1 -0
  110. package/dist/src/lib/constants.d.ts +32 -0
  111. package/dist/src/lib/constants.js +53 -0
  112. package/dist/src/lib/constants.js.map +1 -0
  113. package/dist/src/lib/credentials.d.ts +19 -0
  114. package/dist/src/lib/credentials.js +55 -0
  115. package/dist/src/lib/credentials.js.map +1 -0
  116. package/dist/src/lib/env-writer.d.ts +14 -0
  117. package/dist/src/lib/env-writer.js +39 -0
  118. package/dist/src/lib/env-writer.js.map +1 -0
  119. package/dist/src/lib/events.d.ts +114 -0
  120. package/dist/src/lib/events.js +19 -0
  121. package/dist/src/lib/events.js.map +1 -0
  122. package/dist/src/lib/framework-config.d.ts +108 -0
  123. package/dist/src/lib/framework-config.js +11 -0
  124. package/dist/src/lib/framework-config.js.map +1 -0
  125. package/dist/src/lib/helper-functions.d.ts +1 -0
  126. package/dist/src/lib/helper-functions.js +2 -0
  127. package/dist/src/lib/helper-functions.js.map +1 -0
  128. package/dist/src/lib/port-detection.d.ts +7 -0
  129. package/dist/src/lib/port-detection.js +112 -0
  130. package/dist/src/lib/port-detection.js.map +1 -0
  131. package/dist/src/lib/progress-tracker.d.ts +22 -0
  132. package/dist/src/lib/progress-tracker.js +47 -0
  133. package/dist/src/lib/progress-tracker.js.map +1 -0
  134. package/dist/src/lib/run-with-core.d.ts +2 -0
  135. package/dist/src/lib/run-with-core.js +266 -0
  136. package/dist/src/lib/run-with-core.js.map +1 -0
  137. package/dist/src/lib/safe-tools.d.ts +2 -0
  138. package/dist/src/lib/safe-tools.js +212 -0
  139. package/dist/src/lib/safe-tools.js.map +1 -0
  140. package/dist/src/lib/settings.d.ts +59 -0
  141. package/dist/src/lib/settings.js +36 -0
  142. package/dist/src/lib/settings.js.map +1 -0
  143. package/dist/src/lib/token-refresh.d.ts +12 -0
  144. package/dist/src/lib/token-refresh.js +26 -0
  145. package/dist/src/lib/token-refresh.js.map +1 -0
  146. package/dist/src/lib/validation/build-validator.d.ts +9 -0
  147. package/dist/src/lib/validation/build-validator.js +118 -0
  148. package/dist/src/lib/validation/build-validator.js.map +1 -0
  149. package/dist/src/lib/validation/index.d.ts +3 -0
  150. package/dist/src/lib/validation/index.js +3 -0
  151. package/dist/src/lib/validation/index.js.map +1 -0
  152. package/dist/src/lib/validation/types.d.ts +41 -0
  153. package/dist/src/lib/validation/types.js +2 -0
  154. package/dist/src/lib/validation/types.js.map +1 -0
  155. package/dist/src/lib/validation/validator.d.ts +6 -0
  156. package/dist/src/lib/validation/validator.js +647 -0
  157. package/dist/src/lib/validation/validator.js.map +1 -0
  158. package/dist/src/lib/wizard-core.d.ts +200 -0
  159. package/dist/src/lib/wizard-core.js +392 -0
  160. package/dist/src/lib/wizard-core.js.map +1 -0
  161. package/dist/src/lib/wizard-core.types.d.ts +73 -0
  162. package/dist/src/lib/wizard-core.types.js +2 -0
  163. package/dist/src/lib/wizard-core.types.js.map +1 -0
  164. package/dist/src/lib/workos-management.d.ts +32 -0
  165. package/dist/src/lib/workos-management.js +142 -0
  166. package/dist/src/lib/workos-management.js.map +1 -0
  167. package/dist/src/nextjs/nextjs-wizard-agent.d.ts +6 -0
  168. package/dist/src/nextjs/nextjs-wizard-agent.js +97 -0
  169. package/dist/src/nextjs/nextjs-wizard-agent.js.map +1 -0
  170. package/dist/src/nextjs/utils.d.ts +8 -0
  171. package/dist/src/nextjs/utils.js +53 -0
  172. package/dist/src/nextjs/utils.js.map +1 -0
  173. package/dist/src/react/react-wizard-agent.d.ts +2 -0
  174. package/dist/src/react/react-wizard-agent.js +47 -0
  175. package/dist/src/react/react-wizard-agent.js.map +1 -0
  176. package/dist/src/react-router/react-router-wizard-agent.d.ts +6 -0
  177. package/dist/src/react-router/react-router-wizard-agent.js +103 -0
  178. package/dist/src/react-router/react-router-wizard-agent.js.map +1 -0
  179. package/dist/src/react-router/utils.d.ts +19 -0
  180. package/dist/src/react-router/utils.js +210 -0
  181. package/dist/src/react-router/utils.js.map +1 -0
  182. package/dist/src/run.d.ts +24 -0
  183. package/dist/src/run.js +48 -0
  184. package/dist/src/run.js.map +1 -0
  185. package/dist/src/steps/add-or-update-environment-variables.d.ts +10 -0
  186. package/dist/src/steps/add-or-update-environment-variables.js +155 -0
  187. package/dist/src/steps/add-or-update-environment-variables.js.map +1 -0
  188. package/dist/src/steps/index.d.ts +3 -0
  189. package/dist/src/steps/index.js +4 -0
  190. package/dist/src/steps/index.js.map +1 -0
  191. package/dist/src/steps/run-prettier.d.ts +5 -0
  192. package/dist/src/steps/run-prettier.js +54 -0
  193. package/dist/src/steps/run-prettier.js.map +1 -0
  194. package/dist/src/steps/upload-environment-variables/EnvironmentProvider.d.ts +8 -0
  195. package/dist/src/steps/upload-environment-variables/EnvironmentProvider.js +7 -0
  196. package/dist/src/steps/upload-environment-variables/EnvironmentProvider.js.map +1 -0
  197. package/dist/src/steps/upload-environment-variables/index.d.ts +6 -0
  198. package/dist/src/steps/upload-environment-variables/index.js +57 -0
  199. package/dist/src/steps/upload-environment-variables/index.js.map +1 -0
  200. package/dist/src/steps/upload-environment-variables/providers/vercel.d.ts +14 -0
  201. package/dist/src/steps/upload-environment-variables/providers/vercel.js +104 -0
  202. package/dist/src/steps/upload-environment-variables/providers/vercel.js.map +1 -0
  203. package/dist/src/tanstack-start/tanstack-start-wizard-agent.d.ts +2 -0
  204. package/dist/src/tanstack-start/tanstack-start-wizard-agent.js +49 -0
  205. package/dist/src/tanstack-start/tanstack-start-wizard-agent.js.map +1 -0
  206. package/dist/src/telemetry.d.ts +2 -0
  207. package/dist/src/telemetry.js +29 -0
  208. package/dist/src/telemetry.js.map +1 -0
  209. package/dist/src/utils/analytics.d.ts +24 -0
  210. package/dist/src/utils/analytics.js +139 -0
  211. package/dist/src/utils/analytics.js.map +1 -0
  212. package/dist/src/utils/bash.d.ts +2 -0
  213. package/dist/src/utils/bash.js +17 -0
  214. package/dist/src/utils/bash.js.map +1 -0
  215. package/dist/src/utils/clack-utils.d.ts +93 -0
  216. package/dist/src/utils/clack-utils.js +397 -0
  217. package/dist/src/utils/clack-utils.js.map +1 -0
  218. package/dist/src/utils/clack.d.ts +5 -0
  219. package/dist/src/utils/clack.js +34 -0
  220. package/dist/src/utils/clack.js.map +1 -0
  221. package/dist/src/utils/cli-symbols.d.ts +32 -0
  222. package/dist/src/utils/cli-symbols.js +46 -0
  223. package/dist/src/utils/cli-symbols.js.map +1 -0
  224. package/dist/src/utils/debug.d.ts +7 -0
  225. package/dist/src/utils/debug.js +88 -0
  226. package/dist/src/utils/debug.js.map +1 -0
  227. package/dist/src/utils/env-parser.d.ts +5 -0
  228. package/dist/src/utils/env-parser.js +18 -0
  229. package/dist/src/utils/env-parser.js.map +1 -0
  230. package/dist/src/utils/environment.d.ts +4 -0
  231. package/dist/src/utils/environment.js +69 -0
  232. package/dist/src/utils/environment.js.map +1 -0
  233. package/dist/src/utils/errors.d.ts +3 -0
  234. package/dist/src/utils/errors.js +7 -0
  235. package/dist/src/utils/errors.js.map +1 -0
  236. package/dist/src/utils/logging.d.ts +9 -0
  237. package/dist/src/utils/logging.js +36 -0
  238. package/dist/src/utils/logging.js.map +1 -0
  239. package/dist/src/utils/package-json.d.ts +25 -0
  240. package/dist/src/utils/package-json.js +21 -0
  241. package/dist/src/utils/package-json.js.map +1 -0
  242. package/dist/src/utils/package-manager.d.ts +21 -0
  243. package/dist/src/utils/package-manager.js +167 -0
  244. package/dist/src/utils/package-manager.js.map +1 -0
  245. package/dist/src/utils/redact.d.ts +5 -0
  246. package/dist/src/utils/redact.js +29 -0
  247. package/dist/src/utils/redact.js.map +1 -0
  248. package/dist/src/utils/semver.d.ts +10 -0
  249. package/dist/src/utils/semver.js +43 -0
  250. package/dist/src/utils/semver.js.map +1 -0
  251. package/dist/src/utils/string.d.ts +1 -0
  252. package/dist/src/utils/string.js +6 -0
  253. package/dist/src/utils/string.js.map +1 -0
  254. package/dist/src/utils/telemetry-client.d.ts +15 -0
  255. package/dist/src/utils/telemetry-client.js +57 -0
  256. package/dist/src/utils/telemetry-client.js.map +1 -0
  257. package/dist/src/utils/telemetry-types.d.ts +51 -0
  258. package/dist/src/utils/telemetry-types.js +6 -0
  259. package/dist/src/utils/telemetry-types.js.map +1 -0
  260. package/dist/src/utils/types.d.ts +80 -0
  261. package/dist/src/utils/types.js +2 -0
  262. package/dist/src/utils/types.js.map +1 -0
  263. package/dist/src/utils/urls.d.ts +7 -0
  264. package/dist/src/utils/urls.js +8 -0
  265. package/dist/src/utils/urls.js.map +1 -0
  266. package/dist/src/utils/vendor/is-unicorn-supported.d.ts +1 -0
  267. package/dist/src/utils/vendor/is-unicorn-supported.js +21 -0
  268. package/dist/src/utils/vendor/is-unicorn-supported.js.map +1 -0
  269. package/dist/src/vanilla-js/vanilla-js-wizard-agent.d.ts +2 -0
  270. package/dist/src/vanilla-js/vanilla-js-wizard-agent.js +47 -0
  271. package/dist/src/vanilla-js/vanilla-js-wizard-agent.js.map +1 -0
  272. package/package.json +76 -84
  273. package/skills/workos-authkit-base/SKILL.md +113 -0
  274. package/skills/workos-authkit-nextjs/SKILL.md +115 -0
  275. package/skills/workos-authkit-react/SKILL.md +91 -0
  276. package/skills/workos-authkit-react-router/SKILL.md +106 -0
  277. package/skills/workos-authkit-tanstack-start/SKILL.md +104 -0
  278. package/skills/workos-authkit-vanilla-js/SKILL.md +81 -0
  279. package/build/apps/index.js +0 -50
  280. package/build/apps/slack.js +0 -151
  281. package/build/cli.js +0 -42
  282. package/build/config.js +0 -34
  283. package/build/dev.js +0 -5
  284. package/build/enable-api-access.png +0 -0
  285. package/build/groups.js +0 -480
  286. package/build/index.js +0 -3
  287. package/build/info.js +0 -69
  288. package/build/login.js +0 -161
  289. package/build/main.js +0 -214
  290. package/build/users.js +0 -402
  291. package/build/util.js +0 -157
  292. package/coverage/clover.xml +0 -66
  293. package/coverage/coverage-final.json +0 -4
  294. package/coverage/lcov-report/base.css +0 -212
  295. package/coverage/lcov-report/cli.ts.html +0 -329
  296. package/coverage/lcov-report/config.ts.html +0 -152
  297. package/coverage/lcov-report/index.html +0 -119
  298. package/coverage/lcov-report/prettify.css +0 -1
  299. package/coverage/lcov-report/prettify.js +0 -1
  300. package/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  301. package/coverage/lcov-report/sorter.js +0 -158
  302. package/coverage/lcov-report/util.ts.html +0 -350
  303. package/coverage/lcov.info +0 -121
  304. package/package-lock.json +0 -7617
  305. package/tests/util.test.ts +0 -35
@@ -0,0 +1,13 @@
1
+ import React from 'react';
2
+ interface TextInputProps {
3
+ label: string;
4
+ placeholder?: string;
5
+ mask?: boolean;
6
+ value: string;
7
+ onChange: (value: string) => void;
8
+ onSubmit: () => void;
9
+ focused?: boolean;
10
+ error?: string;
11
+ }
12
+ export declare function TextInput({ label, placeholder, mask, value, onChange, onSubmit, focused, error, }: TextInputProps): React.ReactElement;
13
+ export {};
@@ -0,0 +1,57 @@
1
+ import { jsxs as _jsxs, jsx as _jsx } from "react/jsx-runtime";
2
+ import { useState, useEffect, useRef } from 'react';
3
+ import { Box, Text, useInput } from 'ink';
4
+ export function TextInput({ label, placeholder = '', mask = false, value, onChange, onSubmit, focused = true, error, }) {
5
+ const [cursorVisible, setCursorVisible] = useState(true);
6
+ // Use ref for synchronous value tracking (fixes snippet tool race conditions)
7
+ const valueRef = useRef(value);
8
+ valueRef.current = value;
9
+ // Blink cursor
10
+ useEffect(() => {
11
+ if (!focused)
12
+ return;
13
+ const interval = setInterval(() => {
14
+ setCursorVisible((v) => !v);
15
+ }, 500);
16
+ return () => clearInterval(interval);
17
+ }, [focused]);
18
+ useInput((input, key) => {
19
+ if (!focused)
20
+ return;
21
+ if (key.return) {
22
+ onSubmit();
23
+ }
24
+ else if (key.ctrl && input === 'u') {
25
+ // Ctrl+U: clear entire line
26
+ valueRef.current = '';
27
+ onChange('');
28
+ }
29
+ else if (key.backspace || key.delete) {
30
+ // Use ref for current value to avoid stale closure
31
+ const newValue = valueRef.current.slice(0, -1);
32
+ valueRef.current = newValue;
33
+ onChange(newValue);
34
+ }
35
+ else if (!key.ctrl && !key.meta && input && !key.upArrow && !key.downArrow) {
36
+ // Handle snippet tool expansion: input may contain backspaces (\x7f or \x08)
37
+ // followed by the replacement text
38
+ let newValue = valueRef.current;
39
+ for (const char of input) {
40
+ if (char === '\x7f' || char === '\x08') {
41
+ // Backspace - delete last character
42
+ newValue = newValue.slice(0, -1);
43
+ }
44
+ else if (char.charCodeAt(0) >= 32) {
45
+ // Only add printable characters
46
+ newValue += char;
47
+ }
48
+ }
49
+ valueRef.current = newValue;
50
+ onChange(newValue);
51
+ }
52
+ }, { isActive: focused });
53
+ const displayValue = mask ? '*'.repeat(value.length) : value;
54
+ const showPlaceholder = value.length === 0;
55
+ return (_jsxs(Box, { flexDirection: "column", children: [_jsxs(Box, { children: [_jsxs(Text, { color: "cyan", bold: true, children: [label, ":", ' '] }), _jsx(Text, { dimColor: showPlaceholder, children: showPlaceholder ? placeholder : displayValue }), focused && cursorVisible && _jsx(Text, { color: "cyan", children: "|" })] }), error && (_jsx(Text, { color: "red", dimColor: true, children: error }))] }));
56
+ }
57
+ //# sourceMappingURL=TextInput.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TextInput.js","sourceRoot":"","sources":["../../../../src/dashboard/components/TextInput.tsx"],"names":[],"mappings":"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","sourcesContent":["import React, { useState, useEffect, useRef } from 'react';\nimport { Box, Text, useInput } from 'ink';\n\ninterface TextInputProps {\n label: string;\n placeholder?: string;\n mask?: boolean;\n value: string;\n onChange: (value: string) => void;\n onSubmit: () => void;\n focused?: boolean;\n error?: string;\n}\n\nexport function TextInput({\n label,\n placeholder = '',\n mask = false,\n value,\n onChange,\n onSubmit,\n focused = true,\n error,\n}: TextInputProps): React.ReactElement {\n const [cursorVisible, setCursorVisible] = useState(true);\n // Use ref for synchronous value tracking (fixes snippet tool race conditions)\n const valueRef = useRef(value);\n valueRef.current = value;\n\n // Blink cursor\n useEffect(() => {\n if (!focused) return;\n const interval = setInterval(() => {\n setCursorVisible((v) => !v);\n }, 500);\n return () => clearInterval(interval);\n }, [focused]);\n\n useInput(\n (input, key) => {\n if (!focused) return;\n\n if (key.return) {\n onSubmit();\n } else if (key.ctrl && input === 'u') {\n // Ctrl+U: clear entire line\n valueRef.current = '';\n onChange('');\n } else if (key.backspace || key.delete) {\n // Use ref for current value to avoid stale closure\n const newValue = valueRef.current.slice(0, -1);\n valueRef.current = newValue;\n onChange(newValue);\n } else if (!key.ctrl && !key.meta && input && !key.upArrow && !key.downArrow) {\n // Handle snippet tool expansion: input may contain backspaces (\\x7f or \\x08)\n // followed by the replacement text\n let newValue = valueRef.current;\n\n for (const char of input) {\n if (char === '\\x7f' || char === '\\x08') {\n // Backspace - delete last character\n newValue = newValue.slice(0, -1);\n } else if (char.charCodeAt(0) >= 32) {\n // Only add printable characters\n newValue += char;\n }\n }\n\n valueRef.current = newValue;\n onChange(newValue);\n }\n },\n { isActive: focused },\n );\n\n const displayValue = mask ? '*'.repeat(value.length) : value;\n const showPlaceholder = value.length === 0;\n\n return (\n <Box flexDirection=\"column\">\n <Box>\n <Text color=\"cyan\" bold>\n {label}:{' '}\n </Text>\n <Text dimColor={showPlaceholder}>{showPlaceholder ? placeholder : displayValue}</Text>\n {focused && cursorVisible && <Text color=\"cyan\">|</Text>}\n </Box>\n {error && (\n <Text color=\"red\" dimColor>\n {error}\n </Text>\n )}\n </Box>\n );\n}\n"]}
@@ -0,0 +1,2 @@
1
+ import React from 'react';
2
+ export declare function WelcomeArt(): React.ReactElement;
@@ -0,0 +1,9 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ import { Box, Text } from 'ink';
3
+ import { getWelcomeArt } from '../lib/welcome-art.js';
4
+ export function WelcomeArt() {
5
+ const art = getWelcomeArt();
6
+ const lines = art.split('\n');
7
+ return (_jsx(Box, { flexDirection: "column", alignItems: "center", justifyContent: "center", flexGrow: 1, children: _jsx(Box, { flexDirection: "column", children: lines.map((line, i) => (_jsx(Text, { color: "cyan", children: line }, i))) }) }));
8
+ }
9
+ //# sourceMappingURL=WelcomeArt.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"WelcomeArt.js","sourceRoot":"","sources":["../../../../src/dashboard/components/WelcomeArt.tsx"],"names":[],"mappings":";AACA,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,MAAM,KAAK,CAAC;AAChC,OAAO,EAAE,aAAa,EAAE,MAAM,uBAAuB,CAAC;AAEtD,MAAM,UAAU,UAAU;IACxB,MAAM,GAAG,GAAG,aAAa,EAAE,CAAC;IAC5B,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IAE9B,OAAO,CACL,KAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,EAAC,UAAU,EAAC,QAAQ,EAAC,cAAc,EAAC,QAAQ,EAAC,QAAQ,EAAE,CAAC,YACjF,KAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,YACxB,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,CAAC,EAAE,EAAE,CAAC,CACtB,KAAC,IAAI,IAAS,KAAK,EAAC,MAAM,YACvB,IAAI,IADI,CAAC,CAEL,CACR,CAAC,GACE,GACF,CACP,CAAC;AACJ,CAAC","sourcesContent":["import React from 'react';\nimport { Box, Text } from 'ink';\nimport { getWelcomeArt } from '../lib/welcome-art.js';\n\nexport function WelcomeArt(): React.ReactElement {\n const art = getWelcomeArt();\n const lines = art.split('\\n');\n\n return (\n <Box flexDirection=\"column\" alignItems=\"center\" justifyContent=\"center\" flexGrow={1}>\n <Box flexDirection=\"column\">\n {lines.map((line, i) => (\n <Text key={i} color=\"cyan\">\n {line}\n </Text>\n ))}\n </Box>\n </Box>\n );\n}\n"]}
@@ -0,0 +1,7 @@
1
+ interface UseAnimationOptions {
2
+ frameCount: number;
3
+ frameDelayMs: number;
4
+ paused?: boolean;
5
+ }
6
+ export declare function useAnimation({ frameCount, frameDelayMs, paused }: UseAnimationOptions): number;
7
+ export {};
@@ -0,0 +1,24 @@
1
+ import { useState, useEffect, useRef } from 'react';
2
+ export function useAnimation({ frameCount, frameDelayMs, paused = false }) {
3
+ const [frameIndex, setFrameIndex] = useState(0);
4
+ const intervalRef = useRef(null);
5
+ useEffect(() => {
6
+ if (paused) {
7
+ if (intervalRef.current) {
8
+ clearInterval(intervalRef.current);
9
+ intervalRef.current = null;
10
+ }
11
+ return;
12
+ }
13
+ intervalRef.current = setInterval(() => {
14
+ setFrameIndex((prev) => (prev + 1) % frameCount);
15
+ }, frameDelayMs);
16
+ return () => {
17
+ if (intervalRef.current) {
18
+ clearInterval(intervalRef.current);
19
+ }
20
+ };
21
+ }, [frameCount, frameDelayMs, paused]);
22
+ return frameIndex;
23
+ }
24
+ //# sourceMappingURL=useAnimation.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useAnimation.js","sourceRoot":"","sources":["../../../../src/dashboard/hooks/useAnimation.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,MAAM,EAAE,MAAM,OAAO,CAAC;AAQpD,MAAM,UAAU,YAAY,CAAC,EAAE,UAAU,EAAE,YAAY,EAAE,MAAM,GAAG,KAAK,EAAuB;IAC5F,MAAM,CAAC,UAAU,EAAE,aAAa,CAAC,GAAG,QAAQ,CAAC,CAAC,CAAC,CAAC;IAChD,MAAM,WAAW,GAAG,MAAM,CAAwB,IAAI,CAAC,CAAC;IAExD,SAAS,CAAC,GAAG,EAAE;QACb,IAAI,MAAM,EAAE,CAAC;YACX,IAAI,WAAW,CAAC,OAAO,EAAE,CAAC;gBACxB,aAAa,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;gBACnC,WAAW,CAAC,OAAO,GAAG,IAAI,CAAC;YAC7B,CAAC;YACD,OAAO;QACT,CAAC;QAED,WAAW,CAAC,OAAO,GAAG,WAAW,CAAC,GAAG,EAAE;YACrC,aAAa,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC,GAAG,UAAU,CAAC,CAAC;QACnD,CAAC,EAAE,YAAY,CAAC,CAAC;QAEjB,OAAO,GAAG,EAAE;YACV,IAAI,WAAW,CAAC,OAAO,EAAE,CAAC;gBACxB,aAAa,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;YACrC,CAAC;QACH,CAAC,CAAC;IACJ,CAAC,EAAE,CAAC,UAAU,EAAE,YAAY,EAAE,MAAM,CAAC,CAAC,CAAC;IAEvC,OAAO,UAAU,CAAC;AACpB,CAAC","sourcesContent":["import { useState, useEffect, useRef } from 'react';\n\ninterface UseAnimationOptions {\n frameCount: number;\n frameDelayMs: number;\n paused?: boolean;\n}\n\nexport function useAnimation({ frameCount, frameDelayMs, paused = false }: UseAnimationOptions): number {\n const [frameIndex, setFrameIndex] = useState(0);\n const intervalRef = useRef<NodeJS.Timeout | null>(null);\n\n useEffect(() => {\n if (paused) {\n if (intervalRef.current) {\n clearInterval(intervalRef.current);\n intervalRef.current = null;\n }\n return;\n }\n\n intervalRef.current = setInterval(() => {\n setFrameIndex((prev) => (prev + 1) % frameCount);\n }, frameDelayMs);\n\n return () => {\n if (intervalRef.current) {\n clearInterval(intervalRef.current);\n }\n };\n }, [frameCount, frameDelayMs, paused]);\n\n return frameIndex;\n}\n"]}
@@ -0,0 +1,8 @@
1
+ interface UseScrollOptions {
2
+ totalItems: number;
3
+ visibleItems: number;
4
+ scrollOffset: number;
5
+ setScrollOffset: (offset: number) => void;
6
+ }
7
+ export declare function useScrollNavigation({ totalItems, visibleItems, scrollOffset, setScrollOffset, }: UseScrollOptions): void;
8
+ export {};
@@ -0,0 +1,18 @@
1
+ import { useInput } from 'ink';
2
+ export function useScrollNavigation({ totalItems, visibleItems, scrollOffset, setScrollOffset, }) {
3
+ useInput((input, key) => {
4
+ if (key.upArrow) {
5
+ setScrollOffset(Math.max(0, scrollOffset - 1));
6
+ }
7
+ else if (key.downArrow) {
8
+ setScrollOffset(Math.min(totalItems - visibleItems, scrollOffset + 1));
9
+ }
10
+ else if (key.pageUp) {
11
+ setScrollOffset(Math.max(0, scrollOffset - visibleItems));
12
+ }
13
+ else if (key.pageDown) {
14
+ setScrollOffset(Math.min(totalItems - visibleItems, scrollOffset + visibleItems));
15
+ }
16
+ });
17
+ }
18
+ //# sourceMappingURL=useKeyboard.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useKeyboard.js","sourceRoot":"","sources":["../../../../src/dashboard/hooks/useKeyboard.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,KAAK,CAAC;AAS/B,MAAM,UAAU,mBAAmB,CAAC,EAClC,UAAU,EACV,YAAY,EACZ,YAAY,EACZ,eAAe,GACE;IACjB,QAAQ,CAAC,CAAC,KAAK,EAAE,GAAG,EAAE,EAAE;QACtB,IAAI,GAAG,CAAC,OAAO,EAAE,CAAC;YAChB,eAAe,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,YAAY,GAAG,CAAC,CAAC,CAAC,CAAC;QACjD,CAAC;aAAM,IAAI,GAAG,CAAC,SAAS,EAAE,CAAC;YACzB,eAAe,CAAC,IAAI,CAAC,GAAG,CAAC,UAAU,GAAG,YAAY,EAAE,YAAY,GAAG,CAAC,CAAC,CAAC,CAAC;QACzE,CAAC;aAAM,IAAI,GAAG,CAAC,MAAM,EAAE,CAAC;YACtB,eAAe,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,YAAY,GAAG,YAAY,CAAC,CAAC,CAAC;QAC5D,CAAC;aAAM,IAAI,GAAG,CAAC,QAAQ,EAAE,CAAC;YACxB,eAAe,CAAC,IAAI,CAAC,GAAG,CAAC,UAAU,GAAG,YAAY,EAAE,YAAY,GAAG,YAAY,CAAC,CAAC,CAAC;QACpF,CAAC;IACH,CAAC,CAAC,CAAC;AACL,CAAC","sourcesContent":["import { useInput } from 'ink';\n\ninterface UseScrollOptions {\n totalItems: number;\n visibleItems: number;\n scrollOffset: number;\n setScrollOffset: (offset: number) => void;\n}\n\nexport function useScrollNavigation({\n totalItems,\n visibleItems,\n scrollOffset,\n setScrollOffset,\n}: UseScrollOptions): void {\n useInput((input, key) => {\n if (key.upArrow) {\n setScrollOffset(Math.max(0, scrollOffset - 1));\n } else if (key.downArrow) {\n setScrollOffset(Math.min(totalItems - visibleItems, scrollOffset + 1));\n } else if (key.pageUp) {\n setScrollOffset(Math.max(0, scrollOffset - visibleItems));\n } else if (key.pageDown) {\n setScrollOffset(Math.min(totalItems - visibleItems, scrollOffset + visibleItems));\n }\n });\n}\n"]}
@@ -0,0 +1,8 @@
1
+ interface TerminalSize {
2
+ columns: number;
3
+ rows: number;
4
+ }
5
+ export declare function useTerminalSize(): TerminalSize;
6
+ export declare const MIN_COLUMNS = 80;
7
+ export declare const MIN_ROWS = 24;
8
+ export {};
@@ -0,0 +1,23 @@
1
+ import { useState, useEffect } from 'react';
2
+ export function useTerminalSize() {
3
+ const [size, setSize] = useState({
4
+ columns: process.stdout.columns || 80,
5
+ rows: process.stdout.rows || 24,
6
+ });
7
+ useEffect(() => {
8
+ const handleResize = () => {
9
+ setSize({
10
+ columns: process.stdout.columns || 80,
11
+ rows: process.stdout.rows || 24,
12
+ });
13
+ };
14
+ process.stdout.on('resize', handleResize);
15
+ return () => {
16
+ process.stdout.off('resize', handleResize);
17
+ };
18
+ }, []);
19
+ return size;
20
+ }
21
+ export const MIN_COLUMNS = 80;
22
+ export const MIN_ROWS = 24;
23
+ //# sourceMappingURL=useTerminalSize.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useTerminalSize.js","sourceRoot":"","sources":["../../../../src/dashboard/hooks/useTerminalSize.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,MAAM,OAAO,CAAC;AAO5C,MAAM,UAAU,eAAe;IAC7B,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,QAAQ,CAAe;QAC7C,OAAO,EAAE,OAAO,CAAC,MAAM,CAAC,OAAO,IAAI,EAAE;QACrC,IAAI,EAAE,OAAO,CAAC,MAAM,CAAC,IAAI,IAAI,EAAE;KAChC,CAAC,CAAC;IAEH,SAAS,CAAC,GAAG,EAAE;QACb,MAAM,YAAY,GAAG,GAAG,EAAE;YACxB,OAAO,CAAC;gBACN,OAAO,EAAE,OAAO,CAAC,MAAM,CAAC,OAAO,IAAI,EAAE;gBACrC,IAAI,EAAE,OAAO,CAAC,MAAM,CAAC,IAAI,IAAI,EAAE;aAChC,CAAC,CAAC;QACL,CAAC,CAAC;QAEF,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,QAAQ,EAAE,YAAY,CAAC,CAAC;QAC1C,OAAO,GAAG,EAAE;YACV,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,QAAQ,EAAE,YAAY,CAAC,CAAC;QAC7C,CAAC,CAAC;IACJ,CAAC,EAAE,EAAE,CAAC,CAAC;IAEP,OAAO,IAAI,CAAC;AACd,CAAC;AAED,MAAM,CAAC,MAAM,WAAW,GAAG,EAAE,CAAC;AAC9B,MAAM,CAAC,MAAM,QAAQ,GAAG,EAAE,CAAC","sourcesContent":["import { useState, useEffect } from 'react';\n\ninterface TerminalSize {\n columns: number;\n rows: number;\n}\n\nexport function useTerminalSize(): TerminalSize {\n const [size, setSize] = useState<TerminalSize>({\n columns: process.stdout.columns || 80,\n rows: process.stdout.rows || 24,\n });\n\n useEffect(() => {\n const handleResize = () => {\n setSize({\n columns: process.stdout.columns || 80,\n rows: process.stdout.rows || 24,\n });\n };\n\n process.stdout.on('resize', handleResize);\n return () => {\n process.stdout.off('resize', handleResize);\n };\n }, []);\n\n return size;\n}\n\nexport const MIN_COLUMNS = 80;\nexport const MIN_ROWS = 24;\n"]}
@@ -0,0 +1,6 @@
1
+ import type { WizardEventEmitter } from '../lib/events.js';
2
+ export interface DashboardOptions {
3
+ emitter: WizardEventEmitter;
4
+ }
5
+ export declare function startDashboard(options: DashboardOptions): Promise<void>;
6
+ export declare function stopDashboard(): Promise<void>;
@@ -0,0 +1,36 @@
1
+ let cleanup = null;
2
+ let isRunning = false;
3
+ // Enter alternate screen buffer for fullscreen TUI
4
+ function enterFullscreen() {
5
+ process.stdout.write('\x1b[?1049h'); // Enter alternate screen
6
+ process.stdout.write('\x1b[2J'); // Clear entire screen
7
+ process.stdout.write('\x1b[H'); // Move cursor to home position
8
+ process.stdout.write('\x1b[?25l'); // Hide cursor
9
+ }
10
+ // Exit alternate screen buffer
11
+ function exitFullscreen() {
12
+ process.stdout.write('\x1b[?25h'); // Show cursor
13
+ process.stdout.write('\x1b[?1049l'); // Exit alternate screen
14
+ }
15
+ export async function startDashboard(options) {
16
+ if (isRunning)
17
+ return; // Prevent double initialization
18
+ isRunning = true;
19
+ const { render } = await import('ink');
20
+ const { createElement } = await import('react');
21
+ const { Dashboard } = await import('./components/Dashboard.js');
22
+ enterFullscreen();
23
+ const instance = render(createElement(Dashboard, { emitter: options.emitter }), {
24
+ exitOnCtrlC: false,
25
+ });
26
+ cleanup = () => {
27
+ instance.unmount();
28
+ exitFullscreen();
29
+ isRunning = false;
30
+ };
31
+ }
32
+ export async function stopDashboard() {
33
+ cleanup?.();
34
+ cleanup = null;
35
+ }
36
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/dashboard/index.ts"],"names":[],"mappings":"AAMA,IAAI,OAAO,GAAwB,IAAI,CAAC;AACxC,IAAI,SAAS,GAAG,KAAK,CAAC;AAEtB,mDAAmD;AACnD,SAAS,eAAe;IACtB,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,yBAAyB;IAC9D,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,sBAAsB;IACvD,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,+BAA+B;IAC/D,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC,CAAC,cAAc;AACnD,CAAC;AAED,+BAA+B;AAC/B,SAAS,cAAc;IACrB,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC,CAAC,cAAc;IACjD,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,wBAAwB;AAC/D,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,cAAc,CAAC,OAAyB;IAC5D,IAAI,SAAS;QAAE,OAAO,CAAC,gCAAgC;IACvD,SAAS,GAAG,IAAI,CAAC;IAEjB,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,MAAM,CAAC,KAAK,CAAC,CAAC;IACvC,MAAM,EAAE,aAAa,EAAE,GAAG,MAAM,MAAM,CAAC,OAAO,CAAC,CAAC;IAChD,MAAM,EAAE,SAAS,EAAE,GAAG,MAAM,MAAM,CAAC,2BAA2B,CAAC,CAAC;IAEhE,eAAe,EAAE,CAAC;IAElB,MAAM,QAAQ,GAAG,MAAM,CAAC,aAAa,CAAC,SAAS,EAAE,EAAE,OAAO,EAAE,OAAO,CAAC,OAAO,EAAE,CAAC,EAAE;QAC9E,WAAW,EAAE,KAAK;KACnB,CAAC,CAAC;IACH,OAAO,GAAG,GAAG,EAAE;QACb,QAAQ,CAAC,OAAO,EAAE,CAAC;QACnB,cAAc,EAAE,CAAC;QACjB,SAAS,GAAG,KAAK,CAAC;IACpB,CAAC,CAAC;AACJ,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,aAAa;IACjC,OAAO,EAAE,EAAE,CAAC;IACZ,OAAO,GAAG,IAAI,CAAC;AACjB,CAAC","sourcesContent":["import type { WizardEventEmitter } from '../lib/events.js';\n\nexport interface DashboardOptions {\n emitter: WizardEventEmitter;\n}\n\nlet cleanup: (() => void) | null = null;\nlet isRunning = false;\n\n// Enter alternate screen buffer for fullscreen TUI\nfunction enterFullscreen(): void {\n process.stdout.write('\\x1b[?1049h'); // Enter alternate screen\n process.stdout.write('\\x1b[2J'); // Clear entire screen\n process.stdout.write('\\x1b[H'); // Move cursor to home position\n process.stdout.write('\\x1b[?25l'); // Hide cursor\n}\n\n// Exit alternate screen buffer\nfunction exitFullscreen(): void {\n process.stdout.write('\\x1b[?25h'); // Show cursor\n process.stdout.write('\\x1b[?1049l'); // Exit alternate screen\n}\n\nexport async function startDashboard(options: DashboardOptions): Promise<void> {\n if (isRunning) return; // Prevent double initialization\n isRunning = true;\n\n const { render } = await import('ink');\n const { createElement } = await import('react');\n const { Dashboard } = await import('./components/Dashboard.js');\n\n enterFullscreen();\n\n const instance = render(createElement(Dashboard, { emitter: options.emitter }), {\n exitOnCtrlC: false,\n });\n cleanup = () => {\n instance.unmount();\n exitFullscreen();\n isRunning = false;\n };\n}\n\nexport async function stopDashboard(): Promise<void> {\n cleanup?.();\n cleanup = null;\n}\n"]}
@@ -0,0 +1,21 @@
1
+ export interface FileDiff {
2
+ path: string;
3
+ isNew: boolean;
4
+ changes: DiffLine[];
5
+ timestamp: number;
6
+ }
7
+ export interface DiffLine {
8
+ type: 'add' | 'remove' | 'unchanged';
9
+ content: string;
10
+ lineNumber?: number;
11
+ }
12
+ export type TokenType = 'keyword' | 'string' | 'comment' | 'number' | 'function' | 'type' | 'operator' | 'punctuation' | 'plain';
13
+ export interface Token {
14
+ type: TokenType;
15
+ value: string;
16
+ }
17
+ export declare function getLanguage(path: string): string;
18
+ export declare function tokenize(line: string, language: string): Token[];
19
+ export declare function computeDiff(path: string, oldContent: string | null, newContent: string): FileDiff;
20
+ export declare function filterWithContext(changes: DiffLine[], contextLines: number): DiffLine[];
21
+ export declare function formatDiffLine(change: DiffLine): string;
@@ -0,0 +1,271 @@
1
+ import { diffLines } from 'diff';
2
+ // Detect file language from path
3
+ export function getLanguage(path) {
4
+ const ext = path.split('.').pop()?.toLowerCase() || '';
5
+ const langMap = {
6
+ ts: 'typescript',
7
+ tsx: 'typescript',
8
+ js: 'javascript',
9
+ jsx: 'javascript',
10
+ mjs: 'javascript',
11
+ json: 'json',
12
+ css: 'css',
13
+ scss: 'css',
14
+ html: 'html',
15
+ md: 'markdown',
16
+ env: 'env',
17
+ };
18
+ return langMap[ext] || 'plain';
19
+ }
20
+ // Keywords for JS/TS
21
+ const JS_KEYWORDS = new Set([
22
+ 'async',
23
+ 'await',
24
+ 'break',
25
+ 'case',
26
+ 'catch',
27
+ 'class',
28
+ 'const',
29
+ 'continue',
30
+ 'debugger',
31
+ 'default',
32
+ 'delete',
33
+ 'do',
34
+ 'else',
35
+ 'enum',
36
+ 'export',
37
+ 'extends',
38
+ 'false',
39
+ 'finally',
40
+ 'for',
41
+ 'from',
42
+ 'function',
43
+ 'if',
44
+ 'import',
45
+ 'in',
46
+ 'instanceof',
47
+ 'interface',
48
+ 'let',
49
+ 'new',
50
+ 'null',
51
+ 'of',
52
+ 'private',
53
+ 'protected',
54
+ 'public',
55
+ 'readonly',
56
+ 'return',
57
+ 'static',
58
+ 'super',
59
+ 'switch',
60
+ 'this',
61
+ 'throw',
62
+ 'true',
63
+ 'try',
64
+ 'type',
65
+ 'typeof',
66
+ 'undefined',
67
+ 'var',
68
+ 'void',
69
+ 'while',
70
+ 'with',
71
+ 'yield',
72
+ ]);
73
+ // Tokenize a line of code for syntax highlighting
74
+ export function tokenize(line, language) {
75
+ if (language === 'plain' || language === 'env') {
76
+ // For .env files, highlight key=value
77
+ if (language === 'env') {
78
+ const match = line.match(/^(\w+)(=)(.*)$/);
79
+ if (match) {
80
+ return [
81
+ { type: 'type', value: match[1] },
82
+ { type: 'operator', value: match[2] },
83
+ { type: 'string', value: match[3] },
84
+ ];
85
+ }
86
+ }
87
+ return [{ type: 'plain', value: line }];
88
+ }
89
+ const tokens = [];
90
+ let remaining = line;
91
+ while (remaining.length > 0) {
92
+ // Comments
93
+ const commentMatch = remaining.match(/^(\/\/.*|\/\*[\s\S]*?\*\/)/);
94
+ if (commentMatch) {
95
+ tokens.push({ type: 'comment', value: commentMatch[1] });
96
+ remaining = remaining.slice(commentMatch[1].length);
97
+ continue;
98
+ }
99
+ // Strings (single, double, template)
100
+ const stringMatch = remaining.match(/^("(?:[^"\\]|\\.)*"|'(?:[^'\\]|\\.)*'|`(?:[^`\\]|\\.)*`)/);
101
+ if (stringMatch) {
102
+ tokens.push({ type: 'string', value: stringMatch[1] });
103
+ remaining = remaining.slice(stringMatch[1].length);
104
+ continue;
105
+ }
106
+ // JSX/HTML tags
107
+ const tagMatch = remaining.match(/^(<\/?[A-Za-z][A-Za-z0-9]*)/);
108
+ if (tagMatch) {
109
+ tokens.push({ type: 'keyword', value: tagMatch[1] });
110
+ remaining = remaining.slice(tagMatch[1].length);
111
+ continue;
112
+ }
113
+ // Numbers
114
+ const numberMatch = remaining.match(/^(\d+\.?\d*)/);
115
+ if (numberMatch) {
116
+ tokens.push({ type: 'number', value: numberMatch[1] });
117
+ remaining = remaining.slice(numberMatch[1].length);
118
+ continue;
119
+ }
120
+ // Identifiers and keywords
121
+ const identMatch = remaining.match(/^([a-zA-Z_$][a-zA-Z0-9_$]*)/);
122
+ if (identMatch) {
123
+ const word = identMatch[1];
124
+ if (JS_KEYWORDS.has(word)) {
125
+ tokens.push({ type: 'keyword', value: word });
126
+ }
127
+ else if (word[0] === word[0].toUpperCase() && word !== word.toUpperCase()) {
128
+ // PascalCase = likely a type or component
129
+ tokens.push({ type: 'type', value: word });
130
+ }
131
+ else if (remaining.slice(word.length).match(/^\s*[(<]/)) {
132
+ // Followed by ( or < = function call
133
+ tokens.push({ type: 'function', value: word });
134
+ }
135
+ else {
136
+ tokens.push({ type: 'plain', value: word });
137
+ }
138
+ remaining = remaining.slice(word.length);
139
+ continue;
140
+ }
141
+ // Operators
142
+ const opMatch = remaining.match(/^(=>|===|!==|==|!=|<=|>=|&&|\|\||[+\-*/%=<>!&|^~?:])/);
143
+ if (opMatch) {
144
+ tokens.push({ type: 'operator', value: opMatch[1] });
145
+ remaining = remaining.slice(opMatch[1].length);
146
+ continue;
147
+ }
148
+ // Punctuation
149
+ const punctMatch = remaining.match(/^([{}[\]();,.])/);
150
+ if (punctMatch) {
151
+ tokens.push({ type: 'punctuation', value: punctMatch[1] });
152
+ remaining = remaining.slice(1);
153
+ continue;
154
+ }
155
+ // Whitespace and other characters
156
+ const wsMatch = remaining.match(/^(\s+)/);
157
+ if (wsMatch) {
158
+ tokens.push({ type: 'plain', value: wsMatch[1] });
159
+ remaining = remaining.slice(wsMatch[1].length);
160
+ continue;
161
+ }
162
+ // Single character fallback
163
+ tokens.push({ type: 'plain', value: remaining[0] });
164
+ remaining = remaining.slice(1);
165
+ }
166
+ return tokens;
167
+ }
168
+ export function computeDiff(path, oldContent, newContent) {
169
+ const isNew = oldContent === null || oldContent === '';
170
+ if (isNew) {
171
+ // New file - all lines are additions
172
+ const lines = newContent.split('\n');
173
+ return {
174
+ path,
175
+ isNew: true,
176
+ changes: lines.map((content, i) => ({
177
+ type: 'add',
178
+ content,
179
+ lineNumber: i + 1,
180
+ })),
181
+ timestamp: Date.now(),
182
+ };
183
+ }
184
+ // Compute unified diff
185
+ const changes = diffLines(oldContent, newContent);
186
+ const resultLines = [];
187
+ let lineNumber = 1;
188
+ for (const change of changes) {
189
+ const lines = change.value.split('\n').filter((l, i, arr) =>
190
+ // Filter out empty last line from split
191
+ i < arr.length - 1 || l !== '');
192
+ for (const line of lines) {
193
+ if (change.added) {
194
+ resultLines.push({
195
+ type: 'add',
196
+ content: line,
197
+ lineNumber: lineNumber++,
198
+ });
199
+ }
200
+ else if (change.removed) {
201
+ resultLines.push({ type: 'remove', content: line });
202
+ }
203
+ else {
204
+ resultLines.push({
205
+ type: 'unchanged',
206
+ content: line,
207
+ lineNumber: lineNumber++,
208
+ });
209
+ }
210
+ }
211
+ }
212
+ // Truncate very large diffs
213
+ const MAX_LINES = 500;
214
+ if (resultLines.length > MAX_LINES) {
215
+ return {
216
+ path,
217
+ isNew: false,
218
+ changes: [
219
+ ...resultLines.slice(0, MAX_LINES),
220
+ {
221
+ type: 'unchanged',
222
+ content: `... (${resultLines.length - MAX_LINES} more lines)`,
223
+ },
224
+ ],
225
+ timestamp: Date.now(),
226
+ };
227
+ }
228
+ return {
229
+ path,
230
+ isNew: false,
231
+ changes: resultLines,
232
+ timestamp: Date.now(),
233
+ };
234
+ }
235
+ // Helper: filter diff lines to show only changes with context
236
+ export function filterWithContext(changes, contextLines) {
237
+ const result = [];
238
+ const indices = new Set();
239
+ // Mark indices of changed lines and their context
240
+ changes.forEach((change, i) => {
241
+ if (change.type === 'add' || change.type === 'remove') {
242
+ for (let j = Math.max(0, i - contextLines); j <= Math.min(changes.length - 1, i + contextLines); j++) {
243
+ indices.add(j);
244
+ }
245
+ }
246
+ });
247
+ // Build filtered result
248
+ let lastIncluded = -2;
249
+ for (let i = 0; i < changes.length; i++) {
250
+ if (indices.has(i)) {
251
+ if (lastIncluded < i - 1 && lastIncluded >= 0) {
252
+ result.push({ type: 'unchanged', content: '...' });
253
+ }
254
+ result.push(changes[i]);
255
+ lastIncluded = i;
256
+ }
257
+ }
258
+ return result;
259
+ }
260
+ // Helper: format diff line with prefix
261
+ export function formatDiffLine(change) {
262
+ switch (change.type) {
263
+ case 'add':
264
+ return `+ ${change.content}`;
265
+ case 'remove':
266
+ return `- ${change.content}`;
267
+ default:
268
+ return ` ${change.content}`;
269
+ }
270
+ }
271
+ //# sourceMappingURL=diff-utils.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"diff-utils.js","sourceRoot":"","sources":["../../../../src/dashboard/lib/diff-utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { diffLines } from 'diff';\n\nexport interface FileDiff {\n path: string;\n isNew: boolean;\n changes: DiffLine[];\n timestamp: number;\n}\n\nexport interface DiffLine {\n type: 'add' | 'remove' | 'unchanged';\n content: string;\n lineNumber?: number;\n}\n\n// Token types for syntax highlighting\nexport type TokenType =\n | 'keyword'\n | 'string'\n | 'comment'\n | 'number'\n | 'function'\n | 'type'\n | 'operator'\n | 'punctuation'\n | 'plain';\n\nexport interface Token {\n type: TokenType;\n value: string;\n}\n\n// Detect file language from path\nexport function getLanguage(path: string): string {\n const ext = path.split('.').pop()?.toLowerCase() || '';\n const langMap: Record<string, string> = {\n ts: 'typescript',\n tsx: 'typescript',\n js: 'javascript',\n jsx: 'javascript',\n mjs: 'javascript',\n json: 'json',\n css: 'css',\n scss: 'css',\n html: 'html',\n md: 'markdown',\n env: 'env',\n };\n return langMap[ext] || 'plain';\n}\n\n// Keywords for JS/TS\nconst JS_KEYWORDS = new Set([\n 'async',\n 'await',\n 'break',\n 'case',\n 'catch',\n 'class',\n 'const',\n 'continue',\n 'debugger',\n 'default',\n 'delete',\n 'do',\n 'else',\n 'enum',\n 'export',\n 'extends',\n 'false',\n 'finally',\n 'for',\n 'from',\n 'function',\n 'if',\n 'import',\n 'in',\n 'instanceof',\n 'interface',\n 'let',\n 'new',\n 'null',\n 'of',\n 'private',\n 'protected',\n 'public',\n 'readonly',\n 'return',\n 'static',\n 'super',\n 'switch',\n 'this',\n 'throw',\n 'true',\n 'try',\n 'type',\n 'typeof',\n 'undefined',\n 'var',\n 'void',\n 'while',\n 'with',\n 'yield',\n]);\n\n// Tokenize a line of code for syntax highlighting\nexport function tokenize(line: string, language: string): Token[] {\n if (language === 'plain' || language === 'env') {\n // For .env files, highlight key=value\n if (language === 'env') {\n const match = line.match(/^(\\w+)(=)(.*)$/);\n if (match) {\n return [\n { type: 'type', value: match[1] },\n { type: 'operator', value: match[2] },\n { type: 'string', value: match[3] },\n ];\n }\n }\n return [{ type: 'plain', value: line }];\n }\n\n const tokens: Token[] = [];\n let remaining = line;\n\n while (remaining.length > 0) {\n // Comments\n const commentMatch = remaining.match(/^(\\/\\/.*|\\/\\*[\\s\\S]*?\\*\\/)/);\n if (commentMatch) {\n tokens.push({ type: 'comment', value: commentMatch[1] });\n remaining = remaining.slice(commentMatch[1].length);\n continue;\n }\n\n // Strings (single, double, template)\n const stringMatch = remaining.match(/^(\"(?:[^\"\\\\]|\\\\.)*\"|'(?:[^'\\\\]|\\\\.)*'|`(?:[^`\\\\]|\\\\.)*`)/);\n if (stringMatch) {\n tokens.push({ type: 'string', value: stringMatch[1] });\n remaining = remaining.slice(stringMatch[1].length);\n continue;\n }\n\n // JSX/HTML tags\n const tagMatch = remaining.match(/^(<\\/?[A-Za-z][A-Za-z0-9]*)/);\n if (tagMatch) {\n tokens.push({ type: 'keyword', value: tagMatch[1] });\n remaining = remaining.slice(tagMatch[1].length);\n continue;\n }\n\n // Numbers\n const numberMatch = remaining.match(/^(\\d+\\.?\\d*)/);\n if (numberMatch) {\n tokens.push({ type: 'number', value: numberMatch[1] });\n remaining = remaining.slice(numberMatch[1].length);\n continue;\n }\n\n // Identifiers and keywords\n const identMatch = remaining.match(/^([a-zA-Z_$][a-zA-Z0-9_$]*)/);\n if (identMatch) {\n const word = identMatch[1];\n if (JS_KEYWORDS.has(word)) {\n tokens.push({ type: 'keyword', value: word });\n } else if (word[0] === word[0].toUpperCase() && word !== word.toUpperCase()) {\n // PascalCase = likely a type or component\n tokens.push({ type: 'type', value: word });\n } else if (remaining.slice(word.length).match(/^\\s*[(<]/)) {\n // Followed by ( or < = function call\n tokens.push({ type: 'function', value: word });\n } else {\n tokens.push({ type: 'plain', value: word });\n }\n remaining = remaining.slice(word.length);\n continue;\n }\n\n // Operators\n const opMatch = remaining.match(/^(=>|===|!==|==|!=|<=|>=|&&|\\|\\||[+\\-*/%=<>!&|^~?:])/);\n if (opMatch) {\n tokens.push({ type: 'operator', value: opMatch[1] });\n remaining = remaining.slice(opMatch[1].length);\n continue;\n }\n\n // Punctuation\n const punctMatch = remaining.match(/^([{}[\\]();,.])/);\n if (punctMatch) {\n tokens.push({ type: 'punctuation', value: punctMatch[1] });\n remaining = remaining.slice(1);\n continue;\n }\n\n // Whitespace and other characters\n const wsMatch = remaining.match(/^(\\s+)/);\n if (wsMatch) {\n tokens.push({ type: 'plain', value: wsMatch[1] });\n remaining = remaining.slice(wsMatch[1].length);\n continue;\n }\n\n // Single character fallback\n tokens.push({ type: 'plain', value: remaining[0] });\n remaining = remaining.slice(1);\n }\n\n return tokens;\n}\n\nexport function computeDiff(path: string, oldContent: string | null, newContent: string): FileDiff {\n const isNew = oldContent === null || oldContent === '';\n\n if (isNew) {\n // New file - all lines are additions\n const lines = newContent.split('\\n');\n return {\n path,\n isNew: true,\n changes: lines.map((content, i) => ({\n type: 'add' as const,\n content,\n lineNumber: i + 1,\n })),\n timestamp: Date.now(),\n };\n }\n\n // Compute unified diff\n const changes = diffLines(oldContent, newContent);\n const resultLines: DiffLine[] = [];\n let lineNumber = 1;\n\n for (const change of changes) {\n const lines = change.value.split('\\n').filter(\n (l, i, arr) =>\n // Filter out empty last line from split\n i < arr.length - 1 || l !== '',\n );\n\n for (const line of lines) {\n if (change.added) {\n resultLines.push({\n type: 'add',\n content: line,\n lineNumber: lineNumber++,\n });\n } else if (change.removed) {\n resultLines.push({ type: 'remove', content: line });\n } else {\n resultLines.push({\n type: 'unchanged',\n content: line,\n lineNumber: lineNumber++,\n });\n }\n }\n }\n\n // Truncate very large diffs\n const MAX_LINES = 500;\n if (resultLines.length > MAX_LINES) {\n return {\n path,\n isNew: false,\n changes: [\n ...resultLines.slice(0, MAX_LINES),\n {\n type: 'unchanged' as const,\n content: `... (${resultLines.length - MAX_LINES} more lines)`,\n },\n ],\n timestamp: Date.now(),\n };\n }\n\n return {\n path,\n isNew: false,\n changes: resultLines,\n timestamp: Date.now(),\n };\n}\n\n// Helper: filter diff lines to show only changes with context\nexport function filterWithContext(changes: DiffLine[], contextLines: number): DiffLine[] {\n const result: DiffLine[] = [];\n const indices = new Set<number>();\n\n // Mark indices of changed lines and their context\n changes.forEach((change, i) => {\n if (change.type === 'add' || change.type === 'remove') {\n for (let j = Math.max(0, i - contextLines); j <= Math.min(changes.length - 1, i + contextLines); j++) {\n indices.add(j);\n }\n }\n });\n\n // Build filtered result\n let lastIncluded = -2;\n for (let i = 0; i < changes.length; i++) {\n if (indices.has(i)) {\n if (lastIncluded < i - 1 && lastIncluded >= 0) {\n result.push({ type: 'unchanged', content: '...' });\n }\n result.push(changes[i]);\n lastIncluded = i;\n }\n }\n\n return result;\n}\n\n// Helper: format diff line with prefix\nexport function formatDiffLine(change: DiffLine): string {\n switch (change.type) {\n case 'add':\n return `+ ${change.content}`;\n case 'remove':\n return `- ${change.content}`;\n default:\n return ` ${change.content}`;\n }\n}\n"]}
@@ -0,0 +1,20 @@
1
+ export declare const COLORS: {
2
+ brightest: string;
3
+ bright: string;
4
+ medium: string;
5
+ dark: string;
6
+ darker: string;
7
+ darkest: string;
8
+ };
9
+ export type AnimationMode = 'spin' | 'pulse';
10
+ export interface LogoFrame {
11
+ lines: string[];
12
+ color: string;
13
+ }
14
+ export declare function generateSpinFrames(): LogoFrame[];
15
+ export declare function generatePulseFrames(): LogoFrame[];
16
+ export declare function getFrames(mode: AnimationMode): LogoFrame[];
17
+ export declare const FRAME_DELAYS: {
18
+ readonly spin: 55;
19
+ readonly pulse: 80;
20
+ };