@swarmclawai/swarmclaw 0.2.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 (319) hide show
  1. package/README.md +577 -0
  2. package/bin/server-cmd.js +359 -0
  3. package/bin/swarmclaw.js +29 -0
  4. package/bin/swarmclaw.mjs +1504 -0
  5. package/next.config.ts +33 -0
  6. package/package.json +112 -0
  7. package/postcss.config.mjs +7 -0
  8. package/public/branding/swarmclaw-org-avatar.png +0 -0
  9. package/public/branding/swarmclaw-org-avatar.svg +58 -0
  10. package/public/file.svg +1 -0
  11. package/public/globe.svg +1 -0
  12. package/public/next.svg +1 -0
  13. package/public/screenshots/agents.png +0 -0
  14. package/public/screenshots/connectors.png +0 -0
  15. package/public/screenshots/dashboard.png +0 -0
  16. package/public/screenshots/new-session-openclaw.png +0 -0
  17. package/public/screenshots/providers.png +0 -0
  18. package/public/screenshots/schedules.png +0 -0
  19. package/public/screenshots/tasks.png +0 -0
  20. package/public/vercel.svg +1 -0
  21. package/public/window.svg +1 -0
  22. package/src/app/api/agents/[id]/route.ts +30 -0
  23. package/src/app/api/agents/[id]/thread/route.ts +66 -0
  24. package/src/app/api/agents/generate/route.ts +42 -0
  25. package/src/app/api/agents/route.ts +33 -0
  26. package/src/app/api/auth/route.ts +25 -0
  27. package/src/app/api/claude-skills/route.ts +42 -0
  28. package/src/app/api/clawhub/install/route.ts +39 -0
  29. package/src/app/api/clawhub/search/route.ts +11 -0
  30. package/src/app/api/connectors/[id]/route.ts +79 -0
  31. package/src/app/api/connectors/route.ts +60 -0
  32. package/src/app/api/credentials/[id]/route.ts +14 -0
  33. package/src/app/api/credentials/route.ts +31 -0
  34. package/src/app/api/daemon/health-check/route.ts +11 -0
  35. package/src/app/api/daemon/route.ts +22 -0
  36. package/src/app/api/dirs/pick/route.ts +60 -0
  37. package/src/app/api/dirs/route.ts +29 -0
  38. package/src/app/api/documents/[id]/route.ts +47 -0
  39. package/src/app/api/documents/route.ts +93 -0
  40. package/src/app/api/files/serve/route.ts +69 -0
  41. package/src/app/api/generate/info/route.ts +12 -0
  42. package/src/app/api/generate/route.ts +106 -0
  43. package/src/app/api/ip/route.ts +6 -0
  44. package/src/app/api/knowledge/[id]/route.ts +61 -0
  45. package/src/app/api/knowledge/route.ts +48 -0
  46. package/src/app/api/knowledge/upload/route.ts +86 -0
  47. package/src/app/api/logs/route.ts +65 -0
  48. package/src/app/api/mcp-servers/[id]/route.ts +32 -0
  49. package/src/app/api/mcp-servers/[id]/test/route.ts +23 -0
  50. package/src/app/api/mcp-servers/[id]/tools/route.ts +32 -0
  51. package/src/app/api/mcp-servers/route.ts +27 -0
  52. package/src/app/api/memory/[id]/route.ts +126 -0
  53. package/src/app/api/memory/maintenance/route.ts +63 -0
  54. package/src/app/api/memory/route.ts +111 -0
  55. package/src/app/api/memory-images/[filename]/route.ts +36 -0
  56. package/src/app/api/orchestrator/run/route.ts +43 -0
  57. package/src/app/api/plugins/install/route.ts +58 -0
  58. package/src/app/api/plugins/marketplace/route.ts +33 -0
  59. package/src/app/api/plugins/route.ts +21 -0
  60. package/src/app/api/preview-server/route.ts +339 -0
  61. package/src/app/api/providers/[id]/models/route.ts +29 -0
  62. package/src/app/api/providers/[id]/route.ts +34 -0
  63. package/src/app/api/providers/configs/route.ts +7 -0
  64. package/src/app/api/providers/ollama/route.ts +30 -0
  65. package/src/app/api/providers/openclaw/health/route.ts +23 -0
  66. package/src/app/api/providers/route.ts +28 -0
  67. package/src/app/api/runs/[id]/route.ts +9 -0
  68. package/src/app/api/runs/route.ts +13 -0
  69. package/src/app/api/schedules/[id]/route.ts +28 -0
  70. package/src/app/api/schedules/[id]/run/route.ts +104 -0
  71. package/src/app/api/schedules/route.ts +78 -0
  72. package/src/app/api/secrets/[id]/route.ts +29 -0
  73. package/src/app/api/secrets/route.ts +42 -0
  74. package/src/app/api/sessions/[id]/browser/route.ts +13 -0
  75. package/src/app/api/sessions/[id]/chat/route.ts +96 -0
  76. package/src/app/api/sessions/[id]/clear/route.ts +19 -0
  77. package/src/app/api/sessions/[id]/deploy/route.ts +34 -0
  78. package/src/app/api/sessions/[id]/devserver/route.ts +69 -0
  79. package/src/app/api/sessions/[id]/mailbox/route.ts +70 -0
  80. package/src/app/api/sessions/[id]/main-loop/route.ts +94 -0
  81. package/src/app/api/sessions/[id]/messages/route.ts +9 -0
  82. package/src/app/api/sessions/[id]/retry/route.ts +28 -0
  83. package/src/app/api/sessions/[id]/route.ts +103 -0
  84. package/src/app/api/sessions/[id]/stop/route.ts +13 -0
  85. package/src/app/api/sessions/heartbeat/route.ts +26 -0
  86. package/src/app/api/sessions/route.ts +85 -0
  87. package/src/app/api/settings/route.ts +58 -0
  88. package/src/app/api/setup/check-provider/route.ts +326 -0
  89. package/src/app/api/setup/doctor/route.ts +250 -0
  90. package/src/app/api/skills/[id]/route.ts +40 -0
  91. package/src/app/api/skills/import/route.ts +69 -0
  92. package/src/app/api/skills/route.ts +28 -0
  93. package/src/app/api/tasks/[id]/route.ts +102 -0
  94. package/src/app/api/tasks/route.ts +115 -0
  95. package/src/app/api/tts/route.ts +40 -0
  96. package/src/app/api/upload/route.ts +18 -0
  97. package/src/app/api/uploads/[filename]/route.ts +59 -0
  98. package/src/app/api/usage/route.ts +35 -0
  99. package/src/app/api/version/route.ts +81 -0
  100. package/src/app/api/version/update/route.ts +95 -0
  101. package/src/app/api/webhooks/[id]/history/route.ts +13 -0
  102. package/src/app/api/webhooks/[id]/route.ts +204 -0
  103. package/src/app/api/webhooks/route.ts +37 -0
  104. package/src/app/favicon.ico +0 -0
  105. package/src/app/globals.css +370 -0
  106. package/src/app/layout.tsx +52 -0
  107. package/src/app/page.tsx +172 -0
  108. package/src/cli/index.js +1232 -0
  109. package/src/cli/index.test.js +281 -0
  110. package/src/cli/index.ts +1158 -0
  111. package/src/cli/spec.js +284 -0
  112. package/src/components/agents/agent-card.tsx +219 -0
  113. package/src/components/agents/agent-chat-list.tsx +165 -0
  114. package/src/components/agents/agent-list.tsx +110 -0
  115. package/src/components/agents/agent-sheet.tsx +1220 -0
  116. package/src/components/auth/access-key-gate.tsx +248 -0
  117. package/src/components/auth/setup-wizard.tsx +940 -0
  118. package/src/components/auth/user-picker.tsx +88 -0
  119. package/src/components/chat/chat-area.tsx +406 -0
  120. package/src/components/chat/chat-header.tsx +491 -0
  121. package/src/components/chat/chat-tool-toggles.tsx +161 -0
  122. package/src/components/chat/code-block.tsx +146 -0
  123. package/src/components/chat/dev-server-bar.tsx +39 -0
  124. package/src/components/chat/message-bubble.tsx +486 -0
  125. package/src/components/chat/message-list.tsx +299 -0
  126. package/src/components/chat/session-debug-panel.tsx +196 -0
  127. package/src/components/chat/streaming-bubble.tsx +85 -0
  128. package/src/components/chat/thinking-indicator.tsx +26 -0
  129. package/src/components/chat/tool-call-bubble.tsx +438 -0
  130. package/src/components/chat/tool-request-banner.tsx +103 -0
  131. package/src/components/connectors/connector-list.tsx +196 -0
  132. package/src/components/connectors/connector-sheet.tsx +804 -0
  133. package/src/components/input/chat-input.tsx +235 -0
  134. package/src/components/knowledge/knowledge-list.tsx +206 -0
  135. package/src/components/knowledge/knowledge-sheet.tsx +316 -0
  136. package/src/components/layout/app-layout.tsx +1016 -0
  137. package/src/components/layout/daemon-indicator.tsx +56 -0
  138. package/src/components/layout/mobile-header.tsx +31 -0
  139. package/src/components/layout/network-banner.tsx +17 -0
  140. package/src/components/layout/update-banner.tsx +130 -0
  141. package/src/components/logs/log-list.tsx +358 -0
  142. package/src/components/mcp-servers/mcp-server-list.tsx +122 -0
  143. package/src/components/mcp-servers/mcp-server-sheet.tsx +243 -0
  144. package/src/components/memory/memory-card.tsx +63 -0
  145. package/src/components/memory/memory-detail.tsx +339 -0
  146. package/src/components/memory/memory-list.tsx +198 -0
  147. package/src/components/memory/memory-sheet.tsx +70 -0
  148. package/src/components/plugins/plugin-list.tsx +60 -0
  149. package/src/components/plugins/plugin-sheet.tsx +311 -0
  150. package/src/components/providers/provider-list.tsx +96 -0
  151. package/src/components/providers/provider-sheet.tsx +542 -0
  152. package/src/components/runs/run-list.tsx +231 -0
  153. package/src/components/schedules/schedule-card.tsx +63 -0
  154. package/src/components/schedules/schedule-list.tsx +76 -0
  155. package/src/components/schedules/schedule-sheet.tsx +336 -0
  156. package/src/components/secrets/secret-sheet.tsx +180 -0
  157. package/src/components/secrets/secrets-list.tsx +91 -0
  158. package/src/components/sessions/new-session-sheet.tsx +478 -0
  159. package/src/components/sessions/session-card.tsx +144 -0
  160. package/src/components/sessions/session-list.tsx +202 -0
  161. package/src/components/shared/ai-gen-block.tsx +77 -0
  162. package/src/components/shared/avatar.tsx +48 -0
  163. package/src/components/shared/bottom-sheet.tsx +30 -0
  164. package/src/components/shared/confirm-dialog.tsx +47 -0
  165. package/src/components/shared/connector-platform-icon.tsx +113 -0
  166. package/src/components/shared/dir-browser.tsx +285 -0
  167. package/src/components/shared/dropdown.tsx +55 -0
  168. package/src/components/shared/icon-button.tsx +25 -0
  169. package/src/components/shared/settings/plugin-manager.tsx +207 -0
  170. package/src/components/shared/settings/section-capability-policy.tsx +93 -0
  171. package/src/components/shared/settings/section-embedding.tsx +99 -0
  172. package/src/components/shared/settings/section-heartbeat.tsx +168 -0
  173. package/src/components/shared/settings/section-memory.tsx +77 -0
  174. package/src/components/shared/settings/section-orchestrator.tsx +108 -0
  175. package/src/components/shared/settings/section-providers.tsx +181 -0
  176. package/src/components/shared/settings/section-runtime-loop.tsx +183 -0
  177. package/src/components/shared/settings/section-secrets.tsx +132 -0
  178. package/src/components/shared/settings/section-user-preferences.tsx +24 -0
  179. package/src/components/shared/settings/section-voice.tsx +53 -0
  180. package/src/components/shared/settings/settings-sheet.tsx +88 -0
  181. package/src/components/shared/settings/types.ts +7 -0
  182. package/src/components/shared/settings/utils.ts +13 -0
  183. package/src/components/shared/settings-sheet.tsx +1 -0
  184. package/src/components/shared/skeleton.tsx +19 -0
  185. package/src/components/shared/usage-badge.tsx +28 -0
  186. package/src/components/skills/clawhub-browser.tsx +225 -0
  187. package/src/components/skills/skill-list.tsx +70 -0
  188. package/src/components/skills/skill-sheet.tsx +254 -0
  189. package/src/components/tasks/task-board.tsx +96 -0
  190. package/src/components/tasks/task-card.tsx +179 -0
  191. package/src/components/tasks/task-column.tsx +73 -0
  192. package/src/components/tasks/task-list.tsx +118 -0
  193. package/src/components/tasks/task-sheet.tsx +415 -0
  194. package/src/components/ui/avatar.tsx +109 -0
  195. package/src/components/ui/badge.tsx +48 -0
  196. package/src/components/ui/button.tsx +64 -0
  197. package/src/components/ui/card.tsx +92 -0
  198. package/src/components/ui/dialog.tsx +158 -0
  199. package/src/components/ui/dropdown-menu.tsx +257 -0
  200. package/src/components/ui/input.tsx +21 -0
  201. package/src/components/ui/scroll-area.tsx +58 -0
  202. package/src/components/ui/select.tsx +190 -0
  203. package/src/components/ui/separator.tsx +28 -0
  204. package/src/components/ui/sheet.tsx +143 -0
  205. package/src/components/ui/sonner.tsx +22 -0
  206. package/src/components/ui/textarea.tsx +18 -0
  207. package/src/components/ui/tooltip.tsx +56 -0
  208. package/src/components/usage/usage-list.tsx +105 -0
  209. package/src/components/webhooks/webhook-list.tsx +166 -0
  210. package/src/components/webhooks/webhook-sheet.tsx +402 -0
  211. package/src/hooks/use-auto-resize.ts +20 -0
  212. package/src/hooks/use-media-query.ts +21 -0
  213. package/src/hooks/use-speech-recognition.ts +83 -0
  214. package/src/instrumentation.ts +8 -0
  215. package/src/lib/agents.ts +13 -0
  216. package/src/lib/api-client.ts +100 -0
  217. package/src/lib/chat.ts +60 -0
  218. package/src/lib/memory.ts +42 -0
  219. package/src/lib/openclaw-endpoint.test.ts +48 -0
  220. package/src/lib/openclaw-endpoint.ts +67 -0
  221. package/src/lib/provider-config.ts +13 -0
  222. package/src/lib/providers/anthropic.ts +135 -0
  223. package/src/lib/providers/claude-cli.ts +202 -0
  224. package/src/lib/providers/codex-cli.ts +260 -0
  225. package/src/lib/providers/index.ts +351 -0
  226. package/src/lib/providers/ollama.ts +131 -0
  227. package/src/lib/providers/openai.ts +164 -0
  228. package/src/lib/providers/openclaw.ts +330 -0
  229. package/src/lib/providers/opencode-cli.ts +164 -0
  230. package/src/lib/runtime-loop.ts +15 -0
  231. package/src/lib/schedule-dedupe.test.ts +84 -0
  232. package/src/lib/schedule-dedupe.ts +174 -0
  233. package/src/lib/schedule-name.ts +62 -0
  234. package/src/lib/schedules.ts +16 -0
  235. package/src/lib/server/agent-registry.ts +70 -0
  236. package/src/lib/server/api-routes.test.ts +362 -0
  237. package/src/lib/server/autonomy-contract.ts +200 -0
  238. package/src/lib/server/build-llm.ts +155 -0
  239. package/src/lib/server/capability-router.test.ts +21 -0
  240. package/src/lib/server/capability-router.ts +172 -0
  241. package/src/lib/server/chat-execution.ts +894 -0
  242. package/src/lib/server/clawhub-client.test.ts +161 -0
  243. package/src/lib/server/clawhub-client.ts +26 -0
  244. package/src/lib/server/connectors/connector-routing.test.ts +243 -0
  245. package/src/lib/server/connectors/discord.ts +116 -0
  246. package/src/lib/server/connectors/googlechat.ts +66 -0
  247. package/src/lib/server/connectors/manager.ts +559 -0
  248. package/src/lib/server/connectors/matrix.ts +78 -0
  249. package/src/lib/server/connectors/media.ts +149 -0
  250. package/src/lib/server/connectors/openclaw.test.ts +375 -0
  251. package/src/lib/server/connectors/openclaw.ts +1132 -0
  252. package/src/lib/server/connectors/signal.ts +183 -0
  253. package/src/lib/server/connectors/slack.ts +258 -0
  254. package/src/lib/server/connectors/teams.ts +94 -0
  255. package/src/lib/server/connectors/telegram.ts +221 -0
  256. package/src/lib/server/connectors/types.ts +62 -0
  257. package/src/lib/server/connectors/whatsapp.ts +349 -0
  258. package/src/lib/server/context-manager.ts +232 -0
  259. package/src/lib/server/cost.ts +31 -0
  260. package/src/lib/server/daemon-state.ts +354 -0
  261. package/src/lib/server/data-dir.ts +3 -0
  262. package/src/lib/server/embeddings.ts +111 -0
  263. package/src/lib/server/execution-log.ts +257 -0
  264. package/src/lib/server/gateway/protocol.test.ts +54 -0
  265. package/src/lib/server/gateway/protocol.ts +114 -0
  266. package/src/lib/server/heartbeat-service.ts +366 -0
  267. package/src/lib/server/knowledge-db.test.ts +441 -0
  268. package/src/lib/server/logger.ts +47 -0
  269. package/src/lib/server/main-agent-loop.ts +1017 -0
  270. package/src/lib/server/mcp-client.test.ts +342 -0
  271. package/src/lib/server/mcp-client.ts +130 -0
  272. package/src/lib/server/memory-db.ts +1078 -0
  273. package/src/lib/server/memory-graph.test.ts +153 -0
  274. package/src/lib/server/memory-graph.ts +138 -0
  275. package/src/lib/server/openclaw-health.ts +245 -0
  276. package/src/lib/server/orchestrator-lg.ts +431 -0
  277. package/src/lib/server/orchestrator.ts +364 -0
  278. package/src/lib/server/playwright-proxy.mjs +70 -0
  279. package/src/lib/server/plugins.ts +229 -0
  280. package/src/lib/server/process-manager.ts +327 -0
  281. package/src/lib/server/provider-health.ts +113 -0
  282. package/src/lib/server/queue.ts +859 -0
  283. package/src/lib/server/runtime-settings.ts +119 -0
  284. package/src/lib/server/scheduler.ts +196 -0
  285. package/src/lib/server/session-mailbox.ts +129 -0
  286. package/src/lib/server/session-run-manager.ts +512 -0
  287. package/src/lib/server/session-tools/connector.ts +124 -0
  288. package/src/lib/server/session-tools/context-mgmt.ts +103 -0
  289. package/src/lib/server/session-tools/context.ts +114 -0
  290. package/src/lib/server/session-tools/crud.ts +673 -0
  291. package/src/lib/server/session-tools/delegate.ts +708 -0
  292. package/src/lib/server/session-tools/file.ts +264 -0
  293. package/src/lib/server/session-tools/index.ts +164 -0
  294. package/src/lib/server/session-tools/memory.ts +230 -0
  295. package/src/lib/server/session-tools/session-info.ts +422 -0
  296. package/src/lib/server/session-tools/session-tools-wiring.test.ts +166 -0
  297. package/src/lib/server/session-tools/shell.ts +171 -0
  298. package/src/lib/server/session-tools/web.ts +408 -0
  299. package/src/lib/server/session-tools.ts +9 -0
  300. package/src/lib/server/skills-normalize.ts +130 -0
  301. package/src/lib/server/storage-mcp.test.ts +161 -0
  302. package/src/lib/server/storage.ts +670 -0
  303. package/src/lib/server/stream-agent-chat.ts +571 -0
  304. package/src/lib/server/task-reports.ts +122 -0
  305. package/src/lib/server/task-result.ts +161 -0
  306. package/src/lib/server/task-validation.test.ts +27 -0
  307. package/src/lib/server/task-validation.ts +90 -0
  308. package/src/lib/server/tool-capability-policy.test.ts +58 -0
  309. package/src/lib/server/tool-capability-policy.ts +262 -0
  310. package/src/lib/sessions.ts +68 -0
  311. package/src/lib/tasks.ts +20 -0
  312. package/src/lib/tts.ts +42 -0
  313. package/src/lib/upload.ts +10 -0
  314. package/src/lib/utils.ts +6 -0
  315. package/src/proxy.ts +43 -0
  316. package/src/stores/use-app-store.ts +468 -0
  317. package/src/stores/use-chat-store.ts +323 -0
  318. package/src/types/index.ts +621 -0
  319. package/tsconfig.json +34 -0
@@ -0,0 +1,370 @@
1
+ @import "tailwindcss";
2
+ @import "tw-animate-css";
3
+ @import "shadcn/tailwind.css";
4
+ @import "highlight.js/styles/github-dark-dimmed.min.css";
5
+
6
+ @custom-variant dark (&:is(.dark *));
7
+
8
+ @theme inline {
9
+ --color-background: var(--background);
10
+ --color-foreground: var(--foreground);
11
+ --font-sans: var(--font-dm-sans), system-ui, sans-serif;
12
+ --font-display: var(--font-sora), system-ui, sans-serif;
13
+ --font-mono: var(--font-jetbrains-mono), 'SF Mono', monospace;
14
+ --color-sidebar-ring: var(--sidebar-ring);
15
+ --color-sidebar-border: var(--sidebar-border);
16
+ --color-sidebar-accent-foreground: var(--sidebar-accent-foreground);
17
+ --color-sidebar-accent: var(--sidebar-accent);
18
+ --color-sidebar-primary-foreground: var(--sidebar-primary-foreground);
19
+ --color-sidebar-primary: var(--sidebar-primary);
20
+ --color-sidebar-foreground: var(--sidebar-foreground);
21
+ --color-sidebar: var(--sidebar);
22
+ --color-chart-5: var(--chart-5);
23
+ --color-chart-4: var(--chart-4);
24
+ --color-chart-3: var(--chart-3);
25
+ --color-chart-2: var(--chart-2);
26
+ --color-chart-1: var(--chart-1);
27
+ --color-ring: var(--ring);
28
+ --color-input: var(--input);
29
+ --color-border: var(--border);
30
+ --color-destructive: var(--destructive);
31
+ --color-accent-foreground: var(--accent-foreground);
32
+ --color-accent: var(--accent);
33
+ --color-muted-foreground: var(--muted-foreground);
34
+ --color-muted: var(--muted);
35
+ --color-secondary-foreground: var(--secondary-foreground);
36
+ --color-secondary: var(--secondary);
37
+ --color-primary-foreground: var(--primary-foreground);
38
+ --color-primary: var(--primary);
39
+ --color-popover-foreground: var(--popover-foreground);
40
+ --color-popover: var(--popover);
41
+ --color-card-foreground: var(--card-foreground);
42
+ --color-card: var(--card);
43
+ --radius-sm: calc(var(--radius) - 4px);
44
+ --radius-md: calc(var(--radius) - 2px);
45
+ --radius-lg: var(--radius);
46
+ --radius-xl: calc(var(--radius) + 4px);
47
+ --radius-2xl: calc(var(--radius) + 8px);
48
+ --radius-3xl: calc(var(--radius) + 12px);
49
+ --radius-4xl: calc(var(--radius) + 16px);
50
+
51
+ /* ===== Midnight Glass Palette ===== */
52
+ --color-bg: #08080d;
53
+ --color-raised: #0d0d14;
54
+ --color-surface: #13131e;
55
+ --color-surface-2: #1e1e30;
56
+ --color-surface-3: #23233a;
57
+ --color-border-hi: rgba(255,255,255,0.07);
58
+ --color-border-focus: rgba(99,102,241,0.5);
59
+ --color-text: #e2e2ec;
60
+ --color-text-2: #8e8ea8;
61
+ --color-text-3: #5c5c78;
62
+ --color-accent-soft: rgba(99,102,241,0.08);
63
+ --color-accent-glow: rgba(99,102,241,0.18);
64
+ --color-accent-bright: #818CF8;
65
+ --color-user-text: #fff;
66
+ --color-success: #34D399;
67
+ --color-success-soft: rgba(52,211,153,0.08);
68
+ --color-danger: #F43F5E;
69
+ --color-danger-soft: rgba(244,63,94,0.08);
70
+ --color-shereen: #EC4899;
71
+ --color-user-bubble: #4338CA;
72
+ --color-user-bubble-2: #6366F1;
73
+ --color-ai-bubble: #12121e;
74
+ --color-glass: rgba(16,16,28,0.75);
75
+ --color-glass-border: rgba(255,255,255,0.05);
76
+ --radius-pill: 9999px;
77
+ --ease-out-expo: cubic-bezier(0.16, 1, 0.3, 1);
78
+ --ease-spring: cubic-bezier(0.34, 1.56, 0.64, 1);
79
+ }
80
+
81
+ :root {
82
+ --radius: 0.625rem;
83
+ --background: #08080d;
84
+ --foreground: #e2e2ec;
85
+ --card: #0d0d14;
86
+ --card-foreground: #e2e2ec;
87
+ --popover: #0d0d14;
88
+ --popover-foreground: #e2e2ec;
89
+ --primary: #6366F1;
90
+ --primary-foreground: #ffffff;
91
+ --secondary: #13131e;
92
+ --secondary-foreground: #e2e2ec;
93
+ --muted: #13131e;
94
+ --muted-foreground: #8e8ea8;
95
+ --accent: #6366F1;
96
+ --accent-foreground: #ffffff;
97
+ --destructive: #F43F5E;
98
+ --border: rgba(255,255,255,0.04);
99
+ --input: rgba(255,255,255,0.04);
100
+ --ring: rgba(99,102,241,0.4);
101
+ --sidebar: #0d0d14;
102
+ --sidebar-foreground: #e2e2ec;
103
+ --sidebar-primary: #6366F1;
104
+ --sidebar-primary-foreground: #ffffff;
105
+ --sidebar-accent: rgba(99,102,241,0.08);
106
+ --sidebar-accent-foreground: #e2e2ec;
107
+ --sidebar-border: rgba(255,255,255,0.04);
108
+ --sidebar-ring: rgba(99,102,241,0.4);
109
+ }
110
+
111
+ @layer base {
112
+ *, *::before, *::after { margin: 0; padding: 0; box-sizing: border-box; }
113
+ }
114
+
115
+ body {
116
+ font-family: var(--font-dm-sans), 'DM Sans', -apple-system, BlinkMacSystemFont, system-ui, sans-serif;
117
+ background: var(--color-bg);
118
+ color: var(--color-text);
119
+ height: 100vh;
120
+ height: 100dvh;
121
+ overflow: hidden;
122
+ -webkit-font-smoothing: antialiased;
123
+ -webkit-tap-highlight-color: transparent;
124
+ letter-spacing: -0.01em;
125
+ }
126
+
127
+ /* Scrollbar */
128
+ ::-webkit-scrollbar { width: 4px; }
129
+ ::-webkit-scrollbar-track { background: transparent; }
130
+ ::-webkit-scrollbar-thumb { background: rgba(255,255,255,0.04); border-radius: 4px; }
131
+ ::-webkit-scrollbar-thumb:hover { background: rgba(255,255,255,0.1); }
132
+
133
+ /* Selection */
134
+ ::selection { background: rgba(99,102,241,0.3); }
135
+
136
+ /* ===== Animations ===== */
137
+ @keyframes msg-in {
138
+ from { opacity: 0; transform: translateY(12px); }
139
+ to { opacity: 1; transform: translateY(0); }
140
+ }
141
+ @keyframes msg-in-right {
142
+ from { opacity: 0; transform: translateX(12px); }
143
+ to { opacity: 1; transform: translateX(0); }
144
+ }
145
+ @keyframes msg-in-left {
146
+ from { opacity: 0; transform: translateX(-12px); }
147
+ to { opacity: 1; transform: translateX(0); }
148
+ }
149
+ @keyframes blink { 0%,100% { opacity: 1; } 50% { opacity: 0; } }
150
+ @keyframes pulse { 0%,100% { opacity: 1; } 50% { opacity: 0.3; } }
151
+ @keyframes dot-bounce {
152
+ 0%,80%,100% { transform: translateY(0); opacity: 0.25; }
153
+ 40% { transform: translateY(-6px); opacity: 1; }
154
+ }
155
+ @keyframes mic-pulse {
156
+ 0% { box-shadow: 0 0 0 0 rgba(244,63,94,0.3); }
157
+ 70% { box-shadow: 0 0 0 14px rgba(244,63,94,0); }
158
+ 100% { box-shadow: 0 0 0 0 rgba(244,63,94,0); }
159
+ }
160
+ @keyframes sheet-up {
161
+ from { transform: translateY(100%) scale(0.97); opacity: 0; }
162
+ to { transform: translateY(0) scale(1); opacity: 1; }
163
+ }
164
+ @keyframes modal-in {
165
+ from { transform: scale(0.92) translateY(12px); opacity: 0; }
166
+ to { transform: scale(1) translateY(0); opacity: 1; }
167
+ }
168
+ @keyframes fade-in {
169
+ from { opacity: 0; transform: scale(0.97) translateY(6px); }
170
+ to { opacity: 1; transform: scale(1) translateY(0); }
171
+ }
172
+ @keyframes spin { to { transform: rotate(360deg); } }
173
+ @keyframes slide-in-left {
174
+ from { transform: translateX(-100%); }
175
+ to { transform: translateX(0); }
176
+ }
177
+ @keyframes panel-in {
178
+ from { opacity: 0; transform: translateX(-8px); }
179
+ to { opacity: 1; transform: translateX(0); }
180
+ }
181
+ @keyframes glow-pulse {
182
+ 0%, 100% { opacity: 0.4; transform: scale(1); }
183
+ 50% { opacity: 0.7; transform: scale(1.05); }
184
+ }
185
+ @keyframes sparkle-spin {
186
+ 0% { transform: rotate(0deg) scale(1); }
187
+ 25% { transform: rotate(90deg) scale(1.1); }
188
+ 50% { transform: rotate(180deg) scale(1); }
189
+ 75% { transform: rotate(270deg) scale(1.1); }
190
+ 100% { transform: rotate(360deg) scale(1); }
191
+ }
192
+ @keyframes float {
193
+ 0%, 100% { transform: translateY(0); }
194
+ 50% { transform: translateY(-8px); }
195
+ }
196
+ @keyframes shimmer {
197
+ 0% { background-position: -200% center; }
198
+ 100% { background-position: 200% center; }
199
+ }
200
+ @keyframes gradient-drift {
201
+ 0% { background-position: 0% 50%; }
202
+ 50% { background-position: 100% 50%; }
203
+ 100% { background-position: 0% 50%; }
204
+ }
205
+
206
+ /* ===== Markdown content ===== */
207
+ .msg-content pre {
208
+ background: #0a0a12 !important;
209
+ padding: 0 !important;
210
+ border-radius: 12px;
211
+ overflow: hidden;
212
+ margin: 12px 0;
213
+ border: 1px solid rgba(255,255,255,0.04);
214
+ }
215
+
216
+ .msg-content pre code {
217
+ display: block;
218
+ padding: 14px 16px;
219
+ overflow-x: auto;
220
+ font-size: 12.5px;
221
+ line-height: 1.6;
222
+ -webkit-overflow-scrolling: touch;
223
+ background: transparent !important;
224
+ }
225
+
226
+ .msg-content code {
227
+ font-family: var(--font-jetbrains-mono), 'JetBrains Mono', 'SF Mono', monospace;
228
+ font-size: 12.5px;
229
+ font-feature-settings: 'liga' on, 'calt' on;
230
+ }
231
+
232
+ .msg-content :not(pre) > code {
233
+ background: rgba(99,102,241,0.08);
234
+ padding: 2px 7px;
235
+ border-radius: 5px;
236
+ font-size: 0.87em;
237
+ color: var(--color-accent-bright);
238
+ }
239
+
240
+ .msg-content h2, .msg-content h3, .msg-content h4 {
241
+ font-family: var(--font-sora), 'Sora', system-ui, sans-serif;
242
+ font-size: 14px;
243
+ font-weight: 600;
244
+ margin: 16px 0 6px;
245
+ letter-spacing: -0.03em;
246
+ color: var(--color-text);
247
+ }
248
+
249
+ .msg-content h2:first-child, .msg-content h3:first-child { margin-top: 0; }
250
+
251
+ .msg-content p { margin-bottom: 10px; line-height: 1.7; }
252
+ .msg-content p:last-child { margin-bottom: 0; }
253
+
254
+ .msg-content ul, .msg-content ol { margin: 8px 0 10px 20px; }
255
+ .msg-content li { margin-bottom: 4px; line-height: 1.65; }
256
+ .msg-content li::marker { color: var(--color-accent-bright); }
257
+
258
+ .msg-content a {
259
+ color: var(--color-accent-bright);
260
+ text-decoration: none;
261
+ border-bottom: 1px solid rgba(99,102,241,0.3);
262
+ transition: border-color 0.15s;
263
+ }
264
+ .msg-content a:hover { border-color: var(--color-accent-bright); }
265
+
266
+ .msg-content table { border-collapse: collapse; margin: 10px 0; font-size: 12.5px; width: 100%; }
267
+ .msg-content th, .msg-content td { border: 1px solid var(--color-border-hi); padding: 8px 12px; text-align: left; }
268
+ .msg-content th { background: var(--color-surface); font-weight: 600; font-size: 11px; text-transform: uppercase; letter-spacing: 0.05em; color: var(--color-text-2); }
269
+
270
+ .msg-content blockquote {
271
+ border-left: 2px solid var(--color-accent-bright);
272
+ padding-left: 16px;
273
+ margin: 10px 0;
274
+ color: var(--color-text-2);
275
+ }
276
+
277
+ .msg-content hr { border: none; height: 1px; background: var(--color-border-hi); margin: 16px 0; }
278
+
279
+ .msg-content img { max-width: 100%; border-radius: 12px; margin: 8px 0; border: 1px solid rgba(255,255,255,0.08); }
280
+
281
+ /* Streaming cursor */
282
+ .streaming-cursor::after {
283
+ content: '';
284
+ display: inline-block;
285
+ width: 2px;
286
+ height: 1.1em;
287
+ background: var(--color-accent-bright);
288
+ margin-left: 3px;
289
+ vertical-align: text-bottom;
290
+ animation: blink 0.8s step-end infinite;
291
+ border-radius: 1px;
292
+ box-shadow: 0 0 12px var(--color-accent-glow);
293
+ }
294
+
295
+ /* Focus glow */
296
+ .focus-glow:focus-within {
297
+ border-color: var(--color-border-focus) !important;
298
+ box-shadow: 0 0 0 3px rgba(99,102,241,0.06);
299
+ }
300
+
301
+ /* Glass effect */
302
+ .glass {
303
+ background: var(--color-glass);
304
+ backdrop-filter: blur(20px) saturate(1.2);
305
+ -webkit-backdrop-filter: blur(20px) saturate(1.2);
306
+ border: 1px solid var(--color-glass-border);
307
+ }
308
+
309
+ /* Icon rail button */
310
+ .rail-btn {
311
+ width: 40px;
312
+ height: 40px;
313
+ border-radius: 10px;
314
+ border: none;
315
+ background: transparent;
316
+ color: var(--color-text-3);
317
+ display: flex;
318
+ align-items: center;
319
+ justify-content: center;
320
+ cursor: pointer;
321
+ transition: all 0.2s cubic-bezier(0.16, 1, 0.3, 1);
322
+ }
323
+ .rail-btn:hover { background: rgba(255,255,255,0.04); color: var(--color-text-2); }
324
+ .rail-btn.active { background: var(--color-accent-soft); color: var(--color-accent-bright); }
325
+
326
+ /* Message bubble styles */
327
+ .bubble-user {
328
+ background: linear-gradient(135deg, var(--color-user-bubble) 0%, var(--color-user-bubble-2) 100%);
329
+ border-radius: 20px 20px 6px 20px;
330
+ color: white;
331
+ box-shadow: 0 2px 16px rgba(99,102,241,0.15);
332
+ }
333
+
334
+ .bubble-ai {
335
+ background: #1c1c2e;
336
+ border: 1px solid rgba(255,255,255,0.08);
337
+ border-radius: 20px 20px 20px 6px;
338
+ }
339
+
340
+ /* Suggestion card */
341
+ .suggestion-card {
342
+ position: relative;
343
+ overflow: hidden;
344
+ transition: all 0.25s cubic-bezier(0.16, 1, 0.3, 1);
345
+ }
346
+ .suggestion-card::before {
347
+ content: '';
348
+ position: absolute;
349
+ inset: 0;
350
+ background: linear-gradient(135deg, rgba(99,102,241,0.06), rgba(129,140,248,0.02));
351
+ opacity: 0;
352
+ transition: opacity 0.25s;
353
+ }
354
+ .suggestion-card:hover::before { opacity: 1; }
355
+ .suggestion-card:hover {
356
+ border-color: rgba(99,102,241,0.2);
357
+ transform: translateY(-2px);
358
+ box-shadow: 0 8px 32px rgba(0,0,0,0.3);
359
+ }
360
+
361
+ /* Display font utility */
362
+ .font-display {
363
+ font-family: var(--font-sora), 'Sora', system-ui, sans-serif;
364
+ }
365
+
366
+ @layer base {
367
+ * {
368
+ @apply border-border outline-ring/50;
369
+ }
370
+ }
@@ -0,0 +1,52 @@
1
+ import type { Metadata, Viewport } from "next"
2
+ import { Sora, DM_Sans, JetBrains_Mono } from "next/font/google"
3
+ import { TooltipProvider } from "@/components/ui/tooltip"
4
+ import { Toaster } from "@/components/ui/sonner"
5
+ import "./globals.css"
6
+
7
+ const sora = Sora({
8
+ variable: "--font-sora",
9
+ subsets: ["latin"],
10
+ weight: ["400", "500", "600", "700", "800"],
11
+ })
12
+
13
+ const dmSans = DM_Sans({
14
+ variable: "--font-dm-sans",
15
+ subsets: ["latin"],
16
+ weight: ["400", "500", "600", "700"],
17
+ })
18
+
19
+ const jetbrainsMono = JetBrains_Mono({
20
+ variable: "--font-jetbrains-mono",
21
+ subsets: ["latin"],
22
+ weight: ["400", "500"],
23
+ })
24
+
25
+ export const metadata: Metadata = {
26
+ title: "SwarmClaw",
27
+ description: "AI agent orchestration dashboard with multi-provider support",
28
+ }
29
+
30
+ export const viewport: Viewport = {
31
+ width: "device-width",
32
+ initialScale: 1,
33
+ maximumScale: 1,
34
+ viewportFit: "cover",
35
+ }
36
+
37
+ export default function RootLayout({
38
+ children,
39
+ }: Readonly<{
40
+ children: React.ReactNode
41
+ }>) {
42
+ return (
43
+ <html lang="en" className="dark">
44
+ <body className={`${sora.variable} ${dmSans.variable} ${jetbrainsMono.variable} antialiased`} cz-shortcut-listen="true">
45
+ <TooltipProvider>
46
+ {children}
47
+ <Toaster />
48
+ </TooltipProvider>
49
+ </body>
50
+ </html>
51
+ )
52
+ }
@@ -0,0 +1,172 @@
1
+ 'use client'
2
+
3
+ import { useEffect, useState, useCallback } from 'react'
4
+ import { useAppStore } from '@/stores/use-app-store'
5
+ import { initAudioContext } from '@/lib/tts'
6
+ import { getStoredAccessKey, clearStoredAccessKey, api } from '@/lib/api-client'
7
+ import { AccessKeyGate } from '@/components/auth/access-key-gate'
8
+ import { UserPicker } from '@/components/auth/user-picker'
9
+ import { SetupWizard } from '@/components/auth/setup-wizard'
10
+ import { AppLayout } from '@/components/layout/app-layout'
11
+
12
+ export default function Home() {
13
+ const currentUser = useAppStore((s) => s.currentUser)
14
+ const setUser = useAppStore((s) => s.setUser)
15
+ const hydrated = useAppStore((s) => s._hydrated)
16
+ const hydrate = useAppStore((s) => s.hydrate)
17
+ const loadNetworkInfo = useAppStore((s) => s.loadNetworkInfo)
18
+ const sessions = useAppStore((s) => s.sessions)
19
+ const loadSessions = useAppStore((s) => s.loadSessions)
20
+ const loadSettings = useAppStore((s) => s.loadSettings)
21
+
22
+ const [authChecked, setAuthChecked] = useState(false)
23
+ const [authenticated, setAuthenticated] = useState(false)
24
+ const [setupDone, setSetupDone] = useState<boolean | null>(null)
25
+
26
+ const checkAuth = useCallback(async () => {
27
+ const key = getStoredAccessKey()
28
+ if (!key) {
29
+ setAuthChecked(true)
30
+ setAuthenticated(false)
31
+ return
32
+ }
33
+
34
+ try {
35
+ const res = await fetch('/api/auth', {
36
+ method: 'POST',
37
+ headers: { 'Content-Type': 'application/json' },
38
+ body: JSON.stringify({ key }),
39
+ })
40
+ if (res.ok) {
41
+ setAuthenticated(true)
42
+ } else {
43
+ clearStoredAccessKey()
44
+ setAuthenticated(false)
45
+ }
46
+ } catch {
47
+ setAuthenticated(true)
48
+ }
49
+ setAuthChecked(true)
50
+ }, [])
51
+
52
+ // After auth, try to restore username from server settings
53
+ const syncUserFromServer = useCallback(async () => {
54
+ if (currentUser) return // already have a name locally
55
+ try {
56
+ const settings = await api<{ userName?: string }>('GET', '/settings')
57
+ if (settings.userName) {
58
+ setUser(settings.userName)
59
+ }
60
+ } catch { /* ignore */ }
61
+ }, [currentUser, setUser])
62
+
63
+ useEffect(() => {
64
+ hydrate()
65
+ }, [])
66
+
67
+ useEffect(() => {
68
+ if (hydrated) checkAuth()
69
+ }, [hydrated, checkAuth])
70
+
71
+ useEffect(() => {
72
+ if (!authenticated) return
73
+ syncUserFromServer()
74
+ loadNetworkInfo()
75
+ loadSettings()
76
+ loadSessions()
77
+ const interval = setInterval(loadSessions, 5000)
78
+ return () => clearInterval(interval)
79
+ }, [authenticated])
80
+
81
+ // Auto-select default agent's thread on load
82
+ useEffect(() => {
83
+ if (!authenticated || !currentUser) return
84
+ const state = useAppStore.getState()
85
+ // Only auto-select if no agent is selected yet
86
+ if (state.currentAgentId) return
87
+
88
+ // Load agents and select 'default' agent
89
+ let cancelled = false
90
+ ;(async () => {
91
+ try {
92
+ await state.loadAgents()
93
+ if (cancelled) return
94
+ const agents = useAppStore.getState().agents
95
+ // Try 'default' agent first, then fall back to first agent
96
+ const defaultAgent = agents['default'] || Object.values(agents)[0]
97
+ if (defaultAgent) {
98
+ await useAppStore.getState().setCurrentAgent(defaultAgent.id)
99
+ }
100
+ } catch { /* ignore */ }
101
+ })()
102
+ return () => { cancelled = true }
103
+ }, [authenticated, currentUser])
104
+
105
+ // Keep __main__ session for backward compat — create if missing
106
+ useEffect(() => {
107
+ if (!authenticated || !currentUser) return
108
+ const sessionList = Object.values(sessions)
109
+ const mainSession = sessionList.find((s: any) => s.name === '__main__' && s.user === currentUser)
110
+ if (mainSession) return
111
+ let cancelled = false
112
+ ;(async () => {
113
+ try {
114
+ const mainId = `main-${currentUser}`
115
+ await api<any>('POST', '/sessions', {
116
+ id: mainId,
117
+ name: '__main__',
118
+ user: currentUser,
119
+ agentId: 'default',
120
+ heartbeatEnabled: true,
121
+ })
122
+ if (!cancelled) await loadSessions()
123
+ } catch { /* ignore */ }
124
+ })()
125
+ return () => { cancelled = true }
126
+ }, [authenticated, currentUser, sessions, loadSessions])
127
+
128
+ // Check if first-run setup is needed
129
+ useEffect(() => {
130
+ if (!authenticated || !currentUser) return
131
+ let cancelled = false
132
+ ;(async () => {
133
+ try {
134
+ const [settings, creds] = await Promise.all([
135
+ api<{ setupCompleted?: boolean }>('GET', '/settings'),
136
+ api<Record<string, unknown>>('GET', '/credentials'),
137
+ ])
138
+ if (cancelled) return
139
+ const hasCreds = Object.keys(creds).length > 0
140
+ setSetupDone(settings.setupCompleted === true || hasCreds)
141
+ } catch {
142
+ if (!cancelled) setSetupDone(true) // on error, skip wizard
143
+ }
144
+ })()
145
+ return () => { cancelled = true }
146
+ }, [authenticated, currentUser])
147
+
148
+ useEffect(() => {
149
+ const handler = () => {
150
+ initAudioContext()
151
+ document.removeEventListener('click', handler)
152
+ }
153
+ document.addEventListener('click', handler)
154
+ return () => document.removeEventListener('click', handler)
155
+ }, [])
156
+
157
+ useEffect(() => {
158
+ const handler = () => {
159
+ setAuthenticated(false)
160
+ setAuthChecked(true)
161
+ }
162
+ window.addEventListener('sc_auth_required', handler)
163
+ return () => window.removeEventListener('sc_auth_required', handler)
164
+ }, [])
165
+
166
+ if (!hydrated || !authChecked) return null
167
+ if (!authenticated) return <AccessKeyGate onAuthenticated={() => setAuthenticated(true)} />
168
+ if (!currentUser) return <UserPicker />
169
+ if (setupDone === null) return null
170
+ if (!setupDone) return <SetupWizard onComplete={() => setSetupDone(true)} />
171
+ return <AppLayout />
172
+ }