cowork-os 0.3.21

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 (526) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +1638 -0
  3. package/bin/cowork.js +42 -0
  4. package/build/entitlements.mac.plist +16 -0
  5. package/build/icon.icns +0 -0
  6. package/build/icon.png +0 -0
  7. package/dist/electron/electron/activity/ActivityRepository.js +190 -0
  8. package/dist/electron/electron/agent/browser/browser-service.js +639 -0
  9. package/dist/electron/electron/agent/context-manager.js +225 -0
  10. package/dist/electron/electron/agent/custom-skill-loader.js +566 -0
  11. package/dist/electron/electron/agent/daemon.js +975 -0
  12. package/dist/electron/electron/agent/executor.js +3561 -0
  13. package/dist/electron/electron/agent/llm/anthropic-provider.js +155 -0
  14. package/dist/electron/electron/agent/llm/bedrock-provider.js +202 -0
  15. package/dist/electron/electron/agent/llm/gemini-provider.js +375 -0
  16. package/dist/electron/electron/agent/llm/index.js +34 -0
  17. package/dist/electron/electron/agent/llm/ollama-provider.js +263 -0
  18. package/dist/electron/electron/agent/llm/openai-oauth.js +101 -0
  19. package/dist/electron/electron/agent/llm/openai-provider.js +657 -0
  20. package/dist/electron/electron/agent/llm/openrouter-provider.js +232 -0
  21. package/dist/electron/electron/agent/llm/pricing.js +160 -0
  22. package/dist/electron/electron/agent/llm/provider-factory.js +880 -0
  23. package/dist/electron/electron/agent/llm/types.js +178 -0
  24. package/dist/electron/electron/agent/queue-manager.js +378 -0
  25. package/dist/electron/electron/agent/sandbox/docker-sandbox.js +402 -0
  26. package/dist/electron/electron/agent/sandbox/macos-sandbox.js +407 -0
  27. package/dist/electron/electron/agent/sandbox/runner.js +410 -0
  28. package/dist/electron/electron/agent/sandbox/sandbox-factory.js +228 -0
  29. package/dist/electron/electron/agent/sandbox/security-utils.js +258 -0
  30. package/dist/electron/electron/agent/search/brave-provider.js +119 -0
  31. package/dist/electron/electron/agent/search/google-provider.js +100 -0
  32. package/dist/electron/electron/agent/search/index.js +28 -0
  33. package/dist/electron/electron/agent/search/provider-factory.js +395 -0
  34. package/dist/electron/electron/agent/search/serpapi-provider.js +112 -0
  35. package/dist/electron/electron/agent/search/tavily-provider.js +90 -0
  36. package/dist/electron/electron/agent/search/types.js +40 -0
  37. package/dist/electron/electron/agent/security/index.js +12 -0
  38. package/dist/electron/electron/agent/security/input-sanitizer.js +303 -0
  39. package/dist/electron/electron/agent/security/output-filter.js +217 -0
  40. package/dist/electron/electron/agent/skill-eligibility.js +281 -0
  41. package/dist/electron/electron/agent/skill-registry.js +396 -0
  42. package/dist/electron/electron/agent/skills/document.js +878 -0
  43. package/dist/electron/electron/agent/skills/image-generator.js +225 -0
  44. package/dist/electron/electron/agent/skills/organizer.js +141 -0
  45. package/dist/electron/electron/agent/skills/presentation.js +367 -0
  46. package/dist/electron/electron/agent/skills/spreadsheet.js +165 -0
  47. package/dist/electron/electron/agent/tools/browser-tools.js +523 -0
  48. package/dist/electron/electron/agent/tools/builtin-settings.js +384 -0
  49. package/dist/electron/electron/agent/tools/canvas-tools.js +530 -0
  50. package/dist/electron/electron/agent/tools/cron-tools.js +577 -0
  51. package/dist/electron/electron/agent/tools/edit-tools.js +194 -0
  52. package/dist/electron/electron/agent/tools/file-tools.js +719 -0
  53. package/dist/electron/electron/agent/tools/glob-tools.js +283 -0
  54. package/dist/electron/electron/agent/tools/grep-tools.js +387 -0
  55. package/dist/electron/electron/agent/tools/image-tools.js +111 -0
  56. package/dist/electron/electron/agent/tools/mention-tools.js +282 -0
  57. package/dist/electron/electron/agent/tools/node-tools.js +476 -0
  58. package/dist/electron/electron/agent/tools/registry.js +2719 -0
  59. package/dist/electron/electron/agent/tools/search-tools.js +91 -0
  60. package/dist/electron/electron/agent/tools/shell-tools.js +574 -0
  61. package/dist/electron/electron/agent/tools/skill-tools.js +274 -0
  62. package/dist/electron/electron/agent/tools/system-tools.js +578 -0
  63. package/dist/electron/electron/agent/tools/web-fetch-tools.js +444 -0
  64. package/dist/electron/electron/agent/tools/x-tools.js +264 -0
  65. package/dist/electron/electron/agents/AgentRoleRepository.js +420 -0
  66. package/dist/electron/electron/agents/HeartbeatService.js +356 -0
  67. package/dist/electron/electron/agents/MentionRepository.js +197 -0
  68. package/dist/electron/electron/agents/TaskSubscriptionRepository.js +168 -0
  69. package/dist/electron/electron/agents/WorkingStateRepository.js +229 -0
  70. package/dist/electron/electron/canvas/canvas-manager.js +714 -0
  71. package/dist/electron/electron/canvas/canvas-preload.js +53 -0
  72. package/dist/electron/electron/canvas/canvas-protocol.js +195 -0
  73. package/dist/electron/electron/canvas/canvas-store.js +174 -0
  74. package/dist/electron/electron/canvas/index.js +13 -0
  75. package/dist/electron/electron/control-plane/client.js +364 -0
  76. package/dist/electron/electron/control-plane/handlers.js +572 -0
  77. package/dist/electron/electron/control-plane/index.js +41 -0
  78. package/dist/electron/electron/control-plane/node-manager.js +264 -0
  79. package/dist/electron/electron/control-plane/protocol.js +194 -0
  80. package/dist/electron/electron/control-plane/remote-client.js +437 -0
  81. package/dist/electron/electron/control-plane/server.js +640 -0
  82. package/dist/electron/electron/control-plane/settings.js +369 -0
  83. package/dist/electron/electron/control-plane/ssh-tunnel.js +549 -0
  84. package/dist/electron/electron/cron/index.js +30 -0
  85. package/dist/electron/electron/cron/schedule.js +190 -0
  86. package/dist/electron/electron/cron/service.js +614 -0
  87. package/dist/electron/electron/cron/store.js +155 -0
  88. package/dist/electron/electron/cron/types.js +82 -0
  89. package/dist/electron/electron/cron/webhook.js +258 -0
  90. package/dist/electron/electron/database/SecureSettingsRepository.js +444 -0
  91. package/dist/electron/electron/database/TaskLabelRepository.js +120 -0
  92. package/dist/electron/electron/database/repositories.js +1781 -0
  93. package/dist/electron/electron/database/schema.js +978 -0
  94. package/dist/electron/electron/extensions/index.js +33 -0
  95. package/dist/electron/electron/extensions/loader.js +313 -0
  96. package/dist/electron/electron/extensions/registry.js +485 -0
  97. package/dist/electron/electron/extensions/types.js +11 -0
  98. package/dist/electron/electron/gateway/channel-registry.js +1102 -0
  99. package/dist/electron/electron/gateway/channels/bluebubbles-client.js +479 -0
  100. package/dist/electron/electron/gateway/channels/bluebubbles.js +432 -0
  101. package/dist/electron/electron/gateway/channels/discord.js +975 -0
  102. package/dist/electron/electron/gateway/channels/email-client.js +593 -0
  103. package/dist/electron/electron/gateway/channels/email.js +443 -0
  104. package/dist/electron/electron/gateway/channels/google-chat.js +631 -0
  105. package/dist/electron/electron/gateway/channels/imessage-client.js +363 -0
  106. package/dist/electron/electron/gateway/channels/imessage.js +465 -0
  107. package/dist/electron/electron/gateway/channels/index.js +36 -0
  108. package/dist/electron/electron/gateway/channels/line-client.js +470 -0
  109. package/dist/electron/electron/gateway/channels/line.js +479 -0
  110. package/dist/electron/electron/gateway/channels/matrix-client.js +432 -0
  111. package/dist/electron/electron/gateway/channels/matrix.js +592 -0
  112. package/dist/electron/electron/gateway/channels/mattermost-client.js +394 -0
  113. package/dist/electron/electron/gateway/channels/mattermost.js +496 -0
  114. package/dist/electron/electron/gateway/channels/signal-client.js +500 -0
  115. package/dist/electron/electron/gateway/channels/signal.js +582 -0
  116. package/dist/electron/electron/gateway/channels/slack.js +415 -0
  117. package/dist/electron/electron/gateway/channels/teams.js +596 -0
  118. package/dist/electron/electron/gateway/channels/telegram.js +1390 -0
  119. package/dist/electron/electron/gateway/channels/twitch-client.js +502 -0
  120. package/dist/electron/electron/gateway/channels/twitch.js +396 -0
  121. package/dist/electron/electron/gateway/channels/types.js +8 -0
  122. package/dist/electron/electron/gateway/channels/whatsapp.js +953 -0
  123. package/dist/electron/electron/gateway/context-policy.js +268 -0
  124. package/dist/electron/electron/gateway/index.js +1063 -0
  125. package/dist/electron/electron/gateway/infrastructure.js +496 -0
  126. package/dist/electron/electron/gateway/router.js +2700 -0
  127. package/dist/electron/electron/gateway/security.js +375 -0
  128. package/dist/electron/electron/gateway/session.js +115 -0
  129. package/dist/electron/electron/gateway/tunnel.js +503 -0
  130. package/dist/electron/electron/guardrails/guardrail-manager.js +348 -0
  131. package/dist/electron/electron/hooks/gmail-watcher.js +300 -0
  132. package/dist/electron/electron/hooks/index.js +46 -0
  133. package/dist/electron/electron/hooks/mappings.js +381 -0
  134. package/dist/electron/electron/hooks/server.js +480 -0
  135. package/dist/electron/electron/hooks/settings.js +447 -0
  136. package/dist/electron/electron/hooks/types.js +41 -0
  137. package/dist/electron/electron/ipc/canvas-handlers.js +158 -0
  138. package/dist/electron/electron/ipc/handlers.js +3138 -0
  139. package/dist/electron/electron/ipc/mission-control-handlers.js +141 -0
  140. package/dist/electron/electron/main.js +448 -0
  141. package/dist/electron/electron/mcp/client/MCPClientManager.js +330 -0
  142. package/dist/electron/electron/mcp/client/MCPServerConnection.js +437 -0
  143. package/dist/electron/electron/mcp/client/transports/SSETransport.js +304 -0
  144. package/dist/electron/electron/mcp/client/transports/StdioTransport.js +307 -0
  145. package/dist/electron/electron/mcp/client/transports/WebSocketTransport.js +329 -0
  146. package/dist/electron/electron/mcp/host/MCPHostServer.js +354 -0
  147. package/dist/electron/electron/mcp/host/ToolAdapter.js +100 -0
  148. package/dist/electron/electron/mcp/registry/MCPRegistryManager.js +497 -0
  149. package/dist/electron/electron/mcp/settings.js +446 -0
  150. package/dist/electron/electron/mcp/types.js +59 -0
  151. package/dist/electron/electron/memory/MemoryService.js +435 -0
  152. package/dist/electron/electron/notifications/index.js +17 -0
  153. package/dist/electron/electron/notifications/service.js +118 -0
  154. package/dist/electron/electron/notifications/store.js +144 -0
  155. package/dist/electron/electron/preload.js +842 -0
  156. package/dist/electron/electron/reports/StandupReportService.js +272 -0
  157. package/dist/electron/electron/security/concurrency.js +293 -0
  158. package/dist/electron/electron/security/index.js +15 -0
  159. package/dist/electron/electron/security/policy-manager.js +435 -0
  160. package/dist/electron/electron/settings/appearance-manager.js +193 -0
  161. package/dist/electron/electron/settings/personality-manager.js +724 -0
  162. package/dist/electron/electron/settings/x-manager.js +58 -0
  163. package/dist/electron/electron/tailscale/exposure.js +188 -0
  164. package/dist/electron/electron/tailscale/index.js +28 -0
  165. package/dist/electron/electron/tailscale/settings.js +205 -0
  166. package/dist/electron/electron/tailscale/tailscale.js +355 -0
  167. package/dist/electron/electron/tray/QuickInputWindow.js +568 -0
  168. package/dist/electron/electron/tray/TrayManager.js +895 -0
  169. package/dist/electron/electron/tray/index.js +9 -0
  170. package/dist/electron/electron/updater/index.js +6 -0
  171. package/dist/electron/electron/updater/update-manager.js +418 -0
  172. package/dist/electron/electron/utils/env-migration.js +209 -0
  173. package/dist/electron/electron/utils/process.js +102 -0
  174. package/dist/electron/electron/utils/rate-limiter.js +104 -0
  175. package/dist/electron/electron/utils/validation.js +419 -0
  176. package/dist/electron/electron/utils/x-cli.js +177 -0
  177. package/dist/electron/electron/voice/VoiceService.js +507 -0
  178. package/dist/electron/electron/voice/index.js +14 -0
  179. package/dist/electron/electron/voice/voice-settings-manager.js +359 -0
  180. package/dist/electron/shared/channelMessages.js +170 -0
  181. package/dist/electron/shared/types.js +1185 -0
  182. package/package.json +159 -0
  183. package/resources/skills/1password.json +10 -0
  184. package/resources/skills/add-documentation.json +31 -0
  185. package/resources/skills/analyze-csv.json +17 -0
  186. package/resources/skills/apple-notes.json +10 -0
  187. package/resources/skills/apple-reminders.json +10 -0
  188. package/resources/skills/auto-commenter.json +10 -0
  189. package/resources/skills/bear-notes.json +10 -0
  190. package/resources/skills/bird.json +35 -0
  191. package/resources/skills/blogwatcher.json +10 -0
  192. package/resources/skills/blucli.json +10 -0
  193. package/resources/skills/bluebubbles.json +10 -0
  194. package/resources/skills/camsnap.json +10 -0
  195. package/resources/skills/clean-imports.json +18 -0
  196. package/resources/skills/code-review.json +18 -0
  197. package/resources/skills/coding-agent.json +10 -0
  198. package/resources/skills/compare-files.json +23 -0
  199. package/resources/skills/convert-code.json +34 -0
  200. package/resources/skills/create-changelog.json +24 -0
  201. package/resources/skills/debug-error.json +17 -0
  202. package/resources/skills/dependency-check.json +10 -0
  203. package/resources/skills/discord.json +10 -0
  204. package/resources/skills/eightctl.json +10 -0
  205. package/resources/skills/explain-code.json +29 -0
  206. package/resources/skills/extract-todos.json +18 -0
  207. package/resources/skills/food-order.json +10 -0
  208. package/resources/skills/gemini.json +10 -0
  209. package/resources/skills/generate-readme.json +10 -0
  210. package/resources/skills/gifgrep.json +10 -0
  211. package/resources/skills/git-commit.json +10 -0
  212. package/resources/skills/github.json +10 -0
  213. package/resources/skills/gog.json +10 -0
  214. package/resources/skills/goplaces.json +10 -0
  215. package/resources/skills/himalaya.json +10 -0
  216. package/resources/skills/imsg.json +10 -0
  217. package/resources/skills/karpathy-guidelines.json +12 -0
  218. package/resources/skills/last30days.json +26 -0
  219. package/resources/skills/local-places.json +10 -0
  220. package/resources/skills/mcporter.json +10 -0
  221. package/resources/skills/model-usage.json +10 -0
  222. package/resources/skills/nano-banana-pro.json +10 -0
  223. package/resources/skills/nano-pdf.json +10 -0
  224. package/resources/skills/notion.json +10 -0
  225. package/resources/skills/obsidian.json +10 -0
  226. package/resources/skills/openai-image-gen.json +10 -0
  227. package/resources/skills/openai-whisper-api.json +10 -0
  228. package/resources/skills/openai-whisper.json +10 -0
  229. package/resources/skills/openhue.json +10 -0
  230. package/resources/skills/oracle.json +10 -0
  231. package/resources/skills/ordercli.json +10 -0
  232. package/resources/skills/peekaboo.json +10 -0
  233. package/resources/skills/project-structure.json +10 -0
  234. package/resources/skills/proofread.json +17 -0
  235. package/resources/skills/refactor-code.json +31 -0
  236. package/resources/skills/rename-symbol.json +23 -0
  237. package/resources/skills/sag.json +10 -0
  238. package/resources/skills/security-audit.json +18 -0
  239. package/resources/skills/session-logs.json +10 -0
  240. package/resources/skills/sherpa-onnx-tts.json +10 -0
  241. package/resources/skills/skill-creator.json +15 -0
  242. package/resources/skills/skill-hub.json +29 -0
  243. package/resources/skills/slack.json +10 -0
  244. package/resources/skills/songsee.json +10 -0
  245. package/resources/skills/sonoscli.json +10 -0
  246. package/resources/skills/spotify-player.json +10 -0
  247. package/resources/skills/startup-cfo.json +55 -0
  248. package/resources/skills/summarize-folder.json +18 -0
  249. package/resources/skills/summarize.json +10 -0
  250. package/resources/skills/things-mac.json +10 -0
  251. package/resources/skills/tmux.json +10 -0
  252. package/resources/skills/translate.json +36 -0
  253. package/resources/skills/trello.json +10 -0
  254. package/resources/skills/video-frames.json +10 -0
  255. package/resources/skills/voice-call.json +10 -0
  256. package/resources/skills/wacli.json +10 -0
  257. package/resources/skills/weather.json +10 -0
  258. package/resources/skills/write-tests.json +31 -0
  259. package/src/electron/activity/ActivityRepository.ts +238 -0
  260. package/src/electron/agent/browser/browser-service.ts +721 -0
  261. package/src/electron/agent/context-manager.ts +257 -0
  262. package/src/electron/agent/custom-skill-loader.ts +634 -0
  263. package/src/electron/agent/daemon.ts +1097 -0
  264. package/src/electron/agent/executor.ts +4017 -0
  265. package/src/electron/agent/llm/anthropic-provider.ts +175 -0
  266. package/src/electron/agent/llm/bedrock-provider.ts +236 -0
  267. package/src/electron/agent/llm/gemini-provider.ts +422 -0
  268. package/src/electron/agent/llm/index.ts +9 -0
  269. package/src/electron/agent/llm/ollama-provider.ts +347 -0
  270. package/src/electron/agent/llm/openai-oauth.ts +127 -0
  271. package/src/electron/agent/llm/openai-provider.ts +686 -0
  272. package/src/electron/agent/llm/openrouter-provider.ts +273 -0
  273. package/src/electron/agent/llm/pricing.ts +180 -0
  274. package/src/electron/agent/llm/provider-factory.ts +971 -0
  275. package/src/electron/agent/llm/types.ts +291 -0
  276. package/src/electron/agent/queue-manager.ts +408 -0
  277. package/src/electron/agent/sandbox/docker-sandbox.ts +453 -0
  278. package/src/electron/agent/sandbox/macos-sandbox.ts +426 -0
  279. package/src/electron/agent/sandbox/runner.ts +453 -0
  280. package/src/electron/agent/sandbox/sandbox-factory.ts +337 -0
  281. package/src/electron/agent/sandbox/security-utils.ts +251 -0
  282. package/src/electron/agent/search/brave-provider.ts +141 -0
  283. package/src/electron/agent/search/google-provider.ts +131 -0
  284. package/src/electron/agent/search/index.ts +6 -0
  285. package/src/electron/agent/search/provider-factory.ts +450 -0
  286. package/src/electron/agent/search/serpapi-provider.ts +138 -0
  287. package/src/electron/agent/search/tavily-provider.ts +108 -0
  288. package/src/electron/agent/search/types.ts +118 -0
  289. package/src/electron/agent/security/index.ts +20 -0
  290. package/src/electron/agent/security/input-sanitizer.ts +380 -0
  291. package/src/electron/agent/security/output-filter.ts +259 -0
  292. package/src/electron/agent/skill-eligibility.ts +334 -0
  293. package/src/electron/agent/skill-registry.ts +457 -0
  294. package/src/electron/agent/skills/document.ts +1070 -0
  295. package/src/electron/agent/skills/image-generator.ts +272 -0
  296. package/src/electron/agent/skills/organizer.ts +131 -0
  297. package/src/electron/agent/skills/presentation.ts +418 -0
  298. package/src/electron/agent/skills/spreadsheet.ts +166 -0
  299. package/src/electron/agent/tools/browser-tools.ts +546 -0
  300. package/src/electron/agent/tools/builtin-settings.ts +422 -0
  301. package/src/electron/agent/tools/canvas-tools.ts +572 -0
  302. package/src/electron/agent/tools/cron-tools.ts +723 -0
  303. package/src/electron/agent/tools/edit-tools.ts +196 -0
  304. package/src/electron/agent/tools/file-tools.ts +811 -0
  305. package/src/electron/agent/tools/glob-tools.ts +303 -0
  306. package/src/electron/agent/tools/grep-tools.ts +432 -0
  307. package/src/electron/agent/tools/image-tools.ts +126 -0
  308. package/src/electron/agent/tools/mention-tools.ts +371 -0
  309. package/src/electron/agent/tools/node-tools.ts +550 -0
  310. package/src/electron/agent/tools/registry.ts +3052 -0
  311. package/src/electron/agent/tools/search-tools.ts +111 -0
  312. package/src/electron/agent/tools/shell-tools.ts +651 -0
  313. package/src/electron/agent/tools/skill-tools.ts +340 -0
  314. package/src/electron/agent/tools/system-tools.ts +665 -0
  315. package/src/electron/agent/tools/web-fetch-tools.ts +528 -0
  316. package/src/electron/agent/tools/x-tools.ts +267 -0
  317. package/src/electron/agents/AgentRoleRepository.ts +557 -0
  318. package/src/electron/agents/HeartbeatService.ts +469 -0
  319. package/src/electron/agents/MentionRepository.ts +242 -0
  320. package/src/electron/agents/TaskSubscriptionRepository.ts +231 -0
  321. package/src/electron/agents/WorkingStateRepository.ts +278 -0
  322. package/src/electron/canvas/canvas-manager.ts +818 -0
  323. package/src/electron/canvas/canvas-preload.ts +102 -0
  324. package/src/electron/canvas/canvas-protocol.ts +174 -0
  325. package/src/electron/canvas/canvas-store.ts +200 -0
  326. package/src/electron/canvas/index.ts +8 -0
  327. package/src/electron/control-plane/client.ts +527 -0
  328. package/src/electron/control-plane/handlers.ts +723 -0
  329. package/src/electron/control-plane/index.ts +51 -0
  330. package/src/electron/control-plane/node-manager.ts +322 -0
  331. package/src/electron/control-plane/protocol.ts +269 -0
  332. package/src/electron/control-plane/remote-client.ts +517 -0
  333. package/src/electron/control-plane/server.ts +853 -0
  334. package/src/electron/control-plane/settings.ts +401 -0
  335. package/src/electron/control-plane/ssh-tunnel.ts +624 -0
  336. package/src/electron/cron/index.ts +9 -0
  337. package/src/electron/cron/schedule.ts +217 -0
  338. package/src/electron/cron/service.ts +743 -0
  339. package/src/electron/cron/store.ts +165 -0
  340. package/src/electron/cron/types.ts +291 -0
  341. package/src/electron/cron/webhook.ts +303 -0
  342. package/src/electron/database/SecureSettingsRepository.ts +514 -0
  343. package/src/electron/database/TaskLabelRepository.ts +148 -0
  344. package/src/electron/database/repositories.ts +2397 -0
  345. package/src/electron/database/schema.ts +1017 -0
  346. package/src/electron/extensions/index.ts +18 -0
  347. package/src/electron/extensions/loader.ts +336 -0
  348. package/src/electron/extensions/registry.ts +546 -0
  349. package/src/electron/extensions/types.ts +372 -0
  350. package/src/electron/gateway/channel-registry.ts +1267 -0
  351. package/src/electron/gateway/channels/bluebubbles-client.ts +641 -0
  352. package/src/electron/gateway/channels/bluebubbles.ts +509 -0
  353. package/src/electron/gateway/channels/discord.ts +1150 -0
  354. package/src/electron/gateway/channels/email-client.ts +708 -0
  355. package/src/electron/gateway/channels/email.ts +516 -0
  356. package/src/electron/gateway/channels/google-chat.ts +760 -0
  357. package/src/electron/gateway/channels/imessage-client.ts +473 -0
  358. package/src/electron/gateway/channels/imessage.ts +520 -0
  359. package/src/electron/gateway/channels/index.ts +21 -0
  360. package/src/electron/gateway/channels/line-client.ts +598 -0
  361. package/src/electron/gateway/channels/line.ts +559 -0
  362. package/src/electron/gateway/channels/matrix-client.ts +632 -0
  363. package/src/electron/gateway/channels/matrix.ts +655 -0
  364. package/src/electron/gateway/channels/mattermost-client.ts +526 -0
  365. package/src/electron/gateway/channels/mattermost.ts +550 -0
  366. package/src/electron/gateway/channels/signal-client.ts +722 -0
  367. package/src/electron/gateway/channels/signal.ts +666 -0
  368. package/src/electron/gateway/channels/slack.ts +458 -0
  369. package/src/electron/gateway/channels/teams.ts +681 -0
  370. package/src/electron/gateway/channels/telegram.ts +1727 -0
  371. package/src/electron/gateway/channels/twitch-client.ts +665 -0
  372. package/src/electron/gateway/channels/twitch.ts +468 -0
  373. package/src/electron/gateway/channels/types.ts +1002 -0
  374. package/src/electron/gateway/channels/whatsapp.ts +1101 -0
  375. package/src/electron/gateway/context-policy.ts +382 -0
  376. package/src/electron/gateway/index.ts +1274 -0
  377. package/src/electron/gateway/infrastructure.ts +645 -0
  378. package/src/electron/gateway/router.ts +3206 -0
  379. package/src/electron/gateway/security.ts +422 -0
  380. package/src/electron/gateway/session.ts +144 -0
  381. package/src/electron/gateway/tunnel.ts +626 -0
  382. package/src/electron/guardrails/guardrail-manager.ts +380 -0
  383. package/src/electron/hooks/gmail-watcher.ts +355 -0
  384. package/src/electron/hooks/index.ts +30 -0
  385. package/src/electron/hooks/mappings.ts +404 -0
  386. package/src/electron/hooks/server.ts +574 -0
  387. package/src/electron/hooks/settings.ts +466 -0
  388. package/src/electron/hooks/types.ts +245 -0
  389. package/src/electron/ipc/canvas-handlers.ts +223 -0
  390. package/src/electron/ipc/handlers.ts +3661 -0
  391. package/src/electron/ipc/mission-control-handlers.ts +182 -0
  392. package/src/electron/main.ts +496 -0
  393. package/src/electron/mcp/client/MCPClientManager.ts +406 -0
  394. package/src/electron/mcp/client/MCPServerConnection.ts +514 -0
  395. package/src/electron/mcp/client/transports/SSETransport.ts +360 -0
  396. package/src/electron/mcp/client/transports/StdioTransport.ts +355 -0
  397. package/src/electron/mcp/client/transports/WebSocketTransport.ts +384 -0
  398. package/src/electron/mcp/host/MCPHostServer.ts +388 -0
  399. package/src/electron/mcp/host/ToolAdapter.ts +140 -0
  400. package/src/electron/mcp/registry/MCPRegistryManager.ts +565 -0
  401. package/src/electron/mcp/settings.ts +468 -0
  402. package/src/electron/mcp/types.ts +371 -0
  403. package/src/electron/memory/MemoryService.ts +523 -0
  404. package/src/electron/notifications/index.ts +16 -0
  405. package/src/electron/notifications/service.ts +161 -0
  406. package/src/electron/notifications/store.ts +163 -0
  407. package/src/electron/preload.ts +2845 -0
  408. package/src/electron/reports/StandupReportService.ts +356 -0
  409. package/src/electron/security/concurrency.ts +333 -0
  410. package/src/electron/security/index.ts +17 -0
  411. package/src/electron/security/policy-manager.ts +539 -0
  412. package/src/electron/settings/appearance-manager.ts +182 -0
  413. package/src/electron/settings/personality-manager.ts +800 -0
  414. package/src/electron/settings/x-manager.ts +62 -0
  415. package/src/electron/tailscale/exposure.ts +262 -0
  416. package/src/electron/tailscale/index.ts +34 -0
  417. package/src/electron/tailscale/settings.ts +218 -0
  418. package/src/electron/tailscale/tailscale.ts +379 -0
  419. package/src/electron/tray/QuickInputWindow.ts +609 -0
  420. package/src/electron/tray/TrayManager.ts +1005 -0
  421. package/src/electron/tray/index.ts +6 -0
  422. package/src/electron/updater/index.ts +1 -0
  423. package/src/electron/updater/update-manager.ts +447 -0
  424. package/src/electron/utils/env-migration.ts +203 -0
  425. package/src/electron/utils/process.ts +124 -0
  426. package/src/electron/utils/rate-limiter.ts +130 -0
  427. package/src/electron/utils/validation.ts +493 -0
  428. package/src/electron/utils/x-cli.ts +198 -0
  429. package/src/electron/voice/VoiceService.ts +583 -0
  430. package/src/electron/voice/index.ts +9 -0
  431. package/src/electron/voice/voice-settings-manager.ts +403 -0
  432. package/src/renderer/App.tsx +775 -0
  433. package/src/renderer/components/ActivityFeed.tsx +407 -0
  434. package/src/renderer/components/ActivityFeedItem.tsx +285 -0
  435. package/src/renderer/components/AgentRoleCard.tsx +343 -0
  436. package/src/renderer/components/AgentRoleEditor.tsx +805 -0
  437. package/src/renderer/components/AgentSquadSettings.tsx +295 -0
  438. package/src/renderer/components/AgentWorkingStatePanel.tsx +411 -0
  439. package/src/renderer/components/AppearanceSettings.tsx +122 -0
  440. package/src/renderer/components/ApprovalDialog.tsx +100 -0
  441. package/src/renderer/components/BlueBubblesSettings.tsx +505 -0
  442. package/src/renderer/components/BuiltinToolsSettings.tsx +307 -0
  443. package/src/renderer/components/CanvasPreview.tsx +1189 -0
  444. package/src/renderer/components/CommandOutput.tsx +202 -0
  445. package/src/renderer/components/ContextPolicySettings.tsx +523 -0
  446. package/src/renderer/components/ControlPlaneSettings.tsx +1134 -0
  447. package/src/renderer/components/DisclaimerModal.tsx +124 -0
  448. package/src/renderer/components/DiscordSettings.tsx +436 -0
  449. package/src/renderer/components/EmailSettings.tsx +606 -0
  450. package/src/renderer/components/ExtensionsSettings.tsx +542 -0
  451. package/src/renderer/components/FileViewer.tsx +224 -0
  452. package/src/renderer/components/GoogleChatSettings.tsx +535 -0
  453. package/src/renderer/components/GuardrailSettings.tsx +487 -0
  454. package/src/renderer/components/HooksSettings.tsx +581 -0
  455. package/src/renderer/components/ImessageSettings.tsx +484 -0
  456. package/src/renderer/components/LineSettings.tsx +483 -0
  457. package/src/renderer/components/MCPRegistryBrowser.tsx +386 -0
  458. package/src/renderer/components/MCPSettings.tsx +943 -0
  459. package/src/renderer/components/MainContent.tsx +2433 -0
  460. package/src/renderer/components/MatrixSettings.tsx +510 -0
  461. package/src/renderer/components/MattermostSettings.tsx +473 -0
  462. package/src/renderer/components/MemorySettings.tsx +247 -0
  463. package/src/renderer/components/MentionBadge.tsx +87 -0
  464. package/src/renderer/components/MentionInput.tsx +409 -0
  465. package/src/renderer/components/MentionList.tsx +476 -0
  466. package/src/renderer/components/MissionControlPanel.tsx +1995 -0
  467. package/src/renderer/components/NodesSettings.tsx +316 -0
  468. package/src/renderer/components/NotificationPanel.tsx +481 -0
  469. package/src/renderer/components/Onboarding/AwakeningOrb.tsx +44 -0
  470. package/src/renderer/components/Onboarding/Onboarding.tsx +443 -0
  471. package/src/renderer/components/Onboarding/TypewriterText.tsx +102 -0
  472. package/src/renderer/components/Onboarding/index.ts +3 -0
  473. package/src/renderer/components/OnboardingModal.tsx +698 -0
  474. package/src/renderer/components/PairingCodeDisplay.tsx +324 -0
  475. package/src/renderer/components/PersonalitySettings.tsx +597 -0
  476. package/src/renderer/components/QueueSettings.tsx +119 -0
  477. package/src/renderer/components/QuickTaskFAB.tsx +71 -0
  478. package/src/renderer/components/RightPanel.tsx +413 -0
  479. package/src/renderer/components/ScheduledTasksSettings.tsx +1328 -0
  480. package/src/renderer/components/SearchSettings.tsx +328 -0
  481. package/src/renderer/components/Settings.tsx +1504 -0
  482. package/src/renderer/components/Sidebar.tsx +344 -0
  483. package/src/renderer/components/SignalSettings.tsx +673 -0
  484. package/src/renderer/components/SkillHubBrowser.tsx +458 -0
  485. package/src/renderer/components/SkillParameterModal.tsx +185 -0
  486. package/src/renderer/components/SkillsSettings.tsx +451 -0
  487. package/src/renderer/components/SlackSettings.tsx +442 -0
  488. package/src/renderer/components/StandupReportViewer.tsx +614 -0
  489. package/src/renderer/components/TaskBoard.tsx +498 -0
  490. package/src/renderer/components/TaskBoardCard.tsx +357 -0
  491. package/src/renderer/components/TaskBoardColumn.tsx +211 -0
  492. package/src/renderer/components/TaskLabelManager.tsx +472 -0
  493. package/src/renderer/components/TaskQueuePanel.tsx +144 -0
  494. package/src/renderer/components/TaskQuickActions.tsx +492 -0
  495. package/src/renderer/components/TaskTimeline.tsx +216 -0
  496. package/src/renderer/components/TaskView.tsx +162 -0
  497. package/src/renderer/components/TeamsSettings.tsx +518 -0
  498. package/src/renderer/components/TelegramSettings.tsx +421 -0
  499. package/src/renderer/components/Toast.tsx +76 -0
  500. package/src/renderer/components/TraySettings.tsx +189 -0
  501. package/src/renderer/components/TwitchSettings.tsx +511 -0
  502. package/src/renderer/components/UpdateSettings.tsx +295 -0
  503. package/src/renderer/components/VoiceIndicator.tsx +270 -0
  504. package/src/renderer/components/VoiceSettings.tsx +867 -0
  505. package/src/renderer/components/WhatsAppSettings.tsx +721 -0
  506. package/src/renderer/components/WorkingStateEditor.tsx +309 -0
  507. package/src/renderer/components/WorkingStateHistory.tsx +481 -0
  508. package/src/renderer/components/WorkspaceSelector.tsx +150 -0
  509. package/src/renderer/components/XSettings.tsx +311 -0
  510. package/src/renderer/global.d.ts +9 -0
  511. package/src/renderer/hooks/useAgentContext.ts +153 -0
  512. package/src/renderer/hooks/useOnboardingFlow.ts +548 -0
  513. package/src/renderer/hooks/useVoiceInput.ts +268 -0
  514. package/src/renderer/index.html +12 -0
  515. package/src/renderer/main.tsx +10 -0
  516. package/src/renderer/public/cowork-os-logo.png +0 -0
  517. package/src/renderer/quick-input.html +164 -0
  518. package/src/renderer/styles/index.css +14504 -0
  519. package/src/renderer/utils/agentMessages.ts +749 -0
  520. package/src/renderer/utils/voice-directives.ts +169 -0
  521. package/src/shared/channelMessages.ts +213 -0
  522. package/src/shared/types.ts +3608 -0
  523. package/tsconfig.electron.json +26 -0
  524. package/tsconfig.json +26 -0
  525. package/tsconfig.node.json +10 -0
  526. package/vite.config.ts +23 -0
@@ -0,0 +1,1328 @@
1
+ import { useState, useEffect, useCallback } from 'react';
2
+ import { useAgentContext } from '../hooks/useAgentContext';
3
+
4
+ // Types from preload (duplicated for renderer use)
5
+ type CronSchedule =
6
+ | { kind: 'at'; atMs: number }
7
+ | { kind: 'every'; everyMs: number; anchorMs?: number }
8
+ | { kind: 'cron'; expr: string; tz?: string };
9
+
10
+ interface CronJobState {
11
+ nextRunAtMs?: number;
12
+ runningAtMs?: number;
13
+ lastRunAtMs?: number;
14
+ lastStatus?: 'ok' | 'error' | 'skipped';
15
+ lastError?: string;
16
+ lastDurationMs?: number;
17
+ lastTaskId?: string;
18
+ totalRuns?: number;
19
+ }
20
+
21
+ interface CronJob {
22
+ id: string;
23
+ name: string;
24
+ description?: string;
25
+ enabled: boolean;
26
+ deleteAfterRun?: boolean;
27
+ createdAtMs: number;
28
+ updatedAtMs: number;
29
+ schedule: CronSchedule;
30
+ workspaceId: string;
31
+ taskPrompt: string;
32
+ taskTitle?: string;
33
+ state: CronJobState;
34
+ }
35
+
36
+ interface CronStatusSummary {
37
+ enabled: boolean;
38
+ storePath: string;
39
+ jobCount: number;
40
+ enabledJobCount: number;
41
+ nextWakeAtMs: number | null;
42
+ }
43
+
44
+ // Minimal Workspace type for the UI
45
+ interface Workspace {
46
+ id: string;
47
+ name: string;
48
+ path: string;
49
+ }
50
+
51
+ // Schedule presets
52
+ const SCHEDULE_PRESETS = [
53
+ { label: 'Every 5 minutes', schedule: { kind: 'every' as const, everyMs: 5 * 60 * 1000 } },
54
+ { label: 'Every 15 minutes', schedule: { kind: 'every' as const, everyMs: 15 * 60 * 1000 } },
55
+ { label: 'Every 30 minutes', schedule: { kind: 'every' as const, everyMs: 30 * 60 * 1000 } },
56
+ { label: 'Every hour', schedule: { kind: 'every' as const, everyMs: 60 * 60 * 1000 } },
57
+ { label: 'Every 2 hours', schedule: { kind: 'every' as const, everyMs: 2 * 60 * 60 * 1000 } },
58
+ { label: 'Every 6 hours', schedule: { kind: 'every' as const, everyMs: 6 * 60 * 60 * 1000 } },
59
+ { label: 'Every 12 hours', schedule: { kind: 'every' as const, everyMs: 12 * 60 * 60 * 1000 } },
60
+ { label: 'Daily', schedule: { kind: 'every' as const, everyMs: 24 * 60 * 60 * 1000 } },
61
+ ];
62
+
63
+ const CRON_PRESETS = [
64
+ { label: 'Every minute', value: '* * * * *', desc: 'Runs every minute' },
65
+ { label: 'Every 5 minutes', value: '*/5 * * * *', desc: 'At minutes 0, 5, 10...' },
66
+ { label: 'Every 15 minutes', value: '*/15 * * * *', desc: 'At :00, :15, :30, :45' },
67
+ { label: 'Every hour', value: '0 * * * *', desc: 'At the start of each hour' },
68
+ { label: 'Daily at midnight', value: '0 0 * * *', desc: '12:00 AM every day' },
69
+ { label: 'Daily at 9:00 AM', value: '0 9 * * *', desc: '9:00 AM every day' },
70
+ { label: 'Daily at 6:00 PM', value: '0 18 * * *', desc: '6:00 PM every day' },
71
+ { label: 'Weekdays at 9:00 AM', value: '0 9 * * 1-5', desc: 'Mon-Fri at 9:00 AM' },
72
+ { label: 'Weekly on Monday', value: '0 0 * * 1', desc: 'Every Monday at midnight' },
73
+ { label: 'Monthly on the 1st', value: '0 0 1 * *', desc: '1st day of month at midnight' },
74
+ ];
75
+
76
+ // Icons as inline SVGs
77
+ const Icons = {
78
+ clock: (
79
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
80
+ <circle cx="12" cy="12" r="10" />
81
+ <polyline points="12 6 12 12 16 14" />
82
+ </svg>
83
+ ),
84
+ play: (
85
+ <svg width="14" height="14" viewBox="0 0 24 24" fill="currentColor">
86
+ <polygon points="5 3 19 12 5 21 5 3" />
87
+ </svg>
88
+ ),
89
+ pause: (
90
+ <svg width="14" height="14" viewBox="0 0 24 24" fill="currentColor">
91
+ <rect x="6" y="4" width="4" height="16" />
92
+ <rect x="14" y="4" width="4" height="16" />
93
+ </svg>
94
+ ),
95
+ edit: (
96
+ <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
97
+ <path d="M11 4H4a2 2 0 0 0-2 2v14a2 2 0 0 0 2 2h14a2 2 0 0 0 2-2v-7" />
98
+ <path d="M18.5 2.5a2.121 2.121 0 0 1 3 3L12 15l-4 1 1-4 9.5-9.5z" />
99
+ </svg>
100
+ ),
101
+ trash: (
102
+ <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
103
+ <polyline points="3 6 5 6 21 6" />
104
+ <path d="M19 6v14a2 2 0 0 1-2 2H7a2 2 0 0 1-2-2V6m3 0V4a2 2 0 0 1 2-2h4a2 2 0 0 1 2 2v2" />
105
+ </svg>
106
+ ),
107
+ plus: (
108
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2.5" strokeLinecap="round" strokeLinejoin="round">
109
+ <line x1="12" y1="5" x2="12" y2="19" />
110
+ <line x1="5" y1="12" x2="19" y2="12" />
111
+ </svg>
112
+ ),
113
+ check: (
114
+ <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2.5" strokeLinecap="round" strokeLinejoin="round">
115
+ <polyline points="20 6 9 17 4 12" />
116
+ </svg>
117
+ ),
118
+ x: (
119
+ <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2.5" strokeLinecap="round" strokeLinejoin="round">
120
+ <line x1="18" y1="6" x2="6" y2="18" />
121
+ <line x1="6" y1="6" x2="18" y2="18" />
122
+ </svg>
123
+ ),
124
+ calendar: (
125
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
126
+ <rect x="3" y="4" width="18" height="18" rx="2" ry="2" />
127
+ <line x1="16" y1="2" x2="16" y2="6" />
128
+ <line x1="8" y1="2" x2="8" y2="6" />
129
+ <line x1="3" y1="10" x2="21" y2="10" />
130
+ </svg>
131
+ ),
132
+ repeat: (
133
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
134
+ <polyline points="17 1 21 5 17 9" />
135
+ <path d="M3 11V9a4 4 0 0 1 4-4h14" />
136
+ <polyline points="7 23 3 19 7 15" />
137
+ <path d="M21 13v2a4 4 0 0 1-4 4H3" />
138
+ </svg>
139
+ ),
140
+ chevronDown: (
141
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
142
+ <polyline points="6 9 12 15 18 9" />
143
+ </svg>
144
+ ),
145
+ zap: (
146
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
147
+ <polygon points="13 2 3 14 12 14 11 22 21 10 12 10 13 2" />
148
+ </svg>
149
+ ),
150
+ activity: (
151
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
152
+ <polyline points="22 12 18 12 15 21 9 3 6 12 2 12" />
153
+ </svg>
154
+ ),
155
+ };
156
+
157
+ function describeSchedule(schedule: CronSchedule): string {
158
+ switch (schedule.kind) {
159
+ case 'at': {
160
+ const date = new Date(schedule.atMs);
161
+ return `Once at ${date.toLocaleString()}`;
162
+ }
163
+ case 'every': {
164
+ const ms = schedule.everyMs;
165
+ if (ms >= 86400000) {
166
+ const days = Math.round(ms / 86400000);
167
+ return `Every ${days} day${days > 1 ? 's' : ''}`;
168
+ }
169
+ if (ms >= 3600000) {
170
+ const hours = Math.round(ms / 3600000);
171
+ return `Every ${hours} hour${hours > 1 ? 's' : ''}`;
172
+ }
173
+ if (ms >= 60000) {
174
+ const minutes = Math.round(ms / 60000);
175
+ return `Every ${minutes} minute${minutes > 1 ? 's' : ''}`;
176
+ }
177
+ return `Every ${Math.round(ms / 1000)} seconds`;
178
+ }
179
+ case 'cron': {
180
+ // Try to find a matching preset for friendly name
181
+ const preset = CRON_PRESETS.find((p) => p.value === schedule.expr);
182
+ return preset ? preset.label : schedule.expr;
183
+ }
184
+ }
185
+ }
186
+
187
+ function getScheduleIcon(schedule: CronSchedule) {
188
+ if (schedule.kind === 'at') return Icons.calendar;
189
+ return Icons.repeat;
190
+ }
191
+
192
+ function formatRelativeTime(ms: number): string {
193
+ const now = Date.now();
194
+ const diff = ms - now;
195
+ const absDiff = Math.abs(diff);
196
+ const isPast = diff < 0;
197
+
198
+ if (absDiff < 60000) {
199
+ return isPast ? 'just now' : 'in < 1 min';
200
+ }
201
+ if (absDiff < 3600000) {
202
+ const minutes = Math.round(absDiff / 60000);
203
+ return isPast ? `${minutes}m ago` : `in ${minutes}m`;
204
+ }
205
+ if (absDiff < 86400000) {
206
+ const hours = Math.round(absDiff / 3600000);
207
+ return isPast ? `${hours}h ago` : `in ${hours}h`;
208
+ }
209
+ const days = Math.round(absDiff / 86400000);
210
+ return isPast ? `${days}d ago` : `in ${days}d`;
211
+ }
212
+
213
+ function formatDuration(ms: number): string {
214
+ if (ms < 1000) return `${ms}ms`;
215
+ if (ms < 60000) return `${Math.round(ms / 1000)}s`;
216
+ const minutes = Math.floor(ms / 60000);
217
+ const seconds = Math.round((ms % 60000) / 1000);
218
+ return `${minutes}m ${seconds}s`;
219
+ }
220
+
221
+ // Styles
222
+ const styles = {
223
+ container: {
224
+ padding: '24px',
225
+ maxWidth: '900px',
226
+ } as React.CSSProperties,
227
+ header: {
228
+ marginBottom: '24px',
229
+ } as React.CSSProperties,
230
+ title: {
231
+ fontSize: '24px',
232
+ fontWeight: 600,
233
+ color: 'var(--color-text-primary)',
234
+ marginBottom: '8px',
235
+ display: 'flex',
236
+ alignItems: 'center',
237
+ gap: '12px',
238
+ } as React.CSSProperties,
239
+ subtitle: {
240
+ fontSize: '14px',
241
+ color: 'var(--color-text-muted)',
242
+ } as React.CSSProperties,
243
+ statsGrid: {
244
+ display: 'grid',
245
+ gridTemplateColumns: 'repeat(auto-fit, minmax(160px, 1fr))',
246
+ gap: '12px',
247
+ marginBottom: '24px',
248
+ } as React.CSSProperties,
249
+ statCard: {
250
+ backgroundColor: 'var(--color-bg-glass)',
251
+ border: '1px solid var(--color-border-subtle)',
252
+ borderRadius: 'var(--radius-md)',
253
+ padding: '16px',
254
+ display: 'flex',
255
+ flexDirection: 'column' as const,
256
+ gap: '4px',
257
+ } as React.CSSProperties,
258
+ statLabel: {
259
+ fontSize: '12px',
260
+ color: 'var(--color-text-muted)',
261
+ textTransform: 'uppercase' as const,
262
+ letterSpacing: '0.5px',
263
+ } as React.CSSProperties,
264
+ statValue: {
265
+ fontSize: '24px',
266
+ fontWeight: 600,
267
+ color: 'var(--color-text-primary)',
268
+ } as React.CSSProperties,
269
+ addButton: {
270
+ display: 'inline-flex',
271
+ alignItems: 'center',
272
+ gap: '8px',
273
+ padding: '12px 20px',
274
+ backgroundColor: 'var(--color-accent)',
275
+ color: '#000',
276
+ border: 'none',
277
+ borderRadius: 'var(--radius-md)',
278
+ fontSize: '14px',
279
+ fontWeight: 600,
280
+ cursor: 'pointer',
281
+ transition: 'all 0.2s ease',
282
+ marginBottom: '24px',
283
+ } as React.CSSProperties,
284
+ jobCard: {
285
+ backgroundColor: 'var(--color-bg-glass)',
286
+ border: '1px solid var(--color-border-subtle)',
287
+ borderRadius: 'var(--radius-md)',
288
+ marginBottom: '12px',
289
+ overflow: 'hidden',
290
+ transition: 'all 0.2s ease',
291
+ } as React.CSSProperties,
292
+ jobHeader: {
293
+ display: 'flex',
294
+ alignItems: 'center',
295
+ padding: '16px',
296
+ gap: '12px',
297
+ cursor: 'pointer',
298
+ } as React.CSSProperties,
299
+ statusDot: {
300
+ width: '10px',
301
+ height: '10px',
302
+ borderRadius: '50%',
303
+ flexShrink: 0,
304
+ } as React.CSSProperties,
305
+ jobInfo: {
306
+ flex: 1,
307
+ minWidth: 0,
308
+ } as React.CSSProperties,
309
+ jobName: {
310
+ fontSize: '15px',
311
+ fontWeight: 500,
312
+ color: 'var(--color-text-primary)',
313
+ marginBottom: '4px',
314
+ display: 'flex',
315
+ alignItems: 'center',
316
+ gap: '8px',
317
+ } as React.CSSProperties,
318
+ jobMeta: {
319
+ display: 'flex',
320
+ alignItems: 'center',
321
+ gap: '12px',
322
+ fontSize: '13px',
323
+ color: 'var(--color-text-muted)',
324
+ } as React.CSSProperties,
325
+ scheduleTag: {
326
+ display: 'inline-flex',
327
+ alignItems: 'center',
328
+ gap: '4px',
329
+ padding: '2px 8px',
330
+ backgroundColor: 'var(--color-bg-secondary)',
331
+ borderRadius: '4px',
332
+ fontSize: '12px',
333
+ } as React.CSSProperties,
334
+ nextRun: {
335
+ display: 'flex',
336
+ alignItems: 'center',
337
+ gap: '6px',
338
+ padding: '6px 12px',
339
+ backgroundColor: 'var(--color-accent-subtle)',
340
+ color: 'var(--color-accent)',
341
+ borderRadius: '6px',
342
+ fontSize: '13px',
343
+ fontWeight: 500,
344
+ } as React.CSSProperties,
345
+ actions: {
346
+ display: 'flex',
347
+ gap: '4px',
348
+ } as React.CSSProperties,
349
+ actionBtn: {
350
+ display: 'flex',
351
+ alignItems: 'center',
352
+ justifyContent: 'center',
353
+ width: '32px',
354
+ height: '32px',
355
+ backgroundColor: 'transparent',
356
+ border: '1px solid var(--color-border-subtle)',
357
+ borderRadius: '6px',
358
+ color: 'var(--color-text-secondary)',
359
+ cursor: 'pointer',
360
+ transition: 'all 0.15s ease',
361
+ } as React.CSSProperties,
362
+ expandedContent: {
363
+ borderTop: '1px solid var(--color-border-subtle)',
364
+ padding: '16px',
365
+ backgroundColor: 'var(--color-bg-darker)',
366
+ } as React.CSSProperties,
367
+ detailGrid: {
368
+ display: 'grid',
369
+ gridTemplateColumns: '120px 1fr',
370
+ gap: '8px 16px',
371
+ fontSize: '13px',
372
+ } as React.CSSProperties,
373
+ detailLabel: {
374
+ color: 'var(--color-text-muted)',
375
+ } as React.CSSProperties,
376
+ detailValue: {
377
+ color: 'var(--color-text-primary)',
378
+ wordBreak: 'break-word' as const,
379
+ } as React.CSSProperties,
380
+ emptyState: {
381
+ textAlign: 'center' as const,
382
+ padding: '60px 20px',
383
+ color: 'var(--color-text-muted)',
384
+ } as React.CSSProperties,
385
+ emptyIcon: {
386
+ width: '64px',
387
+ height: '64px',
388
+ margin: '0 auto 16px',
389
+ opacity: 0.3,
390
+ } as React.CSSProperties,
391
+ emptyTitle: {
392
+ fontSize: '18px',
393
+ fontWeight: 500,
394
+ color: 'var(--color-text-secondary)',
395
+ marginBottom: '8px',
396
+ } as React.CSSProperties,
397
+ emptyDesc: {
398
+ fontSize: '14px',
399
+ maxWidth: '400px',
400
+ margin: '0 auto',
401
+ } as React.CSSProperties,
402
+ errorBanner: {
403
+ display: 'flex',
404
+ alignItems: 'center',
405
+ gap: '12px',
406
+ padding: '12px 16px',
407
+ backgroundColor: 'var(--color-error-subtle)',
408
+ border: '1px solid var(--color-error)',
409
+ borderRadius: 'var(--radius-md)',
410
+ marginBottom: '16px',
411
+ color: 'var(--color-error)',
412
+ fontSize: '14px',
413
+ } as React.CSSProperties,
414
+ lastRunBadge: {
415
+ display: 'inline-flex',
416
+ alignItems: 'center',
417
+ gap: '4px',
418
+ padding: '2px 6px',
419
+ borderRadius: '4px',
420
+ fontSize: '11px',
421
+ fontWeight: 500,
422
+ } as React.CSSProperties,
423
+ };
424
+
425
+ export function ScheduledTasksSettings() {
426
+ const [status, setStatus] = useState<CronStatusSummary | null>(null);
427
+ const [jobs, setJobs] = useState<CronJob[]>([]);
428
+ const [workspaces, setWorkspaces] = useState<Workspace[]>([]);
429
+ const [loading, setLoading] = useState(true);
430
+ const [error, setError] = useState<string | null>(null);
431
+ const [showCreateModal, setShowCreateModal] = useState(false);
432
+ const [editingJob, setEditingJob] = useState<CronJob | null>(null);
433
+ const [expandedJobId, setExpandedJobId] = useState<string | null>(null);
434
+
435
+ const loadData = useCallback(async () => {
436
+ try {
437
+ setLoading(true);
438
+ setError(null);
439
+ const [statusResult, jobsResult, workspacesResult] = await Promise.all([
440
+ window.electronAPI.getCronStatus(),
441
+ window.electronAPI.listCronJobs({ includeDisabled: true }),
442
+ window.electronAPI.listWorkspaces(),
443
+ ]);
444
+ setStatus(statusResult);
445
+ setJobs(jobsResult);
446
+ setWorkspaces(workspacesResult);
447
+ } catch (err: any) {
448
+ setError(err.message || 'Failed to load scheduled tasks');
449
+ } finally {
450
+ setLoading(false);
451
+ }
452
+ }, []);
453
+
454
+ useEffect(() => {
455
+ loadData();
456
+
457
+ // Subscribe to cron events
458
+ const unsubscribe = window.electronAPI.onCronEvent((event) => {
459
+ console.log('[ScheduledTasks] Cron event:', event);
460
+ loadData();
461
+ });
462
+
463
+ return unsubscribe;
464
+ }, [loadData]);
465
+
466
+ const handleToggleJob = async (job: CronJob, e: React.MouseEvent) => {
467
+ e.stopPropagation();
468
+ try {
469
+ const result = await window.electronAPI.updateCronJob(job.id, { enabled: !job.enabled });
470
+ if (!result.ok) {
471
+ setError(result.error);
472
+ return;
473
+ }
474
+ await loadData();
475
+ } catch (err: any) {
476
+ setError(err.message);
477
+ }
478
+ };
479
+
480
+ const handleDeleteJob = async (job: CronJob, e: React.MouseEvent) => {
481
+ e.stopPropagation();
482
+ if (!confirm(`Delete "${job.name}"?\n\nThis action cannot be undone.`)) return;
483
+
484
+ try {
485
+ const result = await window.electronAPI.removeCronJob(job.id);
486
+ if (!result.ok) {
487
+ setError((result as any).error || 'Failed to delete job');
488
+ return;
489
+ }
490
+ await loadData();
491
+ } catch (err: any) {
492
+ setError(err.message);
493
+ }
494
+ };
495
+
496
+ const handleRunNow = async (job: CronJob, e: React.MouseEvent) => {
497
+ e.stopPropagation();
498
+ try {
499
+ const result = await window.electronAPI.runCronJob(job.id, 'force');
500
+ if (!result.ok) {
501
+ setError(result.error);
502
+ return;
503
+ }
504
+ if (result.ran) {
505
+ console.log(`[ScheduledTasks] Created task: ${result.taskId}`);
506
+ }
507
+ await loadData();
508
+ } catch (err: any) {
509
+ setError(err.message);
510
+ }
511
+ };
512
+
513
+ const handleEditJob = (job: CronJob, e: React.MouseEvent) => {
514
+ e.stopPropagation();
515
+ setEditingJob(job);
516
+ setShowCreateModal(true);
517
+ };
518
+
519
+ if (loading) {
520
+ return (
521
+ <div style={styles.container}>
522
+ <div style={styles.header}>
523
+ <div style={styles.title}>
524
+ {Icons.clock}
525
+ <span>Scheduled Tasks</span>
526
+ </div>
527
+ </div>
528
+ <div style={{ padding: '40px', textAlign: 'center', color: 'var(--color-text-muted)' }}>
529
+ Loading...
530
+ </div>
531
+ </div>
532
+ );
533
+ }
534
+
535
+ const lastRunJob = jobs.reduce<CronJob | null>((latest, job) => {
536
+ if (!job.state.lastRunAtMs) return latest;
537
+ if (!latest || !latest.state.lastRunAtMs) return job;
538
+ return job.state.lastRunAtMs > latest.state.lastRunAtMs ? job : latest;
539
+ }, null);
540
+
541
+ return (
542
+ <div style={styles.container}>
543
+ {/* Header */}
544
+ <div style={styles.header}>
545
+ <div style={styles.title}>
546
+ {Icons.clock}
547
+ <span>Scheduled Tasks</span>
548
+ </div>
549
+ <div style={styles.subtitle}>
550
+ Automate tasks to run on a schedule. Results appear in your workspace.
551
+ </div>
552
+ </div>
553
+
554
+ {/* Error Banner */}
555
+ {error && (
556
+ <div style={styles.errorBanner}>
557
+ {Icons.x}
558
+ <span style={{ flex: 1 }}>{error}</span>
559
+ <button
560
+ onClick={() => setError(null)}
561
+ style={{
562
+ background: 'none',
563
+ border: 'none',
564
+ color: 'inherit',
565
+ cursor: 'pointer',
566
+ padding: '4px',
567
+ }}
568
+ >
569
+ {Icons.x}
570
+ </button>
571
+ </div>
572
+ )}
573
+
574
+ {/* Stats Cards */}
575
+ <div style={styles.statsGrid}>
576
+ <div style={styles.statCard}>
577
+ <span style={styles.statLabel}>Total Tasks</span>
578
+ <span style={styles.statValue}>{status?.jobCount || 0}</span>
579
+ </div>
580
+ <div style={styles.statCard}>
581
+ <span style={styles.statLabel}>Active</span>
582
+ <span style={{ ...styles.statValue, color: 'var(--color-success)' }}>
583
+ {status?.enabledJobCount || 0}
584
+ </span>
585
+ </div>
586
+ <div style={styles.statCard}>
587
+ <span style={styles.statLabel}>Next Run</span>
588
+ <span style={{ ...styles.statValue, fontSize: '16px' }}>
589
+ {status?.nextWakeAtMs ? formatRelativeTime(status.nextWakeAtMs) : '-'}
590
+ </span>
591
+ </div>
592
+ <div style={styles.statCard}>
593
+ <span style={styles.statLabel}>Last Run</span>
594
+ <span style={{ ...styles.statValue, fontSize: '16px' }}>
595
+ {lastRunJob?.state.lastRunAtMs ? formatRelativeTime(lastRunJob.state.lastRunAtMs) : '-'}
596
+ </span>
597
+ </div>
598
+ </div>
599
+
600
+ {/* Add Button */}
601
+ <button
602
+ style={styles.addButton}
603
+ onClick={() => {
604
+ setEditingJob(null);
605
+ setShowCreateModal(true);
606
+ }}
607
+ onMouseOver={(e) => {
608
+ e.currentTarget.style.backgroundColor = 'var(--color-accent-hover)';
609
+ e.currentTarget.style.transform = 'translateY(-1px)';
610
+ }}
611
+ onMouseOut={(e) => {
612
+ e.currentTarget.style.backgroundColor = 'var(--color-accent)';
613
+ e.currentTarget.style.transform = 'translateY(0)';
614
+ }}
615
+ >
616
+ {Icons.plus}
617
+ <span>New Scheduled Task</span>
618
+ </button>
619
+
620
+ {/* Jobs List */}
621
+ {jobs.length === 0 ? (
622
+ <div style={styles.emptyState}>
623
+ <div style={styles.emptyIcon}>
624
+ <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="1.5">
625
+ <circle cx="12" cy="12" r="10" />
626
+ <polyline points="12 6 12 12 16 14" />
627
+ </svg>
628
+ </div>
629
+ <div style={styles.emptyTitle}>No scheduled tasks yet</div>
630
+ <div style={styles.emptyDesc}>
631
+ Create a scheduled task to automatically run prompts on a schedule.
632
+ Great for daily reports, periodic checks, and automated workflows.
633
+ </div>
634
+ </div>
635
+ ) : (
636
+ <div>
637
+ {jobs.map((job) => {
638
+ const isExpanded = expandedJobId === job.id;
639
+ const workspace = workspaces.find((w) => w.id === job.workspaceId);
640
+ const lastStatus = job.state.lastStatus;
641
+
642
+ return (
643
+ <div
644
+ key={job.id}
645
+ style={{
646
+ ...styles.jobCard,
647
+ borderColor: job.enabled ? 'var(--color-border-subtle)' : 'transparent',
648
+ opacity: job.enabled ? 1 : 0.6,
649
+ }}
650
+ >
651
+ {/* Job Header */}
652
+ <div
653
+ style={styles.jobHeader}
654
+ onClick={() => setExpandedJobId(isExpanded ? null : job.id)}
655
+ >
656
+ {/* Status Indicator */}
657
+ <div
658
+ style={{
659
+ ...styles.statusDot,
660
+ backgroundColor: !job.enabled
661
+ ? 'var(--color-text-muted)'
662
+ : job.state.runningAtMs
663
+ ? 'var(--color-warning)'
664
+ : lastStatus === 'error'
665
+ ? 'var(--color-error)'
666
+ : 'var(--color-success)',
667
+ boxShadow: job.enabled && !job.state.runningAtMs
668
+ ? `0 0 8px ${lastStatus === 'error' ? 'var(--color-error)' : 'var(--color-success)'}`
669
+ : 'none',
670
+ }}
671
+ />
672
+
673
+ {/* Job Info */}
674
+ <div style={styles.jobInfo}>
675
+ <div style={styles.jobName}>
676
+ <span>{job.name}</span>
677
+ {lastStatus && (
678
+ <span
679
+ style={{
680
+ ...styles.lastRunBadge,
681
+ backgroundColor:
682
+ lastStatus === 'ok'
683
+ ? 'var(--color-success-subtle)'
684
+ : 'var(--color-error-subtle)',
685
+ color: lastStatus === 'ok' ? 'var(--color-success)' : 'var(--color-error)',
686
+ }}
687
+ >
688
+ {lastStatus === 'ok' ? Icons.check : Icons.x}
689
+ {lastStatus}
690
+ </span>
691
+ )}
692
+ </div>
693
+ <div style={styles.jobMeta}>
694
+ <span style={styles.scheduleTag}>
695
+ {getScheduleIcon(job.schedule)}
696
+ {describeSchedule(job.schedule)}
697
+ </span>
698
+ {workspace && (
699
+ <span style={{ opacity: 0.7 }}>{workspace.name}</span>
700
+ )}
701
+ </div>
702
+ </div>
703
+
704
+ {/* Next Run */}
705
+ {job.enabled && job.state.nextRunAtMs && (
706
+ <div style={styles.nextRun}>
707
+ {Icons.zap}
708
+ <span>{formatRelativeTime(job.state.nextRunAtMs)}</span>
709
+ </div>
710
+ )}
711
+
712
+ {/* Actions */}
713
+ <div style={styles.actions}>
714
+ <button
715
+ style={{
716
+ ...styles.actionBtn,
717
+ backgroundColor: job.enabled ? 'var(--color-success-subtle)' : 'transparent',
718
+ color: job.enabled ? 'var(--color-success)' : 'var(--color-text-muted)',
719
+ }}
720
+ onClick={(e) => handleToggleJob(job, e)}
721
+ title={job.enabled ? 'Disable' : 'Enable'}
722
+ >
723
+ {job.enabled ? Icons.pause : Icons.play}
724
+ </button>
725
+ <button
726
+ style={styles.actionBtn}
727
+ onClick={(e) => handleRunNow(job, e)}
728
+ title="Run now"
729
+ >
730
+ {Icons.play}
731
+ </button>
732
+ <button
733
+ style={styles.actionBtn}
734
+ onClick={(e) => handleEditJob(job, e)}
735
+ title="Edit"
736
+ >
737
+ {Icons.edit}
738
+ </button>
739
+ <button
740
+ style={{
741
+ ...styles.actionBtn,
742
+ color: 'var(--color-error)',
743
+ }}
744
+ onClick={(e) => handleDeleteJob(job, e)}
745
+ title="Delete"
746
+ >
747
+ {Icons.trash}
748
+ </button>
749
+ </div>
750
+
751
+ {/* Expand Arrow */}
752
+ <span
753
+ style={{
754
+ color: 'var(--color-text-muted)',
755
+ transform: isExpanded ? 'rotate(180deg)' : 'rotate(0)',
756
+ transition: 'transform 0.2s ease',
757
+ }}
758
+ >
759
+ {Icons.chevronDown}
760
+ </span>
761
+ </div>
762
+
763
+ {/* Expanded Content */}
764
+ {isExpanded && (
765
+ <div style={styles.expandedContent}>
766
+ {job.description && (
767
+ <div
768
+ style={{
769
+ marginBottom: '16px',
770
+ padding: '12px',
771
+ backgroundColor: 'var(--color-bg-glass)',
772
+ borderRadius: '6px',
773
+ fontSize: '13px',
774
+ color: 'var(--color-text-secondary)',
775
+ }}
776
+ >
777
+ {job.description}
778
+ </div>
779
+ )}
780
+
781
+ <div style={styles.detailGrid}>
782
+ <span style={styles.detailLabel}>Workspace</span>
783
+ <span style={styles.detailValue}>{workspace?.name || job.workspaceId}</span>
784
+
785
+ <span style={styles.detailLabel}>Prompt</span>
786
+ <span
787
+ style={{
788
+ ...styles.detailValue,
789
+ fontFamily: 'monospace',
790
+ fontSize: '12px',
791
+ backgroundColor: 'var(--color-bg-glass)',
792
+ padding: '8px',
793
+ borderRadius: '4px',
794
+ }}
795
+ >
796
+ {job.taskPrompt}
797
+ </span>
798
+
799
+ {job.schedule.kind === 'cron' && (
800
+ <>
801
+ <span style={styles.detailLabel}>Cron Expression</span>
802
+ <span style={{ ...styles.detailValue, fontFamily: 'monospace' }}>
803
+ {job.schedule.expr}
804
+ </span>
805
+ </>
806
+ )}
807
+
808
+ <span style={styles.detailLabel}>Created</span>
809
+ <span style={styles.detailValue}>
810
+ {new Date(job.createdAtMs).toLocaleString()}
811
+ </span>
812
+
813
+ {job.state.totalRuns !== undefined && job.state.totalRuns > 0 && (
814
+ <>
815
+ <span style={styles.detailLabel}>Total Runs</span>
816
+ <span style={styles.detailValue}>{job.state.totalRuns}</span>
817
+ </>
818
+ )}
819
+
820
+ {job.state.lastRunAtMs && (
821
+ <>
822
+ <span style={styles.detailLabel}>Last Run</span>
823
+ <span style={styles.detailValue}>
824
+ {new Date(job.state.lastRunAtMs).toLocaleString()}
825
+ {job.state.lastDurationMs && (
826
+ <span style={{ color: 'var(--color-text-muted)', marginLeft: '8px' }}>
827
+ ({formatDuration(job.state.lastDurationMs)})
828
+ </span>
829
+ )}
830
+ </span>
831
+ </>
832
+ )}
833
+
834
+ {job.state.lastError && (
835
+ <>
836
+ <span style={styles.detailLabel}>Last Error</span>
837
+ <span style={{ ...styles.detailValue, color: 'var(--color-error)' }}>
838
+ {job.state.lastError}
839
+ </span>
840
+ </>
841
+ )}
842
+ </div>
843
+ </div>
844
+ )}
845
+ </div>
846
+ );
847
+ })}
848
+ </div>
849
+ )}
850
+
851
+ {/* Create/Edit Modal */}
852
+ {showCreateModal && (
853
+ <JobModal
854
+ job={editingJob}
855
+ workspaces={workspaces}
856
+ onClose={() => {
857
+ setShowCreateModal(false);
858
+ setEditingJob(null);
859
+ }}
860
+ onSave={async () => {
861
+ await loadData();
862
+ setShowCreateModal(false);
863
+ setEditingJob(null);
864
+ }}
865
+ />
866
+ )}
867
+ </div>
868
+ );
869
+ }
870
+
871
+ interface JobModalProps {
872
+ job: CronJob | null;
873
+ workspaces: Workspace[];
874
+ onClose: () => void;
875
+ onSave: () => void;
876
+ }
877
+
878
+ function JobModal({ job, workspaces, onClose, onSave }: JobModalProps) {
879
+ const isEditing = job !== null;
880
+ const agentContext = useAgentContext();
881
+
882
+ const [name, setName] = useState(job?.name || '');
883
+ const [description, setDescription] = useState(job?.description || '');
884
+ const [workspaceId, setWorkspaceId] = useState(job?.workspaceId || (workspaces[0]?.id || ''));
885
+ const [taskPrompt, setTaskPrompt] = useState(job?.taskPrompt || '');
886
+ const [taskTitle, setTaskTitle] = useState(job?.taskTitle || '');
887
+ const [enabled, setEnabled] = useState(job?.enabled ?? true);
888
+ const [deleteAfterRun, setDeleteAfterRun] = useState(job?.deleteAfterRun ?? false);
889
+
890
+ // Schedule type and values
891
+ const [scheduleType, setScheduleType] = useState<'every' | 'cron' | 'at'>(
892
+ job?.schedule.kind || 'every'
893
+ );
894
+ const [everyMs, setEveryMs] = useState(
895
+ job?.schedule.kind === 'every' ? job.schedule.everyMs : 60 * 60 * 1000
896
+ );
897
+ const [cronExpr, setCronExpr] = useState(job?.schedule.kind === 'cron' ? job.schedule.expr : '0 9 * * *');
898
+ const [atDateTime, setAtDateTime] = useState(
899
+ job?.schedule.kind === 'at' ? new Date(job.schedule.atMs).toISOString().slice(0, 16) : ''
900
+ );
901
+
902
+ const [saving, setSaving] = useState(false);
903
+ const [error, setError] = useState<string | null>(null);
904
+
905
+ const handleSave = async () => {
906
+ if (!name.trim()) {
907
+ setError('Name is required');
908
+ return;
909
+ }
910
+ if (!workspaceId) {
911
+ setError('Workspace is required');
912
+ return;
913
+ }
914
+ if (!taskPrompt.trim()) {
915
+ setError('Task prompt is required');
916
+ return;
917
+ }
918
+
919
+ let schedule: CronSchedule;
920
+ if (scheduleType === 'every') {
921
+ schedule = { kind: 'every', everyMs, anchorMs: Date.now() };
922
+ } else if (scheduleType === 'cron') {
923
+ schedule = { kind: 'cron', expr: cronExpr };
924
+ } else {
925
+ const atMs = new Date(atDateTime).getTime();
926
+ if (isNaN(atMs) || atMs < Date.now()) {
927
+ setError('Please select a future date and time');
928
+ return;
929
+ }
930
+ schedule = { kind: 'at', atMs };
931
+ }
932
+
933
+ try {
934
+ setSaving(true);
935
+ setError(null);
936
+
937
+ if (isEditing && job) {
938
+ const result = await window.electronAPI.updateCronJob(job.id, {
939
+ name: name.trim(),
940
+ description: description.trim() || undefined,
941
+ workspaceId,
942
+ taskPrompt: taskPrompt.trim(),
943
+ taskTitle: taskTitle.trim() || undefined,
944
+ enabled,
945
+ deleteAfterRun,
946
+ schedule,
947
+ });
948
+ if (!result.ok) {
949
+ setError(result.error);
950
+ return;
951
+ }
952
+ } else {
953
+ const result = await window.electronAPI.addCronJob({
954
+ name: name.trim(),
955
+ description: description.trim() || undefined,
956
+ workspaceId,
957
+ taskPrompt: taskPrompt.trim(),
958
+ taskTitle: taskTitle.trim() || undefined,
959
+ enabled,
960
+ deleteAfterRun,
961
+ schedule,
962
+ });
963
+ if (!result.ok) {
964
+ setError(result.error);
965
+ return;
966
+ }
967
+ }
968
+
969
+ onSave();
970
+ } catch (err: any) {
971
+ setError(err.message);
972
+ } finally {
973
+ setSaving(false);
974
+ }
975
+ };
976
+
977
+ const modalStyles = {
978
+ overlay: {
979
+ position: 'fixed' as const,
980
+ top: 0,
981
+ left: 0,
982
+ right: 0,
983
+ bottom: 0,
984
+ backgroundColor: 'rgba(0, 0, 0, 0.6)',
985
+ backdropFilter: 'blur(4px)',
986
+ display: 'flex',
987
+ alignItems: 'center',
988
+ justifyContent: 'center',
989
+ zIndex: 1000,
990
+ },
991
+ content: {
992
+ backgroundColor: 'var(--color-bg-elevated)',
993
+ border: '1px solid var(--color-border)',
994
+ borderRadius: 'var(--radius-lg)',
995
+ padding: '24px',
996
+ width: '560px',
997
+ maxWidth: '90vw',
998
+ maxHeight: '85vh',
999
+ overflow: 'auto',
1000
+ boxShadow: 'var(--shadow-lg)',
1001
+ },
1002
+ title: {
1003
+ fontSize: '20px',
1004
+ fontWeight: 600,
1005
+ color: 'var(--color-text-primary)',
1006
+ marginBottom: '20px',
1007
+ display: 'flex',
1008
+ alignItems: 'center',
1009
+ gap: '10px',
1010
+ },
1011
+ field: {
1012
+ marginBottom: '20px',
1013
+ },
1014
+ label: {
1015
+ display: 'block',
1016
+ marginBottom: '6px',
1017
+ fontSize: '13px',
1018
+ fontWeight: 500,
1019
+ color: 'var(--color-text-secondary)',
1020
+ },
1021
+ input: {
1022
+ width: '100%',
1023
+ padding: '10px 12px',
1024
+ backgroundColor: 'var(--color-bg-input)',
1025
+ border: '1px solid var(--color-border)',
1026
+ borderRadius: 'var(--radius-sm)',
1027
+ color: 'var(--color-text-primary)',
1028
+ fontSize: '14px',
1029
+ outline: 'none',
1030
+ transition: 'border-color 0.2s',
1031
+ },
1032
+ textarea: {
1033
+ width: '100%',
1034
+ padding: '10px 12px',
1035
+ backgroundColor: 'var(--color-bg-input)',
1036
+ border: '1px solid var(--color-border)',
1037
+ borderRadius: 'var(--radius-sm)',
1038
+ color: 'var(--color-text-primary)',
1039
+ fontSize: '14px',
1040
+ outline: 'none',
1041
+ resize: 'vertical' as const,
1042
+ minHeight: '100px',
1043
+ fontFamily: 'inherit',
1044
+ },
1045
+ select: {
1046
+ width: '100%',
1047
+ padding: '10px 12px',
1048
+ backgroundColor: 'var(--color-bg-input)',
1049
+ border: '1px solid var(--color-border)',
1050
+ borderRadius: 'var(--radius-sm)',
1051
+ color: 'var(--color-text-primary)',
1052
+ fontSize: '14px',
1053
+ outline: 'none',
1054
+ },
1055
+ scheduleToggle: {
1056
+ display: 'flex',
1057
+ gap: '8px',
1058
+ marginBottom: '12px',
1059
+ },
1060
+ toggleBtn: (active: boolean) => ({
1061
+ flex: 1,
1062
+ padding: '10px',
1063
+ backgroundColor: active ? 'var(--color-accent-subtle)' : 'var(--color-bg-glass)',
1064
+ border: `1px solid ${active ? 'var(--color-accent)' : 'var(--color-border-subtle)'}`,
1065
+ borderRadius: 'var(--radius-sm)',
1066
+ color: active ? 'var(--color-accent)' : 'var(--color-text-secondary)',
1067
+ fontSize: '13px',
1068
+ fontWeight: 500,
1069
+ cursor: 'pointer',
1070
+ transition: 'all 0.15s ease',
1071
+ }),
1072
+ checkbox: {
1073
+ display: 'flex',
1074
+ alignItems: 'center',
1075
+ gap: '8px',
1076
+ cursor: 'pointer',
1077
+ fontSize: '14px',
1078
+ color: 'var(--color-text-secondary)',
1079
+ },
1080
+ actions: {
1081
+ display: 'flex',
1082
+ gap: '12px',
1083
+ justifyContent: 'flex-end',
1084
+ marginTop: '24px',
1085
+ paddingTop: '20px',
1086
+ borderTop: '1px solid var(--color-border-subtle)',
1087
+ },
1088
+ cancelBtn: {
1089
+ padding: '10px 20px',
1090
+ backgroundColor: 'transparent',
1091
+ border: '1px solid var(--color-border)',
1092
+ borderRadius: 'var(--radius-sm)',
1093
+ color: 'var(--color-text-secondary)',
1094
+ fontSize: '14px',
1095
+ fontWeight: 500,
1096
+ cursor: 'pointer',
1097
+ },
1098
+ saveBtn: {
1099
+ padding: '10px 24px',
1100
+ backgroundColor: 'var(--color-accent)',
1101
+ border: 'none',
1102
+ borderRadius: 'var(--radius-sm)',
1103
+ color: '#000',
1104
+ fontSize: '14px',
1105
+ fontWeight: 600,
1106
+ cursor: 'pointer',
1107
+ },
1108
+ error: {
1109
+ padding: '10px 12px',
1110
+ backgroundColor: 'var(--color-error-subtle)',
1111
+ border: '1px solid var(--color-error)',
1112
+ borderRadius: 'var(--radius-sm)',
1113
+ color: 'var(--color-error)',
1114
+ fontSize: '13px',
1115
+ marginBottom: '16px',
1116
+ },
1117
+ };
1118
+
1119
+ return (
1120
+ <div
1121
+ style={modalStyles.overlay}
1122
+ onClick={(e) => {
1123
+ if (e.target === e.currentTarget) onClose();
1124
+ }}
1125
+ >
1126
+ <div style={modalStyles.content}>
1127
+ <div style={modalStyles.title}>
1128
+ {Icons.clock}
1129
+ {isEditing ? 'Edit Scheduled Task' : 'New Scheduled Task'}
1130
+ </div>
1131
+
1132
+ {error && <div style={modalStyles.error}>{error}</div>}
1133
+
1134
+ {/* Name */}
1135
+ <div style={modalStyles.field}>
1136
+ <label style={modalStyles.label}>Name *</label>
1137
+ <input
1138
+ type="text"
1139
+ value={name}
1140
+ onChange={(e) => setName(e.target.value)}
1141
+ placeholder="e.g., Daily AI News Report"
1142
+ style={modalStyles.input}
1143
+ />
1144
+ </div>
1145
+
1146
+ {/* Description */}
1147
+ <div style={modalStyles.field}>
1148
+ <label style={modalStyles.label}>Description</label>
1149
+ <input
1150
+ type="text"
1151
+ value={description}
1152
+ onChange={(e) => setDescription(e.target.value)}
1153
+ placeholder="Optional description"
1154
+ style={modalStyles.input}
1155
+ />
1156
+ </div>
1157
+
1158
+ {/* Workspace */}
1159
+ <div style={modalStyles.field}>
1160
+ <label style={modalStyles.label}>Workspace *</label>
1161
+ <select
1162
+ value={workspaceId}
1163
+ onChange={(e) => setWorkspaceId(e.target.value)}
1164
+ style={modalStyles.select}
1165
+ >
1166
+ {workspaces.length === 0 && (
1167
+ <option value="">{agentContext.getUiCopy('scheduledNoWorkspaces')}</option>
1168
+ )}
1169
+ {workspaces.map((w) => (
1170
+ <option key={w.id} value={w.id}>
1171
+ {w.name}
1172
+ </option>
1173
+ ))}
1174
+ </select>
1175
+ </div>
1176
+
1177
+ {/* Task Prompt */}
1178
+ <div style={modalStyles.field}>
1179
+ <label style={modalStyles.label}>Task Prompt *</label>
1180
+ <textarea
1181
+ value={taskPrompt}
1182
+ onChange={(e) => setTaskPrompt(e.target.value)}
1183
+ placeholder="What should the agent do? Be specific about the task..."
1184
+ style={modalStyles.textarea}
1185
+ />
1186
+ </div>
1187
+
1188
+ {/* Task Title */}
1189
+ <div style={modalStyles.field}>
1190
+ <label style={modalStyles.label}>Task Title (optional)</label>
1191
+ <input
1192
+ type="text"
1193
+ value={taskTitle}
1194
+ onChange={(e) => setTaskTitle(e.target.value)}
1195
+ placeholder="Custom title for created tasks"
1196
+ style={modalStyles.input}
1197
+ />
1198
+ </div>
1199
+
1200
+ {/* Schedule Type */}
1201
+ <div style={modalStyles.field}>
1202
+ <label style={modalStyles.label}>Schedule *</label>
1203
+ <div style={modalStyles.scheduleToggle}>
1204
+ <button
1205
+ type="button"
1206
+ style={modalStyles.toggleBtn(scheduleType === 'every')}
1207
+ onClick={() => setScheduleType('every')}
1208
+ >
1209
+ {Icons.repeat}
1210
+ <span style={{ marginLeft: '6px' }}>Interval</span>
1211
+ </button>
1212
+ <button
1213
+ type="button"
1214
+ style={modalStyles.toggleBtn(scheduleType === 'cron')}
1215
+ onClick={() => setScheduleType('cron')}
1216
+ >
1217
+ {Icons.activity}
1218
+ <span style={{ marginLeft: '6px' }}>Cron</span>
1219
+ </button>
1220
+ <button
1221
+ type="button"
1222
+ style={modalStyles.toggleBtn(scheduleType === 'at')}
1223
+ onClick={() => setScheduleType('at')}
1224
+ >
1225
+ {Icons.calendar}
1226
+ <span style={{ marginLeft: '6px' }}>One-time</span>
1227
+ </button>
1228
+ </div>
1229
+
1230
+ {/* Interval Schedule */}
1231
+ {scheduleType === 'every' && (
1232
+ <select
1233
+ value={everyMs}
1234
+ onChange={(e) => setEveryMs(Number(e.target.value))}
1235
+ style={modalStyles.select}
1236
+ >
1237
+ {SCHEDULE_PRESETS.map((preset) => (
1238
+ <option key={preset.label} value={preset.schedule.everyMs}>
1239
+ {preset.label}
1240
+ </option>
1241
+ ))}
1242
+ </select>
1243
+ )}
1244
+
1245
+ {/* Cron Schedule */}
1246
+ {scheduleType === 'cron' && (
1247
+ <div>
1248
+ <select
1249
+ value={CRON_PRESETS.find((p) => p.value === cronExpr)?.value || 'custom'}
1250
+ onChange={(e) => {
1251
+ if (e.target.value !== 'custom') {
1252
+ setCronExpr(e.target.value);
1253
+ }
1254
+ }}
1255
+ style={{ ...modalStyles.select, marginBottom: '8px' }}
1256
+ >
1257
+ {CRON_PRESETS.map((preset) => (
1258
+ <option key={preset.value} value={preset.value}>
1259
+ {preset.label} - {preset.desc}
1260
+ </option>
1261
+ ))}
1262
+ <option value="custom">Custom expression</option>
1263
+ </select>
1264
+ <input
1265
+ type="text"
1266
+ value={cronExpr}
1267
+ onChange={(e) => setCronExpr(e.target.value)}
1268
+ placeholder="minute hour day month weekday (e.g., 0 9 * * 1-5)"
1269
+ style={{ ...modalStyles.input, fontFamily: 'monospace' }}
1270
+ />
1271
+ <div
1272
+ style={{
1273
+ marginTop: '6px',
1274
+ fontSize: '12px',
1275
+ color: 'var(--color-text-muted)',
1276
+ }}
1277
+ >
1278
+ Format: minute (0-59) hour (0-23) day (1-31) month (1-12) weekday (0-6, 0=Sun)
1279
+ </div>
1280
+ </div>
1281
+ )}
1282
+
1283
+ {/* One-time Schedule */}
1284
+ {scheduleType === 'at' && (
1285
+ <input
1286
+ type="datetime-local"
1287
+ value={atDateTime}
1288
+ onChange={(e) => setAtDateTime(e.target.value)}
1289
+ style={modalStyles.input}
1290
+ />
1291
+ )}
1292
+ </div>
1293
+
1294
+ {/* Options */}
1295
+ <div style={modalStyles.field}>
1296
+ <label style={modalStyles.checkbox}>
1297
+ <input
1298
+ type="checkbox"
1299
+ checked={enabled}
1300
+ onChange={(e) => setEnabled(e.target.checked)}
1301
+ />
1302
+ Enable immediately after saving
1303
+ </label>
1304
+ {scheduleType === 'at' && (
1305
+ <label style={{ ...modalStyles.checkbox, marginTop: '8px' }}>
1306
+ <input
1307
+ type="checkbox"
1308
+ checked={deleteAfterRun}
1309
+ onChange={(e) => setDeleteAfterRun(e.target.checked)}
1310
+ />
1311
+ Delete after execution (one-shot)
1312
+ </label>
1313
+ )}
1314
+ </div>
1315
+
1316
+ {/* Actions */}
1317
+ <div style={modalStyles.actions}>
1318
+ <button style={modalStyles.cancelBtn} onClick={onClose} disabled={saving}>
1319
+ Cancel
1320
+ </button>
1321
+ <button style={modalStyles.saveBtn} onClick={handleSave} disabled={saving}>
1322
+ {saving ? 'Saving...' : isEditing ? 'Save Changes' : 'Create Task'}
1323
+ </button>
1324
+ </div>
1325
+ </div>
1326
+ </div>
1327
+ );
1328
+ }