@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,1900 @@
1
+ /**
2
+ * config-field.tsx — React port of the 19 Lit field renderers from config-field.ts.
3
+ *
4
+ * Each renderer is a plain function (props: FieldRenderProps) => JSX.Element.
5
+ * Styling uses Tailwind utility classes + CSS custom properties from base.css.
6
+ *
7
+ * Also exports:
8
+ * - defaultRenderers map (field type name -> renderer)
9
+ * - ConfigField wrapper component (label + renderer + help + errors)
10
+ */
11
+
12
+ import { ChevronDown, X } from "lucide-react";
13
+ import React, { useCallback, useRef, useState } from "react";
14
+ import { useApp } from "../state";
15
+ import type { DynamicValue } from "../types";
16
+ import type { FieldRenderer, FieldRenderProps } from "./config-catalog";
17
+ import { resolveDynamic } from "./config-catalog";
18
+
19
+ // ── Action binding helper ──────────────────────────────────────────────
20
+
21
+ /**
22
+ * Resolve DynamicValue params and fire the onAction callback.
23
+ * No-ops when the binding or onAction is missing.
24
+ */
25
+ function fireAction(props: FieldRenderProps, eventName: string): void {
26
+ const binding = props.hint.on?.[eventName];
27
+ if (!binding || !props.onAction) return;
28
+
29
+ // Resolve any DynamicValue params against a state snapshot built from the
30
+ // current field value (keyed by the field's own key). The resolveDynamic
31
+ // function handles both literal values and { path } references.
32
+ let resolvedParams: Record<string, unknown> | undefined;
33
+ if (binding.params) {
34
+ const state: Record<string, unknown> = { [props.key]: props.value };
35
+ resolvedParams = {};
36
+ for (const [k, v] of Object.entries(binding.params)) {
37
+ resolvedParams[k] = resolveDynamic(v as DynamicValue, state);
38
+ }
39
+ }
40
+
41
+ void props.onAction(binding.action, resolvedParams);
42
+ }
43
+
44
+ // ── Shared Tailwind class constants ─────────────────────────────────────
45
+
46
+ const INPUT_CLS =
47
+ "w-full px-3 py-2 border border-[var(--border)] bg-[var(--card)] text-[13px] font-[var(--mono)] transition-all focus:border-[var(--accent)] focus:outline-none focus:ring-1 focus:ring-[var(--accent)] box-border h-[36px] rounded-sm placeholder:text-[var(--muted)] placeholder:opacity-60";
48
+
49
+ const INPUT_ERROR_CLS =
50
+ "w-full px-3 py-2 border border-[var(--destructive)] bg-[color-mix(in_srgb,var(--destructive)_3%,var(--card))] text-[13px] font-[var(--mono)] transition-all focus:border-[var(--accent)] focus:outline-none focus:ring-1 focus:ring-[var(--accent)] box-border h-[36px] rounded-sm placeholder:text-[var(--muted)] placeholder:opacity-60";
51
+
52
+ function inputCls(hasError: boolean): string {
53
+ return hasError ? INPUT_ERROR_CLS : INPUT_CLS;
54
+ }
55
+
56
+ // ── 1. Text ─────────────────────────────────────────────────────────────
57
+
58
+ /** Single-line text input. Fallback renderer for unresolved field types. */
59
+ export function renderTextField(props: FieldRenderProps) {
60
+ const value = props.isSet ? String(props.value ?? "") : "";
61
+ const placeholder =
62
+ (props.hint.placeholder as string | undefined) ??
63
+ (props.schema.default != null
64
+ ? `Default: ${props.schema.default}`
65
+ : "Enter value...");
66
+
67
+ return (
68
+ <input
69
+ className={inputCls(!!props.errors?.length)}
70
+ type="text"
71
+ defaultValue={value}
72
+ placeholder={placeholder}
73
+ data-config-key={props.key}
74
+ data-field-type="text"
75
+ disabled={props.readonly}
76
+ onChange={(e) => {
77
+ props.onChange(e.target.value);
78
+ fireAction(props, "change");
79
+ }}
80
+ onBlur={() => fireAction(props, "blur")}
81
+ onClick={() => fireAction(props, "click")}
82
+ />
83
+ );
84
+ }
85
+
86
+ // ── 2. Password ─────────────────────────────────────────────────────────
87
+
88
+ /** Masked password input with show/hide toggle and async onReveal for server-backed decryption. */
89
+ export function renderPasswordField(props: FieldRenderProps) {
90
+ return <PasswordFieldInner fp={props} />;
91
+ }
92
+
93
+ function PasswordFieldInner({ fp: props }: { fp: FieldRenderProps }) {
94
+ const maskedValue = props.isSet ? String(props.value ?? "") : "";
95
+ const placeholder = props.isSet
96
+ ? `Current: ${maskedValue || "********"} (leave blank to keep)`
97
+ : ((props.hint.placeholder as string | undefined) ?? "Enter value...");
98
+
99
+ const [visible, setVisible] = useState(false);
100
+ const [busy, setBusy] = useState(false);
101
+ const inputRef = useRef<HTMLInputElement>(null);
102
+ const onReveal = props.onReveal;
103
+
104
+ const handleToggle = useCallback(async () => {
105
+ const input = inputRef.current;
106
+ if (!input) return;
107
+
108
+ if (visible) {
109
+ // Currently showing -- hide it
110
+ setVisible(false);
111
+ input.value = "";
112
+ return;
113
+ }
114
+
115
+ // Reveal: fetch the real value from the server
116
+ if (onReveal) {
117
+ setBusy(true);
118
+ const realValue = await onReveal();
119
+ setBusy(false);
120
+ if (realValue != null) {
121
+ setVisible(true);
122
+ input.value = realValue;
123
+ }
124
+ } else {
125
+ // Fallback: just toggle type (shows whatever is in the input)
126
+ setVisible(true);
127
+ }
128
+ }, [visible, onReveal]);
129
+
130
+ return (
131
+ <div className="flex">
132
+ <input
133
+ ref={inputRef}
134
+ className="flex-1 px-3 py-2 border border-[var(--border)] border-r-0 bg-[var(--card)] text-[13px] font-[var(--mono)] transition-all focus:border-[var(--accent)] focus:outline-none focus:ring-1 focus:ring-[var(--accent)] box-border h-[36px] rounded-l-sm placeholder:text-[var(--muted)] placeholder:opacity-60"
135
+ type={visible ? "text" : "password"}
136
+ defaultValue=""
137
+ placeholder={placeholder}
138
+ data-config-key={props.key}
139
+ data-field-type="password"
140
+ onChange={(e) => {
141
+ props.onChange(e.target.value);
142
+ fireAction(props, "change");
143
+ }}
144
+ onBlur={() => fireAction(props, "blur")}
145
+ />
146
+ <button
147
+ type="button"
148
+ className="px-3 border border-[var(--border)] bg-[var(--bg-hover)] text-[11px] text-[var(--muted)] cursor-pointer whitespace-nowrap min-w-[56px] text-center transition-colors hover:bg-[var(--surface)] hover:text-[var(--text)] h-[36px] font-medium rounded-r-sm"
149
+ onClick={() => {
150
+ void handleToggle();
151
+ fireAction(props, "click");
152
+ }}
153
+ title={visible ? "Hide value" : "Reveal value"}
154
+ >
155
+ {busy ? "\u2026" : visible ? "\u{1F441} Hide" : "\u{1F441} Show"}
156
+ </button>
157
+ </div>
158
+ );
159
+ }
160
+
161
+ // ── 3. Number ───────────────────────────────────────────────────────────
162
+
163
+ /** Numeric input with min/max/step attributes derived from schema and hints. */
164
+ export function renderNumberField(props: FieldRenderProps) {
165
+ return <NumberFieldInner fp={props} />;
166
+ }
167
+
168
+ function NumberFieldInner({ fp: props }: { fp: FieldRenderProps }) {
169
+ const minVal =
170
+ props.schema.minimum ?? (props.hint.min as number | undefined) ?? undefined;
171
+ const maxVal =
172
+ props.schema.maximum ?? (props.hint.max as number | undefined) ?? undefined;
173
+ const stepVal = (props.hint.step as number | undefined) ?? 1;
174
+ const unit = props.hint.unit as string | undefined;
175
+ const placeholder =
176
+ (props.hint.placeholder as string | undefined) ??
177
+ (props.schema.default != null
178
+ ? `Default: ${props.schema.default}`
179
+ : "Enter number...");
180
+
181
+ const initial = props.isSet ? String(props.value ?? "") : "";
182
+ const [val, setVal] = useState(initial);
183
+
184
+ const hasRange = minVal != null || maxVal != null;
185
+
186
+ const step = (direction: 1 | -1) => {
187
+ const current = val === "" ? 0 : Number(val);
188
+ if (Number.isNaN(current)) return;
189
+ let next = current + direction * stepVal;
190
+ if (minVal != null && next < minVal) next = minVal;
191
+ if (maxVal != null && next > maxVal) next = maxVal;
192
+ const s = String(next);
193
+ setVal(s);
194
+ props.onChange(s);
195
+ fireAction(props, "change");
196
+ };
197
+
198
+ return (
199
+ <div>
200
+ <div className="flex items-center gap-1.5">
201
+ {!props.readonly && (
202
+ <button
203
+ type="button"
204
+ className="px-2 py-1.5 border border-[var(--border)] bg-transparent text-sm text-[var(--muted)] cursor-pointer transition-colors hover:bg-[var(--surface)] hover:text-[var(--text)] h-[36px] rounded-sm font-mono select-none"
205
+ onClick={() => step(-1)}
206
+ >
207
+
208
+ </button>
209
+ )}
210
+ <input
211
+ className={`${inputCls(!!props.errors?.length)} ${unit ? "flex-1" : "w-full"} text-center`}
212
+ type="number"
213
+ value={val}
214
+ placeholder={placeholder}
215
+ min={minVal}
216
+ max={maxVal}
217
+ step={stepVal}
218
+ data-config-key={props.key}
219
+ data-field-type="number"
220
+ disabled={props.readonly}
221
+ onChange={(e) => {
222
+ setVal(e.target.value);
223
+ props.onChange(e.target.value);
224
+ fireAction(props, "change");
225
+ }}
226
+ onBlur={() => fireAction(props, "blur")}
227
+ onClick={() => fireAction(props, "click")}
228
+ />
229
+ {!props.readonly && (
230
+ <button
231
+ type="button"
232
+ className="px-2 py-1.5 border border-[var(--border)] bg-transparent text-sm text-[var(--muted)] cursor-pointer transition-colors hover:bg-[var(--surface)] hover:text-[var(--text)] h-[36px] rounded-sm font-mono select-none"
233
+ onClick={() => step(1)}
234
+ >
235
+ +
236
+ </button>
237
+ )}
238
+ {unit && (
239
+ <span className="text-[11px] text-[var(--muted)] font-medium shrink-0 min-w-[20px]">
240
+ {unit}
241
+ </span>
242
+ )}
243
+ </div>
244
+ {hasRange && (
245
+ <div className="text-[10px] text-[var(--muted)] mt-0.5 opacity-70">
246
+ {minVal != null && maxVal != null
247
+ ? `Range: ${minVal}–${maxVal}${unit ? ` ${unit}` : ""}`
248
+ : minVal != null
249
+ ? `Min: ${minVal}${unit ? ` ${unit}` : ""}`
250
+ : `Max: ${maxVal}${unit ? ` ${unit}` : ""}`}
251
+ </div>
252
+ )}
253
+ </div>
254
+ );
255
+ }
256
+
257
+ // ── 4. Boolean ──────────────────────────────────────────────────────────
258
+
259
+ /** Pill-shaped toggle switch. Accepts boolean or string 'true'/'false' values. */
260
+ export function renderBooleanField(props: FieldRenderProps) {
261
+ return <BooleanFieldInner fp={props} />;
262
+ }
263
+
264
+ function BooleanFieldInner({ fp: props }: { fp: FieldRenderProps }) {
265
+ const val =
266
+ props.value === true || props.value === "true" || props.value === "1";
267
+ const initialVal = props.isSet
268
+ ? val
269
+ : props.schema.default === true || props.schema.default === "true";
270
+
271
+ const [localVal, setLocalVal] = useState(initialVal);
272
+
273
+ const handleToggle = () => {
274
+ const next = !localVal;
275
+ setLocalVal(next);
276
+ props.onChange(String(next));
277
+ fireAction(props, "change");
278
+ };
279
+
280
+ return (
281
+ <button
282
+ type="button"
283
+ className="flex items-center gap-2.5 cursor-pointer bg-transparent border-none p-0 group"
284
+ disabled={props.readonly}
285
+ onClick={() => {
286
+ handleToggle();
287
+ fireAction(props, "click");
288
+ }}
289
+ data-config-key={props.key}
290
+ data-field-type="boolean"
291
+ >
292
+ <div
293
+ className={`relative w-10 h-[22px] rounded-full transition-colors duration-200 ${
294
+ localVal ? "bg-[var(--accent)]" : "bg-[var(--muted)] opacity-40"
295
+ }`}
296
+ >
297
+ <div
298
+ className={`absolute top-[3px] w-4 h-4 bg-white rounded-full shadow-sm transition-all duration-200 ${
299
+ localVal ? "left-[21px]" : "left-[3px]"
300
+ }`}
301
+ />
302
+ </div>
303
+ <span
304
+ className={`text-xs transition-colors ${localVal ? "text-[var(--text)] font-medium" : "text-[var(--muted)]"}`}
305
+ >
306
+ {localVal ? "Enabled" : "Disabled"}
307
+ </span>
308
+ </button>
309
+ );
310
+ }
311
+
312
+ // ── 5. URL ──────────────────────────────────────────────────────────────
313
+
314
+ /** URL input with type="url" browser validation. */
315
+ export function renderUrlField(props: FieldRenderProps) {
316
+ const value = props.isSet ? String(props.value ?? "") : "";
317
+ const placeholder =
318
+ (props.hint.placeholder as string | undefined) ??
319
+ (props.schema.default != null
320
+ ? `Default: ${props.schema.default}`
321
+ : "https://...");
322
+
323
+ return (
324
+ <input
325
+ className={inputCls(!!props.errors?.length)}
326
+ type="url"
327
+ defaultValue={value}
328
+ placeholder={placeholder}
329
+ data-config-key={props.key}
330
+ data-field-type="url"
331
+ disabled={props.readonly}
332
+ onChange={(e) => {
333
+ props.onChange(e.target.value);
334
+ fireAction(props, "change");
335
+ }}
336
+ onBlur={() => fireAction(props, "blur")}
337
+ onClick={() => fireAction(props, "click")}
338
+ />
339
+ );
340
+ }
341
+
342
+ // ── 6. Select ───────────────────────────────────────────────────────────
343
+
344
+ /** Dropdown select. Options from hint.options or schema.enum. */
345
+ export function RenderSelectField(props: FieldRenderProps) {
346
+ const { t } = useApp();
347
+ const enhancedOptions = (props.hint as Record<string, unknown>).options as
348
+ | Array<{ value: string; label: string; description?: string }>
349
+ | undefined;
350
+
351
+ const plainOptions: string[] =
352
+ (props.schema.enum as string[]) ??
353
+ props.schema.oneOf?.map((o) => String(o.const ?? o.description ?? "")) ??
354
+ [];
355
+
356
+ const allOptions = enhancedOptions
357
+ ? enhancedOptions.map((o) => ({
358
+ value: o.value,
359
+ label: o.label,
360
+ description: o.description,
361
+ }))
362
+ : plainOptions.map((o) => ({
363
+ value: o,
364
+ label: o,
365
+ description: undefined as string | undefined,
366
+ }));
367
+
368
+ const value = props.isSet ? String(props.value ?? "") : "";
369
+ const effectiveValue = value || String(props.schema.default ?? "");
370
+ const useSearch = allOptions.length >= 5;
371
+ if (useSearch) {
372
+ return (
373
+ <SearchableSelectInner
374
+ fp={props}
375
+ options={allOptions}
376
+ effectiveValue={effectiveValue}
377
+ />
378
+ );
379
+ }
380
+
381
+ return (
382
+ <select
383
+ className={`${INPUT_CLS} appearance-auto`}
384
+ defaultValue={effectiveValue}
385
+ data-config-key={props.key}
386
+ data-field-type="select"
387
+ disabled={props.readonly}
388
+ onChange={(e) => {
389
+ props.onChange(e.target.value);
390
+ fireAction(props, "change");
391
+ }}
392
+ onBlur={() => fireAction(props, "blur")}
393
+ onClick={() => fireAction(props, "click")}
394
+ >
395
+ {!props.required && <option value="">{t("config-field.None")}</option>}
396
+ {allOptions.map((opt) => (
397
+ <option key={opt.value} value={opt.value}>
398
+ {opt.label}
399
+ {opt.description ? ` — ${opt.description}` : ""}
400
+ </option>
401
+ ))}
402
+ </select>
403
+ );
404
+ }
405
+
406
+ function SearchableSelectInner({
407
+ fp: props,
408
+ options,
409
+ effectiveValue,
410
+ }: {
411
+ fp: FieldRenderProps;
412
+ options: Array<{ value: string; label: string; description?: string }>;
413
+ effectiveValue: string;
414
+ }) {
415
+ const { t } = useApp();
416
+ const matchingOpt = options.find((o) => o.value === effectiveValue);
417
+ const [inputVal, setInputVal] = useState(
418
+ matchingOpt?.label ?? effectiveValue,
419
+ );
420
+ const [open, setOpen] = useState(false);
421
+ const [filter, setFilter] = useState("");
422
+ const containerRef = useRef<HTMLDivElement>(null);
423
+ const inputRef = useRef<HTMLInputElement>(null);
424
+
425
+ const filtered = filter
426
+ ? options.filter(
427
+ (o) =>
428
+ o.label.toLowerCase().includes(filter.toLowerCase()) ||
429
+ o.value.toLowerCase().includes(filter.toLowerCase()),
430
+ )
431
+ : options;
432
+
433
+ const select = useCallback(
434
+ (opt: { value: string; label: string }) => {
435
+ props.onChange(opt.value);
436
+ setInputVal(opt.label);
437
+ setOpen(false);
438
+ setFilter("");
439
+ fireAction(props, "change");
440
+ },
441
+ [props],
442
+ );
443
+
444
+ // Close on click outside
445
+ React.useEffect(() => {
446
+ if (!open) return;
447
+ const handler = (e: MouseEvent) => {
448
+ if (
449
+ containerRef.current &&
450
+ !containerRef.current.contains(e.target as Node)
451
+ ) {
452
+ setOpen(false);
453
+ setFilter("");
454
+ }
455
+ };
456
+ document.addEventListener("mousedown", handler);
457
+ return () => document.removeEventListener("mousedown", handler);
458
+ }, [open]);
459
+
460
+ return (
461
+ <div className="relative" ref={containerRef}>
462
+ {/* Trigger button that looks like a select */}
463
+ <button
464
+ type="button"
465
+ className={`${inputCls(!!props.errors?.length)} text-left flex items-center justify-between gap-2 cursor-pointer`}
466
+ disabled={props.readonly}
467
+ onClick={() => {
468
+ setOpen(!open);
469
+ setFilter("");
470
+ }}
471
+ data-config-key={props.key}
472
+ data-field-type="select"
473
+ >
474
+ <span className={inputVal ? "" : "text-[var(--muted)] opacity-60"}>
475
+ {inputVal || "Select..."}
476
+ </span>
477
+ <span className="text-[var(--muted)] text-[10px] shrink-0">
478
+ {open ? "\u25B2" : "\u25BC"}
479
+ </span>
480
+ </button>
481
+
482
+ {/* Dropdown panel */}
483
+ {open && (
484
+ <div className="absolute z-50 left-0 right-0 mt-1 border border-[var(--border)] bg-[var(--card)] shadow-lg max-h-[280px] flex flex-col rounded-sm">
485
+ {/* Search input */}
486
+ <div className="p-1.5 border-b border-[var(--border)]">
487
+ <input
488
+ ref={inputRef}
489
+ className="w-full px-2 py-1.5 border border-[var(--border)] bg-[var(--bg)] text-[12px] font-[var(--mono)] focus:border-[var(--accent)] focus:outline-none rounded-sm"
490
+ type="text"
491
+ value={filter}
492
+ placeholder={`Search ${options.length} models...`}
493
+ onChange={(e) => setFilter(e.target.value)}
494
+ onKeyDown={(e) => {
495
+ if (e.key === "Escape") {
496
+ setOpen(false);
497
+ setFilter("");
498
+ } else if (e.key === "Enter" && filtered.length === 1) {
499
+ select(filtered[0]);
500
+ }
501
+ }}
502
+ />
503
+ </div>
504
+ {/* Options list */}
505
+ <div className="overflow-y-auto flex-1">
506
+ {!props.required && (
507
+ <button
508
+ type="button"
509
+ className="w-full text-left px-3 py-1.5 text-[12px] text-[var(--muted)] hover:bg-[var(--bg-hover)] transition-colors italic"
510
+ onClick={() => {
511
+ props.onChange("");
512
+ setInputVal("");
513
+ setOpen(false);
514
+ setFilter("");
515
+ fireAction(props, "change");
516
+ }}
517
+ >
518
+ {t("config-field.None")}
519
+ </button>
520
+ )}
521
+ {filtered.length === 0 && (
522
+ <div className="px-3 py-3 text-[12px] text-[var(--muted)] text-center">
523
+ {t("config-field.NoMatches")}
524
+ </div>
525
+ )}
526
+ {filtered.map((opt) => (
527
+ <button
528
+ key={opt.value}
529
+ type="button"
530
+ className={`w-full text-left px-3 py-1.5 text-[12px] hover:bg-[var(--bg-hover)] transition-colors ${
531
+ opt.value === effectiveValue
532
+ ? "bg-[color-mix(in_srgb,var(--accent)_10%,var(--card))] text-[var(--accent)] font-medium"
533
+ : ""
534
+ }`}
535
+ onClick={() => select(opt)}
536
+ >
537
+ {opt.label}
538
+ {opt.description && (
539
+ <span className="text-[var(--muted)] ml-1.5 text-[11px]">
540
+ {opt.description}
541
+ </span>
542
+ )}
543
+ </button>
544
+ ))}
545
+ </div>
546
+ <div className="px-3 py-1 border-t border-[var(--border)] text-[10px] text-[var(--muted)]">
547
+ {filtered.length} of {options.length} {t("config-field.models")}
548
+ </div>
549
+ </div>
550
+ )}
551
+ </div>
552
+ );
553
+ }
554
+
555
+ // ── 7. Textarea ─────────────────────────────────────────────────────────
556
+
557
+ /** Multi-line text input with auto-resize. Auto-detected for maxLength > 200. */
558
+ export function renderTextareaField(props: FieldRenderProps) {
559
+ return <TextareaFieldInner fp={props} />;
560
+ }
561
+
562
+ function TextareaFieldInner({ fp: props }: { fp: FieldRenderProps }) {
563
+ const value = props.isSet ? String(props.value ?? "") : "";
564
+ const placeholder =
565
+ (props.hint.placeholder as string | undefined) ?? "Enter text...";
566
+ const textareaRef = useRef<HTMLTextAreaElement>(null);
567
+
568
+ const autoResize = useCallback(() => {
569
+ const el = textareaRef.current;
570
+ if (!el) return;
571
+ // Check if field-sizing is supported — if so, CSS handles it
572
+ if (globalThis.CSS?.supports?.("field-sizing", "content")) return;
573
+ el.style.height = "auto";
574
+ el.style.height = `${Math.max(72, el.scrollHeight)}px`;
575
+ }, []);
576
+
577
+ return (
578
+ <textarea
579
+ ref={textareaRef}
580
+ className="w-full px-3 py-2 border border-[var(--border)] bg-[var(--card)] text-[13px] font-[var(--mono)] transition-all focus:border-[var(--accent)] focus:outline-none focus:ring-1 focus:ring-[var(--accent)] box-border min-h-[72px] max-h-[400px] rounded-sm placeholder:text-[var(--muted)] placeholder:opacity-60"
581
+ style={{ fieldSizing: "content" } as React.CSSProperties}
582
+ defaultValue={value}
583
+ placeholder={placeholder}
584
+ data-config-key={props.key}
585
+ data-field-type="textarea"
586
+ disabled={props.readonly}
587
+ onChange={(e) => {
588
+ props.onChange(e.target.value);
589
+ fireAction(props, "change");
590
+ autoResize();
591
+ }}
592
+ onBlur={() => fireAction(props, "blur")}
593
+ onClick={() => fireAction(props, "click")}
594
+ onFocus={autoResize}
595
+ />
596
+ );
597
+ }
598
+
599
+ // ── 8. Email ────────────────────────────────────────────────────────────
600
+
601
+ /** Email input with type="email" browser validation. */
602
+ export function renderEmailField(props: FieldRenderProps) {
603
+ const value = props.isSet ? String(props.value ?? "") : "";
604
+ const placeholder =
605
+ (props.hint.placeholder as string | undefined) ?? "user@example.com";
606
+
607
+ return (
608
+ <input
609
+ className={inputCls(!!props.errors?.length)}
610
+ type="email"
611
+ defaultValue={value}
612
+ placeholder={placeholder}
613
+ data-config-key={props.key}
614
+ data-field-type="email"
615
+ disabled={props.readonly}
616
+ onChange={(e) => {
617
+ props.onChange(e.target.value);
618
+ fireAction(props, "change");
619
+ }}
620
+ onBlur={() => fireAction(props, "blur")}
621
+ onClick={() => fireAction(props, "click")}
622
+ />
623
+ );
624
+ }
625
+
626
+ // ── 9. Color ────────────────────────────────────────────────────────────
627
+
628
+ /** Color picker swatch paired with a hex text input. */
629
+ export function renderColorField(props: FieldRenderProps) {
630
+ return <ColorFieldInner fp={props} />;
631
+ }
632
+
633
+ function ColorFieldInner({ fp: props }: { fp: FieldRenderProps }) {
634
+ const initial = props.isSet ? String(props.value ?? "#000000") : "#000000";
635
+ const [color, setColor] = useState(initial);
636
+
637
+ const handleChange = (newColor: string) => {
638
+ setColor(newColor);
639
+ props.onChange(newColor);
640
+ fireAction(props, "change");
641
+ };
642
+
643
+ return (
644
+ <div className="flex items-center gap-2">
645
+ <input
646
+ className="w-[36px] h-[36px] border border-[var(--border)] p-0.5 cursor-pointer bg-transparent rounded-sm [&::-webkit-color-swatch-wrapper]:p-0 [&::-webkit-color-swatch]:border-none [&::-webkit-color-swatch]:rounded-sm"
647
+ type="color"
648
+ value={color}
649
+ data-config-key={props.key}
650
+ data-field-type="color"
651
+ disabled={props.readonly}
652
+ onChange={(e) => handleChange(e.target.value)}
653
+ onBlur={() => fireAction(props, "blur")}
654
+ onClick={() => fireAction(props, "click")}
655
+ />
656
+ <input
657
+ className={`${inputCls(!!props.errors?.length)} flex-1`}
658
+ type="text"
659
+ value={color}
660
+ placeholder="#000000"
661
+ data-config-key={props.key}
662
+ data-field-type="color-text"
663
+ disabled={props.readonly}
664
+ onChange={(e) => handleChange(e.target.value)}
665
+ onBlur={() => fireAction(props, "blur")}
666
+ onClick={() => fireAction(props, "click")}
667
+ />
668
+ </div>
669
+ );
670
+ }
671
+
672
+ // ── 10. Radio ───────────────────────────────────────────────────────────
673
+
674
+ /** Vertical radio button group. Supports per-option descriptions. */
675
+ export function renderRadioField(props: FieldRenderProps) {
676
+ return <RadioFieldInner fp={props} />;
677
+ }
678
+
679
+ function RadioFieldInner({ fp: props }: { fp: FieldRenderProps }) {
680
+ const options: Array<{
681
+ value: string;
682
+ label: string;
683
+ description?: string;
684
+ disabled?: boolean;
685
+ }> =
686
+ ((props.hint as Record<string, unknown>).options as
687
+ | typeof options
688
+ | undefined) ??
689
+ ((props.schema.enum as string[]) ?? []).map((v) => ({
690
+ value: String(v),
691
+ label: String(v),
692
+ }));
693
+
694
+ const initial = props.isSet
695
+ ? String(props.value ?? "")
696
+ : String(props.schema.default ?? "");
697
+ const [selected, setSelected] = useState(initial);
698
+
699
+ const handleChange = (val: string) => {
700
+ setSelected(val);
701
+ props.onChange(val);
702
+ fireAction(props, "change");
703
+ };
704
+
705
+ return (
706
+ <div
707
+ className="flex flex-col gap-1.5"
708
+ data-config-key={props.key}
709
+ data-field-type="radio"
710
+ >
711
+ {options.map((opt) => (
712
+ <label
713
+ key={opt.value}
714
+ className="flex items-start gap-2 cursor-pointer text-[13px]"
715
+ >
716
+ <input
717
+ type="radio"
718
+ name={props.key}
719
+ value={opt.value}
720
+ checked={opt.value === selected}
721
+ disabled={props.readonly || opt.disabled}
722
+ onChange={() => handleChange(opt.value)}
723
+ onClick={() => fireAction(props, "click")}
724
+ onBlur={() => fireAction(props, "blur")}
725
+ className="mt-0.5 shrink-0"
726
+ />
727
+ <span>
728
+ {opt.label}
729
+ {opt.description && (
730
+ <div className="text-[11px] text-[var(--muted)] mt-px">
731
+ {opt.description}
732
+ </div>
733
+ )}
734
+ </span>
735
+ </label>
736
+ ))}
737
+ </div>
738
+ );
739
+ }
740
+
741
+ // ── 11. Multiselect ─────────────────────────────────────────────────────
742
+
743
+ /** Checkbox group for selecting multiple values from options. */
744
+ export function renderMultiselectField(props: FieldRenderProps) {
745
+ return <MultiselectFieldInner fp={props} />;
746
+ }
747
+
748
+ function MultiselectFieldInner({ fp: props }: { fp: FieldRenderProps }) {
749
+ const options: Array<{ value: string; label: string }> =
750
+ ((props.hint as Record<string, unknown>).options as
751
+ | typeof options
752
+ | undefined) ??
753
+ ((props.schema.items?.enum as string[]) ?? []).map((v) => ({
754
+ value: String(v),
755
+ label: String(v),
756
+ }));
757
+
758
+ const rawVal = props.isSet ? props.value : [];
759
+ const initialSet = new Set(Array.isArray(rawVal) ? rawVal.map(String) : []);
760
+ const [selected, setSelected] = useState(initialSet);
761
+
762
+ const toggle = (optValue: string) => {
763
+ setSelected((prev) => {
764
+ const next = new Set(prev);
765
+ if (next.has(optValue)) {
766
+ next.delete(optValue);
767
+ } else {
768
+ next.add(optValue);
769
+ }
770
+ props.onChange([...next]);
771
+ fireAction(props, "change");
772
+ return next;
773
+ });
774
+ };
775
+
776
+ const remove = (optValue: string) => {
777
+ setSelected((prev) => {
778
+ const next = new Set(prev);
779
+ next.delete(optValue);
780
+ props.onChange([...next]);
781
+ fireAction(props, "change");
782
+ return next;
783
+ });
784
+ };
785
+
786
+ const selectedOptions = options.filter((o) => selected.has(o.value));
787
+
788
+ return (
789
+ <div
790
+ className="flex flex-col gap-2"
791
+ data-config-key={props.key}
792
+ data-field-type="multiselect"
793
+ >
794
+ {/* Selected tag pills */}
795
+ {selectedOptions.length > 0 && (
796
+ <div className="flex flex-wrap gap-1">
797
+ {selectedOptions.map((opt) => (
798
+ <span
799
+ key={opt.value}
800
+ className="inline-flex items-center gap-1 px-2 py-0.5 text-[11px] bg-[var(--accent-subtle,rgba(59,130,246,0.1))] text-[var(--accent)] border border-[var(--accent)] border-opacity-30 rounded-full"
801
+ >
802
+ {opt.label}
803
+ {!props.readonly && (
804
+ <button
805
+ type="button"
806
+ className="inline-flex items-center justify-center w-3.5 h-3.5 text-[10px] rounded-full hover:bg-[var(--accent)] hover:text-[var(--accent-foreground,#1a1f26)] transition-colors cursor-pointer"
807
+ onClick={() => remove(opt.value)}
808
+ >
809
+ ×
810
+ </button>
811
+ )}
812
+ </span>
813
+ ))}
814
+ </div>
815
+ )}
816
+ {/* Checkbox list */}
817
+ <div className="flex flex-col gap-1">
818
+ {options.map((opt) => (
819
+ <label
820
+ key={opt.value}
821
+ className="flex items-center gap-2 cursor-pointer text-[13px]"
822
+ >
823
+ <input
824
+ type="checkbox"
825
+ value={opt.value}
826
+ checked={selected.has(opt.value)}
827
+ disabled={props.readonly}
828
+ onChange={() => toggle(opt.value)}
829
+ onClick={() => fireAction(props, "click")}
830
+ onBlur={() => fireAction(props, "blur")}
831
+ />
832
+ <span>{opt.label}</span>
833
+ </label>
834
+ ))}
835
+ </div>
836
+ </div>
837
+ );
838
+ }
839
+
840
+ // ── 12. Date ────────────────────────────────────────────────────────────
841
+
842
+ /** Native date picker input. */
843
+ export function renderDateField(props: FieldRenderProps) {
844
+ const value = props.isSet ? String(props.value ?? "") : "";
845
+ const inputType =
846
+ props.schema.format === "date-time" ? "datetime-local" : "date";
847
+
848
+ return (
849
+ <input
850
+ className={inputCls(!!props.errors?.length)}
851
+ type={inputType}
852
+ defaultValue={value}
853
+ data-config-key={props.key}
854
+ data-field-type="date"
855
+ disabled={props.readonly}
856
+ onChange={(e) => {
857
+ props.onChange(e.target.value);
858
+ fireAction(props, "change");
859
+ }}
860
+ onBlur={() => fireAction(props, "blur")}
861
+ onClick={() => fireAction(props, "click")}
862
+ />
863
+ );
864
+ }
865
+
866
+ // ── 13. JSON ────────────────────────────────────────────────────────────
867
+
868
+ /** JSON editor textarea with parse validation on blur. Shows inline error for invalid JSON. */
869
+ export function renderJsonField(props: FieldRenderProps) {
870
+ return <JsonFieldInner fp={props} />;
871
+ }
872
+
873
+ function JsonFieldInner({ fp: props }: { fp: FieldRenderProps }) {
874
+ const { t } = useApp();
875
+ const initial = props.isSet ? String(props.value ?? "") : "";
876
+ const [jsonError, setJsonError] = useState<string | null>(null);
877
+
878
+ const handleBlur = (e: React.FocusEvent<HTMLTextAreaElement>) => {
879
+ const val = e.target.value.trim();
880
+ if (val) {
881
+ try {
882
+ const parsed = JSON.parse(val);
883
+ const jsonStr = JSON.stringify(parsed);
884
+ if (/__proto__|constructor\s*:/.test(jsonStr)) {
885
+ setJsonError("Unsafe JSON: contains dangerous keys");
886
+ return;
887
+ }
888
+ setJsonError(null);
889
+ } catch (err) {
890
+ setJsonError(err instanceof Error ? err.message : "Invalid JSON");
891
+ }
892
+ } else {
893
+ setJsonError(null);
894
+ }
895
+ fireAction(props, "blur");
896
+ };
897
+
898
+ return (
899
+ <div>
900
+ <textarea
901
+ className={`w-full px-2.5 py-[7px] border ${
902
+ jsonError || props.errors?.length
903
+ ? "border-[var(--destructive)]"
904
+ : "border-[var(--border)]"
905
+ } bg-[var(--card)] text-[13px] font-mono transition-all focus:border-[var(--accent)] focus:outline-none focus:ring-1 focus:ring-[var(--accent)] box-border min-h-[100px] resize-y rounded-sm`}
906
+ defaultValue={initial}
907
+ placeholder={t("config-field.KeyValue")}
908
+ rows={6}
909
+ data-config-key={props.key}
910
+ data-field-type="json"
911
+ disabled={props.readonly}
912
+ onChange={(e) => {
913
+ props.onChange(e.target.value);
914
+ fireAction(props, "change");
915
+ }}
916
+ onBlur={handleBlur}
917
+ onClick={() => fireAction(props, "click")}
918
+ />
919
+ {jsonError && (
920
+ <div className="text-[11px] text-[var(--destructive)] mt-1 leading-snug">
921
+ {jsonError}
922
+ </div>
923
+ )}
924
+ </div>
925
+ );
926
+ }
927
+
928
+ // ── 14. Code ────────────────────────────────────────────────────────────
929
+
930
+ /** Monospaced code editor textarea for templates and snippets. */
931
+ export function renderCodeField(props: FieldRenderProps) {
932
+ const value = props.isSet ? String(props.value ?? "") : "";
933
+ const placeholder =
934
+ (props.hint.placeholder as string | undefined) ?? "Enter code...";
935
+
936
+ return (
937
+ <textarea
938
+ className={`w-full px-2.5 py-[7px] border ${
939
+ props.errors?.length
940
+ ? "border-[var(--destructive)]"
941
+ : "border-[var(--border)]"
942
+ } bg-[var(--card)] text-[13px] font-mono transition-all focus:border-[var(--accent)] focus:outline-none focus:ring-1 focus:ring-[var(--accent)] box-border min-h-[100px] resize-y rounded-sm`}
943
+ defaultValue={value}
944
+ placeholder={placeholder}
945
+ rows={6}
946
+ data-config-key={props.key}
947
+ data-field-type="code"
948
+ disabled={props.readonly}
949
+ onChange={(e) => {
950
+ props.onChange(e.target.value);
951
+ fireAction(props, "change");
952
+ }}
953
+ onBlur={() => fireAction(props, "blur")}
954
+ onClick={() => fireAction(props, "click")}
955
+ />
956
+ );
957
+ }
958
+
959
+ // ── 15. Array ───────────────────────────────────────────────────────────
960
+
961
+ /** Add/remove items list. Max 100 items. Parses comma-separated strings. */
962
+ export function renderArrayField(props: FieldRenderProps) {
963
+ return <ArrayFieldInner fp={props} />;
964
+ }
965
+
966
+ function ArrayItem({
967
+ index,
968
+ value,
969
+ total,
970
+ hasError,
971
+ readonly,
972
+ onChange,
973
+ onRemove,
974
+ onMoveUp,
975
+ onMoveDown,
976
+ onBlur,
977
+ }: {
978
+ index: number;
979
+ value: string;
980
+ total: number;
981
+ hasError: boolean;
982
+ readonly?: boolean;
983
+ onChange: (value: string) => void;
984
+ onRemove: () => void;
985
+ onMoveUp: () => void;
986
+ onMoveDown: () => void;
987
+ onBlur: () => void;
988
+ }) {
989
+ const { t } = useApp();
990
+ return (
991
+ <div className="flex items-center gap-1">
992
+ {!readonly && (
993
+ <div className="flex flex-col shrink-0">
994
+ <button
995
+ type="button"
996
+ className="px-1 py-0 text-[10px] leading-tight text-[var(--muted)] cursor-pointer hover:text-[var(--text)] disabled:opacity-30 disabled:cursor-not-allowed"
997
+ onClick={onMoveUp}
998
+ disabled={index === 0}
999
+ title={t("config-field.MoveUp")}
1000
+ >
1001
+
1002
+ </button>
1003
+ <button
1004
+ type="button"
1005
+ className="px-1 py-0 text-[10px] leading-tight text-[var(--muted)] cursor-pointer hover:text-[var(--text)] disabled:opacity-30 disabled:cursor-not-allowed"
1006
+ onClick={onMoveDown}
1007
+ disabled={index === total - 1}
1008
+ title={t("config-field.MoveDown")}
1009
+ >
1010
+ <ChevronDown className="w-3 h-3" />
1011
+ </button>
1012
+ </div>
1013
+ )}
1014
+ <input
1015
+ className={`${inputCls(hasError)} flex-1`}
1016
+ type="text"
1017
+ value={value}
1018
+ placeholder={`Item ${index + 1}`}
1019
+ disabled={readonly}
1020
+ onChange={(e) => onChange(e.target.value)}
1021
+ onBlur={onBlur}
1022
+ />
1023
+ {!readonly && (
1024
+ <button
1025
+ type="button"
1026
+ className="px-2 py-1.5 border border-[var(--border)] bg-[var(--bg-hover)] text-xs text-[var(--muted)] cursor-pointer transition-colors hover:bg-[var(--surface)] hover:text-[var(--destructive)] h-[36px] rounded-sm"
1027
+ onClick={onRemove}
1028
+ >
1029
+ <X className="w-3 h-3" />
1030
+ </button>
1031
+ )}
1032
+ </div>
1033
+ );
1034
+ }
1035
+
1036
+ function ArrayFieldInner({ fp: props }: { fp: FieldRenderProps }) {
1037
+ const { t } = useApp();
1038
+ const rawVal = props.isSet ? props.value : [];
1039
+ const initialItems: string[] = Array.isArray(rawVal)
1040
+ ? rawVal.map(String)
1041
+ : [];
1042
+ const [items, setItems] = useState<string[]>(initialItems);
1043
+
1044
+ const updateItems = (nextItems: string[]) => {
1045
+ setItems(nextItems);
1046
+ props.onChange(nextItems);
1047
+ fireAction(props, "change");
1048
+ };
1049
+
1050
+ const MAX_ARRAY_ITEMS = 100;
1051
+ const addItem = () => {
1052
+ if (items.length >= MAX_ARRAY_ITEMS) return;
1053
+ updateItems([...items, ""]);
1054
+ };
1055
+
1056
+ const removeItem = (index: number) => {
1057
+ updateItems(items.filter((_, i) => i !== index));
1058
+ };
1059
+
1060
+ const changeItem = (index: number, value: string) => {
1061
+ const next = [...items];
1062
+ next[index] = value;
1063
+ updateItems(next);
1064
+ };
1065
+
1066
+ const moveItem = (from: number, to: number) => {
1067
+ const next = [...items];
1068
+ const [moved] = next.splice(from, 1);
1069
+ next.splice(to, 0, moved);
1070
+ updateItems(next);
1071
+ };
1072
+
1073
+ return (
1074
+ <div
1075
+ className="flex flex-col gap-1.5"
1076
+ data-config-key={props.key}
1077
+ data-field-type="array"
1078
+ >
1079
+ {items.map((item, index) => (
1080
+ <ArrayItem
1081
+ key={`${index}-${items.length}`}
1082
+ index={index}
1083
+ value={item}
1084
+ total={items.length}
1085
+ hasError={!!props.errors?.length}
1086
+ readonly={props.readonly}
1087
+ onChange={(v) => changeItem(index, v)}
1088
+ onRemove={() => {
1089
+ removeItem(index);
1090
+ fireAction(props, "click");
1091
+ }}
1092
+ onMoveUp={() => moveItem(index, index - 1)}
1093
+ onMoveDown={() => moveItem(index, index + 1)}
1094
+ onBlur={() => fireAction(props, "blur")}
1095
+ />
1096
+ ))}
1097
+ {!props.readonly && (
1098
+ <button
1099
+ type="button"
1100
+ className="self-start px-3 py-1.5 border border-dashed border-[var(--border)] bg-transparent text-[11px] text-[var(--muted)] cursor-pointer transition-colors hover:bg-[var(--bg-hover)] hover:text-[var(--text)] hover:border-[var(--accent)] rounded-sm"
1101
+ onClick={() => {
1102
+ addItem();
1103
+ fireAction(props, "click");
1104
+ }}
1105
+ >
1106
+ {t("config-field.AddItem")}
1107
+ </button>
1108
+ )}
1109
+ </div>
1110
+ );
1111
+ }
1112
+
1113
+ // ── 16. Key-Value ───────────────────────────────────────────────────────
1114
+
1115
+ /** Key-value pair editor with add/remove rows. Blocks prototype pollution keys. */
1116
+ export function renderKeyValueField(props: FieldRenderProps) {
1117
+ return <KeyValueFieldInner fp={props} />;
1118
+ }
1119
+
1120
+ function KeyValueFieldInner({ fp: props }: { fp: FieldRenderProps }) {
1121
+ const { t } = useApp();
1122
+ const rawVal = props.isSet ? props.value : {};
1123
+ const initialPairs: Array<{ key: string; value: string }> =
1124
+ rawVal && typeof rawVal === "object" && !Array.isArray(rawVal)
1125
+ ? Object.entries(rawVal as Record<string, unknown>).map(([k, v]) => ({
1126
+ key: k,
1127
+ value: String(v ?? ""),
1128
+ }))
1129
+ : [];
1130
+ const [pairs, setPairs] = useState(initialPairs);
1131
+
1132
+ const emitChange = (next: Array<{ key: string; value: string }>) => {
1133
+ setPairs(next);
1134
+ const obj: Record<string, string> = {};
1135
+ for (const p of next) {
1136
+ if (p.key) obj[p.key] = p.value;
1137
+ }
1138
+ props.onChange(obj);
1139
+ fireAction(props, "change");
1140
+ };
1141
+
1142
+ const addRow = () => {
1143
+ emitChange([...pairs, { key: "", value: "" }]);
1144
+ };
1145
+
1146
+ const removeRow = (index: number) => {
1147
+ emitChange(pairs.filter((_, i) => i !== index));
1148
+ };
1149
+
1150
+ const BLOCKED_KEYS = ["__proto__", "constructor", "prototype"];
1151
+ const updateRow = (index: number, field: "key" | "value", val: string) => {
1152
+ if (field === "key" && BLOCKED_KEYS.includes(val)) return;
1153
+ const next = [...pairs];
1154
+ next[index] = { ...next[index], [field]: val };
1155
+ emitChange(next);
1156
+ };
1157
+
1158
+ return (
1159
+ <div
1160
+ className="flex flex-col gap-1.5"
1161
+ data-config-key={props.key}
1162
+ data-field-type="keyvalue"
1163
+ >
1164
+ {pairs.map((pair, index) => (
1165
+ <div
1166
+ key={`${pair.key}:${pair.value}`}
1167
+ className="flex items-center gap-1"
1168
+ >
1169
+ <input
1170
+ className={`${inputCls(!!props.errors?.length)} flex-1`}
1171
+ type="text"
1172
+ value={pair.key}
1173
+ placeholder={t("config-field.Key")}
1174
+ disabled={props.readonly}
1175
+ onChange={(e) => updateRow(index, "key", e.target.value)}
1176
+ onBlur={() => fireAction(props, "blur")}
1177
+ />
1178
+ <input
1179
+ className={`${inputCls(!!props.errors?.length)} flex-1`}
1180
+ type="text"
1181
+ value={pair.value}
1182
+ placeholder={t("config-field.Value")}
1183
+ disabled={props.readonly}
1184
+ onChange={(e) => updateRow(index, "value", e.target.value)}
1185
+ onBlur={() => fireAction(props, "blur")}
1186
+ />
1187
+ {!props.readonly && (
1188
+ <button
1189
+ type="button"
1190
+ className="px-2 py-1.5 border border-[var(--border)] bg-[var(--bg-hover)] text-xs text-[var(--muted)] cursor-pointer transition-colors hover:bg-[var(--surface)] hover:text-[var(--destructive)] h-[36px] rounded-sm"
1191
+ onClick={() => {
1192
+ removeRow(index);
1193
+ fireAction(props, "click");
1194
+ }}
1195
+ >
1196
+ <X className="w-3 h-3" />
1197
+ </button>
1198
+ )}
1199
+ </div>
1200
+ ))}
1201
+ {!props.readonly && (
1202
+ <button
1203
+ type="button"
1204
+ className="self-start px-3 py-1.5 border border-dashed border-[var(--border)] bg-transparent text-[11px] text-[var(--muted)] cursor-pointer transition-colors hover:bg-[var(--bg-hover)] hover:text-[var(--text)] hover:border-[var(--accent)] rounded-sm"
1205
+ onClick={() => {
1206
+ addRow();
1207
+ fireAction(props, "click");
1208
+ }}
1209
+ >
1210
+ {t("config-field.AddRow")}
1211
+ </button>
1212
+ )}
1213
+ </div>
1214
+ );
1215
+ }
1216
+
1217
+ // ── 17. Datetime ────────────────────────────────────────────────────────
1218
+
1219
+ /** Combined date and time picker (datetime-local). */
1220
+ export function renderDatetimeField(props: FieldRenderProps) {
1221
+ const value = props.isSet ? String(props.value ?? "") : "";
1222
+
1223
+ return (
1224
+ <input
1225
+ className={inputCls(!!props.errors?.length)}
1226
+ type="datetime-local"
1227
+ defaultValue={value}
1228
+ data-config-key={props.key}
1229
+ data-field-type="datetime"
1230
+ disabled={props.readonly}
1231
+ onChange={(e) => {
1232
+ props.onChange(e.target.value);
1233
+ fireAction(props, "change");
1234
+ }}
1235
+ onBlur={() => fireAction(props, "blur")}
1236
+ onClick={() => fireAction(props, "click")}
1237
+ />
1238
+ );
1239
+ }
1240
+
1241
+ // ── 18. File ────────────────────────────────────────────────────────────
1242
+
1243
+ /** File path text input with path traversal guard. */
1244
+ export function RenderFileField(props: FieldRenderProps) {
1245
+ const { t } = useApp();
1246
+ const value = props.isSet ? String(props.value ?? "") : "";
1247
+ const placeholder =
1248
+ (props.hint.placeholder as string | undefined) ?? "/path/to/file";
1249
+
1250
+ return (
1251
+ <div>
1252
+ <input
1253
+ className={inputCls(!!props.errors?.length)}
1254
+ type="text"
1255
+ defaultValue={value}
1256
+ placeholder={placeholder}
1257
+ data-config-key={props.key}
1258
+ data-field-type="file"
1259
+ disabled={props.readonly}
1260
+ onChange={(e) => {
1261
+ const v = e.target.value;
1262
+ if (v.includes("..")) return; // block path traversal
1263
+ props.onChange(v);
1264
+ fireAction(props, "change");
1265
+ }}
1266
+ onBlur={() => fireAction(props, "blur")}
1267
+ onClick={() => fireAction(props, "click")}
1268
+ />
1269
+ <div className="text-[11px] text-[var(--muted)] mt-0.5">
1270
+ {t("config-field.EnterAFilePathOr")}
1271
+ </div>
1272
+ </div>
1273
+ );
1274
+ }
1275
+
1276
+ // ── 19. Custom ──────────────────────────────────────────────────────────
1277
+
1278
+ /** Placeholder for plugin-provided custom React components. */
1279
+ export function RenderCustomField(props: FieldRenderProps) {
1280
+ const { t } = useApp();
1281
+ const componentName = (props.hint as Record<string, unknown>).component as
1282
+ | string
1283
+ | undefined;
1284
+
1285
+ return (
1286
+ <div
1287
+ className="px-3 py-4 border border-dashed border-[var(--border)] bg-[var(--bg-hover)] text-[13px] text-[var(--muted)]"
1288
+ data-config-key={props.key}
1289
+ data-field-type="custom"
1290
+ >
1291
+ {t("config-field.CustomComponent")} {componentName ?? props.fieldType}
1292
+ </div>
1293
+ );
1294
+ }
1295
+
1296
+ // ── 20. Markdown ─────────────────────────────────────────────────────────
1297
+
1298
+ /**
1299
+ * Simple markdown renderer for preview mode.
1300
+ * Converts basic markdown patterns to React elements without external dependencies.
1301
+ */
1302
+ function renderMarkdown(text: string): React.ReactNode {
1303
+ if (!text) return null;
1304
+
1305
+ const blocks = text.split(/\n\n+/);
1306
+ const elements: React.ReactNode[] = [];
1307
+
1308
+ blocks.forEach((block) => {
1309
+ const trimmed = block.trim();
1310
+ if (!trimmed) return;
1311
+
1312
+ // Fenced code block
1313
+ if (trimmed.startsWith("```")) {
1314
+ const lines = trimmed.split("\n");
1315
+ const code = lines.slice(1, -1).join("\n");
1316
+ elements.push(
1317
+ <pre
1318
+ key={`code:${code}`}
1319
+ className="bg-[var(--bg-hover)] px-3 py-2 rounded-sm overflow-x-auto my-2"
1320
+ >
1321
+ <code className="font-mono text-[12px]">{code}</code>
1322
+ </pre>,
1323
+ );
1324
+ return;
1325
+ }
1326
+
1327
+ // Heading
1328
+ const headingMatch = trimmed.match(/^(#{1,6})\s+(.+)$/);
1329
+ if (headingMatch) {
1330
+ const level = headingMatch[1].length;
1331
+ const content = processInline(headingMatch[2]);
1332
+ const sizes = [
1333
+ "text-xl",
1334
+ "text-lg",
1335
+ "text-base",
1336
+ "text-sm",
1337
+ "text-sm",
1338
+ "text-xs",
1339
+ ];
1340
+ const cls = `${sizes[level - 1]} font-bold mb-1 mt-2`;
1341
+ elements.push(
1342
+ React.createElement(
1343
+ `h${level}`,
1344
+ { key: `heading:${headingMatch[2]}`, className: cls },
1345
+ content,
1346
+ ),
1347
+ );
1348
+ return;
1349
+ }
1350
+
1351
+ // Unordered list
1352
+ if (/^[-*]\s/.test(trimmed)) {
1353
+ const items = trimmed.split("\n").filter((l) => /^[-*]\s/.test(l));
1354
+ elements.push(
1355
+ <ul key={`list:${trimmed}`} className="list-disc pl-4 my-2 space-y-1">
1356
+ {items.map((item) => (
1357
+ <li key={item}>{processInline(item.replace(/^[-*]\s/, ""))}</li>
1358
+ ))}
1359
+ </ul>,
1360
+ );
1361
+ return;
1362
+ }
1363
+
1364
+ // Regular paragraph
1365
+ elements.push(
1366
+ <p key={`paragraph:${trimmed}`} className="my-2">
1367
+ {processInline(trimmed)}
1368
+ </p>,
1369
+ );
1370
+ });
1371
+
1372
+ return <>{elements}</>;
1373
+ }
1374
+
1375
+ /**
1376
+ * Process inline markdown patterns (bold, italic, code, links).
1377
+ */
1378
+ function processInline(text: string): React.ReactNode {
1379
+ const parts: React.ReactNode[] = [];
1380
+ let remaining = text;
1381
+ let key = 0;
1382
+
1383
+ while (remaining) {
1384
+ // Link: [text](url)
1385
+ const linkMatch = remaining.match(/\[([^\]]+)\]\(([^)]+)\)/);
1386
+ if (linkMatch) {
1387
+ const before = remaining.substring(0, linkMatch.index);
1388
+ if (before) parts.push(processSimpleInline(before, key++));
1389
+ parts.push(
1390
+ <a
1391
+ key={key++}
1392
+ href={linkMatch[2]}
1393
+ target="_blank"
1394
+ rel="noopener noreferrer"
1395
+ className="text-[var(--accent)] underline"
1396
+ >
1397
+ {linkMatch[1]}
1398
+ </a>,
1399
+ );
1400
+ const linkIndex = linkMatch.index ?? 0;
1401
+ remaining = remaining.substring(linkIndex + linkMatch[0].length);
1402
+ continue;
1403
+ }
1404
+
1405
+ // Inline code: `code`
1406
+ const codeMatch = remaining.match(/`([^`]+)`/);
1407
+ if (codeMatch) {
1408
+ const before = remaining.substring(0, codeMatch.index);
1409
+ if (before) parts.push(processSimpleInline(before, key++));
1410
+ parts.push(
1411
+ <code
1412
+ key={key++}
1413
+ className="bg-[var(--bg-hover)] px-1 py-0.5 rounded font-mono text-[12px]"
1414
+ >
1415
+ {codeMatch[1]}
1416
+ </code>,
1417
+ );
1418
+ const codeIndex = codeMatch.index ?? 0;
1419
+ remaining = remaining.substring(codeIndex + codeMatch[0].length);
1420
+ continue;
1421
+ }
1422
+
1423
+ // Bold: **text**
1424
+ const boldMatch = remaining.match(/\*\*([^*]+)\*\*/);
1425
+ if (boldMatch) {
1426
+ const before = remaining.substring(0, boldMatch.index);
1427
+ if (before) parts.push(processSimpleInline(before, key++));
1428
+ parts.push(<strong key={key++}>{boldMatch[1]}</strong>);
1429
+ const boldIndex = boldMatch.index ?? 0;
1430
+ remaining = remaining.substring(boldIndex + boldMatch[0].length);
1431
+ continue;
1432
+ }
1433
+
1434
+ // Italic: *text* or _text_
1435
+ const italicMatch = remaining.match(/(\*|_)([^*_]+)\1/);
1436
+ if (italicMatch) {
1437
+ const before = remaining.substring(0, italicMatch.index);
1438
+ if (before) parts.push(processSimpleInline(before, key++));
1439
+ parts.push(<em key={key++}>{italicMatch[2]}</em>);
1440
+ const italicIndex = italicMatch.index ?? 0;
1441
+ remaining = remaining.substring(italicIndex + italicMatch[0].length);
1442
+ continue;
1443
+ }
1444
+
1445
+ // No more patterns found
1446
+ parts.push(remaining);
1447
+ break;
1448
+ }
1449
+
1450
+ return <>{parts}</>;
1451
+ }
1452
+
1453
+ /**
1454
+ * Helper to handle plain text without additional patterns.
1455
+ */
1456
+ function processSimpleInline(text: string, key: number): React.ReactNode {
1457
+ return <span key={key}>{text}</span>;
1458
+ }
1459
+
1460
+ function MarkdownFieldInner(props: FieldRenderProps) {
1461
+ const { t } = useApp();
1462
+ const [preview, setPreview] = useState(false);
1463
+ const value = typeof props.value === "string" ? props.value : "";
1464
+
1465
+ return (
1466
+ <div className="flex flex-col gap-1.5">
1467
+ <div className="flex items-center gap-2 mb-1">
1468
+ <button
1469
+ type="button"
1470
+ className={`text-[11px] px-2 py-0.5 border transition-colors ${
1471
+ !preview
1472
+ ? "bg-[var(--accent)] text-[var(--accent-fg)] border-[var(--accent)]"
1473
+ : "bg-transparent text-[var(--muted)] border-[var(--border)] hover:text-[var(--txt)]"
1474
+ }`}
1475
+ onClick={() => setPreview(false)}
1476
+ >
1477
+ {t("config-field.Edit")}
1478
+ </button>
1479
+ <button
1480
+ type="button"
1481
+ className={`text-[11px] px-2 py-0.5 border transition-colors ${
1482
+ preview
1483
+ ? "bg-[var(--accent)] text-[var(--accent-fg)] border-[var(--accent)]"
1484
+ : "bg-transparent text-[var(--muted)] border-[var(--border)] hover:text-[var(--txt)]"
1485
+ }`}
1486
+ onClick={() => setPreview(true)}
1487
+ >
1488
+ {t("config-field.Preview")}
1489
+ </button>
1490
+ </div>
1491
+ {preview ? (
1492
+ <div
1493
+ className="min-h-[100px] px-3 py-2 border border-[var(--border)] bg-[var(--surface)] text-[13px] leading-relaxed"
1494
+ data-config-key={props.key}
1495
+ data-field-type="markdown"
1496
+ >
1497
+ {value ? (
1498
+ renderMarkdown(value)
1499
+ ) : (
1500
+ <span className="text-[var(--muted)] italic">
1501
+ {t("config-field.NothingToPreview")}
1502
+ </span>
1503
+ )}
1504
+ </div>
1505
+ ) : (
1506
+ <textarea
1507
+ className={`${inputCls(!!props.errors?.length)} min-h-[100px] h-auto resize-y font-[var(--mono)]`}
1508
+ defaultValue={value}
1509
+ placeholder={props.hint.placeholder ?? "Markdown content..."}
1510
+ data-config-key={props.key}
1511
+ data-field-type="markdown"
1512
+ disabled={props.readonly}
1513
+ onChange={(e) => {
1514
+ props.onChange(e.target.value);
1515
+ fireAction(props, "change");
1516
+ }}
1517
+ onBlur={() => fireAction(props, "blur")}
1518
+ />
1519
+ )}
1520
+ </div>
1521
+ );
1522
+ }
1523
+ /** Markdown textarea with Edit/Preview toggle. */
1524
+ export const renderMarkdownField: FieldRenderer = (props) => (
1525
+ <MarkdownFieldInner {...props} />
1526
+ );
1527
+
1528
+ // ── 21. Checkbox Group ───────────────────────────────────────────────────
1529
+
1530
+ function CheckboxGroupInner(props: FieldRenderProps) {
1531
+ const { t } = useApp();
1532
+ const selected = new Set(
1533
+ Array.isArray(props.value)
1534
+ ? (props.value as string[])
1535
+ : typeof props.value === "string" && props.value
1536
+ ? props.value.split(",").map((s) => s.trim())
1537
+ : [],
1538
+ );
1539
+ const options: Array<{
1540
+ value: string;
1541
+ label: string;
1542
+ description?: string;
1543
+ disabled?: boolean;
1544
+ }> =
1545
+ props.hint.options ??
1546
+ (props.schema.enum as string[] | undefined)?.map((v) => ({
1547
+ value: v,
1548
+ label: v,
1549
+ })) ??
1550
+ [];
1551
+
1552
+ const toggle = (val: string) => {
1553
+ const next = new Set(selected);
1554
+ if (next.has(val)) next.delete(val);
1555
+ else next.add(val);
1556
+ props.onChange([...next]);
1557
+ fireAction(props, "change");
1558
+ };
1559
+
1560
+ return (
1561
+ <div
1562
+ className="flex flex-col gap-1.5"
1563
+ data-config-key={props.key}
1564
+ data-field-type="checkbox-group"
1565
+ >
1566
+ {options.map((opt) => (
1567
+ <label
1568
+ key={opt.value}
1569
+ className={`flex items-start gap-2.5 px-3 py-2 border border-[var(--border)] bg-[var(--card)] cursor-pointer transition-colors hover:bg-[var(--bg-hover)] ${
1570
+ selected.has(opt.value)
1571
+ ? "border-[var(--accent)] bg-[color-mix(in_srgb,var(--accent)_5%,var(--card))]"
1572
+ : ""
1573
+ } ${opt.disabled ? "opacity-50 pointer-events-none" : ""}`}
1574
+ >
1575
+ <input
1576
+ type="checkbox"
1577
+ checked={selected.has(opt.value)}
1578
+ disabled={props.readonly || opt.disabled}
1579
+ onChange={() => toggle(opt.value)}
1580
+ className="mt-0.5 accent-[var(--accent)]"
1581
+ />
1582
+ <div className="flex flex-col">
1583
+ <span className="text-[13px]">{opt.label}</span>
1584
+ {opt.description && (
1585
+ <span className="text-[11px] text-[var(--muted)] mt-0.5">
1586
+ {opt.description}
1587
+ </span>
1588
+ )}
1589
+ </div>
1590
+ </label>
1591
+ ))}
1592
+ {options.length === 0 && (
1593
+ <span className="text-[11px] text-[var(--muted)] italic">
1594
+ {t("config-field.NoOptionsDefined")}
1595
+ </span>
1596
+ )}
1597
+ </div>
1598
+ );
1599
+ }
1600
+ /** Vertical checkbox list with per-option descriptions and accent highlighting. */
1601
+ export const renderCheckboxGroupField: FieldRenderer = (props) => (
1602
+ <CheckboxGroupInner {...props} />
1603
+ );
1604
+
1605
+ // ── 22. Group ────────────────────────────────────────────────────────────
1606
+
1607
+ /** Fieldset container with legend label. */
1608
+ export const renderGroupField: FieldRenderer = (props) => {
1609
+ const value = typeof props.value === "string" ? props.value : "";
1610
+ return (
1611
+ <fieldset
1612
+ className="border border-[var(--border)] px-4 py-3 bg-[var(--surface)]"
1613
+ data-config-key={props.key}
1614
+ data-field-type="group"
1615
+ >
1616
+ <legend className="text-[12px] font-semibold text-[var(--muted)] px-1.5">
1617
+ {props.hint.label ?? props.key}
1618
+ </legend>
1619
+ <textarea
1620
+ className={`${inputCls(!!props.errors?.length)} min-h-[60px] h-auto resize-y`}
1621
+ defaultValue={value}
1622
+ placeholder={props.hint.placeholder ?? "Group configuration..."}
1623
+ disabled={props.readonly}
1624
+ onChange={(e) => {
1625
+ props.onChange(e.target.value);
1626
+ fireAction(props, "change");
1627
+ }}
1628
+ onBlur={() => fireAction(props, "blur")}
1629
+ />
1630
+ </fieldset>
1631
+ );
1632
+ };
1633
+
1634
+ // ── 23. Table ────────────────────────────────────────────────────────────
1635
+
1636
+ function TableFieldInner(props: FieldRenderProps) {
1637
+ const { t } = useApp();
1638
+ const MAX_TABLE_ROWS = 50;
1639
+ const columns: Array<{ key: string; label: string }> = ((
1640
+ props.hint as Record<string, unknown>
1641
+ ).columns as Array<{ key: string; label: string }>) ?? [
1642
+ { key: "key", label: "Key" },
1643
+ { key: "value", label: "Value" },
1644
+ ];
1645
+
1646
+ const rawRows = Array.isArray(props.value)
1647
+ ? (props.value as Record<string, string>[])
1648
+ : [];
1649
+ const [rows, setRows] = useState<Record<string, string>[]>(
1650
+ rawRows.length > 0
1651
+ ? rawRows
1652
+ : [Object.fromEntries(columns.map((c) => [c.key, ""]))],
1653
+ );
1654
+
1655
+ const emit = (next: Record<string, string>[]) => {
1656
+ setRows(next);
1657
+ props.onChange(next.filter((r) => columns.some((c) => r[c.key]?.trim())));
1658
+ fireAction(props, "change");
1659
+ };
1660
+
1661
+ const updateCell = (rowIdx: number, colKey: string, val: string) => {
1662
+ if (["__proto__", "constructor", "prototype"].includes(colKey)) return;
1663
+ const next = [...rows];
1664
+ next[rowIdx] = { ...next[rowIdx], [colKey]: val };
1665
+ emit(next);
1666
+ };
1667
+
1668
+ const addRow = () => {
1669
+ if (rows.length >= MAX_TABLE_ROWS) return;
1670
+ emit([...rows, Object.fromEntries(columns.map((c) => [c.key, ""]))]);
1671
+ };
1672
+
1673
+ const removeRow = (idx: number) => {
1674
+ if (rows.length <= 1) return;
1675
+ emit(rows.filter((_, i) => i !== idx));
1676
+ };
1677
+
1678
+ return (
1679
+ <div
1680
+ className="flex flex-col gap-1.5"
1681
+ data-config-key={props.key}
1682
+ data-field-type="table"
1683
+ >
1684
+ <div className="border border-[var(--border)] overflow-x-auto">
1685
+ <table className="w-full text-[13px] border-collapse">
1686
+ <thead>
1687
+ <tr className="bg-[var(--surface)]">
1688
+ {columns.map((col) => (
1689
+ <th
1690
+ key={col.key}
1691
+ className="text-left text-[11px] font-semibold text-[var(--muted)] px-3 py-1.5 border-b border-[var(--border)]"
1692
+ >
1693
+ {col.label}
1694
+ </th>
1695
+ ))}
1696
+ <th className="w-[36px] border-b border-[var(--border)]" />
1697
+ </tr>
1698
+ </thead>
1699
+ <tbody>
1700
+ {rows.map((row, ri) => (
1701
+ <tr
1702
+ key={JSON.stringify(row)}
1703
+ className="border-b border-[var(--border)] last:border-b-0"
1704
+ >
1705
+ {columns.map((col) => (
1706
+ <td key={col.key} className="px-1 py-0.5">
1707
+ <input
1708
+ className="w-full px-2 py-1 bg-transparent text-[13px] border-none outline-none focus:bg-[var(--bg-hover)]"
1709
+ value={row[col.key] ?? ""}
1710
+ placeholder={col.label}
1711
+ disabled={props.readonly}
1712
+ onChange={(e) => updateCell(ri, col.key, e.target.value)}
1713
+ />
1714
+ </td>
1715
+ ))}
1716
+ <td className="text-center">
1717
+ {!props.readonly && rows.length > 1 && (
1718
+ <button
1719
+ type="button"
1720
+ className="text-[var(--muted)] hover:text-[var(--destructive)] text-[14px] px-1"
1721
+ onClick={() => removeRow(ri)}
1722
+ title={t("config-field.RemoveRow")}
1723
+ >
1724
+ {t("config-field.Times")}
1725
+ </button>
1726
+ )}
1727
+ </td>
1728
+ </tr>
1729
+ ))}
1730
+ </tbody>
1731
+ </table>
1732
+ </div>
1733
+ {!props.readonly && rows.length < MAX_TABLE_ROWS && (
1734
+ <button
1735
+ type="button"
1736
+ className="self-start text-[11px] text-[var(--accent)] hover:underline"
1737
+ onClick={addRow}
1738
+ >
1739
+ {t("config-field.AddRow")}
1740
+ </button>
1741
+ )}
1742
+ </div>
1743
+ );
1744
+ }
1745
+ /** Tabular data editor with configurable columns. Max 50 rows. */
1746
+ export const renderTableField: FieldRenderer = (props) => (
1747
+ <TableFieldInner {...props} />
1748
+ );
1749
+
1750
+ // ── Default renderers map ───────────────────────────────────────────────
1751
+
1752
+ export const defaultRenderers: Record<string, FieldRenderer> = {
1753
+ text: renderTextField,
1754
+ password: renderPasswordField,
1755
+ number: renderNumberField,
1756
+ boolean: renderBooleanField,
1757
+ url: renderUrlField,
1758
+ select: RenderSelectField,
1759
+ textarea: renderTextareaField,
1760
+ email: renderEmailField,
1761
+ color: renderColorField,
1762
+ radio: renderRadioField,
1763
+ multiselect: renderMultiselectField,
1764
+ date: renderDateField,
1765
+ json: renderJsonField,
1766
+ code: renderCodeField,
1767
+ array: renderArrayField,
1768
+ keyvalue: renderKeyValueField,
1769
+ datetime: renderDatetimeField,
1770
+ file: RenderFileField,
1771
+ custom: RenderCustomField,
1772
+ markdown: renderMarkdownField,
1773
+ "checkbox-group": renderCheckboxGroupField,
1774
+ group: renderGroupField,
1775
+ table: renderTableField,
1776
+ };
1777
+
1778
+ // ── ConfigField wrapper component ───────────────────────────────────────
1779
+
1780
+ /**
1781
+ * Wraps a field renderer with the standard label row, env key display,
1782
+ * help text, and error messages.
1783
+ */
1784
+ export function ConfigField({
1785
+ renderProps,
1786
+ renderer,
1787
+ pluginId,
1788
+ }: {
1789
+ renderProps: FieldRenderProps;
1790
+ renderer: FieldRenderer;
1791
+ pluginId?: string;
1792
+ }) {
1793
+ const { t } = useApp();
1794
+ const label = renderProps.hint.label ?? renderProps.key;
1795
+ const envKey = renderProps.key;
1796
+ const labelDiffersFromKey = label !== envKey;
1797
+ const errors = renderProps.errors ?? [];
1798
+ const hasError = errors.length > 0;
1799
+ const isRequiredEmpty = renderProps.required && !renderProps.isSet;
1800
+
1801
+ const renderFn =
1802
+ renderer ??
1803
+ defaultRenderers[renderProps.fieldType] ??
1804
+ defaultRenderers.text;
1805
+
1806
+ return (
1807
+ <div
1808
+ id={
1809
+ pluginId
1810
+ ? `field-${pluginId}-${renderProps.key}`
1811
+ : `field-${renderProps.key}`
1812
+ }
1813
+ className={`py-2.5 group/field ${
1814
+ renderProps.readonly ? "opacity-50 pointer-events-none" : ""
1815
+ } ${isRequiredEmpty ? "relative" : ""}`}
1816
+ >
1817
+ {/* Required-but-empty accent bar */}
1818
+ {isRequiredEmpty && (
1819
+ <div className="absolute left-0 top-2.5 bottom-2.5 w-[2px] bg-[var(--destructive)] opacity-40 rounded-full" />
1820
+ )}
1821
+
1822
+ <div className={isRequiredEmpty ? "pl-2.5" : ""}>
1823
+ {/* Label row */}
1824
+ <div className="flex items-center gap-2 mb-1.5">
1825
+ <span
1826
+ className="font-semibold leading-tight truncate"
1827
+ style={{
1828
+ fontSize: "var(--plugin-label-size)",
1829
+ color: "var(--plugin-label)",
1830
+ }}
1831
+ >
1832
+ {label}
1833
+ </span>
1834
+ {renderProps.required && !renderProps.isSet && (
1835
+ <span className="text-[10px] text-[var(--destructive)] font-semibold px-1.5 py-px bg-[color-mix(in_srgb,var(--destructive)_10%,transparent)] rounded-sm shrink-0">
1836
+ {t("config-field.Required")}
1837
+ </span>
1838
+ )}
1839
+ {renderProps.isSet && (
1840
+ <span className="inline-flex items-center gap-1 text-[10px] text-[var(--ok)] font-medium shrink-0">
1841
+ <span className="inline-block w-1.5 h-1.5 rounded-full bg-[var(--ok)]" />
1842
+
1843
+ {t("config-field.Configured")}
1844
+ </span>
1845
+ )}
1846
+ {/* Env key — right-aligned, subtle, only when label differs */}
1847
+ {labelDiffersFromKey && (
1848
+ <code className="text-[10px] font-mono text-[var(--muted)] opacity-0 group-hover/field:opacity-50 transition-opacity truncate ml-auto">
1849
+ {envKey}
1850
+ </code>
1851
+ )}
1852
+ </div>
1853
+
1854
+ {/* Field renderer */}
1855
+ {renderFn(renderProps)}
1856
+
1857
+ {/* Errors */}
1858
+ {hasError && (
1859
+ <div className="mt-1.5 flex flex-col gap-0.5">
1860
+ {errors.map((err) => (
1861
+ <div
1862
+ key={err}
1863
+ className="leading-snug flex items-start gap-1"
1864
+ style={{
1865
+ fontSize: "var(--plugin-error-size)",
1866
+ color: "var(--plugin-error)",
1867
+ }}
1868
+ >
1869
+ <span className="shrink-0 mt-px">
1870
+ {t("config-field.Times")}
1871
+ </span>
1872
+ <span>{err}</span>
1873
+ </div>
1874
+ ))}
1875
+ </div>
1876
+ )}
1877
+
1878
+ {/* Help text */}
1879
+ {(renderProps.hint.help || renderProps.schema.description) && (
1880
+ <div
1881
+ className="mt-1 leading-relaxed line-clamp-2"
1882
+ style={{
1883
+ fontSize: "var(--plugin-help-size)",
1884
+ color: "var(--plugin-help)",
1885
+ }}
1886
+ >
1887
+ {renderProps.hint.help ?? renderProps.schema.description}
1888
+ {renderProps.schema.default != null && (
1889
+ <span className="opacity-70">
1890
+ {" "}
1891
+ {t("config-field.Default")} {String(renderProps.schema.default)}
1892
+ )
1893
+ </span>
1894
+ )}
1895
+ </div>
1896
+ )}
1897
+ </div>
1898
+ </div>
1899
+ );
1900
+ }