@frontmcp/ui 0.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (393) hide show
  1. package/LICENSE +201 -0
  2. package/README.md +438 -0
  3. package/package.json +147 -0
  4. package/src/adapters/index.d.ts +10 -0
  5. package/src/adapters/index.js +18 -0
  6. package/src/adapters/index.js.map +1 -0
  7. package/src/adapters/platform-meta.d.ts +165 -0
  8. package/src/adapters/platform-meta.js +310 -0
  9. package/src/adapters/platform-meta.js.map +1 -0
  10. package/src/base-template/bridge.d.ts +89 -0
  11. package/src/base-template/bridge.js +452 -0
  12. package/src/base-template/bridge.js.map +1 -0
  13. package/src/base-template/default-base-template.d.ts +91 -0
  14. package/src/base-template/default-base-template.js +435 -0
  15. package/src/base-template/default-base-template.js.map +1 -0
  16. package/src/base-template/index.d.ts +14 -0
  17. package/src/base-template/index.js +30 -0
  18. package/src/base-template/index.js.map +1 -0
  19. package/src/base-template/polyfills.d.ts +30 -0
  20. package/src/base-template/polyfills.js +190 -0
  21. package/src/base-template/polyfills.js.map +1 -0
  22. package/src/base-template/theme-styles.d.ts +73 -0
  23. package/src/base-template/theme-styles.js +95 -0
  24. package/src/base-template/theme-styles.js.map +1 -0
  25. package/src/bridge/adapters/base-adapter.d.ts +103 -0
  26. package/src/bridge/adapters/base-adapter.js +314 -0
  27. package/src/bridge/adapters/base-adapter.js.map +1 -0
  28. package/src/bridge/adapters/claude.adapter.d.ts +66 -0
  29. package/src/bridge/adapters/claude.adapter.js +145 -0
  30. package/src/bridge/adapters/claude.adapter.js.map +1 -0
  31. package/src/bridge/adapters/ext-apps.adapter.d.ts +142 -0
  32. package/src/bridge/adapters/ext-apps.adapter.js +416 -0
  33. package/src/bridge/adapters/ext-apps.adapter.js.map +1 -0
  34. package/src/bridge/adapters/gemini.adapter.d.ts +63 -0
  35. package/src/bridge/adapters/gemini.adapter.js +160 -0
  36. package/src/bridge/adapters/gemini.adapter.js.map +1 -0
  37. package/src/bridge/adapters/generic.adapter.d.ts +55 -0
  38. package/src/bridge/adapters/generic.adapter.js +108 -0
  39. package/src/bridge/adapters/generic.adapter.js.map +1 -0
  40. package/src/bridge/adapters/index.d.ts +25 -0
  41. package/src/bridge/adapters/index.js +65 -0
  42. package/src/bridge/adapters/index.js.map +1 -0
  43. package/src/bridge/adapters/openai.adapter.d.ts +64 -0
  44. package/src/bridge/adapters/openai.adapter.js +194 -0
  45. package/src/bridge/adapters/openai.adapter.js.map +1 -0
  46. package/src/bridge/core/adapter-registry.d.ts +121 -0
  47. package/src/bridge/core/adapter-registry.js +271 -0
  48. package/src/bridge/core/adapter-registry.js.map +1 -0
  49. package/src/bridge/core/bridge-factory.d.ts +198 -0
  50. package/src/bridge/core/bridge-factory.js +428 -0
  51. package/src/bridge/core/bridge-factory.js.map +1 -0
  52. package/src/bridge/core/index.d.ts +9 -0
  53. package/src/bridge/core/index.js +22 -0
  54. package/src/bridge/core/index.js.map +1 -0
  55. package/src/bridge/index.d.ts +61 -0
  56. package/src/bridge/index.js +94 -0
  57. package/src/bridge/index.js.map +1 -0
  58. package/src/bridge/runtime/iife-generator.d.ts +61 -0
  59. package/src/bridge/runtime/iife-generator.js +940 -0
  60. package/src/bridge/runtime/iife-generator.js.map +1 -0
  61. package/src/bridge/runtime/index.d.ts +8 -0
  62. package/src/bridge/runtime/index.js +16 -0
  63. package/src/bridge/runtime/index.js.map +1 -0
  64. package/src/bridge/types.d.ts +385 -0
  65. package/src/bridge/types.js +11 -0
  66. package/src/bridge/types.js.map +1 -0
  67. package/src/build/cdn-resources.d.ts +140 -0
  68. package/src/build/cdn-resources.js +314 -0
  69. package/src/build/cdn-resources.js.map +1 -0
  70. package/src/build/index.d.ts +294 -0
  71. package/src/build/index.js +325 -0
  72. package/src/build/index.js.map +1 -0
  73. package/src/build/widget-manifest.d.ts +212 -0
  74. package/src/build/widget-manifest.js +652 -0
  75. package/src/build/widget-manifest.js.map +1 -0
  76. package/src/bundler/bundler.d.ts +110 -0
  77. package/src/bundler/bundler.js +432 -0
  78. package/src/bundler/bundler.js.map +1 -0
  79. package/src/bundler/cache.d.ts +172 -0
  80. package/src/bundler/cache.js +250 -0
  81. package/src/bundler/cache.js.map +1 -0
  82. package/src/bundler/index.d.ts +41 -0
  83. package/src/bundler/index.js +73 -0
  84. package/src/bundler/index.js.map +1 -0
  85. package/src/bundler/sandbox/enclave-adapter.d.ts +120 -0
  86. package/src/bundler/sandbox/enclave-adapter.js +339 -0
  87. package/src/bundler/sandbox/enclave-adapter.js.map +1 -0
  88. package/src/bundler/sandbox/executor.d.ts +13 -0
  89. package/src/bundler/sandbox/executor.js +22 -0
  90. package/src/bundler/sandbox/executor.js.map +1 -0
  91. package/src/bundler/sandbox/policy.d.ts +61 -0
  92. package/src/bundler/sandbox/policy.js +238 -0
  93. package/src/bundler/sandbox/policy.js.map +1 -0
  94. package/src/bundler/types.d.ts +347 -0
  95. package/src/bundler/types.js +132 -0
  96. package/src/bundler/types.js.map +1 -0
  97. package/src/components/alert.d.ts +71 -0
  98. package/src/components/alert.js +189 -0
  99. package/src/components/alert.js.map +1 -0
  100. package/src/components/alert.schema.d.ts +114 -0
  101. package/src/components/alert.schema.js +105 -0
  102. package/src/components/alert.schema.js.map +1 -0
  103. package/src/components/avatar.d.ts +76 -0
  104. package/src/components/avatar.js +176 -0
  105. package/src/components/avatar.js.map +1 -0
  106. package/src/components/avatar.schema.d.ts +169 -0
  107. package/src/components/avatar.schema.js +103 -0
  108. package/src/components/avatar.schema.js.map +1 -0
  109. package/src/components/badge.d.ts +70 -0
  110. package/src/components/badge.js +149 -0
  111. package/src/components/badge.js.map +1 -0
  112. package/src/components/badge.schema.d.ts +109 -0
  113. package/src/components/badge.schema.js +96 -0
  114. package/src/components/badge.schema.js.map +1 -0
  115. package/src/components/button.d.ts +111 -0
  116. package/src/components/button.js +336 -0
  117. package/src/components/button.js.map +1 -0
  118. package/src/components/button.schema.d.ts +148 -0
  119. package/src/components/button.schema.js +121 -0
  120. package/src/components/button.schema.js.map +1 -0
  121. package/src/components/card.d.ts +60 -0
  122. package/src/components/card.js +117 -0
  123. package/src/components/card.js.map +1 -0
  124. package/src/components/card.schema.d.ts +113 -0
  125. package/src/components/card.schema.js +98 -0
  126. package/src/components/card.schema.js.map +1 -0
  127. package/src/components/form.d.ts +239 -0
  128. package/src/components/form.js +420 -0
  129. package/src/components/form.js.map +1 -0
  130. package/src/components/form.schema.d.ts +441 -0
  131. package/src/components/form.schema.js +406 -0
  132. package/src/components/form.schema.js.map +1 -0
  133. package/src/components/index.d.ts +29 -0
  134. package/src/components/index.js +98 -0
  135. package/src/components/index.js.map +1 -0
  136. package/src/components/list.d.ts +127 -0
  137. package/src/components/list.js +279 -0
  138. package/src/components/list.js.map +1 -0
  139. package/src/components/list.schema.d.ts +134 -0
  140. package/src/components/list.schema.js +168 -0
  141. package/src/components/list.schema.js.map +1 -0
  142. package/src/components/modal.d.ts +111 -0
  143. package/src/components/modal.js +260 -0
  144. package/src/components/modal.js.map +1 -0
  145. package/src/components/modal.schema.d.ts +186 -0
  146. package/src/components/modal.schema.js +167 -0
  147. package/src/components/modal.schema.js.map +1 -0
  148. package/src/components/table.d.ts +105 -0
  149. package/src/components/table.js +283 -0
  150. package/src/components/table.js.map +1 -0
  151. package/src/components/table.schema.d.ts +159 -0
  152. package/src/components/table.schema.js +173 -0
  153. package/src/components/table.schema.js.map +1 -0
  154. package/src/handlebars/helpers.d.ts +348 -0
  155. package/src/handlebars/helpers.js +605 -0
  156. package/src/handlebars/helpers.js.map +1 -0
  157. package/src/handlebars/index.d.ts +193 -0
  158. package/src/handlebars/index.js +350 -0
  159. package/src/handlebars/index.js.map +1 -0
  160. package/src/index.d.ts +50 -0
  161. package/src/index.js +192 -0
  162. package/src/index.js.map +1 -0
  163. package/src/layouts/base.d.ts +88 -0
  164. package/src/layouts/base.js +227 -0
  165. package/src/layouts/base.js.map +1 -0
  166. package/src/layouts/index.d.ts +7 -0
  167. package/src/layouts/index.js +25 -0
  168. package/src/layouts/index.js.map +1 -0
  169. package/src/layouts/presets.d.ts +133 -0
  170. package/src/layouts/presets.js +277 -0
  171. package/src/layouts/presets.js.map +1 -0
  172. package/src/pages/consent.d.ts +116 -0
  173. package/src/pages/consent.js +218 -0
  174. package/src/pages/consent.js.map +1 -0
  175. package/src/pages/error.d.ts +100 -0
  176. package/src/pages/error.js +263 -0
  177. package/src/pages/error.js.map +1 -0
  178. package/src/pages/index.d.ts +8 -0
  179. package/src/pages/index.js +27 -0
  180. package/src/pages/index.js.map +1 -0
  181. package/src/react/Alert.d.ts +101 -0
  182. package/src/react/Alert.js +51 -0
  183. package/src/react/Alert.js.map +1 -0
  184. package/src/react/Badge.d.ts +100 -0
  185. package/src/react/Badge.js +55 -0
  186. package/src/react/Badge.js.map +1 -0
  187. package/src/react/Button.d.ts +108 -0
  188. package/src/react/Button.js +52 -0
  189. package/src/react/Button.js.map +1 -0
  190. package/src/react/Card.d.ts +103 -0
  191. package/src/react/Card.js +55 -0
  192. package/src/react/Card.js.map +1 -0
  193. package/src/react/hooks/context.d.ts +178 -0
  194. package/src/react/hooks/context.js +287 -0
  195. package/src/react/hooks/context.js.map +1 -0
  196. package/src/react/hooks/index.d.ts +41 -0
  197. package/src/react/hooks/index.js +61 -0
  198. package/src/react/hooks/index.js.map +1 -0
  199. package/src/react/hooks/tools.d.ts +283 -0
  200. package/src/react/hooks/tools.js +465 -0
  201. package/src/react/hooks/tools.js.map +1 -0
  202. package/src/react/index.d.ts +80 -0
  203. package/src/react/index.js +113 -0
  204. package/src/react/index.js.map +1 -0
  205. package/src/react/types.d.ts +105 -0
  206. package/src/react/types.js +12 -0
  207. package/src/react/types.js.map +1 -0
  208. package/src/react/utils.d.ts +42 -0
  209. package/src/react/utils.js +99 -0
  210. package/src/react/utils.js.map +1 -0
  211. package/src/registry/index.d.ts +45 -0
  212. package/src/registry/index.js +67 -0
  213. package/src/registry/index.js.map +1 -0
  214. package/src/registry/render-template.d.ts +86 -0
  215. package/src/registry/render-template.js +239 -0
  216. package/src/registry/render-template.js.map +1 -0
  217. package/src/registry/tool-ui.registry.d.ts +260 -0
  218. package/src/registry/tool-ui.registry.js +438 -0
  219. package/src/registry/tool-ui.registry.js.map +1 -0
  220. package/src/registry/uri-utils.d.ts +55 -0
  221. package/src/registry/uri-utils.js +97 -0
  222. package/src/registry/uri-utils.js.map +1 -0
  223. package/src/render/index.d.ts +7 -0
  224. package/src/render/index.js +14 -0
  225. package/src/render/index.js.map +1 -0
  226. package/src/render/prerender.d.ts +56 -0
  227. package/src/render/prerender.js +98 -0
  228. package/src/render/prerender.js.map +1 -0
  229. package/src/renderers/cache.d.ts +144 -0
  230. package/src/renderers/cache.js +240 -0
  231. package/src/renderers/cache.js.map +1 -0
  232. package/src/renderers/html.renderer.d.ts +122 -0
  233. package/src/renderers/html.renderer.js +204 -0
  234. package/src/renderers/html.renderer.js.map +1 -0
  235. package/src/renderers/index.d.ts +35 -0
  236. package/src/renderers/index.js +70 -0
  237. package/src/renderers/index.js.map +1 -0
  238. package/src/renderers/mdx.renderer.d.ts +119 -0
  239. package/src/renderers/mdx.renderer.js +305 -0
  240. package/src/renderers/mdx.renderer.js.map +1 -0
  241. package/src/renderers/react.renderer.d.ts +95 -0
  242. package/src/renderers/react.renderer.js +260 -0
  243. package/src/renderers/react.renderer.js.map +1 -0
  244. package/src/renderers/registry.d.ts +133 -0
  245. package/src/renderers/registry.js +232 -0
  246. package/src/renderers/registry.js.map +1 -0
  247. package/src/renderers/types.d.ts +341 -0
  248. package/src/renderers/types.js +9 -0
  249. package/src/renderers/types.js.map +1 -0
  250. package/src/renderers/utils/detect.d.ts +106 -0
  251. package/src/renderers/utils/detect.js +267 -0
  252. package/src/renderers/utils/detect.js.map +1 -0
  253. package/src/renderers/utils/hash.d.ts +39 -0
  254. package/src/renderers/utils/hash.js +75 -0
  255. package/src/renderers/utils/hash.js.map +1 -0
  256. package/src/renderers/utils/index.d.ts +8 -0
  257. package/src/renderers/utils/index.js +28 -0
  258. package/src/renderers/utils/index.js.map +1 -0
  259. package/src/renderers/utils/transpiler.d.ts +88 -0
  260. package/src/renderers/utils/transpiler.js +215 -0
  261. package/src/renderers/utils/transpiler.js.map +1 -0
  262. package/src/runtime/adapters/html.adapter.d.ts +58 -0
  263. package/src/runtime/adapters/html.adapter.js +131 -0
  264. package/src/runtime/adapters/html.adapter.js.map +1 -0
  265. package/src/runtime/adapters/index.d.ts +25 -0
  266. package/src/runtime/adapters/index.js +54 -0
  267. package/src/runtime/adapters/index.js.map +1 -0
  268. package/src/runtime/adapters/mdx.adapter.d.ts +72 -0
  269. package/src/runtime/adapters/mdx.adapter.js +241 -0
  270. package/src/runtime/adapters/mdx.adapter.js.map +1 -0
  271. package/src/runtime/adapters/react.adapter.d.ts +69 -0
  272. package/src/runtime/adapters/react.adapter.js +245 -0
  273. package/src/runtime/adapters/react.adapter.js.map +1 -0
  274. package/src/runtime/adapters/types.d.ts +94 -0
  275. package/src/runtime/adapters/types.js +11 -0
  276. package/src/runtime/adapters/types.js.map +1 -0
  277. package/src/runtime/csp.d.ts +37 -0
  278. package/src/runtime/csp.js +140 -0
  279. package/src/runtime/csp.js.map +1 -0
  280. package/src/runtime/index.d.ts +16 -0
  281. package/src/runtime/index.js +72 -0
  282. package/src/runtime/index.js.map +1 -0
  283. package/src/runtime/mcp-bridge.d.ts +100 -0
  284. package/src/runtime/mcp-bridge.js +581 -0
  285. package/src/runtime/mcp-bridge.js.map +1 -0
  286. package/src/runtime/renderer-runtime.d.ts +132 -0
  287. package/src/runtime/renderer-runtime.js +389 -0
  288. package/src/runtime/renderer-runtime.js.map +1 -0
  289. package/src/runtime/sanitizer.d.ts +171 -0
  290. package/src/runtime/sanitizer.js +318 -0
  291. package/src/runtime/sanitizer.js.map +1 -0
  292. package/src/runtime/types.d.ts +414 -0
  293. package/src/runtime/types.js +12 -0
  294. package/src/runtime/types.js.map +1 -0
  295. package/src/runtime/wrapper.d.ts +375 -0
  296. package/src/runtime/wrapper.js +1793 -0
  297. package/src/runtime/wrapper.js.map +1 -0
  298. package/src/styles/index.d.ts +7 -0
  299. package/src/styles/index.js +11 -0
  300. package/src/styles/index.js.map +1 -0
  301. package/src/styles/variants.d.ts +50 -0
  302. package/src/styles/variants.js +175 -0
  303. package/src/styles/variants.js.map +1 -0
  304. package/src/theme/cdn.d.ts +194 -0
  305. package/src/theme/cdn.js +375 -0
  306. package/src/theme/cdn.js.map +1 -0
  307. package/src/theme/index.d.ts +17 -0
  308. package/src/theme/index.js +57 -0
  309. package/src/theme/index.js.map +1 -0
  310. package/src/theme/platforms.d.ts +106 -0
  311. package/src/theme/platforms.js +161 -0
  312. package/src/theme/platforms.js.map +1 -0
  313. package/src/theme/presets/github-openai.d.ts +49 -0
  314. package/src/theme/presets/github-openai.js +189 -0
  315. package/src/theme/presets/github-openai.js.map +1 -0
  316. package/src/theme/presets/index.d.ts +10 -0
  317. package/src/theme/presets/index.js +17 -0
  318. package/src/theme/presets/index.js.map +1 -0
  319. package/src/theme/theme.d.ts +395 -0
  320. package/src/theme/theme.js +332 -0
  321. package/src/theme/theme.js.map +1 -0
  322. package/src/tool-template/builder.d.ts +212 -0
  323. package/src/tool-template/builder.js +397 -0
  324. package/src/tool-template/builder.js.map +1 -0
  325. package/src/tool-template/index.d.ts +15 -0
  326. package/src/tool-template/index.js +38 -0
  327. package/src/tool-template/index.js.map +1 -0
  328. package/src/types/index.d.ts +13 -0
  329. package/src/types/index.js +26 -0
  330. package/src/types/index.js.map +1 -0
  331. package/src/types/ui-config.d.ts +357 -0
  332. package/src/types/ui-config.js +12 -0
  333. package/src/types/ui-config.js.map +1 -0
  334. package/src/types/ui-runtime.d.ts +965 -0
  335. package/src/types/ui-runtime.js +117 -0
  336. package/src/types/ui-runtime.js.map +1 -0
  337. package/src/validation/error-box.d.ts +55 -0
  338. package/src/validation/error-box.js +75 -0
  339. package/src/validation/error-box.js.map +1 -0
  340. package/src/validation/index.d.ts +12 -0
  341. package/src/validation/index.js +21 -0
  342. package/src/validation/index.js.map +1 -0
  343. package/src/validation/wrapper.d.ts +96 -0
  344. package/src/validation/wrapper.js +117 -0
  345. package/src/validation/wrapper.js.map +1 -0
  346. package/src/web-components/core/attribute-parser.d.ts +85 -0
  347. package/src/web-components/core/attribute-parser.js +189 -0
  348. package/src/web-components/core/attribute-parser.js.map +1 -0
  349. package/src/web-components/core/base-element.d.ts +197 -0
  350. package/src/web-components/core/base-element.js +289 -0
  351. package/src/web-components/core/base-element.js.map +1 -0
  352. package/src/web-components/core/index.d.ts +8 -0
  353. package/src/web-components/core/index.js +18 -0
  354. package/src/web-components/core/index.js.map +1 -0
  355. package/src/web-components/elements/fmcp-alert.d.ts +45 -0
  356. package/src/web-components/elements/fmcp-alert.js +93 -0
  357. package/src/web-components/elements/fmcp-alert.js.map +1 -0
  358. package/src/web-components/elements/fmcp-badge.d.ts +46 -0
  359. package/src/web-components/elements/fmcp-badge.js +99 -0
  360. package/src/web-components/elements/fmcp-badge.js.map +1 -0
  361. package/src/web-components/elements/fmcp-button.d.ts +124 -0
  362. package/src/web-components/elements/fmcp-button.js +233 -0
  363. package/src/web-components/elements/fmcp-button.js.map +1 -0
  364. package/src/web-components/elements/fmcp-card.d.ts +52 -0
  365. package/src/web-components/elements/fmcp-card.js +115 -0
  366. package/src/web-components/elements/fmcp-card.js.map +1 -0
  367. package/src/web-components/elements/fmcp-input.d.ts +95 -0
  368. package/src/web-components/elements/fmcp-input.js +248 -0
  369. package/src/web-components/elements/fmcp-input.js.map +1 -0
  370. package/src/web-components/elements/fmcp-select.d.ts +99 -0
  371. package/src/web-components/elements/fmcp-select.js +243 -0
  372. package/src/web-components/elements/fmcp-select.js.map +1 -0
  373. package/src/web-components/elements/index.d.ts +12 -0
  374. package/src/web-components/elements/index.js +34 -0
  375. package/src/web-components/elements/index.js.map +1 -0
  376. package/src/web-components/index.d.ts +49 -0
  377. package/src/web-components/index.js +75 -0
  378. package/src/web-components/index.js.map +1 -0
  379. package/src/web-components/register.d.ts +56 -0
  380. package/src/web-components/register.js +80 -0
  381. package/src/web-components/register.js.map +1 -0
  382. package/src/web-components/types.d.ts +121 -0
  383. package/src/web-components/types.js +25 -0
  384. package/src/web-components/types.js.map +1 -0
  385. package/src/widgets/index.d.ts +7 -0
  386. package/src/widgets/index.js +24 -0
  387. package/src/widgets/index.js.map +1 -0
  388. package/src/widgets/progress.d.ts +132 -0
  389. package/src/widgets/progress.js +303 -0
  390. package/src/widgets/progress.js.map +1 -0
  391. package/src/widgets/resource.d.ts +162 -0
  392. package/src/widgets/resource.js +340 -0
  393. package/src/widgets/resource.js.map +1 -0
@@ -0,0 +1,142 @@
1
+ /**
2
+ * ext-apps (SEP-1865) Platform Adapter
3
+ *
4
+ * Implements the MCP Apps Extension protocol (SEP-1865) for embedded
5
+ * widget communication with AI hosts via JSON-RPC 2.0 over postMessage.
6
+ *
7
+ * @see https://github.com/modelcontextprotocol/ext-apps
8
+ * @packageDocumentation
9
+ */
10
+ import type { DisplayMode, AdapterConfig } from '../types';
11
+ import { BaseAdapter } from './base-adapter';
12
+ /**
13
+ * Configuration options for ext-apps adapter.
14
+ */
15
+ export interface ExtAppsAdapterConfig extends AdapterConfig {
16
+ options?: {
17
+ /** Trusted origins for postMessage security (trust-on-first-use if not specified) */
18
+ trustedOrigins?: string[];
19
+ /** Application name for handshake */
20
+ appName?: string;
21
+ /** Application version for handshake */
22
+ appVersion?: string;
23
+ /** Protocol version (defaults to '2024-11-05') */
24
+ protocolVersion?: string;
25
+ /** Timeout for initialization handshake (ms) */
26
+ initTimeout?: number;
27
+ };
28
+ }
29
+ /**
30
+ * ext-apps (SEP-1865) adapter.
31
+ *
32
+ * Provides communication between embedded widgets and AI hosts using
33
+ * the standardized JSON-RPC 2.0 over postMessage protocol.
34
+ *
35
+ * @example
36
+ * ```typescript
37
+ * import { ExtAppsAdapter } from '@frontmcp/ui/bridge';
38
+ *
39
+ * const adapter = new ExtAppsAdapter({
40
+ * options: {
41
+ * trustedOrigins: ['https://claude.ai'],
42
+ * }
43
+ * });
44
+ * if (adapter.canHandle()) {
45
+ * await adapter.initialize();
46
+ * }
47
+ * ```
48
+ */
49
+ export declare class ExtAppsAdapter extends BaseAdapter {
50
+ readonly id = "ext-apps";
51
+ readonly name = "ext-apps (SEP-1865)";
52
+ readonly priority = 80;
53
+ private _config;
54
+ private _messageListener;
55
+ private _pendingRequests;
56
+ private _requestId;
57
+ private _trustedOrigin;
58
+ private _hostCapabilities;
59
+ constructor(config?: ExtAppsAdapterConfig);
60
+ /**
61
+ * Check if we're in an iframe (potential ext-apps context).
62
+ */
63
+ canHandle(): boolean;
64
+ /**
65
+ * Initialize the ext-apps adapter with protocol handshake.
66
+ */
67
+ initialize(): Promise<void>;
68
+ /**
69
+ * Dispose adapter resources.
70
+ */
71
+ dispose(): void;
72
+ callTool(name: string, args: Record<string, unknown>): Promise<unknown>;
73
+ sendMessage(content: string): Promise<void>;
74
+ openLink(url: string): Promise<void>;
75
+ requestDisplayMode(mode: DisplayMode): Promise<void>;
76
+ requestClose(): Promise<void>;
77
+ /**
78
+ * Setup postMessage listener for incoming messages.
79
+ */
80
+ private _setupMessageListener;
81
+ /**
82
+ * Handle incoming postMessage events.
83
+ */
84
+ private _handleMessage;
85
+ /**
86
+ * Handle JSON-RPC response.
87
+ */
88
+ private _handleResponse;
89
+ /**
90
+ * Handle JSON-RPC notification from host.
91
+ */
92
+ private _handleNotification;
93
+ /**
94
+ * Handle tool input notification.
95
+ */
96
+ private _handleToolInput;
97
+ /**
98
+ * Handle partial tool input (streaming).
99
+ */
100
+ private _handleToolInputPartial;
101
+ /**
102
+ * Handle tool result notification.
103
+ */
104
+ private _handleToolResult;
105
+ /**
106
+ * Handle host context change notification.
107
+ */
108
+ private _handleHostContextChange;
109
+ /**
110
+ * Handle cancellation notification.
111
+ */
112
+ private _handleCancelled;
113
+ /**
114
+ * Send a JSON-RPC request to the host.
115
+ */
116
+ private _sendRequest;
117
+ /**
118
+ * Send a JSON-RPC notification (no response expected).
119
+ */
120
+ private _sendNotification;
121
+ /**
122
+ * Post a message to the parent window.
123
+ */
124
+ private _postMessage;
125
+ /**
126
+ * Perform the ui/initialize handshake with the host.
127
+ */
128
+ private _performHandshake;
129
+ /**
130
+ * Check if an origin is trusted.
131
+ * Uses trust-on-first-use if no explicit origins configured.
132
+ */
133
+ private _isOriginTrusted;
134
+ /**
135
+ * Emit a bridge event via CustomEvent.
136
+ */
137
+ private _emitBridgeEvent;
138
+ }
139
+ /**
140
+ * Factory function for creating ext-apps adapter instances.
141
+ */
142
+ export declare function createExtAppsAdapter(config?: ExtAppsAdapterConfig): ExtAppsAdapter;
@@ -0,0 +1,416 @@
1
+ "use strict";
2
+ /**
3
+ * ext-apps (SEP-1865) Platform Adapter
4
+ *
5
+ * Implements the MCP Apps Extension protocol (SEP-1865) for embedded
6
+ * widget communication with AI hosts via JSON-RPC 2.0 over postMessage.
7
+ *
8
+ * @see https://github.com/modelcontextprotocol/ext-apps
9
+ * @packageDocumentation
10
+ */
11
+ Object.defineProperty(exports, "__esModule", { value: true });
12
+ exports.ExtAppsAdapter = void 0;
13
+ exports.createExtAppsAdapter = createExtAppsAdapter;
14
+ const base_adapter_1 = require("./base-adapter");
15
+ /**
16
+ * ext-apps (SEP-1865) adapter.
17
+ *
18
+ * Provides communication between embedded widgets and AI hosts using
19
+ * the standardized JSON-RPC 2.0 over postMessage protocol.
20
+ *
21
+ * @example
22
+ * ```typescript
23
+ * import { ExtAppsAdapter } from '@frontmcp/ui/bridge';
24
+ *
25
+ * const adapter = new ExtAppsAdapter({
26
+ * options: {
27
+ * trustedOrigins: ['https://claude.ai'],
28
+ * }
29
+ * });
30
+ * if (adapter.canHandle()) {
31
+ * await adapter.initialize();
32
+ * }
33
+ * ```
34
+ */
35
+ class ExtAppsAdapter extends base_adapter_1.BaseAdapter {
36
+ id = 'ext-apps';
37
+ name = 'ext-apps (SEP-1865)';
38
+ priority = 80; // High priority, but below OpenAI native
39
+ _config;
40
+ _messageListener;
41
+ _pendingRequests = new Map();
42
+ _requestId = 0;
43
+ _trustedOrigin;
44
+ _hostCapabilities = {};
45
+ constructor(config) {
46
+ super();
47
+ this._config = config || {};
48
+ // Start with minimal capabilities, updated after handshake
49
+ this._capabilities = {
50
+ ...base_adapter_1.DEFAULT_CAPABILITIES,
51
+ canPersistState: true,
52
+ hasNetworkAccess: true, // ext-apps usually allows network
53
+ supportsTheme: true,
54
+ };
55
+ }
56
+ /**
57
+ * Check if we're in an iframe (potential ext-apps context).
58
+ */
59
+ canHandle() {
60
+ if (typeof window === 'undefined')
61
+ return false;
62
+ // Check if we're in an iframe
63
+ const inIframe = window.parent !== window;
64
+ if (!inIframe)
65
+ return false;
66
+ // Check we're not already detected as OpenAI
67
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
68
+ const win = window;
69
+ if (win.openai?.canvas)
70
+ return false;
71
+ // Check for explicit ext-apps marker
72
+ if (win.__mcpPlatform === 'ext-apps')
73
+ return true;
74
+ // In an iframe without OpenAI SDK = likely ext-apps context
75
+ return true;
76
+ }
77
+ /**
78
+ * Initialize the ext-apps adapter with protocol handshake.
79
+ */
80
+ async initialize() {
81
+ if (this._initialized)
82
+ return;
83
+ // Setup message listener
84
+ this._setupMessageListener();
85
+ // Call base initialization
86
+ await super.initialize();
87
+ // Perform ui/initialize handshake
88
+ await this._performHandshake();
89
+ this._initialized = true;
90
+ }
91
+ /**
92
+ * Dispose adapter resources.
93
+ */
94
+ dispose() {
95
+ // Remove message listener
96
+ if (this._messageListener && typeof window !== 'undefined') {
97
+ window.removeEventListener('message', this._messageListener);
98
+ this._messageListener = undefined;
99
+ }
100
+ // Reject all pending requests
101
+ for (const [id, pending] of this._pendingRequests) {
102
+ clearTimeout(pending.timeout);
103
+ pending.reject(new Error('Adapter disposed'));
104
+ }
105
+ this._pendingRequests.clear();
106
+ super.dispose();
107
+ }
108
+ // ============================================
109
+ // Actions (via JSON-RPC)
110
+ // ============================================
111
+ async callTool(name, args) {
112
+ if (!this._hostCapabilities.serverToolProxy) {
113
+ throw new Error('Server tool proxy not supported by host');
114
+ }
115
+ return this._sendRequest('ui/callServerTool', {
116
+ name,
117
+ arguments: args,
118
+ });
119
+ }
120
+ async sendMessage(content) {
121
+ await this._sendRequest('ui/message', { content });
122
+ }
123
+ async openLink(url) {
124
+ if (!this._hostCapabilities.openLink) {
125
+ // Fallback to window.open
126
+ return super.openLink(url);
127
+ }
128
+ await this._sendRequest('ui/openLink', { url });
129
+ }
130
+ async requestDisplayMode(mode) {
131
+ await this._sendRequest('ui/setDisplayMode', { mode });
132
+ this._hostContext = { ...this._hostContext, displayMode: mode };
133
+ }
134
+ async requestClose() {
135
+ await this._sendRequest('ui/close', {});
136
+ }
137
+ // ============================================
138
+ // Private: Message Handling
139
+ // ============================================
140
+ /**
141
+ * Setup postMessage listener for incoming messages.
142
+ */
143
+ _setupMessageListener() {
144
+ if (typeof window === 'undefined')
145
+ return;
146
+ this._messageListener = (event) => {
147
+ this._handleMessage(event);
148
+ };
149
+ window.addEventListener('message', this._messageListener);
150
+ }
151
+ /**
152
+ * Handle incoming postMessage events.
153
+ */
154
+ _handleMessage(event) {
155
+ // Validate origin
156
+ if (!this._isOriginTrusted(event.origin)) {
157
+ return;
158
+ }
159
+ const data = event.data;
160
+ if (!data || typeof data !== 'object')
161
+ return;
162
+ if (data.jsonrpc !== '2.0')
163
+ return;
164
+ // Handle response to our request
165
+ if ('id' in data && (data.result !== undefined || data.error !== undefined)) {
166
+ this._handleResponse(data);
167
+ return;
168
+ }
169
+ // Handle notification from host
170
+ if ('method' in data && !('id' in data)) {
171
+ this._handleNotification(data);
172
+ return;
173
+ }
174
+ }
175
+ /**
176
+ * Handle JSON-RPC response.
177
+ */
178
+ _handleResponse(response) {
179
+ const pending = this._pendingRequests.get(response.id);
180
+ if (!pending)
181
+ return;
182
+ clearTimeout(pending.timeout);
183
+ this._pendingRequests.delete(response.id);
184
+ if (response.error) {
185
+ pending.reject(new Error(`${response.error.message} (code: ${response.error.code})`));
186
+ }
187
+ else {
188
+ pending.resolve(response.result);
189
+ }
190
+ }
191
+ /**
192
+ * Handle JSON-RPC notification from host.
193
+ */
194
+ _handleNotification(notification) {
195
+ switch (notification.method) {
196
+ case 'ui/notifications/tool-input':
197
+ this._handleToolInput(notification.params);
198
+ break;
199
+ case 'ui/notifications/tool-input-partial':
200
+ this._handleToolInputPartial(notification.params);
201
+ break;
202
+ case 'ui/notifications/tool-result':
203
+ this._handleToolResult(notification.params);
204
+ break;
205
+ case 'ui/notifications/host-context-changed':
206
+ this._handleHostContextChange(notification.params);
207
+ break;
208
+ case 'ui/notifications/initialized':
209
+ // Host confirms initialization complete
210
+ break;
211
+ case 'ui/notifications/cancelled':
212
+ this._handleCancelled(notification.params);
213
+ break;
214
+ }
215
+ }
216
+ /**
217
+ * Handle tool input notification.
218
+ */
219
+ _handleToolInput(params) {
220
+ this._toolInput = params.arguments || {};
221
+ // Emit tool:input event
222
+ this._emitBridgeEvent('tool:input', { arguments: this._toolInput });
223
+ }
224
+ /**
225
+ * Handle partial tool input (streaming).
226
+ */
227
+ _handleToolInputPartial(params) {
228
+ this._toolInput = { ...this._toolInput, ...params.arguments };
229
+ // Emit tool:input-partial event
230
+ this._emitBridgeEvent('tool:input-partial', { arguments: this._toolInput });
231
+ }
232
+ /**
233
+ * Handle tool result notification.
234
+ */
235
+ _handleToolResult(params) {
236
+ this._toolOutput = params.content;
237
+ this._structuredContent = params.structuredContent;
238
+ // Notify listeners
239
+ this._notifyToolResult(params.content);
240
+ // Emit tool:result event
241
+ this._emitBridgeEvent('tool:result', {
242
+ content: params.content,
243
+ structuredContent: params.structuredContent,
244
+ });
245
+ }
246
+ /**
247
+ * Handle host context change notification.
248
+ */
249
+ _handleHostContextChange(params) {
250
+ const changes = {};
251
+ if (params.theme !== undefined) {
252
+ changes.theme = params.theme;
253
+ }
254
+ if (params.displayMode !== undefined) {
255
+ changes.displayMode = params.displayMode;
256
+ }
257
+ if (params.viewport !== undefined) {
258
+ changes.viewport = params.viewport;
259
+ }
260
+ if (params.locale !== undefined) {
261
+ changes.locale = params.locale;
262
+ }
263
+ if (params.timezone !== undefined) {
264
+ changes.timezone = params.timezone;
265
+ }
266
+ this._notifyContextChange(changes);
267
+ }
268
+ /**
269
+ * Handle cancellation notification.
270
+ */
271
+ _handleCancelled(params) {
272
+ const reason = params?.reason;
273
+ this._emitBridgeEvent('tool:cancelled', { reason });
274
+ }
275
+ // ============================================
276
+ // Private: JSON-RPC Transport
277
+ // ============================================
278
+ /**
279
+ * Send a JSON-RPC request to the host.
280
+ */
281
+ _sendRequest(method, params) {
282
+ return new Promise((resolve, reject) => {
283
+ const id = ++this._requestId;
284
+ const timeout = this._config.options?.initTimeout || 10000;
285
+ const request = {
286
+ jsonrpc: '2.0',
287
+ id,
288
+ method,
289
+ params,
290
+ };
291
+ const timeoutHandle = setTimeout(() => {
292
+ this._pendingRequests.delete(id);
293
+ reject(new Error(`Request ${method} timed out after ${timeout}ms`));
294
+ }, timeout);
295
+ this._pendingRequests.set(id, {
296
+ resolve,
297
+ reject,
298
+ timeout: timeoutHandle,
299
+ });
300
+ this._postMessage(request);
301
+ });
302
+ }
303
+ /**
304
+ * Send a JSON-RPC notification (no response expected).
305
+ */
306
+ _sendNotification(method, params) {
307
+ const notification = {
308
+ jsonrpc: '2.0',
309
+ method,
310
+ params,
311
+ };
312
+ this._postMessage(notification);
313
+ }
314
+ /**
315
+ * Post a message to the parent window.
316
+ */
317
+ _postMessage(message) {
318
+ if (typeof window === 'undefined')
319
+ return;
320
+ const targetOrigin = this._trustedOrigin || '*';
321
+ window.parent.postMessage(message, targetOrigin);
322
+ }
323
+ // ============================================
324
+ // Private: Handshake
325
+ // ============================================
326
+ /**
327
+ * Perform the ui/initialize handshake with the host.
328
+ */
329
+ async _performHandshake() {
330
+ const params = {
331
+ appInfo: {
332
+ name: this._config.options?.appName || 'FrontMCP Widget',
333
+ version: this._config.options?.appVersion || '1.0.0',
334
+ },
335
+ appCapabilities: {
336
+ tools: {
337
+ listChanged: false,
338
+ },
339
+ },
340
+ protocolVersion: this._config.options?.protocolVersion || '2024-11-05',
341
+ };
342
+ try {
343
+ const result = (await this._sendRequest('ui/initialize', params));
344
+ // Store host capabilities
345
+ this._hostCapabilities = result.hostCapabilities || {};
346
+ // Update adapter capabilities based on host
347
+ this._capabilities = {
348
+ ...this._capabilities,
349
+ canCallTools: Boolean(this._hostCapabilities.serverToolProxy),
350
+ canSendMessages: true,
351
+ canOpenLinks: Boolean(this._hostCapabilities.openLink),
352
+ supportsDisplayModes: true,
353
+ };
354
+ // Update host context
355
+ if (result.hostContext) {
356
+ this._hostContext = {
357
+ ...this._hostContext,
358
+ ...result.hostContext,
359
+ };
360
+ }
361
+ // Trust the origin that successfully completed handshake
362
+ // (trust-on-first-use if no explicit trusted origins)
363
+ if (!this._config.options?.trustedOrigins?.length) {
364
+ // Origin is already set from first successful message
365
+ }
366
+ }
367
+ catch (error) {
368
+ throw new Error(`ext-apps handshake failed: ${error}`);
369
+ }
370
+ }
371
+ // ============================================
372
+ // Private: Origin Security
373
+ // ============================================
374
+ /**
375
+ * Check if an origin is trusted.
376
+ * Uses trust-on-first-use if no explicit origins configured.
377
+ */
378
+ _isOriginTrusted(origin) {
379
+ // Explicit trusted origins from config
380
+ const trustedOrigins = this._config.options?.trustedOrigins;
381
+ if (trustedOrigins && trustedOrigins.length > 0) {
382
+ return trustedOrigins.includes(origin);
383
+ }
384
+ // Trust-on-first-use: trust the first origin we receive from
385
+ if (!this._trustedOrigin) {
386
+ this._trustedOrigin = origin;
387
+ return true;
388
+ }
389
+ return this._trustedOrigin === origin;
390
+ }
391
+ // ============================================
392
+ // Private: Events
393
+ // ============================================
394
+ /**
395
+ * Emit a bridge event via CustomEvent.
396
+ */
397
+ _emitBridgeEvent(type, detail) {
398
+ if (typeof window !== 'undefined' && typeof CustomEvent !== 'undefined') {
399
+ try {
400
+ const event = new CustomEvent(type, { detail });
401
+ window.dispatchEvent(event);
402
+ }
403
+ catch {
404
+ // Ignore event dispatch errors
405
+ }
406
+ }
407
+ }
408
+ }
409
+ exports.ExtAppsAdapter = ExtAppsAdapter;
410
+ /**
411
+ * Factory function for creating ext-apps adapter instances.
412
+ */
413
+ function createExtAppsAdapter(config) {
414
+ return new ExtAppsAdapter(config);
415
+ }
416
+ //# sourceMappingURL=ext-apps.adapter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ext-apps.adapter.js","sourceRoot":"","sources":["../../../../src/bridge/adapters/ext-apps.adapter.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ext-apps (SEP-1865) Platform Adapter\n *\n * Implements the MCP Apps Extension protocol (SEP-1865) for embedded\n * widget communication with AI hosts via JSON-RPC 2.0 over postMessage.\n *\n * @see https://github.com/modelcontextprotocol/ext-apps\n * @packageDocumentation\n */\n\nimport type {\n DisplayMode,\n HostContext,\n JsonRpcRequest,\n JsonRpcResponse,\n JsonRpcNotification,\n ExtAppsInitializeParams,\n ExtAppsInitializeResult,\n ExtAppsToolInputParams,\n ExtAppsToolResultParams,\n ExtAppsHostContextChangeParams,\n AdapterConfig,\n} from '../types';\nimport { BaseAdapter, DEFAULT_CAPABILITIES } from './base-adapter';\n\n/**\n * Configuration options for ext-apps adapter.\n */\nexport interface ExtAppsAdapterConfig extends AdapterConfig {\n options?: {\n /** Trusted origins for postMessage security (trust-on-first-use if not specified) */\n trustedOrigins?: string[];\n /** Application name for handshake */\n appName?: string;\n /** Application version for handshake */\n appVersion?: string;\n /** Protocol version (defaults to '2024-11-05') */\n protocolVersion?: string;\n /** Timeout for initialization handshake (ms) */\n initTimeout?: number;\n };\n}\n\n/**\n * ext-apps (SEP-1865) adapter.\n *\n * Provides communication between embedded widgets and AI hosts using\n * the standardized JSON-RPC 2.0 over postMessage protocol.\n *\n * @example\n * ```typescript\n * import { ExtAppsAdapter } from '@frontmcp/ui/bridge';\n *\n * const adapter = new ExtAppsAdapter({\n * options: {\n * trustedOrigins: ['https://claude.ai'],\n * }\n * });\n * if (adapter.canHandle()) {\n * await adapter.initialize();\n * }\n * ```\n */\nexport class ExtAppsAdapter extends BaseAdapter {\n readonly id = 'ext-apps';\n readonly name = 'ext-apps (SEP-1865)';\n readonly priority = 80; // High priority, but below OpenAI native\n\n private _config: ExtAppsAdapterConfig;\n private _messageListener: ((event: MessageEvent) => void) | undefined;\n private _pendingRequests: Map<\n string | number,\n {\n resolve: (value: unknown) => void;\n reject: (error: Error) => void;\n timeout: ReturnType<typeof setTimeout>;\n }\n > = new Map();\n private _requestId = 0;\n private _trustedOrigin: string | undefined;\n private _hostCapabilities: ExtAppsInitializeResult['hostCapabilities'] = {};\n\n constructor(config?: ExtAppsAdapterConfig) {\n super();\n this._config = config || {};\n\n // Start with minimal capabilities, updated after handshake\n this._capabilities = {\n ...DEFAULT_CAPABILITIES,\n canPersistState: true,\n hasNetworkAccess: true, // ext-apps usually allows network\n supportsTheme: true,\n };\n }\n\n /**\n * Check if we're in an iframe (potential ext-apps context).\n */\n canHandle(): boolean {\n if (typeof window === 'undefined') return false;\n\n // Check if we're in an iframe\n const inIframe = window.parent !== window;\n if (!inIframe) return false;\n\n // Check we're not already detected as OpenAI\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const win = window as any;\n if (win.openai?.canvas) return false;\n\n // Check for explicit ext-apps marker\n if (win.__mcpPlatform === 'ext-apps') return true;\n\n // In an iframe without OpenAI SDK = likely ext-apps context\n return true;\n }\n\n /**\n * Initialize the ext-apps adapter with protocol handshake.\n */\n override async initialize(): Promise<void> {\n if (this._initialized) return;\n\n // Setup message listener\n this._setupMessageListener();\n\n // Call base initialization\n await super.initialize();\n\n // Perform ui/initialize handshake\n await this._performHandshake();\n\n this._initialized = true;\n }\n\n /**\n * Dispose adapter resources.\n */\n override dispose(): void {\n // Remove message listener\n if (this._messageListener && typeof window !== 'undefined') {\n window.removeEventListener('message', this._messageListener);\n this._messageListener = undefined;\n }\n\n // Reject all pending requests\n for (const [id, pending] of this._pendingRequests) {\n clearTimeout(pending.timeout);\n pending.reject(new Error('Adapter disposed'));\n }\n this._pendingRequests.clear();\n\n super.dispose();\n }\n\n // ============================================\n // Actions (via JSON-RPC)\n // ============================================\n\n override async callTool(name: string, args: Record<string, unknown>): Promise<unknown> {\n if (!this._hostCapabilities.serverToolProxy) {\n throw new Error('Server tool proxy not supported by host');\n }\n\n return this._sendRequest('ui/callServerTool', {\n name,\n arguments: args,\n });\n }\n\n override async sendMessage(content: string): Promise<void> {\n await this._sendRequest('ui/message', { content });\n }\n\n override async openLink(url: string): Promise<void> {\n if (!this._hostCapabilities.openLink) {\n // Fallback to window.open\n return super.openLink(url);\n }\n\n await this._sendRequest('ui/openLink', { url });\n }\n\n override async requestDisplayMode(mode: DisplayMode): Promise<void> {\n await this._sendRequest('ui/setDisplayMode', { mode });\n this._hostContext = { ...this._hostContext, displayMode: mode };\n }\n\n override async requestClose(): Promise<void> {\n await this._sendRequest('ui/close', {});\n }\n\n // ============================================\n // Private: Message Handling\n // ============================================\n\n /**\n * Setup postMessage listener for incoming messages.\n */\n private _setupMessageListener(): void {\n if (typeof window === 'undefined') return;\n\n this._messageListener = (event: MessageEvent) => {\n this._handleMessage(event);\n };\n\n window.addEventListener('message', this._messageListener);\n }\n\n /**\n * Handle incoming postMessage events.\n */\n private _handleMessage(event: MessageEvent): void {\n // Validate origin\n if (!this._isOriginTrusted(event.origin)) {\n return;\n }\n\n const data = event.data;\n if (!data || typeof data !== 'object') return;\n if (data.jsonrpc !== '2.0') return;\n\n // Handle response to our request\n if ('id' in data && (data.result !== undefined || data.error !== undefined)) {\n this._handleResponse(data as JsonRpcResponse);\n return;\n }\n\n // Handle notification from host\n if ('method' in data && !('id' in data)) {\n this._handleNotification(data as JsonRpcNotification);\n return;\n }\n }\n\n /**\n * Handle JSON-RPC response.\n */\n private _handleResponse(response: JsonRpcResponse): void {\n const pending = this._pendingRequests.get(response.id);\n if (!pending) return;\n\n clearTimeout(pending.timeout);\n this._pendingRequests.delete(response.id);\n\n if (response.error) {\n pending.reject(new Error(`${response.error.message} (code: ${response.error.code})`));\n } else {\n pending.resolve(response.result);\n }\n }\n\n /**\n * Handle JSON-RPC notification from host.\n */\n private _handleNotification(notification: JsonRpcNotification): void {\n switch (notification.method) {\n case 'ui/notifications/tool-input':\n this._handleToolInput(notification.params as ExtAppsToolInputParams);\n break;\n\n case 'ui/notifications/tool-input-partial':\n this._handleToolInputPartial(notification.params as ExtAppsToolInputParams);\n break;\n\n case 'ui/notifications/tool-result':\n this._handleToolResult(notification.params as ExtAppsToolResultParams);\n break;\n\n case 'ui/notifications/host-context-changed':\n this._handleHostContextChange(notification.params as ExtAppsHostContextChangeParams);\n break;\n\n case 'ui/notifications/initialized':\n // Host confirms initialization complete\n break;\n\n case 'ui/notifications/cancelled':\n this._handleCancelled(notification.params);\n break;\n }\n }\n\n /**\n * Handle tool input notification.\n */\n private _handleToolInput(params: ExtAppsToolInputParams): void {\n this._toolInput = params.arguments || {};\n\n // Emit tool:input event\n this._emitBridgeEvent('tool:input', { arguments: this._toolInput });\n }\n\n /**\n * Handle partial tool input (streaming).\n */\n private _handleToolInputPartial(params: ExtAppsToolInputParams): void {\n this._toolInput = { ...this._toolInput, ...params.arguments };\n\n // Emit tool:input-partial event\n this._emitBridgeEvent('tool:input-partial', { arguments: this._toolInput });\n }\n\n /**\n * Handle tool result notification.\n */\n private _handleToolResult(params: ExtAppsToolResultParams): void {\n this._toolOutput = params.content;\n this._structuredContent = params.structuredContent;\n\n // Notify listeners\n this._notifyToolResult(params.content);\n\n // Emit tool:result event\n this._emitBridgeEvent('tool:result', {\n content: params.content,\n structuredContent: params.structuredContent,\n });\n }\n\n /**\n * Handle host context change notification.\n */\n private _handleHostContextChange(params: ExtAppsHostContextChangeParams): void {\n const changes: Partial<HostContext> = {};\n\n if (params.theme !== undefined) {\n changes.theme = params.theme;\n }\n if (params.displayMode !== undefined) {\n changes.displayMode = params.displayMode;\n }\n if (params.viewport !== undefined) {\n changes.viewport = params.viewport;\n }\n if (params.locale !== undefined) {\n changes.locale = params.locale;\n }\n if (params.timezone !== undefined) {\n changes.timezone = params.timezone;\n }\n\n this._notifyContextChange(changes);\n }\n\n /**\n * Handle cancellation notification.\n */\n private _handleCancelled(params: unknown): void {\n const reason = (params as { reason?: string })?.reason;\n this._emitBridgeEvent('tool:cancelled', { reason });\n }\n\n // ============================================\n // Private: JSON-RPC Transport\n // ============================================\n\n /**\n * Send a JSON-RPC request to the host.\n */\n private _sendRequest(method: string, params?: unknown): Promise<unknown> {\n return new Promise((resolve, reject) => {\n const id = ++this._requestId;\n const timeout = this._config.options?.initTimeout || 10000;\n\n const request: JsonRpcRequest = {\n jsonrpc: '2.0',\n id,\n method,\n params,\n };\n\n const timeoutHandle = setTimeout(() => {\n this._pendingRequests.delete(id);\n reject(new Error(`Request ${method} timed out after ${timeout}ms`));\n }, timeout);\n\n this._pendingRequests.set(id, {\n resolve,\n reject,\n timeout: timeoutHandle,\n });\n\n this._postMessage(request);\n });\n }\n\n /**\n * Send a JSON-RPC notification (no response expected).\n */\n private _sendNotification(method: string, params?: unknown): void {\n const notification: JsonRpcNotification = {\n jsonrpc: '2.0',\n method,\n params,\n };\n this._postMessage(notification);\n }\n\n /**\n * Post a message to the parent window.\n */\n private _postMessage(message: JsonRpcRequest | JsonRpcResponse | JsonRpcNotification): void {\n if (typeof window === 'undefined') return;\n\n const targetOrigin = this._trustedOrigin || '*';\n window.parent.postMessage(message, targetOrigin);\n }\n\n // ============================================\n // Private: Handshake\n // ============================================\n\n /**\n * Perform the ui/initialize handshake with the host.\n */\n private async _performHandshake(): Promise<void> {\n const params: ExtAppsInitializeParams = {\n appInfo: {\n name: this._config.options?.appName || 'FrontMCP Widget',\n version: this._config.options?.appVersion || '1.0.0',\n },\n appCapabilities: {\n tools: {\n listChanged: false,\n },\n },\n protocolVersion: this._config.options?.protocolVersion || '2024-11-05',\n };\n\n try {\n const result = (await this._sendRequest('ui/initialize', params)) as ExtAppsInitializeResult;\n\n // Store host capabilities\n this._hostCapabilities = result.hostCapabilities || {};\n\n // Update adapter capabilities based on host\n this._capabilities = {\n ...this._capabilities,\n canCallTools: Boolean(this._hostCapabilities.serverToolProxy),\n canSendMessages: true,\n canOpenLinks: Boolean(this._hostCapabilities.openLink),\n supportsDisplayModes: true,\n };\n\n // Update host context\n if (result.hostContext) {\n this._hostContext = {\n ...this._hostContext,\n ...result.hostContext,\n };\n }\n\n // Trust the origin that successfully completed handshake\n // (trust-on-first-use if no explicit trusted origins)\n if (!this._config.options?.trustedOrigins?.length) {\n // Origin is already set from first successful message\n }\n } catch (error) {\n throw new Error(`ext-apps handshake failed: ${error}`);\n }\n }\n\n // ============================================\n // Private: Origin Security\n // ============================================\n\n /**\n * Check if an origin is trusted.\n * Uses trust-on-first-use if no explicit origins configured.\n */\n private _isOriginTrusted(origin: string): boolean {\n // Explicit trusted origins from config\n const trustedOrigins = this._config.options?.trustedOrigins;\n if (trustedOrigins && trustedOrigins.length > 0) {\n return trustedOrigins.includes(origin);\n }\n\n // Trust-on-first-use: trust the first origin we receive from\n if (!this._trustedOrigin) {\n this._trustedOrigin = origin;\n return true;\n }\n\n return this._trustedOrigin === origin;\n }\n\n // ============================================\n // Private: Events\n // ============================================\n\n /**\n * Emit a bridge event via CustomEvent.\n */\n private _emitBridgeEvent(type: string, detail: unknown): void {\n if (typeof window !== 'undefined' && typeof CustomEvent !== 'undefined') {\n try {\n const event = new CustomEvent(type, { detail });\n window.dispatchEvent(event);\n } catch {\n // Ignore event dispatch errors\n }\n }\n }\n}\n\n/**\n * Factory function for creating ext-apps adapter instances.\n */\nexport function createExtAppsAdapter(config?: ExtAppsAdapterConfig): ExtAppsAdapter {\n return new ExtAppsAdapter(config);\n}\n"]}
@@ -0,0 +1,63 @@
1
+ /**
2
+ * Gemini Platform Adapter
3
+ *
4
+ * Adapter for Google's Gemini AI platform.
5
+ * Provides integration with Gemini-specific features and APIs.
6
+ *
7
+ * @packageDocumentation
8
+ */
9
+ import { BaseAdapter } from './base-adapter';
10
+ /**
11
+ * Gemini adapter for Google's AI platform.
12
+ *
13
+ * Features:
14
+ * - Theme detection from Gemini SDK or system
15
+ * - Network access for external resources
16
+ * - LocalStorage persistence
17
+ * - Link opening capability
18
+ *
19
+ * @example
20
+ * ```typescript
21
+ * import { GeminiAdapter } from '@frontmcp/ui/bridge';
22
+ *
23
+ * const adapter = new GeminiAdapter();
24
+ * if (adapter.canHandle()) {
25
+ * await adapter.initialize();
26
+ * }
27
+ * ```
28
+ */
29
+ export declare class GeminiAdapter extends BaseAdapter {
30
+ readonly id = "gemini";
31
+ readonly name = "Google Gemini";
32
+ readonly priority = 40;
33
+ private _gemini;
34
+ constructor();
35
+ /**
36
+ * Check if we're running in a Gemini context.
37
+ */
38
+ canHandle(): boolean;
39
+ /**
40
+ * Initialize the Gemini adapter.
41
+ */
42
+ initialize(): Promise<void>;
43
+ /**
44
+ * Get current theme.
45
+ */
46
+ getTheme(): 'light' | 'dark';
47
+ /**
48
+ * Send a message (if supported by SDK).
49
+ */
50
+ sendMessage(content: string): Promise<void>;
51
+ /**
52
+ * Open a link.
53
+ */
54
+ openLink(url: string): Promise<void>;
55
+ /**
56
+ * Setup listener for system theme changes.
57
+ */
58
+ private _setupThemeListener;
59
+ }
60
+ /**
61
+ * Factory function for creating Gemini adapter instances.
62
+ */
63
+ export declare function createGeminiAdapter(): GeminiAdapter;