@elizaos/app-core 2.0.0-alpha.37

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 (1082) hide show
  1. package/.turbo/turbo-build.log +2 -0
  2. package/LICENSE +21 -0
  3. package/dist/App.d.ts +5 -0
  4. package/dist/App.d.ts.map +1 -0
  5. package/dist/App.js +198 -0
  6. package/dist/actions/character.d.ts +27 -0
  7. package/dist/actions/character.d.ts.map +1 -0
  8. package/dist/actions/character.js +97 -0
  9. package/dist/actions/chat-helpers.d.ts +47 -0
  10. package/dist/actions/chat-helpers.d.ts.map +1 -0
  11. package/dist/actions/chat-helpers.js +79 -0
  12. package/dist/actions/cloud.d.ts +17 -0
  13. package/dist/actions/cloud.d.ts.map +1 -0
  14. package/dist/actions/cloud.js +43 -0
  15. package/dist/actions/index.d.ts +12 -0
  16. package/dist/actions/index.d.ts.map +1 -0
  17. package/dist/actions/index.js +11 -0
  18. package/dist/actions/lifecycle.d.ts +43 -0
  19. package/dist/actions/lifecycle.d.ts.map +1 -0
  20. package/dist/actions/lifecycle.js +118 -0
  21. package/dist/actions/onboarding.d.ts +12 -0
  22. package/dist/actions/onboarding.d.ts.map +1 -0
  23. package/dist/actions/onboarding.js +28 -0
  24. package/dist/actions/triggers.d.ts +23 -0
  25. package/dist/actions/triggers.d.ts.map +1 -0
  26. package/dist/actions/triggers.js +148 -0
  27. package/dist/api/client.d.ts +2823 -0
  28. package/dist/api/client.d.ts.map +1 -0
  29. package/dist/api/client.js +2392 -0
  30. package/dist/api/index.d.ts +2 -0
  31. package/dist/api/index.d.ts.map +1 -0
  32. package/dist/api/index.js +1 -0
  33. package/dist/autonomy/index.d.ts +48 -0
  34. package/dist/autonomy/index.d.ts.map +1 -0
  35. package/dist/autonomy/index.js +330 -0
  36. package/dist/bridge/capacitor-bridge.d.ts +153 -0
  37. package/dist/bridge/capacitor-bridge.d.ts.map +1 -0
  38. package/dist/bridge/capacitor-bridge.js +193 -0
  39. package/dist/bridge/electrobun-rpc.d.ts +19 -0
  40. package/dist/bridge/electrobun-rpc.d.ts.map +1 -0
  41. package/dist/bridge/electrobun-rpc.js +27 -0
  42. package/dist/bridge/electrobun-runtime.d.ts +3 -0
  43. package/dist/bridge/electrobun-runtime.d.ts.map +1 -0
  44. package/dist/bridge/electrobun-runtime.js +17 -0
  45. package/dist/bridge/index.d.ts +6 -0
  46. package/dist/bridge/index.d.ts.map +1 -0
  47. package/dist/bridge/index.js +5 -0
  48. package/dist/bridge/native-plugins.d.ts +82 -0
  49. package/dist/bridge/native-plugins.d.ts.map +1 -0
  50. package/dist/bridge/native-plugins.js +39 -0
  51. package/dist/bridge/plugin-bridge.d.ts +116 -0
  52. package/dist/bridge/plugin-bridge.d.ts.map +1 -0
  53. package/dist/bridge/plugin-bridge.js +203 -0
  54. package/dist/bridge/storage-bridge.d.ts +39 -0
  55. package/dist/bridge/storage-bridge.d.ts.map +1 -0
  56. package/dist/bridge/storage-bridge.js +135 -0
  57. package/dist/chat/index.d.ts +57 -0
  58. package/dist/chat/index.d.ts.map +1 -0
  59. package/dist/chat/index.js +161 -0
  60. package/dist/coding/index.d.ts +25 -0
  61. package/dist/coding/index.d.ts.map +1 -0
  62. package/dist/coding/index.js +25 -0
  63. package/dist/components/AdvancedPageView.d.ts +17 -0
  64. package/dist/components/AdvancedPageView.d.ts.map +1 -0
  65. package/dist/components/AdvancedPageView.js +137 -0
  66. package/dist/components/AgentActivityBox.d.ts +7 -0
  67. package/dist/components/AgentActivityBox.d.ts.map +1 -0
  68. package/dist/components/AgentActivityBox.js +25 -0
  69. package/dist/components/ApiKeyConfig.d.ts +26 -0
  70. package/dist/components/ApiKeyConfig.d.ts.map +1 -0
  71. package/dist/components/ApiKeyConfig.js +121 -0
  72. package/dist/components/AppsPageView.d.ts +7 -0
  73. package/dist/components/AppsPageView.d.ts.map +1 -0
  74. package/dist/components/AppsPageView.js +31 -0
  75. package/dist/components/AppsView.d.ts +8 -0
  76. package/dist/components/AppsView.d.ts.map +1 -0
  77. package/dist/components/AppsView.js +149 -0
  78. package/dist/components/AvatarLoader.d.ts +9 -0
  79. package/dist/components/AvatarLoader.d.ts.map +1 -0
  80. package/dist/components/AvatarLoader.js +45 -0
  81. package/dist/components/AvatarSelector.d.ts +23 -0
  82. package/dist/components/AvatarSelector.d.ts.map +1 -0
  83. package/dist/components/AvatarSelector.js +105 -0
  84. package/dist/components/BscTradePanel.d.ts +22 -0
  85. package/dist/components/BscTradePanel.d.ts.map +1 -0
  86. package/dist/components/BscTradePanel.js +221 -0
  87. package/dist/components/BugReportModal.d.ts +2 -0
  88. package/dist/components/BugReportModal.d.ts.map +1 -0
  89. package/dist/components/BugReportModal.js +218 -0
  90. package/dist/components/CharacterView.d.ts +8 -0
  91. package/dist/components/CharacterView.d.ts.map +1 -0
  92. package/dist/components/CharacterView.js +703 -0
  93. package/dist/components/ChatAvatar.d.ts +8 -0
  94. package/dist/components/ChatAvatar.d.ts.map +1 -0
  95. package/dist/components/ChatAvatar.js +89 -0
  96. package/dist/components/ChatComposer.d.ts +37 -0
  97. package/dist/components/ChatComposer.d.ts.map +1 -0
  98. package/dist/components/ChatComposer.js +136 -0
  99. package/dist/components/ChatMessage.d.ts +24 -0
  100. package/dist/components/ChatMessage.d.ts.map +1 -0
  101. package/dist/components/ChatMessage.js +167 -0
  102. package/dist/components/ChatModalView.d.ts +10 -0
  103. package/dist/components/ChatModalView.d.ts.map +1 -0
  104. package/dist/components/ChatModalView.js +57 -0
  105. package/dist/components/ChatView.d.ts +14 -0
  106. package/dist/components/ChatView.d.ts.map +1 -0
  107. package/dist/components/ChatView.js +511 -0
  108. package/dist/components/CloudSourceControls.d.ts +13 -0
  109. package/dist/components/CloudSourceControls.d.ts.map +1 -0
  110. package/dist/components/CloudSourceControls.js +14 -0
  111. package/dist/components/CodingAgentSettingsSection.d.ts +2 -0
  112. package/dist/components/CodingAgentSettingsSection.d.ts.map +1 -0
  113. package/dist/components/CodingAgentSettingsSection.js +268 -0
  114. package/dist/components/CommandPalette.d.ts +2 -0
  115. package/dist/components/CommandPalette.d.ts.map +1 -0
  116. package/dist/components/CommandPalette.js +181 -0
  117. package/dist/components/CompanionSceneHost.d.ts +15 -0
  118. package/dist/components/CompanionSceneHost.d.ts.map +1 -0
  119. package/dist/components/CompanionSceneHost.js +343 -0
  120. package/dist/components/CompanionShell.d.ts +17 -0
  121. package/dist/components/CompanionShell.d.ts.map +1 -0
  122. package/dist/components/CompanionShell.js +15 -0
  123. package/dist/components/CompanionView.d.ts +2 -0
  124. package/dist/components/CompanionView.d.ts.map +1 -0
  125. package/dist/components/CompanionView.js +22 -0
  126. package/dist/components/ConfigPageView.d.ts +11 -0
  127. package/dist/components/ConfigPageView.d.ts.map +1 -0
  128. package/dist/components/ConfigPageView.js +275 -0
  129. package/dist/components/ConfigSaveFooter.d.ts +8 -0
  130. package/dist/components/ConfigSaveFooter.d.ts.map +1 -0
  131. package/dist/components/ConfigSaveFooter.js +10 -0
  132. package/dist/components/ConfirmModal.d.ts +61 -0
  133. package/dist/components/ConfirmModal.d.ts.map +1 -0
  134. package/dist/components/ConfirmModal.js +164 -0
  135. package/dist/components/ConnectionFailedBanner.d.ts +6 -0
  136. package/dist/components/ConnectionFailedBanner.d.ts.map +1 -0
  137. package/dist/components/ConnectionFailedBanner.js +22 -0
  138. package/dist/components/ConnectorsPageView.d.ts +7 -0
  139. package/dist/components/ConnectorsPageView.d.ts.map +1 -0
  140. package/dist/components/ConnectorsPageView.js +8 -0
  141. package/dist/components/ConversationsSidebar.d.ts +9 -0
  142. package/dist/components/ConversationsSidebar.d.ts.map +1 -0
  143. package/dist/components/ConversationsSidebar.js +116 -0
  144. package/dist/components/CustomActionEditor.d.ts +10 -0
  145. package/dist/components/CustomActionEditor.d.ts.map +1 -0
  146. package/dist/components/CustomActionEditor.js +578 -0
  147. package/dist/components/CustomActionsPanel.d.ts +9 -0
  148. package/dist/components/CustomActionsPanel.d.ts.map +1 -0
  149. package/dist/components/CustomActionsPanel.js +107 -0
  150. package/dist/components/CustomActionsView.d.ts +2 -0
  151. package/dist/components/CustomActionsView.d.ts.map +1 -0
  152. package/dist/components/CustomActionsView.js +134 -0
  153. package/dist/components/DatabasePageView.d.ts +5 -0
  154. package/dist/components/DatabasePageView.d.ts.map +1 -0
  155. package/dist/components/DatabasePageView.js +28 -0
  156. package/dist/components/DatabaseView.d.ts +9 -0
  157. package/dist/components/DatabaseView.d.ts.map +1 -0
  158. package/dist/components/DatabaseView.js +311 -0
  159. package/dist/components/ElizaCloudDashboard.d.ts +2 -0
  160. package/dist/components/ElizaCloudDashboard.d.ts.map +1 -0
  161. package/dist/components/ElizaCloudDashboard.js +657 -0
  162. package/dist/components/EmotePicker.d.ts +2 -0
  163. package/dist/components/EmotePicker.d.ts.map +1 -0
  164. package/dist/components/EmotePicker.js +343 -0
  165. package/dist/components/ErrorBoundary.d.ts +22 -0
  166. package/dist/components/ErrorBoundary.d.ts.map +1 -0
  167. package/dist/components/ErrorBoundary.js +31 -0
  168. package/dist/components/FineTuningView.d.ts +2 -0
  169. package/dist/components/FineTuningView.d.ts.map +1 -0
  170. package/dist/components/FineTuningView.js +433 -0
  171. package/dist/components/GameView.d.ts +11 -0
  172. package/dist/components/GameView.d.ts.map +1 -0
  173. package/dist/components/GameView.js +295 -0
  174. package/dist/components/GameViewOverlay.d.ts +8 -0
  175. package/dist/components/GameViewOverlay.d.ts.map +1 -0
  176. package/dist/components/GameViewOverlay.js +70 -0
  177. package/dist/components/GlobalEmoteOverlay.d.ts +2 -0
  178. package/dist/components/GlobalEmoteOverlay.d.ts.map +1 -0
  179. package/dist/components/GlobalEmoteOverlay.js +112 -0
  180. package/dist/components/Header.d.ts +8 -0
  181. package/dist/components/Header.d.ts.map +1 -0
  182. package/dist/components/Header.js +121 -0
  183. package/dist/components/HeartbeatsView.d.ts +2 -0
  184. package/dist/components/HeartbeatsView.d.ts.map +1 -0
  185. package/dist/components/HeartbeatsView.js +378 -0
  186. package/dist/components/InventoryView.d.ts +10 -0
  187. package/dist/components/InventoryView.d.ts.map +1 -0
  188. package/dist/components/InventoryView.js +162 -0
  189. package/dist/components/KnowledgeView.d.ts +20 -0
  190. package/dist/components/KnowledgeView.d.ts.map +1 -0
  191. package/dist/components/KnowledgeView.js +480 -0
  192. package/dist/components/LanguageDropdown.d.ts +30 -0
  193. package/dist/components/LanguageDropdown.d.ts.map +1 -0
  194. package/dist/components/LanguageDropdown.js +98 -0
  195. package/dist/components/LifoMonitorPanel.d.ts +21 -0
  196. package/dist/components/LifoMonitorPanel.d.ts.map +1 -0
  197. package/dist/components/LifoMonitorPanel.js +24 -0
  198. package/dist/components/LifoSandboxView.d.ts +5 -0
  199. package/dist/components/LifoSandboxView.d.ts.map +1 -0
  200. package/dist/components/LifoSandboxView.js +333 -0
  201. package/dist/components/LoadingScreen.d.ts +13 -0
  202. package/dist/components/LoadingScreen.d.ts.map +1 -0
  203. package/dist/components/LoadingScreen.js +39 -0
  204. package/dist/components/LogsPageView.d.ts +2 -0
  205. package/dist/components/LogsPageView.d.ts.map +1 -0
  206. package/dist/components/LogsPageView.js +7 -0
  207. package/dist/components/LogsView.d.ts +5 -0
  208. package/dist/components/LogsView.d.ts.map +1 -0
  209. package/dist/components/LogsView.js +71 -0
  210. package/dist/components/MediaGalleryView.d.ts +9 -0
  211. package/dist/components/MediaGalleryView.d.ts.map +1 -0
  212. package/dist/components/MediaGalleryView.js +236 -0
  213. package/dist/components/MediaSettingsSection.d.ts +11 -0
  214. package/dist/components/MediaSettingsSection.d.ts.map +1 -0
  215. package/dist/components/MediaSettingsSection.js +329 -0
  216. package/dist/components/MessageContent.d.ts +51 -0
  217. package/dist/components/MessageContent.d.ts.map +1 -0
  218. package/dist/components/MessageContent.js +553 -0
  219. package/dist/components/OnboardingWizard.d.ts +2 -0
  220. package/dist/components/OnboardingWizard.d.ts.map +1 -0
  221. package/dist/components/OnboardingWizard.js +59 -0
  222. package/dist/components/PairingView.d.ts +5 -0
  223. package/dist/components/PairingView.d.ts.map +1 -0
  224. package/dist/components/PairingView.js +28 -0
  225. package/dist/components/PermissionsSection.d.ts +20 -0
  226. package/dist/components/PermissionsSection.d.ts.map +1 -0
  227. package/dist/components/PermissionsSection.js +368 -0
  228. package/dist/components/PluginsPageView.d.ts +5 -0
  229. package/dist/components/PluginsPageView.d.ts.map +1 -0
  230. package/dist/components/PluginsPageView.js +8 -0
  231. package/dist/components/PluginsView.d.ts +21 -0
  232. package/dist/components/PluginsView.d.ts.map +1 -0
  233. package/dist/components/PluginsView.js +1531 -0
  234. package/dist/components/ProviderSwitcher.d.ts +42 -0
  235. package/dist/components/ProviderSwitcher.d.ts.map +1 -0
  236. package/dist/components/ProviderSwitcher.js +480 -0
  237. package/dist/components/RestartBanner.d.ts +2 -0
  238. package/dist/components/RestartBanner.d.ts.map +1 -0
  239. package/dist/components/RestartBanner.js +36 -0
  240. package/dist/components/RuntimeView.d.ts +10 -0
  241. package/dist/components/RuntimeView.d.ts.map +1 -0
  242. package/dist/components/RuntimeView.js +165 -0
  243. package/dist/components/SaveCommandModal.d.ts +12 -0
  244. package/dist/components/SaveCommandModal.d.ts.map +1 -0
  245. package/dist/components/SaveCommandModal.js +84 -0
  246. package/dist/components/SecretsView.d.ts +9 -0
  247. package/dist/components/SecretsView.d.ts.map +1 -0
  248. package/dist/components/SecretsView.js +249 -0
  249. package/dist/components/SettingsView.d.ts +9 -0
  250. package/dist/components/SettingsView.d.ts.map +1 -0
  251. package/dist/components/SettingsView.js +230 -0
  252. package/dist/components/ShellOverlays.d.ts +8 -0
  253. package/dist/components/ShellOverlays.d.ts.map +1 -0
  254. package/dist/components/ShellOverlays.js +10 -0
  255. package/dist/components/ShortcutsOverlay.d.ts +2 -0
  256. package/dist/components/ShortcutsOverlay.d.ts.map +1 -0
  257. package/dist/components/ShortcutsOverlay.js +79 -0
  258. package/dist/components/SkillsView.d.ts +11 -0
  259. package/dist/components/SkillsView.d.ts.map +1 -0
  260. package/dist/components/SkillsView.js +358 -0
  261. package/dist/components/StartupFailureView.d.ts +8 -0
  262. package/dist/components/StartupFailureView.d.ts.map +1 -0
  263. package/dist/components/StartupFailureView.js +15 -0
  264. package/dist/components/StreamView.d.ts +16 -0
  265. package/dist/components/StreamView.d.ts.map +1 -0
  266. package/dist/components/StreamView.js +300 -0
  267. package/dist/components/StripeEmbeddedCheckout.d.ts +24 -0
  268. package/dist/components/StripeEmbeddedCheckout.d.ts.map +1 -0
  269. package/dist/components/StripeEmbeddedCheckout.js +101 -0
  270. package/dist/components/SubscriptionStatus.d.ts +22 -0
  271. package/dist/components/SubscriptionStatus.d.ts.map +1 -0
  272. package/dist/components/SubscriptionStatus.js +301 -0
  273. package/dist/components/SystemWarningBanner.d.ts +6 -0
  274. package/dist/components/SystemWarningBanner.d.ts.map +1 -0
  275. package/dist/components/SystemWarningBanner.js +46 -0
  276. package/dist/components/ThemeToggle.d.ts +21 -0
  277. package/dist/components/ThemeToggle.d.ts.map +1 -0
  278. package/dist/components/ThemeToggle.js +24 -0
  279. package/dist/components/TrajectoriesView.d.ts +12 -0
  280. package/dist/components/TrajectoriesView.d.ts.map +1 -0
  281. package/dist/components/TrajectoriesView.js +183 -0
  282. package/dist/components/TrajectoryDetailView.d.ts +13 -0
  283. package/dist/components/TrajectoryDetailView.d.ts.map +1 -0
  284. package/dist/components/TrajectoryDetailView.js +112 -0
  285. package/dist/components/TriggersView.d.ts +2 -0
  286. package/dist/components/TriggersView.d.ts.map +1 -0
  287. package/dist/components/TriggersView.js +1 -0
  288. package/dist/components/VectorBrowserView.d.ts +10 -0
  289. package/dist/components/VectorBrowserView.d.ts.map +1 -0
  290. package/dist/components/VectorBrowserView.js +997 -0
  291. package/dist/components/VoiceConfigView.d.ts +11 -0
  292. package/dist/components/VoiceConfigView.d.ts.map +1 -0
  293. package/dist/components/VoiceConfigView.js +329 -0
  294. package/dist/components/VrmStage.d.ts +21 -0
  295. package/dist/components/VrmStage.d.ts.map +1 -0
  296. package/dist/components/VrmStage.js +252 -0
  297. package/dist/components/WhatsAppQrOverlay.d.ts +8 -0
  298. package/dist/components/WhatsAppQrOverlay.d.ts.map +1 -0
  299. package/dist/components/WhatsAppQrOverlay.js +63 -0
  300. package/dist/components/apps/AppDetailPane.d.ts +15 -0
  301. package/dist/components/apps/AppDetailPane.d.ts.map +1 -0
  302. package/dist/components/apps/AppDetailPane.js +13 -0
  303. package/dist/components/apps/AppsCatalogGrid.d.ts +20 -0
  304. package/dist/components/apps/AppsCatalogGrid.d.ts.map +1 -0
  305. package/dist/components/apps/AppsCatalogGrid.js +18 -0
  306. package/dist/components/apps/extensions/HyperscapeAppDetailPanel.d.ts +3 -0
  307. package/dist/components/apps/extensions/HyperscapeAppDetailPanel.d.ts.map +1 -0
  308. package/dist/components/apps/extensions/HyperscapeAppDetailPanel.js +253 -0
  309. package/dist/components/apps/extensions/registry.d.ts +4 -0
  310. package/dist/components/apps/extensions/registry.d.ts.map +1 -0
  311. package/dist/components/apps/extensions/registry.js +10 -0
  312. package/dist/components/apps/extensions/types.d.ts +7 -0
  313. package/dist/components/apps/extensions/types.d.ts.map +1 -0
  314. package/dist/components/apps/extensions/types.js +1 -0
  315. package/dist/components/apps/helpers.d.ts +7 -0
  316. package/dist/components/apps/helpers.d.ts.map +1 -0
  317. package/dist/components/apps/helpers.js +46 -0
  318. package/dist/components/avatar/VrmAnimationLoader.d.ts +30 -0
  319. package/dist/components/avatar/VrmAnimationLoader.d.ts.map +1 -0
  320. package/dist/components/avatar/VrmAnimationLoader.js +99 -0
  321. package/dist/components/avatar/VrmBlinkController.d.ts +37 -0
  322. package/dist/components/avatar/VrmBlinkController.d.ts.map +1 -0
  323. package/dist/components/avatar/VrmBlinkController.js +98 -0
  324. package/dist/components/avatar/VrmCameraManager.d.ts +57 -0
  325. package/dist/components/avatar/VrmCameraManager.d.ts.map +1 -0
  326. package/dist/components/avatar/VrmCameraManager.js +277 -0
  327. package/dist/components/avatar/VrmEngine.d.ts +229 -0
  328. package/dist/components/avatar/VrmEngine.d.ts.map +1 -0
  329. package/dist/components/avatar/VrmEngine.js +1950 -0
  330. package/dist/components/avatar/VrmFootShadow.d.ts +18 -0
  331. package/dist/components/avatar/VrmFootShadow.d.ts.map +1 -0
  332. package/dist/components/avatar/VrmFootShadow.js +83 -0
  333. package/dist/components/avatar/VrmViewer.d.ts +45 -0
  334. package/dist/components/avatar/VrmViewer.d.ts.map +1 -0
  335. package/dist/components/avatar/VrmViewer.js +341 -0
  336. package/dist/components/avatar/mixamoVRMRigMap.d.ts +3 -0
  337. package/dist/components/avatar/mixamoVRMRigMap.d.ts.map +1 -0
  338. package/dist/components/avatar/mixamoVRMRigMap.js +56 -0
  339. package/dist/components/avatar/retargetMixamoFbxToVrm.d.ts +9 -0
  340. package/dist/components/avatar/retargetMixamoFbxToVrm.d.ts.map +1 -0
  341. package/dist/components/avatar/retargetMixamoFbxToVrm.js +88 -0
  342. package/dist/components/avatar/retargetMixamoGltfToVrm.d.ts +11 -0
  343. package/dist/components/avatar/retargetMixamoGltfToVrm.d.ts.map +1 -0
  344. package/dist/components/avatar/retargetMixamoGltfToVrm.js +80 -0
  345. package/dist/components/chainConfig.d.ts +84 -0
  346. package/dist/components/chainConfig.d.ts.map +1 -0
  347. package/dist/components/chainConfig.js +268 -0
  348. package/dist/components/companion/CompanionHeader.d.ts +15 -0
  349. package/dist/components/companion/CompanionHeader.d.ts.map +1 -0
  350. package/dist/components/companion/CompanionHeader.js +7 -0
  351. package/dist/components/companion/CompanionSceneHost.d.ts +2 -0
  352. package/dist/components/companion/CompanionSceneHost.d.ts.map +1 -0
  353. package/dist/components/companion/CompanionSceneHost.js +1 -0
  354. package/dist/components/companion/VrmStage.d.ts +3 -0
  355. package/dist/components/companion/VrmStage.d.ts.map +1 -0
  356. package/dist/components/companion/VrmStage.js +1 -0
  357. package/dist/components/companion/walletUtils.d.ts +95 -0
  358. package/dist/components/companion/walletUtils.d.ts.map +1 -0
  359. package/dist/components/companion/walletUtils.js +167 -0
  360. package/dist/components/companion-shell-styles.d.ts +38 -0
  361. package/dist/components/companion-shell-styles.d.ts.map +1 -0
  362. package/dist/components/companion-shell-styles.js +248 -0
  363. package/dist/components/confirm-delete-control.d.ts +16 -0
  364. package/dist/components/confirm-delete-control.d.ts.map +1 -0
  365. package/dist/components/confirm-delete-control.js +12 -0
  366. package/dist/components/conversations/ConversationListItem.d.ts +31 -0
  367. package/dist/components/conversations/ConversationListItem.d.ts.map +1 -0
  368. package/dist/components/conversations/ConversationListItem.js +52 -0
  369. package/dist/components/conversations/conversation-utils.d.ts +9 -0
  370. package/dist/components/conversations/conversation-utils.d.ts.map +1 -0
  371. package/dist/components/conversations/conversation-utils.js +138 -0
  372. package/dist/components/format.d.ts +54 -0
  373. package/dist/components/format.d.ts.map +1 -0
  374. package/dist/components/format.js +82 -0
  375. package/dist/components/index.d.ts +93 -0
  376. package/dist/components/index.d.ts.map +1 -0
  377. package/dist/components/index.js +92 -0
  378. package/dist/components/inventory/CopyableAddress.d.ts +8 -0
  379. package/dist/components/inventory/CopyableAddress.d.ts.map +1 -0
  380. package/dist/components/inventory/CopyableAddress.js +18 -0
  381. package/dist/components/inventory/InventoryToolbar.d.ts +25 -0
  382. package/dist/components/inventory/InventoryToolbar.d.ts.map +1 -0
  383. package/dist/components/inventory/InventoryToolbar.js +28 -0
  384. package/dist/components/inventory/NftGrid.d.ts +13 -0
  385. package/dist/components/inventory/NftGrid.d.ts.map +1 -0
  386. package/dist/components/inventory/NftGrid.js +29 -0
  387. package/dist/components/inventory/TokenLogo.d.ts +12 -0
  388. package/dist/components/inventory/TokenLogo.d.ts.map +1 -0
  389. package/dist/components/inventory/TokenLogo.js +33 -0
  390. package/dist/components/inventory/TokensTable.d.ts +24 -0
  391. package/dist/components/inventory/TokensTable.d.ts.map +1 -0
  392. package/dist/components/inventory/TokensTable.js +26 -0
  393. package/dist/components/inventory/constants.d.ts +52 -0
  394. package/dist/components/inventory/constants.d.ts.map +1 -0
  395. package/dist/components/inventory/constants.js +121 -0
  396. package/dist/components/inventory/index.d.ts +9 -0
  397. package/dist/components/inventory/index.d.ts.map +1 -0
  398. package/dist/components/inventory/index.js +8 -0
  399. package/dist/components/inventory/media-url.d.ts +6 -0
  400. package/dist/components/inventory/media-url.d.ts.map +1 -0
  401. package/dist/components/inventory/media-url.js +28 -0
  402. package/dist/components/inventory/useInventoryData.d.ts +53 -0
  403. package/dist/components/inventory/useInventoryData.d.ts.map +1 -0
  404. package/dist/components/inventory/useInventoryData.js +332 -0
  405. package/dist/components/knowledge-upload-image.d.ts +27 -0
  406. package/dist/components/knowledge-upload-image.d.ts.map +1 -0
  407. package/dist/components/knowledge-upload-image.js +146 -0
  408. package/dist/components/labels.d.ts +6 -0
  409. package/dist/components/labels.d.ts.map +1 -0
  410. package/dist/components/labels.js +40 -0
  411. package/dist/components/onboarding/ActivateStep.d.ts +2 -0
  412. package/dist/components/onboarding/ActivateStep.d.ts.map +1 -0
  413. package/dist/components/onboarding/ActivateStep.js +6 -0
  414. package/dist/components/onboarding/ConnectionStep.d.ts +2 -0
  415. package/dist/components/onboarding/ConnectionStep.d.ts.map +1 -0
  416. package/dist/components/onboarding/ConnectionStep.js +552 -0
  417. package/dist/components/onboarding/OnboardingPanel.d.ts +9 -0
  418. package/dist/components/onboarding/OnboardingPanel.d.ts.map +1 -0
  419. package/dist/components/onboarding/OnboardingPanel.js +24 -0
  420. package/dist/components/onboarding/OnboardingStepNav.d.ts +2 -0
  421. package/dist/components/onboarding/OnboardingStepNav.d.ts.map +1 -0
  422. package/dist/components/onboarding/OnboardingStepNav.js +14 -0
  423. package/dist/components/onboarding/PermissionsStep.d.ts +2 -0
  424. package/dist/components/onboarding/PermissionsStep.d.ts.map +1 -0
  425. package/dist/components/onboarding/PermissionsStep.js +7 -0
  426. package/dist/components/onboarding/RpcStep.d.ts +2 -0
  427. package/dist/components/onboarding/RpcStep.d.ts.map +1 -0
  428. package/dist/components/onboarding/RpcStep.js +125 -0
  429. package/dist/components/onboarding/WakeUpStep.d.ts +2 -0
  430. package/dist/components/onboarding/WakeUpStep.d.ts.map +1 -0
  431. package/dist/components/onboarding/WakeUpStep.js +82 -0
  432. package/dist/components/permissions/PermissionIcon.d.ts +4 -0
  433. package/dist/components/permissions/PermissionIcon.d.ts.map +1 -0
  434. package/dist/components/permissions/PermissionIcon.js +12 -0
  435. package/dist/components/permissions/StreamingPermissions.d.ts +20 -0
  436. package/dist/components/permissions/StreamingPermissions.d.ts.map +1 -0
  437. package/dist/components/permissions/StreamingPermissions.js +173 -0
  438. package/dist/components/plugins/showcase-data.d.ts +7 -0
  439. package/dist/components/plugins/showcase-data.d.ts.map +1 -0
  440. package/dist/components/plugins/showcase-data.js +464 -0
  441. package/dist/components/shared/ShellHeaderControls.d.ts +27 -0
  442. package/dist/components/shared/ShellHeaderControls.d.ts.map +1 -0
  443. package/dist/components/shared/ShellHeaderControls.js +60 -0
  444. package/dist/components/skeletons.d.ts +17 -0
  445. package/dist/components/skeletons.d.ts.map +1 -0
  446. package/dist/components/skeletons.js +22 -0
  447. package/dist/components/stream/ActivityFeed.d.ts +5 -0
  448. package/dist/components/stream/ActivityFeed.d.ts.map +1 -0
  449. package/dist/components/stream/ActivityFeed.js +57 -0
  450. package/dist/components/stream/AvatarPip.d.ts +5 -0
  451. package/dist/components/stream/AvatarPip.d.ts.map +1 -0
  452. package/dist/components/stream/AvatarPip.js +6 -0
  453. package/dist/components/stream/ChatContent.d.ts +6 -0
  454. package/dist/components/stream/ChatContent.d.ts.map +1 -0
  455. package/dist/components/stream/ChatContent.js +69 -0
  456. package/dist/components/stream/ChatTicker.d.ts +5 -0
  457. package/dist/components/stream/ChatTicker.d.ts.map +1 -0
  458. package/dist/components/stream/ChatTicker.js +34 -0
  459. package/dist/components/stream/IdleContent.d.ts +5 -0
  460. package/dist/components/stream/IdleContent.d.ts.map +1 -0
  461. package/dist/components/stream/IdleContent.js +17 -0
  462. package/dist/components/stream/StatusBar.d.ts +36 -0
  463. package/dist/components/stream/StatusBar.d.ts.map +1 -0
  464. package/dist/components/stream/StatusBar.js +140 -0
  465. package/dist/components/stream/StreamSettings.d.ts +33 -0
  466. package/dist/components/stream/StreamSettings.d.ts.map +1 -0
  467. package/dist/components/stream/StreamSettings.js +99 -0
  468. package/dist/components/stream/StreamTerminal.d.ts +2 -0
  469. package/dist/components/stream/StreamTerminal.d.ts.map +1 -0
  470. package/dist/components/stream/StreamTerminal.js +52 -0
  471. package/dist/components/stream/StreamVoiceConfig.d.ts +10 -0
  472. package/dist/components/stream/StreamVoiceConfig.d.ts.map +1 -0
  473. package/dist/components/stream/StreamVoiceConfig.js +88 -0
  474. package/dist/components/stream/helpers.d.ts +32 -0
  475. package/dist/components/stream/helpers.d.ts.map +1 -0
  476. package/dist/components/stream/helpers.js +110 -0
  477. package/dist/components/stream/overlays/OverlayLayer.d.ts +20 -0
  478. package/dist/components/stream/overlays/OverlayLayer.d.ts.map +1 -0
  479. package/dist/components/stream/overlays/OverlayLayer.js +24 -0
  480. package/dist/components/stream/overlays/built-in/ActionTickerWidget.d.ts +8 -0
  481. package/dist/components/stream/overlays/built-in/ActionTickerWidget.d.ts.map +1 -0
  482. package/dist/components/stream/overlays/built-in/ActionTickerWidget.js +44 -0
  483. package/dist/components/stream/overlays/built-in/AlertPopupWidget.d.ts +7 -0
  484. package/dist/components/stream/overlays/built-in/AlertPopupWidget.d.ts.map +1 -0
  485. package/dist/components/stream/overlays/built-in/AlertPopupWidget.js +62 -0
  486. package/dist/components/stream/overlays/built-in/BrandingWidget.d.ts +7 -0
  487. package/dist/components/stream/overlays/built-in/BrandingWidget.d.ts.map +1 -0
  488. package/dist/components/stream/overlays/built-in/BrandingWidget.js +36 -0
  489. package/dist/components/stream/overlays/built-in/CustomHtmlWidget.d.ts +26 -0
  490. package/dist/components/stream/overlays/built-in/CustomHtmlWidget.d.ts.map +1 -0
  491. package/dist/components/stream/overlays/built-in/CustomHtmlWidget.js +78 -0
  492. package/dist/components/stream/overlays/built-in/PeonGlassWidget.d.ts +11 -0
  493. package/dist/components/stream/overlays/built-in/PeonGlassWidget.d.ts.map +1 -0
  494. package/dist/components/stream/overlays/built-in/PeonGlassWidget.js +188 -0
  495. package/dist/components/stream/overlays/built-in/PeonHudWidget.d.ts +10 -0
  496. package/dist/components/stream/overlays/built-in/PeonHudWidget.d.ts.map +1 -0
  497. package/dist/components/stream/overlays/built-in/PeonHudWidget.js +168 -0
  498. package/dist/components/stream/overlays/built-in/PeonSakuraWidget.d.ts +11 -0
  499. package/dist/components/stream/overlays/built-in/PeonSakuraWidget.d.ts.map +1 -0
  500. package/dist/components/stream/overlays/built-in/PeonSakuraWidget.js +213 -0
  501. package/dist/components/stream/overlays/built-in/ThoughtBubbleWidget.d.ts +8 -0
  502. package/dist/components/stream/overlays/built-in/ThoughtBubbleWidget.d.ts.map +1 -0
  503. package/dist/components/stream/overlays/built-in/ThoughtBubbleWidget.js +59 -0
  504. package/dist/components/stream/overlays/built-in/ViewerCountWidget.d.ts +7 -0
  505. package/dist/components/stream/overlays/built-in/ViewerCountWidget.d.ts.map +1 -0
  506. package/dist/components/stream/overlays/built-in/ViewerCountWidget.js +34 -0
  507. package/dist/components/stream/overlays/built-in/index.d.ts +13 -0
  508. package/dist/components/stream/overlays/built-in/index.d.ts.map +1 -0
  509. package/dist/components/stream/overlays/built-in/index.js +12 -0
  510. package/dist/components/stream/overlays/registry.d.ts +11 -0
  511. package/dist/components/stream/overlays/registry.d.ts.map +1 -0
  512. package/dist/components/stream/overlays/registry.js +16 -0
  513. package/dist/components/stream/overlays/types.d.ts +67 -0
  514. package/dist/components/stream/overlays/types.d.ts.map +1 -0
  515. package/dist/components/stream/overlays/types.js +7 -0
  516. package/dist/components/stream/overlays/useOverlayLayout.d.ts +27 -0
  517. package/dist/components/stream/overlays/useOverlayLayout.d.ts.map +1 -0
  518. package/dist/components/stream/overlays/useOverlayLayout.js +162 -0
  519. package/dist/components/trajectory-format.d.ts +6 -0
  520. package/dist/components/trajectory-format.d.ts.map +1 -0
  521. package/dist/components/trajectory-format.js +43 -0
  522. package/dist/components/ui-badges.d.ts +23 -0
  523. package/dist/components/ui-badges.d.ts.map +1 -0
  524. package/dist/components/ui-badges.js +38 -0
  525. package/dist/components/ui-switch.d.ts +14 -0
  526. package/dist/components/ui-switch.d.ts.map +1 -0
  527. package/dist/components/ui-switch.js +15 -0
  528. package/dist/components/vector-browser-three.d.ts +4 -0
  529. package/dist/components/vector-browser-three.d.ts.map +1 -0
  530. package/dist/components/vector-browser-three.js +19 -0
  531. package/dist/config/config-catalog.d.ts +376 -0
  532. package/dist/config/config-catalog.d.ts.map +1 -0
  533. package/dist/config/config-catalog.js +724 -0
  534. package/dist/config/config-field.d.ts +68 -0
  535. package/dist/config/config-field.d.ts.map +1 -0
  536. package/dist/config/config-field.js +821 -0
  537. package/dist/config/config-renderer.d.ts +176 -0
  538. package/dist/config/config-renderer.d.ts.map +1 -0
  539. package/dist/config/config-renderer.js +403 -0
  540. package/dist/config/index.d.ts +5 -0
  541. package/dist/config/index.d.ts.map +1 -0
  542. package/dist/config/index.js +4 -0
  543. package/dist/config/ui-renderer.d.ts +26 -0
  544. package/dist/config/ui-renderer.d.ts.map +1 -0
  545. package/dist/config/ui-renderer.js +815 -0
  546. package/dist/config/ui-spec.d.ts +164 -0
  547. package/dist/config/ui-spec.d.ts.map +1 -0
  548. package/dist/config/ui-spec.js +13 -0
  549. package/dist/events/index.d.ts +42 -0
  550. package/dist/events/index.d.ts.map +1 -0
  551. package/dist/events/index.js +41 -0
  552. package/dist/hooks/index.d.ts +14 -0
  553. package/dist/hooks/index.d.ts.map +1 -0
  554. package/dist/hooks/index.js +13 -0
  555. package/dist/hooks/useBugReport.d.ts +14 -0
  556. package/dist/hooks/useBugReport.d.ts.map +1 -0
  557. package/dist/hooks/useBugReport.js +18 -0
  558. package/dist/hooks/useCanvasWindow.d.ts +38 -0
  559. package/dist/hooks/useCanvasWindow.d.ts.map +1 -0
  560. package/dist/hooks/useCanvasWindow.js +273 -0
  561. package/dist/hooks/useChatAvatarVoice.d.ts +10 -0
  562. package/dist/hooks/useChatAvatarVoice.d.ts.map +1 -0
  563. package/dist/hooks/useChatAvatarVoice.js +71 -0
  564. package/dist/hooks/useContextMenu.d.ts +17 -0
  565. package/dist/hooks/useContextMenu.d.ts.map +1 -0
  566. package/dist/hooks/useContextMenu.js +100 -0
  567. package/dist/hooks/useKeyboardShortcuts.d.ts +17 -0
  568. package/dist/hooks/useKeyboardShortcuts.d.ts.map +1 -0
  569. package/dist/hooks/useKeyboardShortcuts.js +67 -0
  570. package/dist/hooks/useLifoSync.d.ts +18 -0
  571. package/dist/hooks/useLifoSync.d.ts.map +1 -0
  572. package/dist/hooks/useLifoSync.js +113 -0
  573. package/dist/hooks/useMemoryMonitor.d.ts +87 -0
  574. package/dist/hooks/useMemoryMonitor.d.ts.map +1 -0
  575. package/dist/hooks/useMemoryMonitor.js +210 -0
  576. package/dist/hooks/useRenderGuard.d.ts +17 -0
  577. package/dist/hooks/useRenderGuard.d.ts.map +1 -0
  578. package/dist/hooks/useRenderGuard.js +36 -0
  579. package/dist/hooks/useRetakeCapture.d.ts +12 -0
  580. package/dist/hooks/useRetakeCapture.d.ts.map +1 -0
  581. package/dist/hooks/useRetakeCapture.js +59 -0
  582. package/dist/hooks/useStreamPopoutNavigation.d.ts +3 -0
  583. package/dist/hooks/useStreamPopoutNavigation.d.ts.map +1 -0
  584. package/dist/hooks/useStreamPopoutNavigation.js +17 -0
  585. package/dist/hooks/useTimeout.d.ts +11 -0
  586. package/dist/hooks/useTimeout.d.ts.map +1 -0
  587. package/dist/hooks/useTimeout.js +32 -0
  588. package/dist/hooks/useVoiceChat.d.ts +129 -0
  589. package/dist/hooks/useVoiceChat.d.ts.map +1 -0
  590. package/dist/hooks/useVoiceChat.js +1071 -0
  591. package/dist/hooks/useWhatsAppPairing.d.ts +11 -0
  592. package/dist/hooks/useWhatsAppPairing.d.ts.map +1 -0
  593. package/dist/hooks/useWhatsAppPairing.js +95 -0
  594. package/dist/i18n/index.d.ts +7 -0
  595. package/dist/i18n/index.d.ts.map +1 -0
  596. package/dist/i18n/index.js +53 -0
  597. package/dist/i18n/locales/en.json +1196 -0
  598. package/dist/i18n/locales/es.json +1196 -0
  599. package/dist/i18n/locales/ko.json +1196 -0
  600. package/dist/i18n/locales/pt.json +1196 -0
  601. package/dist/i18n/locales/zh-CN.json +1196 -0
  602. package/dist/i18n/messages.d.ts +6 -0
  603. package/dist/i18n/messages.d.ts.map +1 -0
  604. package/dist/i18n/messages.js +14 -0
  605. package/dist/index.d.ts +5 -0
  606. package/dist/index.d.ts.map +1 -0
  607. package/dist/index.js +6 -0
  608. package/dist/navigation/index.d.ts +27 -0
  609. package/dist/navigation/index.d.ts.map +1 -0
  610. package/dist/navigation/index.js +230 -0
  611. package/dist/package.json +161 -0
  612. package/dist/platform/browser-launch.d.ts +2 -0
  613. package/dist/platform/browser-launch.d.ts.map +1 -0
  614. package/dist/platform/browser-launch.js +109 -0
  615. package/dist/platform/index.d.ts +64 -0
  616. package/dist/platform/index.d.ts.map +1 -0
  617. package/dist/platform/index.js +155 -0
  618. package/dist/platform/init.d.ts +40 -0
  619. package/dist/platform/init.d.ts.map +1 -0
  620. package/dist/platform/init.js +158 -0
  621. package/dist/providers/index.d.ts +13 -0
  622. package/dist/providers/index.d.ts.map +1 -0
  623. package/dist/providers/index.js +72 -0
  624. package/dist/state/AppContext.d.ts +11 -0
  625. package/dist/state/AppContext.d.ts.map +1 -0
  626. package/dist/state/AppContext.js +4496 -0
  627. package/dist/state/index.d.ts +7 -0
  628. package/dist/state/index.d.ts.map +1 -0
  629. package/dist/state/index.js +6 -0
  630. package/dist/state/internal.d.ts +6 -0
  631. package/dist/state/internal.d.ts.map +1 -0
  632. package/dist/state/internal.js +5 -0
  633. package/dist/state/parsers.d.ts +26 -0
  634. package/dist/state/parsers.d.ts.map +1 -0
  635. package/dist/state/parsers.js +255 -0
  636. package/dist/state/persistence.d.ts +36 -0
  637. package/dist/state/persistence.d.ts.map +1 -0
  638. package/dist/state/persistence.js +254 -0
  639. package/dist/state/types.d.ts +402 -0
  640. package/dist/state/types.d.ts.map +1 -0
  641. package/dist/state/types.js +70 -0
  642. package/dist/state/ui-preferences.d.ts +3 -0
  643. package/dist/state/ui-preferences.d.ts.map +1 -0
  644. package/dist/state/ui-preferences.js +1 -0
  645. package/dist/state/useApp.d.ts +4 -0
  646. package/dist/state/useApp.d.ts.map +1 -0
  647. package/dist/state/useApp.js +22 -0
  648. package/dist/state/vrm.d.ts +19 -0
  649. package/dist/state/vrm.d.ts.map +1 -0
  650. package/dist/state/vrm.js +65 -0
  651. package/dist/stories/AppMockProvider.d.ts +15 -0
  652. package/dist/stories/AppMockProvider.d.ts.map +1 -0
  653. package/dist/stories/AppMockProvider.js +14 -0
  654. package/dist/styles/anime.css +6324 -0
  655. package/dist/styles/base.css +196 -0
  656. package/dist/styles/onboarding-game.css +716 -0
  657. package/dist/styles/styles.css +2085 -0
  658. package/dist/styles/xterm.css +241 -0
  659. package/dist/types/index.d.ts +657 -0
  660. package/dist/types/index.d.ts.map +1 -0
  661. package/dist/types/index.js +1 -0
  662. package/dist/utils/asset-url.d.ts +26 -0
  663. package/dist/utils/asset-url.d.ts.map +1 -0
  664. package/dist/utils/asset-url.js +99 -0
  665. package/dist/utils/assistant-text.d.ts +2 -0
  666. package/dist/utils/assistant-text.d.ts.map +1 -0
  667. package/dist/utils/assistant-text.js +161 -0
  668. package/dist/utils/clipboard.d.ts +2 -0
  669. package/dist/utils/clipboard.d.ts.map +1 -0
  670. package/dist/utils/clipboard.js +38 -0
  671. package/dist/utils/desktop-dialogs.d.ts +19 -0
  672. package/dist/utils/desktop-dialogs.d.ts.map +1 -0
  673. package/dist/utils/desktop-dialogs.js +50 -0
  674. package/dist/utils/index.d.ts +7 -0
  675. package/dist/utils/index.d.ts.map +1 -0
  676. package/dist/utils/index.js +6 -0
  677. package/dist/utils/number-parsing.d.ts +44 -0
  678. package/dist/utils/number-parsing.d.ts.map +1 -0
  679. package/dist/utils/number-parsing.js +56 -0
  680. package/dist/utils/openExternalUrl.d.ts +2 -0
  681. package/dist/utils/openExternalUrl.d.ts.map +1 -0
  682. package/dist/utils/openExternalUrl.js +17 -0
  683. package/dist/utils/spoken-text.d.ts +2 -0
  684. package/dist/utils/spoken-text.d.ts.map +1 -0
  685. package/dist/utils/spoken-text.js +56 -0
  686. package/dist/utils/streaming-text.d.ts +3 -0
  687. package/dist/utils/streaming-text.d.ts.map +1 -0
  688. package/dist/utils/streaming-text.js +87 -0
  689. package/dist/voice/index.d.ts +2 -0
  690. package/dist/voice/index.d.ts.map +1 -0
  691. package/dist/voice/index.js +1 -0
  692. package/dist/voice/types.d.ts +25 -0
  693. package/dist/voice/types.d.ts.map +1 -0
  694. package/dist/voice/types.js +166 -0
  695. package/package.json +86 -0
  696. package/src/App.tsx +469 -0
  697. package/src/actions/character.ts +113 -0
  698. package/src/actions/chat-helpers.ts +100 -0
  699. package/src/actions/cloud.ts +59 -0
  700. package/src/actions/index.ts +12 -0
  701. package/src/actions/lifecycle.ts +175 -0
  702. package/src/actions/onboarding.ts +46 -0
  703. package/src/actions/triggers.ts +190 -0
  704. package/src/ambient.d.ts +16 -0
  705. package/src/api/client.ts +5616 -0
  706. package/src/api/index.ts +1 -0
  707. package/src/autonomy/index.ts +477 -0
  708. package/src/bridge/capacitor-bridge.ts +295 -0
  709. package/src/bridge/electrobun-rpc.ts +58 -0
  710. package/src/bridge/electrobun-runtime.ts +28 -0
  711. package/src/bridge/index.ts +5 -0
  712. package/src/bridge/native-plugins.ts +134 -0
  713. package/src/bridge/plugin-bridge.ts +352 -0
  714. package/src/bridge/storage-bridge.ts +162 -0
  715. package/src/chat/index.ts +251 -0
  716. package/src/coding/index.ts +43 -0
  717. package/src/components/AdvancedPageView.tsx +362 -0
  718. package/src/components/AgentActivityBox.tsx +49 -0
  719. package/src/components/ApiKeyConfig.tsx +224 -0
  720. package/src/components/AppsPageView.tsx +52 -0
  721. package/src/components/AppsView.tsx +293 -0
  722. package/src/components/AvatarLoader.tsx +86 -0
  723. package/src/components/AvatarSelector.tsx +223 -0
  724. package/src/components/BscTradePanel.tsx +549 -0
  725. package/src/components/BugReportModal.tsx +499 -0
  726. package/src/components/CharacterView.tsx +1623 -0
  727. package/src/components/ChatAvatar.test.ts +96 -0
  728. package/src/components/ChatAvatar.tsx +147 -0
  729. package/src/components/ChatComposer.tsx +330 -0
  730. package/src/components/ChatMessage.tsx +448 -0
  731. package/src/components/ChatModalView.test.tsx +118 -0
  732. package/src/components/ChatModalView.tsx +125 -0
  733. package/src/components/ChatView.tsx +999 -0
  734. package/src/components/CloudSourceControls.tsx +80 -0
  735. package/src/components/CodingAgentSettingsSection.tsx +536 -0
  736. package/src/components/CommandPalette.tsx +284 -0
  737. package/src/components/CompanionSceneHost.tsx +498 -0
  738. package/src/components/CompanionShell.tsx +31 -0
  739. package/src/components/CompanionView.tsx +109 -0
  740. package/src/components/ConfigPageView.tsx +722 -0
  741. package/src/components/ConfigSaveFooter.tsx +41 -0
  742. package/src/components/ConfirmModal.tsx +379 -0
  743. package/src/components/ConnectionFailedBanner.tsx +91 -0
  744. package/src/components/ConnectorsPageView.tsx +13 -0
  745. package/src/components/ConversationsSidebar.tsx +279 -0
  746. package/src/components/CustomActionEditor.tsx +1125 -0
  747. package/src/components/CustomActionsPanel.tsx +288 -0
  748. package/src/components/CustomActionsView.tsx +322 -0
  749. package/src/components/DatabasePageView.tsx +55 -0
  750. package/src/components/DatabaseView.tsx +814 -0
  751. package/src/components/ElizaCloudDashboard.tsx +1696 -0
  752. package/src/components/EmotePicker.tsx +529 -0
  753. package/src/components/ErrorBoundary.tsx +76 -0
  754. package/src/components/FineTuningView.tsx +1080 -0
  755. package/src/components/GameView.tsx +551 -0
  756. package/src/components/GameViewOverlay.tsx +133 -0
  757. package/src/components/GlobalEmoteOverlay.tsx +152 -0
  758. package/src/components/Header.test.tsx +413 -0
  759. package/src/components/Header.tsx +400 -0
  760. package/src/components/HeartbeatsView.tsx +1002 -0
  761. package/src/components/InventoryView.tsx +372 -0
  762. package/src/components/KnowledgeView.tsx +1128 -0
  763. package/src/components/LanguageDropdown.tsx +187 -0
  764. package/src/components/LifoMonitorPanel.tsx +196 -0
  765. package/src/components/LifoSandboxView.tsx +499 -0
  766. package/src/components/LoadingScreen.tsx +77 -0
  767. package/src/components/LogsPageView.tsx +17 -0
  768. package/src/components/LogsView.tsx +239 -0
  769. package/src/components/MediaGalleryView.tsx +432 -0
  770. package/src/components/MediaSettingsSection.tsx +893 -0
  771. package/src/components/MessageContent.tsx +815 -0
  772. package/src/components/OnboardingWizard.test.tsx +107 -0
  773. package/src/components/OnboardingWizard.tsx +186 -0
  774. package/src/components/PairingView.tsx +110 -0
  775. package/src/components/PermissionsSection.tsx +818 -0
  776. package/src/components/PluginsPageView.tsx +9 -0
  777. package/src/components/PluginsView.tsx +3152 -0
  778. package/src/components/ProviderSwitcher.tsx +874 -0
  779. package/src/components/RestartBanner.tsx +76 -0
  780. package/src/components/RuntimeView.tsx +460 -0
  781. package/src/components/SaveCommandModal.tsx +211 -0
  782. package/src/components/SecretsView.tsx +569 -0
  783. package/src/components/SettingsView.tsx +825 -0
  784. package/src/components/ShellOverlays.tsx +41 -0
  785. package/src/components/ShortcutsOverlay.tsx +155 -0
  786. package/src/components/SkillsView.tsx +1435 -0
  787. package/src/components/StartupFailureView.tsx +63 -0
  788. package/src/components/StreamView.tsx +483 -0
  789. package/src/components/StripeEmbeddedCheckout.tsx +155 -0
  790. package/src/components/SubscriptionStatus.tsx +634 -0
  791. package/src/components/SystemWarningBanner.tsx +71 -0
  792. package/src/components/ThemeToggle.tsx +100 -0
  793. package/src/components/TrajectoriesView.tsx +526 -0
  794. package/src/components/TrajectoryDetailView.tsx +426 -0
  795. package/src/components/TriggersView.tsx +1 -0
  796. package/src/components/VectorBrowserView.tsx +1633 -0
  797. package/src/components/VoiceConfigView.tsx +674 -0
  798. package/src/components/VrmStage.test.ts +219 -0
  799. package/src/components/VrmStage.tsx +432 -0
  800. package/src/components/WhatsAppQrOverlay.tsx +230 -0
  801. package/src/components/__tests__/chainConfig.test.ts +220 -0
  802. package/src/components/apps/AppDetailPane.tsx +242 -0
  803. package/src/components/apps/AppsCatalogGrid.tsx +137 -0
  804. package/src/components/apps/extensions/HyperscapeAppDetailPanel.tsx +577 -0
  805. package/src/components/apps/extensions/registry.ts +16 -0
  806. package/src/components/apps/extensions/types.ts +9 -0
  807. package/src/components/apps/helpers.ts +44 -0
  808. package/src/components/avatar/VrmAnimationLoader.test.ts +164 -0
  809. package/src/components/avatar/VrmAnimationLoader.ts +151 -0
  810. package/src/components/avatar/VrmBlinkController.ts +118 -0
  811. package/src/components/avatar/VrmCameraManager.ts +407 -0
  812. package/src/components/avatar/VrmEngine.ts +2678 -0
  813. package/src/components/avatar/VrmFootShadow.ts +96 -0
  814. package/src/components/avatar/VrmViewer.tsx +421 -0
  815. package/src/components/avatar/__tests__/VrmCameraManager.test.ts +168 -0
  816. package/src/components/avatar/__tests__/VrmEngine.test.ts +1574 -0
  817. package/src/components/avatar/mixamoVRMRigMap.ts +62 -0
  818. package/src/components/avatar/retargetMixamoFbxToVrm.ts +144 -0
  819. package/src/components/avatar/retargetMixamoGltfToVrm.ts +119 -0
  820. package/src/components/chainConfig.ts +380 -0
  821. package/src/components/companion/CompanionHeader.tsx +47 -0
  822. package/src/components/companion/CompanionSceneHost.tsx +5 -0
  823. package/src/components/companion/VrmStage.tsx +2 -0
  824. package/src/components/companion/__tests__/walletUtils.test.ts +742 -0
  825. package/src/components/companion/walletUtils.ts +290 -0
  826. package/src/components/companion-shell-styles.test.ts +146 -0
  827. package/src/components/companion-shell-styles.ts +270 -0
  828. package/src/components/confirm-delete-control.tsx +69 -0
  829. package/src/components/conversations/ConversationListItem.tsx +185 -0
  830. package/src/components/conversations/conversation-utils.ts +151 -0
  831. package/src/components/format.ts +131 -0
  832. package/src/components/index.ts +92 -0
  833. package/src/components/inventory/CopyableAddress.tsx +41 -0
  834. package/src/components/inventory/InventoryToolbar.tsx +142 -0
  835. package/src/components/inventory/NftGrid.tsx +99 -0
  836. package/src/components/inventory/TokenLogo.tsx +71 -0
  837. package/src/components/inventory/TokensTable.tsx +216 -0
  838. package/src/components/inventory/constants.ts +170 -0
  839. package/src/components/inventory/index.ts +29 -0
  840. package/src/components/inventory/media-url.test.ts +38 -0
  841. package/src/components/inventory/media-url.ts +36 -0
  842. package/src/components/inventory/useInventoryData.ts +460 -0
  843. package/src/components/knowledge-upload-image.ts +215 -0
  844. package/src/components/labels.ts +46 -0
  845. package/src/components/onboarding/ActivateStep.tsx +30 -0
  846. package/src/components/onboarding/ConnectionStep.tsx +1530 -0
  847. package/src/components/onboarding/OnboardingPanel.tsx +39 -0
  848. package/src/components/onboarding/OnboardingStepNav.tsx +31 -0
  849. package/src/components/onboarding/PermissionsStep.tsx +20 -0
  850. package/src/components/onboarding/RpcStep.tsx +402 -0
  851. package/src/components/onboarding/WakeUpStep.tsx +184 -0
  852. package/src/components/permissions/PermissionIcon.tsx +25 -0
  853. package/src/components/permissions/StreamingPermissions.tsx +376 -0
  854. package/src/components/plugins/showcase-data.ts +481 -0
  855. package/src/components/shared/ShellHeaderControls.tsx +193 -0
  856. package/src/components/skeletons.tsx +88 -0
  857. package/src/components/stream/ActivityFeed.tsx +113 -0
  858. package/src/components/stream/AvatarPip.tsx +10 -0
  859. package/src/components/stream/ChatContent.tsx +126 -0
  860. package/src/components/stream/ChatTicker.tsx +55 -0
  861. package/src/components/stream/IdleContent.tsx +73 -0
  862. package/src/components/stream/StatusBar.tsx +469 -0
  863. package/src/components/stream/StreamSettings.tsx +506 -0
  864. package/src/components/stream/StreamTerminal.tsx +94 -0
  865. package/src/components/stream/StreamVoiceConfig.tsx +160 -0
  866. package/src/components/stream/helpers.ts +134 -0
  867. package/src/components/stream/overlays/OverlayLayer.tsx +75 -0
  868. package/src/components/stream/overlays/built-in/ActionTickerWidget.tsx +64 -0
  869. package/src/components/stream/overlays/built-in/AlertPopupWidget.tsx +87 -0
  870. package/src/components/stream/overlays/built-in/BrandingWidget.tsx +51 -0
  871. package/src/components/stream/overlays/built-in/CustomHtmlWidget.tsx +105 -0
  872. package/src/components/stream/overlays/built-in/PeonGlassWidget.tsx +265 -0
  873. package/src/components/stream/overlays/built-in/PeonHudWidget.tsx +247 -0
  874. package/src/components/stream/overlays/built-in/PeonSakuraWidget.tsx +278 -0
  875. package/src/components/stream/overlays/built-in/ThoughtBubbleWidget.tsx +77 -0
  876. package/src/components/stream/overlays/built-in/ViewerCountWidget.tsx +46 -0
  877. package/src/components/stream/overlays/built-in/index.ts +13 -0
  878. package/src/components/stream/overlays/registry.ts +22 -0
  879. package/src/components/stream/overlays/types.ts +90 -0
  880. package/src/components/stream/overlays/useOverlayLayout.ts +218 -0
  881. package/src/components/trajectory-format.ts +50 -0
  882. package/src/components/ui-badges.tsx +109 -0
  883. package/src/components/ui-switch.tsx +57 -0
  884. package/src/components/vector-browser-three.ts +27 -0
  885. package/src/config/config-catalog.ts +1092 -0
  886. package/src/config/config-field.tsx +1900 -0
  887. package/src/config/config-renderer.tsx +730 -0
  888. package/src/config/index.ts +11 -0
  889. package/src/config/ui-renderer.tsx +1751 -0
  890. package/src/config/ui-spec.ts +256 -0
  891. package/src/events/index.ts +89 -0
  892. package/src/hooks/index.ts +13 -0
  893. package/src/hooks/useBugReport.tsx +43 -0
  894. package/src/hooks/useCanvasWindow.ts +372 -0
  895. package/src/hooks/useChatAvatarVoice.ts +111 -0
  896. package/src/hooks/useContextMenu.ts +127 -0
  897. package/src/hooks/useKeyboardShortcuts.ts +86 -0
  898. package/src/hooks/useLifoSync.ts +143 -0
  899. package/src/hooks/useMemoryMonitor.ts +335 -0
  900. package/src/hooks/useRenderGuard.ts +43 -0
  901. package/src/hooks/useRetakeCapture.ts +67 -0
  902. package/src/hooks/useStreamPopoutNavigation.ts +27 -0
  903. package/src/hooks/useTimeout.ts +37 -0
  904. package/src/hooks/useVoiceChat.ts +1443 -0
  905. package/src/hooks/useWhatsAppPairing.ts +123 -0
  906. package/src/i18n/index.ts +76 -0
  907. package/src/i18n/locales/en.json +1196 -0
  908. package/src/i18n/locales/es.json +1196 -0
  909. package/src/i18n/locales/ko.json +1196 -0
  910. package/src/i18n/locales/pt.json +1196 -0
  911. package/src/i18n/locales/zh-CN.json +1196 -0
  912. package/src/i18n/messages.ts +21 -0
  913. package/src/index.ts +6 -0
  914. package/src/navigation/index.ts +282 -0
  915. package/src/navigation.test.ts +189 -0
  916. package/src/platform/browser-launch.test.ts +94 -0
  917. package/src/platform/browser-launch.ts +149 -0
  918. package/src/platform/index.ts +261 -0
  919. package/src/platform/init.ts +236 -0
  920. package/src/providers/index.ts +82 -0
  921. package/src/state/AppContext.tsx +5737 -0
  922. package/src/state/index.ts +6 -0
  923. package/src/state/internal.ts +77 -0
  924. package/src/state/parsers.test.ts +124 -0
  925. package/src/state/parsers.ts +309 -0
  926. package/src/state/persistence.ts +278 -0
  927. package/src/state/types.ts +694 -0
  928. package/src/state/ui-preferences.ts +3 -0
  929. package/src/state/useApp.ts +23 -0
  930. package/src/state/vrm.ts +76 -0
  931. package/src/stories/AppMockProvider.tsx +32 -0
  932. package/src/stories/ChatEmptyState.stories.tsx +27 -0
  933. package/src/stories/ChatMessage.stories.tsx +115 -0
  934. package/src/stories/CompanionHeader.stories.tsx +74 -0
  935. package/src/stories/CompanionView.stories.tsx +33 -0
  936. package/src/stories/ConversationListItem.stories.tsx +102 -0
  937. package/src/stories/TypingIndicator.stories.tsx +28 -0
  938. package/src/styles/anime.css +6324 -0
  939. package/src/styles/base.css +196 -0
  940. package/src/styles/onboarding-game.css +716 -0
  941. package/src/styles/styles.css +2085 -0
  942. package/src/styles/xterm.css +241 -0
  943. package/src/types/index.ts +715 -0
  944. package/src/types/react-test-renderer.d.ts +45 -0
  945. package/src/utils/asset-url.ts +110 -0
  946. package/src/utils/assistant-text.ts +172 -0
  947. package/src/utils/clipboard.ts +41 -0
  948. package/src/utils/desktop-dialogs.ts +80 -0
  949. package/src/utils/index.ts +6 -0
  950. package/src/utils/number-parsing.ts +125 -0
  951. package/src/utils/openExternalUrl.ts +20 -0
  952. package/src/utils/spoken-text.ts +65 -0
  953. package/src/utils/streaming-text.ts +120 -0
  954. package/src/voice/index.ts +1 -0
  955. package/src/voice/types.ts +197 -0
  956. package/test/app/AppContext.pty-sessions.test.tsx +143 -0
  957. package/test/app/MessageContent.test.tsx +326 -0
  958. package/test/app/PermissionsOnboarding.test.tsx +216 -0
  959. package/test/app/PermissionsSection.test.tsx +186 -0
  960. package/test/app/advanced-trajectory-fine-tuning.e2e.test.ts +397 -0
  961. package/test/app/agent-activity-box.test.tsx +132 -0
  962. package/test/app/agent-transfer-lock.test.ts +274 -0
  963. package/test/app/api-client-electron-fallback.test.ts +139 -0
  964. package/test/app/api-client-timeout.test.ts +75 -0
  965. package/test/app/api-client-ws.test.ts +98 -0
  966. package/test/app/api-client.ws-max-reconnect.test.ts +139 -0
  967. package/test/app/api-client.ws-reconnect.test.ts +157 -0
  968. package/test/app/app-context-autonomy-events.test.ts +478 -0
  969. package/test/app/apps-page-view.test.ts +114 -0
  970. package/test/app/apps-view.test.ts +769 -0
  971. package/test/app/autonomous-workflows.e2e.test.ts +765 -0
  972. package/test/app/autonomy-events.test.ts +150 -0
  973. package/test/app/avatar-selector.test.tsx +52 -0
  974. package/test/app/bsc-trade-panel.test.tsx +134 -0
  975. package/test/app/bug-report-modal.test.tsx +353 -0
  976. package/test/app/character-customization.e2e.test.ts +1168 -0
  977. package/test/app/chat-advanced-features.e2e.test.ts +706 -0
  978. package/test/app/chat-composer.test.tsx +181 -0
  979. package/test/app/chat-language-header.test.ts +64 -0
  980. package/test/app/chat-message.test.tsx +222 -0
  981. package/test/app/chat-modal-view.test.tsx +191 -0
  982. package/test/app/chat-routine-filter.test.ts +96 -0
  983. package/test/app/chat-send-lock.test.ts +1302 -0
  984. package/test/app/chat-stream-api-client.test.tsx +390 -0
  985. package/test/app/chat-view-game-modal.test.tsx +661 -0
  986. package/test/app/chat-view.test.tsx +877 -0
  987. package/test/app/cloud-api.e2e.test.ts +258 -0
  988. package/test/app/cloud-login-flow.e2e.test.ts +494 -0
  989. package/test/app/cloud-login-lock.test.ts +411 -0
  990. package/test/app/command-palette.test.tsx +184 -0
  991. package/test/app/command-registry.test.ts +75 -0
  992. package/test/app/companion-greeting-wave.test.tsx +443 -0
  993. package/test/app/companion-stale-conversation.test.tsx +424 -0
  994. package/test/app/companion-view.test.tsx +686 -0
  995. package/test/app/confirm-delete-control.test.ts +79 -0
  996. package/test/app/confirm-modal.test.tsx +219 -0
  997. package/test/app/connectors-ui.e2e.test.ts +508 -0
  998. package/test/app/conversations-sidebar-game-modal.test.tsx +260 -0
  999. package/test/app/conversations-sidebar.test.tsx +160 -0
  1000. package/test/app/custom-actions-smoke.test.ts +387 -0
  1001. package/test/app/custom-avatar-api-client.test.ts +207 -0
  1002. package/test/app/desktop-utils.test.ts +145 -0
  1003. package/test/app/electrobun-rpc-bridge.test.ts +83 -0
  1004. package/test/app/events.test.ts +88 -0
  1005. package/test/app/export-import-flows.e2e.test.ts +700 -0
  1006. package/test/app/fine-tuning-view.test.ts +471 -0
  1007. package/test/app/game-view-auth-session.test.tsx +186 -0
  1008. package/test/app/game-view.test.ts +444 -0
  1009. package/test/app/global-emote-overlay.test.tsx +106 -0
  1010. package/test/app/header-status.test.tsx +149 -0
  1011. package/test/app/i18n.test.ts +152 -0
  1012. package/test/app/inventory-bsc-view.test.ts +908 -0
  1013. package/test/app/knowledge-ui.e2e.test.ts +762 -0
  1014. package/test/app/knowledge-upload-helpers.test.ts +124 -0
  1015. package/test/app/lifecycle-lock.test.ts +267 -0
  1016. package/test/app/lifo-popout-utils.test.ts +208 -0
  1017. package/test/app/lifo-safe-endpoint.test.ts +34 -0
  1018. package/test/app/loading-screen.test.tsx +45 -0
  1019. package/test/app/memory-monitor.test.ts +332 -0
  1020. package/test/app/navigation.test.tsx +29 -0
  1021. package/test/app/onboarding-finish-lock.test.ts +500 -0
  1022. package/test/app/onboarding-language.test.tsx +161 -0
  1023. package/test/app/onboarding-steps.test.tsx +371 -0
  1024. package/test/app/open-external-url.test.ts +65 -0
  1025. package/test/app/pages-navigation-smoke.e2e.test.ts +633 -0
  1026. package/test/app/pairing-lock.test.ts +260 -0
  1027. package/test/app/pairing-view.test.tsx +74 -0
  1028. package/test/app/permissions-section.test.ts +432 -0
  1029. package/test/app/plugin-bridge.test.ts +109 -0
  1030. package/test/app/plugins-ui.e2e.test.ts +605 -0
  1031. package/test/app/plugins-view-game-modal.test.tsx +650 -0
  1032. package/test/app/plugins-view-toggle-restart.test.ts +129 -0
  1033. package/test/app/provider-dropdown-default.test.tsx +164 -0
  1034. package/test/app/restart-banner.test.tsx +197 -0
  1035. package/test/app/retake-capture.test.ts +84 -0
  1036. package/test/app/sandbox-api-client.test.ts +108 -0
  1037. package/test/app/save-command-modal.test.tsx +109 -0
  1038. package/test/app/secrets-view.test.tsx +92 -0
  1039. package/test/app/settings-control-styles.test.tsx +142 -0
  1040. package/test/app/settings-reset.e2e.test.ts +728 -0
  1041. package/test/app/settings-sections.e2e.test.ts +614 -0
  1042. package/test/app/shared-format.test.ts +44 -0
  1043. package/test/app/shared-switch.test.ts +69 -0
  1044. package/test/app/shell-mode-switching.e2e.test.ts +829 -0
  1045. package/test/app/shell-mode-tab-memory.test.tsx +283 -0
  1046. package/test/app/shell-overlays.test.tsx +50 -0
  1047. package/test/app/shortcuts-overlay.test.tsx +111 -0
  1048. package/test/app/sse-interruption.test.ts +122 -0
  1049. package/test/app/startup-asset-missing.e2e.test.ts +126 -0
  1050. package/test/app/startup-backend-missing.e2e.test.ts +118 -0
  1051. package/test/app/startup-chat.e2e.test.ts +305 -0
  1052. package/test/app/startup-conversation-restore.test.tsx +344 -0
  1053. package/test/app/startup-failure-view.test.tsx +103 -0
  1054. package/test/app/startup-onboarding.e2e.test.ts +612 -0
  1055. package/test/app/startup-timeout.test.tsx +80 -0
  1056. package/test/app/startup-token-401.e2e.test.ts +103 -0
  1057. package/test/app/stream-helpers.test.ts +46 -0
  1058. package/test/app/stream-popout-navigation.test.tsx +41 -0
  1059. package/test/app/stream-status-bar.test.tsx +89 -0
  1060. package/test/app/theme-toggle.test.tsx +33 -0
  1061. package/test/app/training-api-client.test.ts +128 -0
  1062. package/test/app/trajectories-view.test.tsx +220 -0
  1063. package/test/app/triggers-api-client.test.ts +77 -0
  1064. package/test/app/triggers-navigation.test.ts +118 -0
  1065. package/test/app/triggers-view.e2e.test.ts +674 -0
  1066. package/test/app/update-channel-lock.test.ts +259 -0
  1067. package/test/app/vector-browser.async-cleanup.test.tsx +367 -0
  1068. package/test/app/vector-browser.e2e.test.ts +653 -0
  1069. package/test/app/vrm-stage.test.tsx +351 -0
  1070. package/test/app/vrm-viewer.test.tsx +298 -0
  1071. package/test/app/wallet-api-save-lock.test.ts +277 -0
  1072. package/test/app/wallet-hooks.test.ts +405 -0
  1073. package/test/app/wallet-ui-flows.e2e.test.ts +556 -0
  1074. package/test/avatar/asset-url.test.ts +90 -0
  1075. package/test/avatar/avatar-selector.test.ts +173 -0
  1076. package/test/avatar/mixamo-vrm-rig-map.test.ts +111 -0
  1077. package/test/avatar/voice-chat-streaming-text.test.ts +96 -0
  1078. package/test/avatar/voice-chat.test.ts +391 -0
  1079. package/test/ui/command-palette-commands.test.ts +57 -0
  1080. package/test/ui/ui-renderer.test.ts +39 -0
  1081. package/tsconfig.build.json +19 -0
  1082. package/tsconfig.json +21 -0
@@ -0,0 +1,1751 @@
1
+ /**
2
+ * ui-renderer.tsx — General-purpose json-render declarative UI renderer.
3
+ *
4
+ * Renders a UiSpec tree into React components. Supports:
5
+ * - 35+ component types (layout, typography, form, data, feedback, nav, viz, interaction)
6
+ * - State binding via statePath
7
+ * - Dynamic values via $path references
8
+ * - Conditional props via $cond expressions
9
+ * - List rendering via repeat config
10
+ * - Event bindings via on.press / on.change
11
+ */
12
+
13
+ import React, {
14
+ createContext,
15
+ useCallback,
16
+ useContext,
17
+ useMemo,
18
+ useState,
19
+ } from "react";
20
+ import { useApp } from "../state";
21
+ import { confirmDesktopAction, resolveAppAssetUrl } from "../utils";
22
+ import { getByPath, setByPath } from "./config-catalog";
23
+ import type {
24
+ AuthState,
25
+ CondExpr,
26
+ UiAction,
27
+ UiElement,
28
+ UiRenderContext,
29
+ UiSpec,
30
+ ValidationCheck,
31
+ VisibilityCondition,
32
+ } from "./ui-spec";
33
+
34
+ const UiContext = createContext<UiRenderContext | null>(null);
35
+
36
+ const BLOCKED_LINK_PROTOCOLS = new Set([
37
+ "javascript",
38
+ "data",
39
+ "vbscript",
40
+ "file",
41
+ ]);
42
+
43
+ function useUiCtx(): UiRenderContext {
44
+ const ctx = useContext(UiContext);
45
+ if (!ctx) throw new Error("UiRenderer context missing");
46
+ return ctx;
47
+ }
48
+
49
+ // ── Dynamic value resolution ────────────────────────────────────────
50
+
51
+ function resolveProp(value: unknown, ctx: UiRenderContext): unknown {
52
+ if (value == null) return value;
53
+
54
+ // $data.path string prefix (simpler syntax for AI)
55
+ if (typeof value === "string" && value.startsWith("$data.")) {
56
+ const path = value.slice(6); // strip "$data."
57
+ if (path.startsWith("$item/") && ctx.repeatItem) {
58
+ return ctx.repeatItem[path.slice(6)];
59
+ }
60
+ return getByPath(ctx.state, path);
61
+ }
62
+
63
+ // $path reference
64
+ if (
65
+ typeof value === "object" &&
66
+ "$path" in (value as Record<string, unknown>)
67
+ ) {
68
+ const path = (value as { $path: string }).$path;
69
+ if (path.startsWith("$item/") && ctx.repeatItem) {
70
+ return ctx.repeatItem[path.slice(6)];
71
+ }
72
+ return getByPath(ctx.state, path);
73
+ }
74
+
75
+ // $cond expression
76
+ if (
77
+ typeof value === "object" &&
78
+ "$cond" in (value as Record<string, unknown>)
79
+ ) {
80
+ const expr = value as CondExpr;
81
+ const cond = expr.$cond;
82
+ let result = false;
83
+
84
+ if (cond.eq) {
85
+ const [a, b] = cond.eq.map((v) => resolveProp(v, ctx));
86
+ result = a === b;
87
+ } else if (cond.neq) {
88
+ const [a, b] = cond.neq.map((v) => resolveProp(v, ctx));
89
+ result = a !== b;
90
+ } else if (cond.gt) {
91
+ const [a, b] = cond.gt.map((v) => resolveProp(v, ctx));
92
+ result = Number(a) > Number(b);
93
+ } else if (cond.lt) {
94
+ const [a, b] = cond.lt.map((v) => resolveProp(v, ctx));
95
+ result = Number(a) < Number(b);
96
+ } else if (cond.truthy) {
97
+ result = !!resolveProp(cond.truthy, ctx);
98
+ } else if (cond.falsy) {
99
+ result = !resolveProp(cond.falsy, ctx);
100
+ } else if (cond.path) {
101
+ result = !!getByPath(ctx.state, cond.path);
102
+ }
103
+
104
+ return result ? resolveProp(expr.$then, ctx) : resolveProp(expr.$else, ctx);
105
+ }
106
+
107
+ // Object with path references
108
+ if (
109
+ typeof value === "object" &&
110
+ value !== null &&
111
+ "path" in (value as Record<string, unknown>)
112
+ ) {
113
+ const p = (value as { path: string }).path;
114
+ if (p.startsWith("$item/") && ctx.repeatItem) {
115
+ return ctx.repeatItem[p.slice(6)];
116
+ }
117
+ return getByPath(ctx.state, p);
118
+ }
119
+
120
+ return value;
121
+ }
122
+
123
+ function resolveProps(
124
+ props: Record<string, unknown>,
125
+ ctx: UiRenderContext,
126
+ ): Record<string, unknown> {
127
+ const resolved: Record<string, unknown> = {};
128
+ for (const [k, v] of Object.entries(props)) {
129
+ resolved[k] = resolveProp(v, ctx);
130
+ }
131
+ return resolved;
132
+ }
133
+
134
+ // ── Visibility evaluation ────────────────────────────────────────────
135
+
136
+ export function evaluateUiVisibility(
137
+ condition: VisibilityCondition | undefined,
138
+ state: Record<string, unknown>,
139
+ auth?: AuthState,
140
+ ): boolean {
141
+ if (!condition) return true;
142
+
143
+ // Path-based
144
+ if ("path" in condition && "operator" in condition) {
145
+ const val = getByPath(state, condition.path);
146
+ const target = condition.value;
147
+ switch (condition.operator) {
148
+ case "eq":
149
+ return val === target;
150
+ case "ne":
151
+ return val !== target;
152
+ case "gt":
153
+ return Number(val) > Number(target);
154
+ case "gte":
155
+ return Number(val) >= Number(target);
156
+ case "lt":
157
+ return Number(val) < Number(target);
158
+ case "lte":
159
+ return Number(val) <= Number(target);
160
+ default:
161
+ return true;
162
+ }
163
+ }
164
+
165
+ // Auth-based
166
+ if ("auth" in condition) {
167
+ if (!auth) return false;
168
+ switch (condition.auth) {
169
+ case "signedIn":
170
+ return auth.isSignedIn;
171
+ case "signedOut":
172
+ return !auth.isSignedIn;
173
+ case "admin":
174
+ return auth.roles?.includes("admin") ?? false;
175
+ default:
176
+ return auth.roles?.includes(condition.auth) ?? false;
177
+ }
178
+ }
179
+
180
+ // Logic combinators
181
+ if ("and" in condition)
182
+ return condition.and.every((c) => evaluateUiVisibility(c, state, auth));
183
+ if ("or" in condition)
184
+ return condition.or.some((c) => evaluateUiVisibility(c, state, auth));
185
+ if ("not" in condition)
186
+ return !evaluateUiVisibility(condition.not, state, auth);
187
+
188
+ return true;
189
+ }
190
+
191
+ export function sanitizeLinkHref(href: unknown): string {
192
+ // Strip ASCII control chars (tab, LF, CR) that browsers silently remove
193
+ // during URL parsing, preventing bypass attacks like "java\nscript:alert(1)".
194
+ const raw = String(href ?? "#")
195
+ .trim()
196
+ .replace(/[\t\n\r]/g, "");
197
+ if (!raw) return "#";
198
+
199
+ // Keep relative/hash links unchanged.
200
+ if (
201
+ raw.startsWith("#") ||
202
+ raw.startsWith("/") ||
203
+ raw.startsWith("./") ||
204
+ raw.startsWith("../") ||
205
+ raw.startsWith("?")
206
+ ) {
207
+ return raw;
208
+ }
209
+
210
+ const match = /^([a-zA-Z][a-zA-Z\d+.-]*):/.exec(raw);
211
+ if (!match) return raw;
212
+
213
+ const protocol = match[1].toLowerCase();
214
+ if (BLOCKED_LINK_PROTOCOLS.has(protocol)) return "#";
215
+
216
+ return raw;
217
+ }
218
+
219
+ // ── Built-in validators ─────────────────────────────────────────────
220
+
221
+ const BUILTIN_VALIDATORS: Record<
222
+ string,
223
+ (value: unknown, args?: Record<string, unknown>) => boolean
224
+ > = {
225
+ required: (v) => v != null && v !== "",
226
+ email: (v) => typeof v === "string" && /^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(v),
227
+ minLength: (v, args) =>
228
+ typeof v === "string" && v.length >= Number(args?.length ?? 0),
229
+ maxLength: (v, args) =>
230
+ typeof v === "string" && v.length <= Number(args?.length ?? Infinity),
231
+ pattern: (v, args) => {
232
+ if (typeof v !== "string" || !args?.pattern) return true;
233
+ try {
234
+ return new RegExp(String(args.pattern)).test(v);
235
+ } catch {
236
+ return true;
237
+ }
238
+ },
239
+ min: (v, args) => Number(v) >= Number(args?.value ?? -Infinity),
240
+ max: (v, args) => Number(v) <= Number(args?.value ?? Infinity),
241
+ };
242
+
243
+ // ── Validation runner ───────────────────────────────────────────────
244
+
245
+ export function runValidation(
246
+ checks: ValidationCheck[],
247
+ value: unknown,
248
+ customValidators?: Record<
249
+ string,
250
+ (
251
+ value: unknown,
252
+ args?: Record<string, unknown>,
253
+ ) => boolean | Promise<boolean>
254
+ >,
255
+ ): string[] {
256
+ const errors: string[] = [];
257
+ for (const check of checks) {
258
+ const fn = BUILTIN_VALIDATORS[check.fn] ?? customValidators?.[check.fn];
259
+ if (fn) {
260
+ const result = fn(value, check.args);
261
+ // Handle sync validators only (async handled separately)
262
+ if (result === false) errors.push(check.message);
263
+ }
264
+ }
265
+ return errors;
266
+ }
267
+
268
+ // ── State helpers ───────────────────────────────────────────────────
269
+
270
+ function useStatePath(statePath: string | undefined, ctx: UiRenderContext) {
271
+ const value = statePath ? getByPath(ctx.state, statePath) : undefined;
272
+ const setValue = useCallback(
273
+ (v: unknown) => {
274
+ if (statePath) ctx.setState(statePath, v);
275
+ },
276
+ [statePath, ctx],
277
+ );
278
+ return [value, setValue] as const;
279
+ }
280
+
281
+ // ── Fire event action ───────────────────────────────────────────────
282
+
283
+ function fireEvent(action: UiAction | undefined, ctx: UiRenderContext) {
284
+ if (!action) return;
285
+
286
+ const execute = () => {
287
+ if (action.action === "setState" && action.params) {
288
+ const p = action.params as { path: string; value: unknown };
289
+ ctx.setState(p.path, p.value);
290
+ if (action.onSuccess && ctx.onAction) {
291
+ ctx.onAction(action.onSuccess.action, action.onSuccess.params);
292
+ }
293
+ } else if (ctx.onAction) {
294
+ try {
295
+ ctx.onAction(action.action, action.params);
296
+ if (action.onSuccess)
297
+ ctx.onAction(action.onSuccess.action, action.onSuccess.params);
298
+ } catch {
299
+ if (action.onError && ctx.onAction)
300
+ ctx.onAction(action.onError.action, action.onError.params);
301
+ }
302
+ }
303
+ };
304
+
305
+ void (async () => {
306
+ if (action.confirm) {
307
+ const ok = await confirmDesktopAction({
308
+ title: action.confirm.title,
309
+ message: action.confirm.message ?? "",
310
+ confirmLabel: "Confirm",
311
+ cancelLabel: "Cancel",
312
+ type: "question",
313
+ });
314
+ if (!ok) return;
315
+ }
316
+ execute();
317
+ })();
318
+ }
319
+
320
+ // ── Gap / size maps ─────────────────────────────────────────────────
321
+
322
+ const GAP: Record<string, string> = {
323
+ none: "gap-0",
324
+ xs: "gap-0.5",
325
+ sm: "gap-1.5",
326
+ md: "gap-3",
327
+ lg: "gap-5",
328
+ xl: "gap-8",
329
+ };
330
+
331
+ const ALIGN: Record<string, string> = {
332
+ start: "items-start",
333
+ center: "items-center",
334
+ end: "items-end",
335
+ stretch: "items-stretch",
336
+ };
337
+
338
+ const JUSTIFY: Record<string, string> = {
339
+ start: "justify-start",
340
+ center: "justify-center",
341
+ end: "justify-end",
342
+ between: "justify-between",
343
+ around: "justify-around",
344
+ };
345
+
346
+ // ── Tailwind class constants ────────────────────────────────────────
347
+
348
+ const INPUT_CLS =
349
+ "w-full px-2 py-[5px] border border-[var(--border)] bg-[var(--card)] text-xs font-[var(--mono)] transition-colors focus:border-[var(--accent)] focus:outline-none box-border h-[30px]";
350
+
351
+ // ══════════════════════════════════════════════════════════════════════
352
+ // COMPONENT REGISTRY
353
+ // ══════════════════════════════════════════════════════════════════════
354
+
355
+ type ComponentFn = (
356
+ props: Record<string, unknown>,
357
+ children: React.ReactNode,
358
+ ctx: UiRenderContext,
359
+ el: UiElement,
360
+ ) => React.ReactNode;
361
+
362
+ // ── Layout ──────────────────────────────────────────────────────────
363
+
364
+ const StackComponent: ComponentFn = (props, children) => {
365
+ const dir = props.direction === "horizontal" ? "flex-row" : "flex-col";
366
+ const gap = GAP[String(props.gap ?? "md")] ?? "gap-3";
367
+ const align = ALIGN[String(props.align ?? "stretch")] ?? "";
368
+ const justify = JUSTIFY[String(props.justify ?? "start")] ?? "";
369
+ return (
370
+ <div className={`flex ${dir} ${gap} ${align} ${justify}`}>{children}</div>
371
+ );
372
+ };
373
+
374
+ const GridComponent: ComponentFn = (props, children) => {
375
+ const cols = Number(props.columns ?? 2);
376
+ const gap = GAP[String(props.gap ?? "md")] ?? "gap-3";
377
+ return (
378
+ <div
379
+ className={`grid ${gap}`}
380
+ style={{ gridTemplateColumns: `repeat(${cols}, 1fr)` }}
381
+ >
382
+ {children}
383
+ </div>
384
+ );
385
+ };
386
+
387
+ const CardComponent: ComponentFn = (props, children) => {
388
+ const maxW = props.maxWidth === "full" ? "max-w-full" : "";
389
+ return (
390
+ <div
391
+ className={`border border-[var(--border)] bg-[var(--card)] p-4 ${maxW}`}
392
+ >
393
+ {props.title ? (
394
+ <div className="font-bold text-sm mb-0.5">{String(props.title)}</div>
395
+ ) : null}
396
+ {props.description ? (
397
+ <div className="text-xs text-[var(--muted)] mb-3">
398
+ {String(props.description)}
399
+ </div>
400
+ ) : null}
401
+ {children}
402
+ </div>
403
+ );
404
+ };
405
+
406
+ const SeparatorComponent: ComponentFn = (props) => {
407
+ const isVert = props.orientation === "vertical";
408
+ return isVert ? (
409
+ <div className="w-px bg-[var(--border)] self-stretch" />
410
+ ) : (
411
+ <hr className="border-t border-[var(--border)] my-2" />
412
+ );
413
+ };
414
+
415
+ // ── Typography ──────────────────────────────────────────────────────
416
+
417
+ const HeadingComponent: ComponentFn = (props) => {
418
+ const text = String(props.text ?? "");
419
+ const level = String(props.level ?? "h2");
420
+ const cls =
421
+ level === "h1"
422
+ ? "text-xl font-bold"
423
+ : level === "h3"
424
+ ? "text-sm font-bold"
425
+ : "text-base font-bold";
426
+ return <div className={cls}>{text}</div>;
427
+ };
428
+
429
+ const TextComponent: ComponentFn = (props) => {
430
+ const text = String(props.text ?? "");
431
+ const variant = String(props.variant ?? "body");
432
+ const cls: Record<string, string> = {
433
+ body: "text-sm",
434
+ caption: "text-xs text-[var(--muted)]",
435
+ muted: "text-sm text-[var(--muted)]",
436
+ lead: "text-sm font-medium",
437
+ code: "text-xs font-mono bg-[var(--bg-hover)] px-1.5 py-0.5 border border-[var(--border)]",
438
+ };
439
+ return <div className={cls[variant] ?? "text-sm"}>{text}</div>;
440
+ };
441
+
442
+ // ── Form ────────────────────────────────────────────────────────────
443
+
444
+ const InputComponent: ComponentFn = (props, _children, ctx, el) => {
445
+ const [value, setValue] = useStatePath(
446
+ props.statePath as string | undefined,
447
+ ctx,
448
+ );
449
+ const sp = props.statePath as string | undefined;
450
+ const errors = sp ? ctx.fieldErrors?.[sp] : undefined;
451
+ const validateOn = el.validation?.validateOn ?? "blur";
452
+
453
+ const handleChange = (v: string) => {
454
+ setValue(v);
455
+ if (validateOn === "change" && sp && ctx.validateField)
456
+ ctx.validateField(sp);
457
+ };
458
+ const handleBlur = () => {
459
+ if (validateOn === "blur" && sp && ctx.validateField) ctx.validateField(sp);
460
+ };
461
+
462
+ return (
463
+ <div className="flex flex-col gap-1">
464
+ {props.label ? (
465
+ <span className="text-xs font-semibold">{String(props.label)}</span>
466
+ ) : null}
467
+ <input
468
+ className={`${INPUT_CLS}${errors?.length ? " border-[var(--destructive)]" : ""}`}
469
+ type={String(props.type ?? "text")}
470
+ name={String(props.name ?? "")}
471
+ placeholder={String(props.placeholder ?? "")}
472
+ value={String(value ?? "")}
473
+ onChange={(e) => handleChange(e.target.value)}
474
+ onBlur={handleBlur}
475
+ />
476
+ {errors?.length ? (
477
+ <div className="flex flex-col gap-0.5">
478
+ {errors.map((err) => (
479
+ <span key={err} className="text-[10px] text-[var(--destructive)]">
480
+ {err}
481
+ </span>
482
+ ))}
483
+ </div>
484
+ ) : null}
485
+ </div>
486
+ );
487
+ };
488
+
489
+ const TextareaComponent: ComponentFn = (props, _children, ctx, el) => {
490
+ const [value, setValue] = useStatePath(
491
+ props.statePath as string | undefined,
492
+ ctx,
493
+ );
494
+ const sp = props.statePath as string | undefined;
495
+ const errors = sp ? ctx.fieldErrors?.[sp] : undefined;
496
+ const validateOn = el.validation?.validateOn ?? "blur";
497
+
498
+ const handleChange = (v: string) => {
499
+ setValue(v);
500
+ if (validateOn === "change" && sp && ctx.validateField)
501
+ ctx.validateField(sp);
502
+ };
503
+ const handleBlur = () => {
504
+ if (validateOn === "blur" && sp && ctx.validateField) ctx.validateField(sp);
505
+ };
506
+
507
+ return (
508
+ <div className="flex flex-col gap-1">
509
+ {props.label ? (
510
+ <span className="text-xs font-semibold">{String(props.label)}</span>
511
+ ) : null}
512
+ <textarea
513
+ className={`w-full px-2 py-[5px] border border-[var(--border)] bg-[var(--card)] text-xs font-[var(--mono)] transition-colors focus:border-[var(--accent)] focus:outline-none box-border min-h-[64px] resize-y${errors?.length ? " border-[var(--destructive)]" : ""}`}
514
+ name={String(props.name ?? "")}
515
+ placeholder={String(props.placeholder ?? "")}
516
+ rows={Number(props.rows ?? 3)}
517
+ value={String(value ?? "")}
518
+ onChange={(e) => handleChange(e.target.value)}
519
+ onBlur={handleBlur}
520
+ />
521
+ {errors?.length ? (
522
+ <div className="flex flex-col gap-0.5">
523
+ {errors.map((err) => (
524
+ <span key={err} className="text-[10px] text-[var(--destructive)]">
525
+ {err}
526
+ </span>
527
+ ))}
528
+ </div>
529
+ ) : null}
530
+ </div>
531
+ );
532
+ };
533
+
534
+ const SelectComponent: ComponentFn = (props, _children, ctx, el) => {
535
+ const [value, setValue] = useStatePath(
536
+ props.statePath as string | undefined,
537
+ ctx,
538
+ );
539
+ const options =
540
+ (props.options as Array<{ label: string; value: string }>) ?? [];
541
+ const sp = props.statePath as string | undefined;
542
+ const errors = sp ? ctx.fieldErrors?.[sp] : undefined;
543
+ const validateOn = el.validation?.validateOn ?? "blur";
544
+
545
+ const handleChange = (v: string) => {
546
+ setValue(v);
547
+ if (validateOn === "change" && sp && ctx.validateField)
548
+ ctx.validateField(sp);
549
+ };
550
+ const handleBlur = () => {
551
+ if (validateOn === "blur" && sp && ctx.validateField) ctx.validateField(sp);
552
+ };
553
+
554
+ return (
555
+ <div className="flex flex-col gap-1">
556
+ {props.label ? (
557
+ <span className="text-xs font-semibold">{String(props.label)}</span>
558
+ ) : null}
559
+ <select
560
+ className={`${INPUT_CLS} appearance-auto${errors?.length ? " border-[var(--destructive)]" : ""}`}
561
+ value={String(value ?? "")}
562
+ onChange={(e) => handleChange(e.target.value)}
563
+ onBlur={handleBlur}
564
+ >
565
+ {props.placeholder ? (
566
+ <option value="">{String(props.placeholder)}</option>
567
+ ) : null}
568
+ {options.map((o) => (
569
+ <option key={o.value} value={o.value}>
570
+ {o.label}
571
+ </option>
572
+ ))}
573
+ </select>
574
+ {errors?.length ? (
575
+ <div className="flex flex-col gap-0.5">
576
+ {errors.map((err) => (
577
+ <span key={err} className="text-[10px] text-[var(--destructive)]">
578
+ {err}
579
+ </span>
580
+ ))}
581
+ </div>
582
+ ) : null}
583
+ </div>
584
+ );
585
+ };
586
+
587
+ const CheckboxComponent: ComponentFn = (props, _children, ctx) => {
588
+ const [value, setValue] = useStatePath(
589
+ props.statePath as string | undefined,
590
+ ctx,
591
+ );
592
+ return (
593
+ <span className="flex items-center gap-2 text-xs cursor-pointer">
594
+ <input
595
+ type="checkbox"
596
+ checked={!!value}
597
+ onChange={(e) => setValue(e.target.checked)}
598
+ />
599
+ <span className="font-semibold">{String(props.label ?? "")}</span>
600
+ </span>
601
+ );
602
+ };
603
+
604
+ const RadioComponent: ComponentFn = (props, _children, ctx) => {
605
+ const [value, setValue] = useStatePath(
606
+ props.statePath as string | undefined,
607
+ ctx,
608
+ );
609
+ const options =
610
+ (props.options as Array<{ label: string; value: string }>) ?? [];
611
+ return (
612
+ <div className="flex flex-col gap-1">
613
+ {props.label ? (
614
+ <span className="text-xs font-semibold mb-0.5">
615
+ {String(props.label)}
616
+ </span>
617
+ ) : null}
618
+ {options.map((o) => (
619
+ <span
620
+ key={o.value}
621
+ className="flex items-center gap-2 text-xs cursor-pointer"
622
+ >
623
+ <input
624
+ type="radio"
625
+ name={String(props.name ?? "")}
626
+ value={o.value}
627
+ checked={value === o.value}
628
+ onChange={() => setValue(o.value)}
629
+ />
630
+ <span>{o.label}</span>
631
+ </span>
632
+ ))}
633
+ </div>
634
+ );
635
+ };
636
+
637
+ const SwitchComponent: ComponentFn = (props, _children, ctx) => {
638
+ const [value, setValue] = useStatePath(
639
+ props.statePath as string | undefined,
640
+ ctx,
641
+ );
642
+ const checked = !!value;
643
+ return (
644
+ <span className="flex items-center gap-2 cursor-pointer">
645
+ <button
646
+ type="button"
647
+ className={`relative w-9 h-[18px] transition-colors ${checked ? "bg-[var(--accent)]" : "bg-[var(--muted)]"}`}
648
+ onClick={() => setValue(!checked)}
649
+ >
650
+ <div
651
+ className={`absolute top-0.5 w-[14px] h-[14px] bg-white transition-all ${checked ? "left-5" : "left-0.5"}`}
652
+ />
653
+ </button>
654
+ <span className="text-xs font-semibold">{String(props.label ?? "")}</span>
655
+ </span>
656
+ );
657
+ };
658
+
659
+ const SliderComponent: ComponentFn = (props, _children, ctx) => {
660
+ const [value, setValue] = useStatePath(
661
+ props.statePath as string | undefined,
662
+ ctx,
663
+ );
664
+ return (
665
+ <div className="flex flex-col gap-1">
666
+ {props.label ? (
667
+ <div className="flex justify-between text-xs">
668
+ <span className="font-semibold">{String(props.label)}</span>
669
+ <span className="text-[var(--muted)]">
670
+ {String(value ?? props.min ?? 0)}
671
+ </span>
672
+ </div>
673
+ ) : null}
674
+ <input
675
+ type="range"
676
+ min={Number(props.min ?? 0)}
677
+ max={Number(props.max ?? 100)}
678
+ step={Number(props.step ?? 1)}
679
+ value={Number(value ?? props.min ?? 0)}
680
+ onChange={(e) => setValue(Number(e.target.value))}
681
+ className="w-full"
682
+ style={{ accentColor: "var(--accent)" }}
683
+ />
684
+ </div>
685
+ );
686
+ };
687
+
688
+ const ToggleComponent: ComponentFn = (props, _children, ctx, el) => {
689
+ const [value, setValue] = useStatePath(
690
+ props.statePath as string | undefined,
691
+ ctx,
692
+ );
693
+ const pressed = !!value;
694
+ return (
695
+ <button
696
+ type="button"
697
+ className={`px-3 py-1.5 text-xs border cursor-pointer transition-colors ${
698
+ pressed
699
+ ? "bg-[var(--accent)] text-[var(--accent-foreground,#1a1f26)] border-[var(--accent)]"
700
+ : "bg-[var(--card)] text-[var(--text)] border-[var(--border)] hover:bg-[var(--bg-hover)]"
701
+ }`}
702
+ onClick={() => {
703
+ setValue(!pressed);
704
+ fireEvent(el.on?.press, ctx);
705
+ }}
706
+ >
707
+ {String(props.label ?? "Toggle")}
708
+ </button>
709
+ );
710
+ };
711
+
712
+ const ToggleGroupComponent: ComponentFn = (props, _children, ctx) => {
713
+ const [value, setValue] = useStatePath(
714
+ props.statePath as string | undefined,
715
+ ctx,
716
+ );
717
+ const items = (props.items as Array<{ label: string; value: string }>) ?? [];
718
+ const isMultiple = props.type === "multiple";
719
+ const selected = new Set(Array.isArray(value) ? (value as string[]) : []);
720
+
721
+ const toggle = (v: string) => {
722
+ if (isMultiple) {
723
+ const next = new Set(selected);
724
+ if (next.has(v)) next.delete(v);
725
+ else next.add(v);
726
+ setValue([...next]);
727
+ } else {
728
+ setValue(v);
729
+ }
730
+ };
731
+
732
+ return (
733
+ <div className="flex gap-1">
734
+ {items.map((item) => {
735
+ const active = isMultiple
736
+ ? selected.has(item.value)
737
+ : value === item.value;
738
+ return (
739
+ <button
740
+ key={item.value}
741
+ type="button"
742
+ className={`px-2.5 py-1 text-xs border cursor-pointer transition-colors ${
743
+ active
744
+ ? "bg-[var(--accent)] text-[var(--accent-foreground,#1a1f26)] border-[var(--accent)]"
745
+ : "bg-[var(--card)] text-[var(--text)] border-[var(--border)] hover:bg-[var(--bg-hover)]"
746
+ }`}
747
+ onClick={() => toggle(item.value)}
748
+ >
749
+ {item.label}
750
+ </button>
751
+ );
752
+ })}
753
+ </div>
754
+ );
755
+ };
756
+
757
+ const ButtonGroupComponent: ComponentFn = (props, _children, ctx) => {
758
+ const [value, setValue] = useStatePath(
759
+ props.statePath as string | undefined,
760
+ ctx,
761
+ );
762
+ const buttons =
763
+ (props.buttons as Array<{ label: string; value: string }>) ?? [];
764
+ return (
765
+ <div className="flex gap-1">
766
+ {buttons.map((btn) => {
767
+ const active = value === btn.value;
768
+ return (
769
+ <button
770
+ key={btn.value}
771
+ type="button"
772
+ className={`px-3 py-1.5 text-xs border cursor-pointer transition-colors ${
773
+ active
774
+ ? "bg-[var(--accent)] text-[var(--accent-foreground,#1a1f26)] border-[var(--accent)]"
775
+ : "bg-[var(--card)] text-[var(--text)] border-[var(--border)] hover:bg-[var(--bg-hover)]"
776
+ }`}
777
+ onClick={() => setValue(btn.value)}
778
+ >
779
+ {btn.label}
780
+ </button>
781
+ );
782
+ })}
783
+ </div>
784
+ );
785
+ };
786
+
787
+ // ── Data Display ────────────────────────────────────────────────────
788
+
789
+ const TableComponent: ComponentFn = (props) => {
790
+ const columns = (props.columns as string[]) ?? [];
791
+ const rows = (props.rows as string[][]) ?? [];
792
+ return (
793
+ <div className="overflow-x-auto">
794
+ {props.caption ? (
795
+ <div className="text-xs font-semibold mb-1.5">
796
+ {String(props.caption)}
797
+ </div>
798
+ ) : null}
799
+ <table className="w-full text-xs border-collapse">
800
+ <thead>
801
+ <tr>
802
+ {columns.map((col) => (
803
+ <th
804
+ key={col}
805
+ className="text-left px-2.5 py-1.5 border-b border-[var(--border)] font-semibold text-[var(--muted)]"
806
+ >
807
+ {col}
808
+ </th>
809
+ ))}
810
+ </tr>
811
+ </thead>
812
+ <tbody>
813
+ {rows.map((row) => (
814
+ <tr
815
+ key={row.join("|")}
816
+ className="border-b border-[var(--border)] last:border-b-0"
817
+ >
818
+ {row.map((cell) => (
819
+ <td key={cell} className="px-2.5 py-1.5">
820
+ {cell}
821
+ </td>
822
+ ))}
823
+ </tr>
824
+ ))}
825
+ </tbody>
826
+ </table>
827
+ </div>
828
+ );
829
+ };
830
+
831
+ const CarouselComponent: ComponentFn = (props) => {
832
+ const { t } = useApp();
833
+ const items =
834
+ (props.items as Array<{ title: string; description: string }>) ?? [];
835
+ const [current, setCurrent] = useState(0);
836
+ return (
837
+ <div className="relative">
838
+ <div className="border border-[var(--border)] bg-[var(--bg-hover)] p-4 min-h-[60px]">
839
+ {items[current] && (
840
+ <div>
841
+ <div className="text-xs font-bold">{items[current].title}</div>
842
+ <div className="text-xs text-[var(--muted)] mt-0.5">
843
+ {items[current].description}
844
+ </div>
845
+ </div>
846
+ )}
847
+ </div>
848
+ <div className="flex justify-center gap-2 mt-2">
849
+ <button
850
+ type="button"
851
+ className="text-xs px-2 py-0.5 border border-[var(--border)] bg-[var(--card)] cursor-pointer hover:bg-[var(--bg-hover)]"
852
+ onClick={() => setCurrent((p) => Math.max(0, p - 1))}
853
+ disabled={current === 0}
854
+ >
855
+ {t("ui-renderer.Larr")}
856
+ </button>
857
+ <span className="text-[10px] text-[var(--muted)] self-center">
858
+ {current + 1} / {items.length}
859
+ </span>
860
+ <button
861
+ type="button"
862
+ className="text-xs px-2 py-0.5 border border-[var(--border)] bg-[var(--card)] cursor-pointer hover:bg-[var(--bg-hover)]"
863
+ onClick={() => setCurrent((p) => Math.min(items.length - 1, p + 1))}
864
+ disabled={current === items.length - 1}
865
+ >
866
+ {t("ui-renderer.Rarr")}
867
+ </button>
868
+ </div>
869
+ </div>
870
+ );
871
+ };
872
+
873
+ const BadgeComponent: ComponentFn = (props) => {
874
+ const variant = String(props.variant ?? "default");
875
+ const cls: Record<string, string> = {
876
+ default: "bg-[var(--surface)] text-[var(--text)] border-[var(--border)]",
877
+ success: "bg-[rgba(22,163,106,0.1)] text-[var(--ok)] border-[var(--ok)]",
878
+ warning:
879
+ "bg-[rgba(243,156,18,0.1)] text-[var(--warn,#f39c12)] border-[var(--warn,#f39c12)]",
880
+ error:
881
+ "bg-[rgba(231,76,60,0.1)] text-[var(--destructive)] border-[var(--destructive)]",
882
+ info: "bg-[rgba(52,152,219,0.1)] text-[var(--accent)] border-[var(--accent)]",
883
+ };
884
+ return (
885
+ <span
886
+ className={`inline-block text-[10px] font-medium px-2 py-0.5 border ${cls[variant] ?? cls.default}`}
887
+ >
888
+ {String(props.text ?? "")}
889
+ </span>
890
+ );
891
+ };
892
+
893
+ const AvatarComponent: ComponentFn = (props) => {
894
+ const name = String(props.name ?? "?");
895
+ const size =
896
+ props.size === "lg"
897
+ ? "w-10 h-10 text-sm"
898
+ : props.size === "sm"
899
+ ? "w-6 h-6 text-[10px]"
900
+ : "w-8 h-8 text-xs";
901
+ const initials = name
902
+ .split(" ")
903
+ .map((w) => w[0])
904
+ .join("")
905
+ .slice(0, 2)
906
+ .toUpperCase();
907
+ return (
908
+ <div
909
+ className={`${size} rounded-full bg-[var(--accent)] text-[var(--accent-foreground,#1a1f26)] flex items-center justify-center font-bold shrink-0`}
910
+ >
911
+ {initials}
912
+ </div>
913
+ );
914
+ };
915
+
916
+ const ImageComponent: ComponentFn = (props) => {
917
+ const src = props.src as string | undefined;
918
+ const resolvedSrc = src ? resolveAppAssetUrl(src) : undefined;
919
+ const alt = String(props.alt ?? "");
920
+ const w = props.width ? `${props.width}px` : "auto";
921
+ const h = props.height ? `${props.height}px` : "auto";
922
+ return resolvedSrc ? (
923
+ <img
924
+ src={resolvedSrc}
925
+ alt={alt}
926
+ style={{ width: w, height: h }}
927
+ className="object-cover border border-[var(--border)]"
928
+ />
929
+ ) : (
930
+ <div
931
+ className="bg-[var(--bg-hover)] border border-[var(--border)] flex items-center justify-center text-xs text-[var(--muted)]"
932
+ style={{ width: w, height: h }}
933
+ >
934
+ {alt || "Image"}
935
+ </div>
936
+ );
937
+ };
938
+
939
+ // ── Feedback ────────────────────────────────────────────────────────
940
+
941
+ const AlertComponent: ComponentFn = (props) => {
942
+ const type = String(props.type ?? "info");
943
+ const borderCls: Record<string, string> = {
944
+ info: "border-[var(--accent)]",
945
+ success: "border-[var(--ok)]",
946
+ warning: "border-[var(--warn,#f39c12)]",
947
+ error: "border-[var(--destructive)]",
948
+ };
949
+ const textCls: Record<string, string> = {
950
+ info: "text-[var(--accent)]",
951
+ success: "text-[var(--ok)]",
952
+ warning: "text-[var(--warn,#f39c12)]",
953
+ error: "text-[var(--destructive)]",
954
+ };
955
+ return (
956
+ <div
957
+ className={`border-l-[3px] ${borderCls[type] ?? ""} bg-[var(--bg-hover)] px-3 py-2`}
958
+ >
959
+ {props.title ? (
960
+ <div className={`text-xs font-bold ${textCls[type] ?? ""}`}>
961
+ {String(props.title)}
962
+ </div>
963
+ ) : null}
964
+ {props.message ? (
965
+ <div className="text-xs text-[var(--text)] mt-0.5">
966
+ {String(props.message)}
967
+ </div>
968
+ ) : null}
969
+ </div>
970
+ );
971
+ };
972
+
973
+ const ProgressComponent: ComponentFn = (props) => {
974
+ const value = Number(props.value ?? 0);
975
+ const max = Number(props.max ?? 100);
976
+ const pct = Math.min(100, Math.max(0, (value / max) * 100));
977
+ return (
978
+ <div className="flex flex-col gap-1">
979
+ {props.label ? (
980
+ <div className="flex justify-between text-xs">
981
+ <span className="font-semibold">{String(props.label)}</span>
982
+ <span className="text-[var(--muted)]">{Math.round(pct)}%</span>
983
+ </div>
984
+ ) : null}
985
+ <div className="w-full h-2 bg-[var(--bg-hover)] border border-[var(--border)] overflow-hidden">
986
+ <div
987
+ className="h-full bg-[var(--accent)] transition-[width] duration-300"
988
+ style={{ width: `${pct}%` }}
989
+ />
990
+ </div>
991
+ </div>
992
+ );
993
+ };
994
+
995
+ const RatingComponent: ComponentFn = (props) => {
996
+ const value = Number(props.value ?? 0);
997
+ const max = Number(props.max ?? 5);
998
+ return (
999
+ <div className="flex flex-col gap-1">
1000
+ {props.label ? (
1001
+ <div className="text-xs font-semibold">{String(props.label)}</div>
1002
+ ) : null}
1003
+ <div className="flex gap-0.5">
1004
+ {Array.from({ length: max }, (_, i) => i + 1).map((starValue) => (
1005
+ <span
1006
+ key={starValue}
1007
+ className={`text-sm ${starValue <= value ? "text-[var(--warn,#f39c12)]" : "text-[var(--muted)] opacity-30"}`}
1008
+ >
1009
+
1010
+ </span>
1011
+ ))}
1012
+ </div>
1013
+ </div>
1014
+ );
1015
+ };
1016
+
1017
+ const SkeletonComponent: ComponentFn = (props) => {
1018
+ const w = props.width ? String(props.width) : "100%";
1019
+ const h = props.height ? String(props.height) : "20px";
1020
+ const rounded = props.rounded ? "rounded" : "";
1021
+ return (
1022
+ <div
1023
+ className={`bg-[var(--bg-hover)] animate-pulse ${rounded}`}
1024
+ style={{ width: w, height: h }}
1025
+ />
1026
+ );
1027
+ };
1028
+
1029
+ const SpinnerComponent: ComponentFn = (props) => {
1030
+ const size =
1031
+ props.size === "lg"
1032
+ ? "w-8 h-8"
1033
+ : props.size === "sm"
1034
+ ? "w-4 h-4"
1035
+ : "w-6 h-6";
1036
+ return (
1037
+ <div className="flex items-center gap-2">
1038
+ <div
1039
+ className={`${size} border-2 border-[var(--border)] border-t-[var(--accent)] rounded-full animate-spin`}
1040
+ />
1041
+ {props.label ? (
1042
+ <span className="text-xs text-[var(--muted)]">
1043
+ {String(props.label)}
1044
+ </span>
1045
+ ) : null}
1046
+ </div>
1047
+ );
1048
+ };
1049
+
1050
+ // ── Navigation ──────────────────────────────────────────────────────
1051
+
1052
+ const ButtonComponent: ComponentFn = (props, _children, ctx, el) => {
1053
+ const variant = String(props.variant ?? "primary");
1054
+ const cls: Record<string, string> = {
1055
+ primary:
1056
+ "bg-[var(--accent)] text-[var(--accent-foreground,#1a1f26)] border-[var(--accent)] hover:opacity-90",
1057
+ secondary:
1058
+ "bg-[var(--card)] text-[var(--text)] border-[var(--border)] hover:bg-[var(--bg-hover)]",
1059
+ danger:
1060
+ "bg-[var(--destructive)] text-white border-[var(--destructive)] hover:opacity-90",
1061
+ ghost:
1062
+ "bg-transparent text-[var(--text)] border-transparent hover:bg-[var(--bg-hover)]",
1063
+ };
1064
+ return (
1065
+ <button
1066
+ type="button"
1067
+ className={`px-3 py-1.5 text-xs font-medium border cursor-pointer transition-colors ${cls[variant] ?? cls.primary}`}
1068
+ disabled={!!props.disabled}
1069
+ onClick={() => fireEvent(el.on?.press, ctx)}
1070
+ >
1071
+ {String(props.label ?? "Button")}
1072
+ </button>
1073
+ );
1074
+ };
1075
+
1076
+ const LinkComponent: ComponentFn = (props, _children, ctx, el) => {
1077
+ const safeHref = sanitizeLinkHref(props.href);
1078
+
1079
+ return (
1080
+ <a
1081
+ href={safeHref}
1082
+ className="text-xs text-[var(--accent)] underline hover:opacity-80"
1083
+ target={props.external ? "_blank" : undefined}
1084
+ rel={props.external ? "noopener noreferrer" : undefined}
1085
+ onClick={(e) => {
1086
+ if (el.on?.press) {
1087
+ e.preventDefault();
1088
+ fireEvent(el.on.press, ctx);
1089
+ }
1090
+ }}
1091
+ >
1092
+ {String(props.label ?? props.href ?? "Link")}
1093
+ </a>
1094
+ );
1095
+ };
1096
+
1097
+ const DropdownMenuComponent: ComponentFn = (props, _children, ctx) => {
1098
+ const [open, setOpen] = useState(false);
1099
+ const items = (props.items as Array<{ label: string; value: string }>) ?? [];
1100
+ return (
1101
+ <div className="relative inline-block">
1102
+ <button
1103
+ type="button"
1104
+ className="px-3 py-1.5 text-xs border border-[var(--border)] bg-[var(--card)] cursor-pointer hover:bg-[var(--bg-hover)]"
1105
+ onClick={() => setOpen(!open)}
1106
+ >
1107
+ {String(props.label ?? "Menu")} ▾
1108
+ </button>
1109
+ {open && (
1110
+ <div className="absolute top-full left-0 mt-1 min-w-[120px] border border-[var(--border)] bg-[var(--card)] shadow-md z-10">
1111
+ {items.map((item) => (
1112
+ <button
1113
+ key={item.value}
1114
+ type="button"
1115
+ className="block w-full text-left px-3 py-1.5 text-xs hover:bg-[var(--bg-hover)] cursor-pointer"
1116
+ onClick={() => {
1117
+ setOpen(false);
1118
+ if (ctx.onAction)
1119
+ ctx.onAction("menuSelect", {
1120
+ value: item.value,
1121
+ label: item.label,
1122
+ });
1123
+ }}
1124
+ >
1125
+ {item.label}
1126
+ </button>
1127
+ ))}
1128
+ </div>
1129
+ )}
1130
+ </div>
1131
+ );
1132
+ };
1133
+
1134
+ const TabsComponent: ComponentFn = (props, _children, ctx) => {
1135
+ const tabs =
1136
+ (props.tabs as Array<{ label: string; value: string; content: string }>) ??
1137
+ [];
1138
+ const [value, setValue] = useStatePath(
1139
+ props.statePath as string | undefined,
1140
+ ctx,
1141
+ );
1142
+ const active = String(value ?? props.defaultValue ?? tabs[0]?.value ?? "");
1143
+ const activeTab = tabs.find((t) => t.value === active);
1144
+ return (
1145
+ <div>
1146
+ <div className="flex border-b border-[var(--border)]">
1147
+ {tabs.map((tab) => (
1148
+ <button
1149
+ key={tab.value}
1150
+ type="button"
1151
+ className={`px-3 py-1.5 text-xs cursor-pointer transition-colors ${
1152
+ tab.value === active
1153
+ ? "border-b-2 border-[var(--accent)] text-[var(--accent)] font-semibold"
1154
+ : "text-[var(--muted)] hover:text-[var(--text)]"
1155
+ }`}
1156
+ onClick={() => setValue(tab.value)}
1157
+ >
1158
+ {tab.label}
1159
+ </button>
1160
+ ))}
1161
+ </div>
1162
+ {activeTab && <div className="py-3 text-xs">{activeTab.content}</div>}
1163
+ </div>
1164
+ );
1165
+ };
1166
+
1167
+ const PaginationComponent: ComponentFn = (props, _children, ctx) => {
1168
+ const total = Number(props.totalPages ?? 1);
1169
+ const [value, setValue] = useStatePath(
1170
+ props.statePath as string | undefined,
1171
+ ctx,
1172
+ );
1173
+ const current = Number(value ?? 1);
1174
+ return (
1175
+ <div className="flex items-center gap-1">
1176
+ <button
1177
+ type="button"
1178
+ className="px-2 py-1 text-xs border border-[var(--border)] bg-[var(--card)] cursor-pointer hover:bg-[var(--bg-hover)] disabled:opacity-40"
1179
+ disabled={current <= 1}
1180
+ onClick={() => setValue(current - 1)}
1181
+ >
1182
+
1183
+ </button>
1184
+ {Array.from({ length: total }, (_, i) => i + 1).map((page) => (
1185
+ <button
1186
+ key={page}
1187
+ type="button"
1188
+ className={`px-2 py-1 text-xs border cursor-pointer ${
1189
+ page === current
1190
+ ? "bg-[var(--accent)] text-[var(--accent-foreground,#1a1f26)] border-[var(--accent)]"
1191
+ : "border-[var(--border)] bg-[var(--card)] hover:bg-[var(--bg-hover)]"
1192
+ }`}
1193
+ onClick={() => setValue(page)}
1194
+ >
1195
+ {page}
1196
+ </button>
1197
+ ))}
1198
+ <button
1199
+ type="button"
1200
+ className="px-2 py-1 text-xs border border-[var(--border)] bg-[var(--card)] cursor-pointer hover:bg-[var(--bg-hover)] disabled:opacity-40"
1201
+ disabled={current >= total}
1202
+ onClick={() => setValue(current + 1)}
1203
+ >
1204
+
1205
+ </button>
1206
+ </div>
1207
+ );
1208
+ };
1209
+
1210
+ // ── Metric / KPI ────────────────────────────────────────────────────
1211
+
1212
+ const MetricComponent: ComponentFn = (props) => {
1213
+ const trend = props.trend as string | undefined;
1214
+ const trendColor =
1215
+ trend === "up"
1216
+ ? "text-green-400"
1217
+ : trend === "down"
1218
+ ? "text-red-400"
1219
+ : "text-[var(--muted)]";
1220
+ return (
1221
+ <div className="flex flex-col gap-0.5 p-3 rounded-lg border border-[var(--border)] bg-[var(--card)]">
1222
+ <div className="text-[10px] text-[var(--muted)] uppercase tracking-wider font-medium">
1223
+ {String(props.label ?? "")}
1224
+ </div>
1225
+ <div className="flex items-baseline gap-1.5">
1226
+ <span className="text-xl font-semibold text-[var(--txt)]">
1227
+ {props.value != null ? String(props.value) : "—"}
1228
+ </span>
1229
+ {props.unit != null && (
1230
+ <span className="text-xs text-[var(--muted)]">
1231
+ {String(props.unit)}
1232
+ </span>
1233
+ )}
1234
+ </div>
1235
+ {props.change != null && (
1236
+ <div className={`text-[11px] font-medium ${trendColor}`}>
1237
+ {String(props.change)}
1238
+ </div>
1239
+ )}
1240
+ </div>
1241
+ );
1242
+ };
1243
+
1244
+ // ── Visualization ───────────────────────────────────────────────────
1245
+
1246
+ const BarGraphComponent: ComponentFn = (props) => {
1247
+ const data = (props.data as Array<{ label: string; value: number }>) ?? [];
1248
+ const maxVal = Math.max(...data.map((d) => d.value), 1);
1249
+ return (
1250
+ <div>
1251
+ {props.title ? (
1252
+ <div className="text-xs font-bold mb-2">{String(props.title)}</div>
1253
+ ) : null}
1254
+ <div className="flex items-end gap-2 h-[100px]">
1255
+ {data.map((d) => (
1256
+ <div
1257
+ key={d.label}
1258
+ className="flex-1 flex flex-col items-center gap-0.5"
1259
+ >
1260
+ <div className="text-[9px] text-[var(--muted)]">{d.value}</div>
1261
+ <div
1262
+ className="w-full bg-[var(--accent)] transition-all duration-300 min-h-[2px]"
1263
+ style={{ height: `${(d.value / maxVal) * 80}px` }}
1264
+ />
1265
+ <div className="text-[9px] text-[var(--muted)] truncate max-w-full">
1266
+ {d.label}
1267
+ </div>
1268
+ </div>
1269
+ ))}
1270
+ </div>
1271
+ </div>
1272
+ );
1273
+ };
1274
+
1275
+ const LineGraphComponent: ComponentFn = (props) => {
1276
+ const data = (props.data as Array<{ label: string; value: number }>) ?? [];
1277
+ const maxVal = Math.max(...data.map((d) => d.value), 1);
1278
+ const h = 80;
1279
+ const w = 100;
1280
+ const points = data.map((d, i) => ({
1281
+ x: (i / Math.max(data.length - 1, 1)) * w,
1282
+ y: h - (d.value / maxVal) * h,
1283
+ }));
1284
+ const pathD = points
1285
+ .map((p, i) => `${i === 0 ? "M" : "L"} ${p.x} ${p.y}`)
1286
+ .join(" ");
1287
+ return (
1288
+ <div>
1289
+ {props.title ? (
1290
+ <div className="text-xs font-bold mb-2">{String(props.title)}</div>
1291
+ ) : null}
1292
+ <svg
1293
+ viewBox={`0 0 ${w} ${h + 20}`}
1294
+ className="w-full h-[100px]"
1295
+ preserveAspectRatio="none"
1296
+ >
1297
+ <title>{String(props.title ?? "Line graph")}</title>
1298
+ <path
1299
+ d={pathD}
1300
+ fill="none"
1301
+ stroke="var(--accent)"
1302
+ strokeWidth="2"
1303
+ vectorEffect="non-scaling-stroke"
1304
+ />
1305
+ {points.map((p) => (
1306
+ <circle
1307
+ key={`${p.x}:${p.y}`}
1308
+ cx={p.x}
1309
+ cy={p.y}
1310
+ r="3"
1311
+ fill="var(--accent)"
1312
+ vectorEffect="non-scaling-stroke"
1313
+ />
1314
+ ))}
1315
+ {data.map((d, i) => (
1316
+ <text
1317
+ key={`${d.label}:${d.value}`}
1318
+ x={points[i].x}
1319
+ y={h + 14}
1320
+ textAnchor="middle"
1321
+ fontSize="8"
1322
+ fill="var(--muted)"
1323
+ >
1324
+ {d.label}
1325
+ </text>
1326
+ ))}
1327
+ </svg>
1328
+ </div>
1329
+ );
1330
+ };
1331
+
1332
+ // ── Interaction ─────────────────────────────────────────────────────
1333
+
1334
+ const TooltipComponent: ComponentFn = (props) => {
1335
+ const [show, setShow] = useState(false);
1336
+ return (
1337
+ <button
1338
+ type="button"
1339
+ className="relative inline-block"
1340
+ onMouseEnter={() => setShow(true)}
1341
+ onMouseLeave={() => setShow(false)}
1342
+ onFocus={() => setShow(true)}
1343
+ onBlur={() => setShow(false)}
1344
+ onClick={() => setShow((prev) => !prev)}
1345
+ >
1346
+ <span className="text-xs text-[var(--accent)] underline cursor-help">
1347
+ {String(props.text ?? "Hover")}
1348
+ </span>
1349
+ {show && (
1350
+ <div className="absolute bottom-full left-1/2 -translate-x-1/2 mb-1 px-2 py-1 text-[10px] bg-[var(--text)] text-[var(--card)] whitespace-nowrap z-10">
1351
+ {String(props.content ?? "")}
1352
+ </div>
1353
+ )}
1354
+ </button>
1355
+ );
1356
+ };
1357
+
1358
+ const PopoverComponent: ComponentFn = (props) => {
1359
+ const [open, setOpen] = useState(false);
1360
+ return (
1361
+ <div className="relative inline-block">
1362
+ <button
1363
+ type="button"
1364
+ className="text-xs text-[var(--accent)] underline cursor-pointer"
1365
+ onClick={() => setOpen(!open)}
1366
+ >
1367
+ {String(props.trigger ?? "Click")}
1368
+ </button>
1369
+ {open && (
1370
+ <div className="absolute top-full left-0 mt-1 p-3 border border-[var(--border)] bg-[var(--card)] shadow-md z-10 min-w-[150px]">
1371
+ <div className="text-xs">{String(props.content ?? "")}</div>
1372
+ <button
1373
+ type="button"
1374
+ className="text-[10px] text-[var(--muted)] mt-1 cursor-pointer hover:text-[var(--text)]"
1375
+ onClick={() => setOpen(false)}
1376
+ >
1377
+ Close
1378
+ </button>
1379
+ </div>
1380
+ )}
1381
+ </div>
1382
+ );
1383
+ };
1384
+
1385
+ const CollapsibleComponent: ComponentFn = (props, children) => {
1386
+ const [open, setOpen] = useState(!!props.defaultOpen);
1387
+ return (
1388
+ <div className="border border-[var(--border)]">
1389
+ <button
1390
+ type="button"
1391
+ className="w-full flex items-center gap-2 px-3 py-2 text-xs font-semibold cursor-pointer hover:bg-[var(--bg-hover)] transition-colors"
1392
+ onClick={() => setOpen(!open)}
1393
+ >
1394
+ <span
1395
+ className="text-[10px] transition-transform"
1396
+ style={{ transform: open ? "rotate(90deg)" : "none" }}
1397
+ >
1398
+ &#9654;
1399
+ </span>
1400
+ {String(props.title ?? "Collapsible")}
1401
+ </button>
1402
+ {open && <div className="px-3 pb-3">{children}</div>}
1403
+ </div>
1404
+ );
1405
+ };
1406
+
1407
+ const AccordionComponent: ComponentFn = (props) => {
1408
+ const items =
1409
+ (props.items as Array<{ title: string; content: string }>) ?? [];
1410
+ const isSingle = props.type === "single";
1411
+ const [openSet, setOpenSet] = useState<Set<number>>(new Set());
1412
+
1413
+ const toggle = (idx: number) => {
1414
+ setOpenSet((prev) => {
1415
+ const next = isSingle ? new Set<number>() : new Set(prev);
1416
+ if (prev.has(idx)) next.delete(idx);
1417
+ else next.add(idx);
1418
+ return next;
1419
+ });
1420
+ };
1421
+
1422
+ return (
1423
+ <div className="border border-[var(--border)] divide-y divide-[var(--border)]">
1424
+ {items.map((item, i) => (
1425
+ <div key={`${item.title}:${item.content}`}>
1426
+ <button
1427
+ type="button"
1428
+ className="w-full flex items-center gap-2 px-3 py-2 text-xs font-semibold cursor-pointer hover:bg-[var(--bg-hover)]"
1429
+ onClick={() => toggle(i)}
1430
+ >
1431
+ <span
1432
+ className="text-[10px] transition-transform"
1433
+ style={{ transform: openSet.has(i) ? "rotate(90deg)" : "none" }}
1434
+ >
1435
+ &#9654;
1436
+ </span>
1437
+ {item.title}
1438
+ </button>
1439
+ {openSet.has(i) && (
1440
+ <div className="px-3 pb-3 text-xs">{item.content}</div>
1441
+ )}
1442
+ </div>
1443
+ ))}
1444
+ </div>
1445
+ );
1446
+ };
1447
+
1448
+ const DialogComponent: ComponentFn = (props, children, ctx) => {
1449
+ const openPath = props.openPath as string | undefined;
1450
+ const isOpen = openPath ? !!getByPath(ctx.state, openPath) : false;
1451
+ if (!isOpen) return null;
1452
+ const close = () => {
1453
+ if (openPath) ctx.setState(openPath, false);
1454
+ };
1455
+ return (
1456
+ <div
1457
+ className="fixed inset-0 z-50 flex items-center justify-center bg-black/50"
1458
+ onClick={(e) => {
1459
+ if (e.target === e.currentTarget) close();
1460
+ }}
1461
+ onKeyDown={(e) => {
1462
+ if (e.key === "Escape" || e.key === "Enter" || e.key === " ") {
1463
+ e.preventDefault();
1464
+ close();
1465
+ }
1466
+ }}
1467
+ role="dialog"
1468
+ aria-modal="true"
1469
+ >
1470
+ <div className="w-full max-w-md border border-[var(--border)] bg-[var(--card)] p-5 shadow-lg">
1471
+ <div className="flex items-center justify-between mb-3">
1472
+ <div>
1473
+ {props.title ? (
1474
+ <div className="font-bold text-sm">{String(props.title)}</div>
1475
+ ) : null}
1476
+ {props.description ? (
1477
+ <div className="text-xs text-[var(--muted)] mt-0.5">
1478
+ {String(props.description)}
1479
+ </div>
1480
+ ) : null}
1481
+ </div>
1482
+ <button
1483
+ type="button"
1484
+ className="text-[var(--muted)] hover:text-[var(--text)] text-lg leading-none px-1 cursor-pointer"
1485
+ onClick={close}
1486
+ >
1487
+ ×
1488
+ </button>
1489
+ </div>
1490
+ {children}
1491
+ </div>
1492
+ </div>
1493
+ );
1494
+ };
1495
+
1496
+ const DrawerComponent: ComponentFn = (props, children, ctx) => {
1497
+ const openPath = props.openPath as string | undefined;
1498
+ const isOpen = openPath ? !!getByPath(ctx.state, openPath) : false;
1499
+ if (!isOpen) return null;
1500
+ const close = () => {
1501
+ if (openPath) ctx.setState(openPath, false);
1502
+ };
1503
+ return (
1504
+ <div
1505
+ className="fixed inset-0 z-50 flex items-end bg-black/50"
1506
+ onClick={(e) => {
1507
+ if (e.target === e.currentTarget) close();
1508
+ }}
1509
+ onKeyDown={(e) => {
1510
+ if (e.key === "Escape" || e.key === "Enter" || e.key === " ") {
1511
+ e.preventDefault();
1512
+ close();
1513
+ }
1514
+ }}
1515
+ role="dialog"
1516
+ aria-modal="true"
1517
+ >
1518
+ <div className="w-full max-h-[80vh] border-t border-[var(--border)] bg-[var(--card)] p-5 shadow-lg overflow-y-auto animate-[slide-up_200ms_ease]">
1519
+ <div className="w-10 h-1 bg-[var(--border)] mx-auto mb-3 rounded-full" />
1520
+ {props.title ? (
1521
+ <div className="font-bold text-sm">{String(props.title)}</div>
1522
+ ) : null}
1523
+ {props.description ? (
1524
+ <div className="text-xs text-[var(--muted)] mt-0.5 mb-3">
1525
+ {String(props.description)}
1526
+ </div>
1527
+ ) : null}
1528
+ {children}
1529
+ </div>
1530
+ </div>
1531
+ );
1532
+ };
1533
+
1534
+ // ── Component map ───────────────────────────────────────────────────
1535
+
1536
+ const COMPONENTS: Record<string, ComponentFn> = {
1537
+ // Layout
1538
+ Stack: StackComponent,
1539
+ Grid: GridComponent,
1540
+ Card: CardComponent,
1541
+ Separator: SeparatorComponent,
1542
+ // Typography
1543
+ Heading: HeadingComponent,
1544
+ Text: TextComponent,
1545
+ // Form
1546
+ Input: InputComponent,
1547
+ Textarea: TextareaComponent,
1548
+ Select: SelectComponent,
1549
+ Checkbox: CheckboxComponent,
1550
+ Radio: RadioComponent,
1551
+ Switch: SwitchComponent,
1552
+ Slider: SliderComponent,
1553
+ Toggle: ToggleComponent,
1554
+ ToggleGroup: ToggleGroupComponent,
1555
+ ButtonGroup: ButtonGroupComponent,
1556
+ // Data
1557
+ Table: TableComponent,
1558
+ Carousel: CarouselComponent,
1559
+ Badge: BadgeComponent,
1560
+ Avatar: AvatarComponent,
1561
+ Image: ImageComponent,
1562
+ // Feedback
1563
+ Alert: AlertComponent,
1564
+ Progress: ProgressComponent,
1565
+ Rating: RatingComponent,
1566
+ Skeleton: SkeletonComponent,
1567
+ Spinner: SpinnerComponent,
1568
+ // Navigation
1569
+ Button: ButtonComponent,
1570
+ Link: LinkComponent,
1571
+ DropdownMenu: DropdownMenuComponent,
1572
+ Tabs: TabsComponent,
1573
+ Pagination: PaginationComponent,
1574
+ // Metric
1575
+ Metric: MetricComponent,
1576
+ // Visualization
1577
+ BarGraph: BarGraphComponent,
1578
+ LineGraph: LineGraphComponent,
1579
+ // Interaction
1580
+ Tooltip: TooltipComponent,
1581
+ Popover: PopoverComponent,
1582
+ Collapsible: CollapsibleComponent,
1583
+ Accordion: AccordionComponent,
1584
+ Dialog: DialogComponent,
1585
+ Drawer: DrawerComponent,
1586
+ };
1587
+
1588
+ // ══════════════════════════════════════════════════════════════════════
1589
+ // ELEMENT RENDERER
1590
+ // ══════════════════════════════════════════════════════════════════════
1591
+
1592
+ function ElementRenderer({ elementId }: { elementId: string }) {
1593
+ const { t } = useApp();
1594
+ const ctx = useUiCtx();
1595
+ const el = ctx.spec.elements[elementId];
1596
+ if (!el) return null;
1597
+
1598
+ // Visibility check
1599
+ if (el.visible && !evaluateUiVisibility(el.visible, ctx.state, ctx.auth)) {
1600
+ return null;
1601
+ }
1602
+
1603
+ const component = COMPONENTS[el.type];
1604
+ if (!component) {
1605
+ return (
1606
+ <div className="text-[10px] text-[var(--destructive)] border border-dashed border-[var(--destructive)] p-2">
1607
+ {t("ui-renderer.UnknownComponent")} {el.type}
1608
+ </div>
1609
+ );
1610
+ }
1611
+
1612
+ const resolvedProps = resolveProps(el.props, ctx);
1613
+
1614
+ // Handle repeat / list rendering
1615
+ if (el.repeat) {
1616
+ const listData = getByPath(ctx.state, el.repeat.path) as
1617
+ | Array<Record<string, unknown>>
1618
+ | undefined;
1619
+ if (!Array.isArray(listData)) return null;
1620
+
1621
+ return (
1622
+ <>
1623
+ {listData.map((item) => {
1624
+ const itemCtx: UiRenderContext = { ...ctx, repeatItem: item };
1625
+ const childNodes = el.children.map((childId) => (
1626
+ <UiContext.Provider key={childId} value={itemCtx}>
1627
+ <ElementRenderer elementId={childId} />
1628
+ </UiContext.Provider>
1629
+ ));
1630
+ const repeatKey = el.repeat?.key;
1631
+ const itemKey = String(
1632
+ repeatKey != null ? item[repeatKey] : Math.random(),
1633
+ );
1634
+ return (
1635
+ <React.Fragment key={itemKey}>
1636
+ {component(resolvedProps, childNodes, itemCtx, el)}
1637
+ </React.Fragment>
1638
+ );
1639
+ })}
1640
+ </>
1641
+ );
1642
+ }
1643
+
1644
+ // Normal rendering: resolve children
1645
+ const childNodes = el.children.map((childId) => (
1646
+ <ElementRenderer key={childId} elementId={childId} />
1647
+ ));
1648
+
1649
+ return <>{component(resolvedProps, childNodes, ctx, el)}</>;
1650
+ }
1651
+
1652
+ // ══════════════════════════════════════════════════════════════════════
1653
+ // ROOT RENDERER
1654
+ // ══════════════════════════════════════════════════════════════════════
1655
+
1656
+ export interface UiRendererProps {
1657
+ spec: UiSpec;
1658
+ onAction?: (action: string, params?: Record<string, unknown>) => void;
1659
+ loading?: boolean;
1660
+ auth?: AuthState;
1661
+ validators?: Record<
1662
+ string,
1663
+ (
1664
+ value: unknown,
1665
+ args?: Record<string, unknown>,
1666
+ ) => boolean | Promise<boolean>
1667
+ >;
1668
+ }
1669
+
1670
+ export function UiRenderer({
1671
+ spec,
1672
+ onAction,
1673
+ loading,
1674
+ auth,
1675
+ validators,
1676
+ }: UiRendererProps) {
1677
+ const [state, setStateRaw] = useState<Record<string, unknown>>(() => ({
1678
+ ...spec.state,
1679
+ }));
1680
+ const [fieldErrors, setFieldErrors] = useState<Record<string, string[]>>({});
1681
+
1682
+ const setState = useCallback((path: string, value: unknown) => {
1683
+ setStateRaw((prev) => {
1684
+ const next = { ...prev };
1685
+ setByPath(next, path, value);
1686
+ return next;
1687
+ });
1688
+ }, []);
1689
+
1690
+ const validateField = useCallback(
1691
+ (statePath: string) => {
1692
+ // Find the element that has this statePath
1693
+ for (const el of Object.values(spec.elements)) {
1694
+ if (el.props.statePath === statePath && el.validation) {
1695
+ const value = getByPath(state, statePath);
1696
+ const errors = runValidation(el.validation.checks, value, validators);
1697
+ setFieldErrors((prev) => ({ ...prev, [statePath]: errors }));
1698
+ return;
1699
+ }
1700
+ }
1701
+ },
1702
+ [spec.elements, state, validators],
1703
+ );
1704
+
1705
+ const ctx = useMemo<UiRenderContext>(
1706
+ () => ({
1707
+ spec,
1708
+ state,
1709
+ setState,
1710
+ onAction,
1711
+ auth,
1712
+ loading,
1713
+ validators,
1714
+ fieldErrors,
1715
+ validateField,
1716
+ }),
1717
+ [
1718
+ spec,
1719
+ state,
1720
+ setState,
1721
+ onAction,
1722
+ auth,
1723
+ loading,
1724
+ validators,
1725
+ fieldErrors,
1726
+ validateField,
1727
+ ],
1728
+ );
1729
+
1730
+ // Loading skeleton when no elements
1731
+ if (loading && Object.keys(spec.elements).length === 0) {
1732
+ return (
1733
+ <div className="flex flex-col gap-3 animate-pulse">
1734
+ <div className="h-4 bg-[var(--bg-hover)] w-3/4" />
1735
+ <div className="h-3 bg-[var(--bg-hover)] w-1/2" />
1736
+ <div className="h-3 bg-[var(--bg-hover)] w-5/6" />
1737
+ </div>
1738
+ );
1739
+ }
1740
+
1741
+ return (
1742
+ <UiContext.Provider value={ctx}>
1743
+ <ElementRenderer elementId={spec.root} />
1744
+ </UiContext.Provider>
1745
+ );
1746
+ }
1747
+
1748
+ /** Get the full list of supported component types. */
1749
+ export function getSupportedComponents(): string[] {
1750
+ return Object.keys(COMPONENTS);
1751
+ }