@frontmcp/uipack 0.12.2 → 1.0.0-beta.1

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 (298) hide show
  1. package/CLAUDE.md +56 -154
  2. package/README.md +367 -62
  3. package/adapters/base-template.d.ts +30 -0
  4. package/adapters/base-template.d.ts.map +1 -0
  5. package/adapters/cdn-info.d.ts +34 -0
  6. package/adapters/cdn-info.d.ts.map +1 -0
  7. package/adapters/constants.d.ts +18 -0
  8. package/adapters/constants.d.ts.map +1 -0
  9. package/adapters/content-detector.d.ts +19 -0
  10. package/adapters/content-detector.d.ts.map +1 -0
  11. package/adapters/content-renderers.d.ts +27 -0
  12. package/adapters/content-renderers.d.ts.map +1 -0
  13. package/adapters/index.d.ts +14 -7
  14. package/adapters/index.d.ts.map +1 -1
  15. package/adapters/index.js +2343 -426
  16. package/adapters/render-failure.d.ts +18 -0
  17. package/adapters/render-failure.d.ts.map +1 -0
  18. package/adapters/response-builder.d.ts +34 -104
  19. package/adapters/response-builder.d.ts.map +1 -1
  20. package/adapters/serving-mode.d.ts +28 -91
  21. package/adapters/serving-mode.d.ts.map +1 -1
  22. package/adapters/template-renderer.d.ts +50 -0
  23. package/adapters/template-renderer.d.ts.map +1 -0
  24. package/adapters/type-detector.d.ts +18 -0
  25. package/adapters/type-detector.d.ts.map +1 -0
  26. package/bridge-runtime/index.js +1 -1
  27. package/component/index.d.ts +14 -0
  28. package/component/index.d.ts.map +1 -0
  29. package/component/index.js +2043 -0
  30. package/component/loader.d.ts +36 -0
  31. package/component/loader.d.ts.map +1 -0
  32. package/component/renderer.d.ts +30 -0
  33. package/component/renderer.d.ts.map +1 -0
  34. package/component/transpiler.d.ts +49 -0
  35. package/component/transpiler.d.ts.map +1 -0
  36. package/component/types.d.ts +82 -0
  37. package/component/types.d.ts.map +1 -0
  38. package/esm/adapters/index.mjs +2337 -422
  39. package/esm/bridge-runtime/index.mjs +1 -1
  40. package/esm/component/index.mjs +2013 -0
  41. package/esm/index.mjs +3446 -13935
  42. package/esm/package.json +3 -12
  43. package/esm/resolver/index.mjs +661 -0
  44. package/esm/shell/index.mjs +1406 -0
  45. package/esm/types/index.mjs +11 -11
  46. package/esm/utils/index.mjs +53 -8
  47. package/index.d.ts +12 -40
  48. package/index.d.ts.map +1 -1
  49. package/index.js +3579 -14218
  50. package/package.json +3 -12
  51. package/resolver/cdn-registry.d.ts +39 -0
  52. package/resolver/cdn-registry.d.ts.map +1 -0
  53. package/resolver/esm-sh.resolver.d.ts +54 -0
  54. package/resolver/esm-sh.resolver.d.ts.map +1 -0
  55. package/resolver/import-map.d.ts +47 -0
  56. package/resolver/import-map.d.ts.map +1 -0
  57. package/resolver/import-parser.d.ts +28 -0
  58. package/resolver/import-parser.d.ts.map +1 -0
  59. package/resolver/import-rewriter.d.ts +29 -0
  60. package/resolver/import-rewriter.d.ts.map +1 -0
  61. package/resolver/index.d.ts +15 -0
  62. package/resolver/index.d.ts.map +1 -0
  63. package/resolver/index.js +708 -0
  64. package/resolver/types.d.ts +191 -0
  65. package/resolver/types.d.ts.map +1 -0
  66. package/shell/builder.d.ts +31 -0
  67. package/shell/builder.d.ts.map +1 -0
  68. package/shell/csp.d.ts +37 -0
  69. package/shell/csp.d.ts.map +1 -0
  70. package/shell/custom-shell-applier.d.ts +33 -0
  71. package/shell/custom-shell-applier.d.ts.map +1 -0
  72. package/shell/custom-shell-resolver.d.ts +47 -0
  73. package/shell/custom-shell-resolver.d.ts.map +1 -0
  74. package/shell/custom-shell-types.d.ts +75 -0
  75. package/shell/custom-shell-types.d.ts.map +1 -0
  76. package/shell/custom-shell-validator.d.ts +26 -0
  77. package/shell/custom-shell-validator.d.ts.map +1 -0
  78. package/shell/data-injector.d.ts +40 -0
  79. package/shell/data-injector.d.ts.map +1 -0
  80. package/shell/index.d.ts +19 -0
  81. package/shell/index.d.ts.map +1 -0
  82. package/shell/index.js +1453 -0
  83. package/shell/types.d.ts +54 -0
  84. package/shell/types.d.ts.map +1 -0
  85. package/types/index.d.ts +1 -3
  86. package/types/index.d.ts.map +1 -1
  87. package/types/index.js +11 -11
  88. package/types/ui-config.d.ts +50 -11
  89. package/types/ui-config.d.ts.map +1 -1
  90. package/types/ui-runtime.d.ts +8 -82
  91. package/types/ui-runtime.d.ts.map +1 -1
  92. package/utils/index.d.ts +9 -3
  93. package/utils/index.d.ts.map +1 -1
  94. package/utils/index.js +59 -7
  95. package/adapters/platform-meta.constants.d.ts +0 -26
  96. package/adapters/platform-meta.constants.d.ts.map +0 -1
  97. package/adapters/platform-meta.d.ts +0 -234
  98. package/adapters/platform-meta.d.ts.map +0 -1
  99. package/base-template/bridge.d.ts +0 -90
  100. package/base-template/bridge.d.ts.map +0 -1
  101. package/base-template/default-base-template.d.ts +0 -91
  102. package/base-template/default-base-template.d.ts.map +0 -1
  103. package/base-template/index.d.ts +0 -15
  104. package/base-template/index.d.ts.map +0 -1
  105. package/base-template/index.js +0 -1393
  106. package/base-template/polyfills.d.ts +0 -31
  107. package/base-template/polyfills.d.ts.map +0 -1
  108. package/base-template/theme-styles.d.ts +0 -74
  109. package/base-template/theme-styles.d.ts.map +0 -1
  110. package/build/builders/base-builder.d.ts +0 -124
  111. package/build/builders/base-builder.d.ts.map +0 -1
  112. package/build/builders/esbuild-config.d.ts +0 -94
  113. package/build/builders/esbuild-config.d.ts.map +0 -1
  114. package/build/builders/hybrid-builder.d.ts +0 -93
  115. package/build/builders/hybrid-builder.d.ts.map +0 -1
  116. package/build/builders/index.d.ts +0 -17
  117. package/build/builders/index.d.ts.map +0 -1
  118. package/build/builders/inline-builder.d.ts +0 -83
  119. package/build/builders/inline-builder.d.ts.map +0 -1
  120. package/build/builders/static-builder.d.ts +0 -78
  121. package/build/builders/static-builder.d.ts.map +0 -1
  122. package/build/builders/types.d.ts +0 -341
  123. package/build/builders/types.d.ts.map +0 -1
  124. package/build/cdn-resources.d.ts +0 -244
  125. package/build/cdn-resources.d.ts.map +0 -1
  126. package/build/hybrid-data.d.ts +0 -127
  127. package/build/hybrid-data.d.ts.map +0 -1
  128. package/build/index.d.ts +0 -299
  129. package/build/index.d.ts.map +0 -1
  130. package/build/index.js +0 -8699
  131. package/build/ui-components-browser.d.ts +0 -64
  132. package/build/ui-components-browser.d.ts.map +0 -1
  133. package/build/widget-manifest.d.ts +0 -362
  134. package/build/widget-manifest.d.ts.map +0 -1
  135. package/bundler/cache.d.ts +0 -173
  136. package/bundler/cache.d.ts.map +0 -1
  137. package/bundler/file-cache/component-builder.d.ts +0 -167
  138. package/bundler/file-cache/component-builder.d.ts.map +0 -1
  139. package/bundler/file-cache/hash-calculator.d.ts +0 -155
  140. package/bundler/file-cache/hash-calculator.d.ts.map +0 -1
  141. package/bundler/file-cache/index.d.ts +0 -12
  142. package/bundler/file-cache/index.d.ts.map +0 -1
  143. package/bundler/file-cache/storage/filesystem.d.ts +0 -149
  144. package/bundler/file-cache/storage/filesystem.d.ts.map +0 -1
  145. package/bundler/file-cache/storage/index.d.ts +0 -11
  146. package/bundler/file-cache/storage/index.d.ts.map +0 -1
  147. package/bundler/file-cache/storage/interface.d.ts +0 -152
  148. package/bundler/file-cache/storage/interface.d.ts.map +0 -1
  149. package/bundler/file-cache/storage/redis.d.ts +0 -139
  150. package/bundler/file-cache/storage/redis.d.ts.map +0 -1
  151. package/bundler/index.d.ts +0 -35
  152. package/bundler/index.d.ts.map +0 -1
  153. package/bundler/index.js +0 -2953
  154. package/bundler/sandbox/enclave-adapter.d.ts +0 -121
  155. package/bundler/sandbox/enclave-adapter.d.ts.map +0 -1
  156. package/bundler/sandbox/executor.d.ts +0 -14
  157. package/bundler/sandbox/executor.d.ts.map +0 -1
  158. package/bundler/sandbox/policy.d.ts +0 -62
  159. package/bundler/sandbox/policy.d.ts.map +0 -1
  160. package/bundler/types.d.ts +0 -702
  161. package/bundler/types.d.ts.map +0 -1
  162. package/dependency/cdn-registry.d.ts +0 -98
  163. package/dependency/cdn-registry.d.ts.map +0 -1
  164. package/dependency/import-map.d.ts +0 -186
  165. package/dependency/import-map.d.ts.map +0 -1
  166. package/dependency/import-parser.d.ts +0 -82
  167. package/dependency/import-parser.d.ts.map +0 -1
  168. package/dependency/index.d.ts +0 -17
  169. package/dependency/index.d.ts.map +0 -1
  170. package/dependency/index.js +0 -3180
  171. package/dependency/resolver.d.ts +0 -164
  172. package/dependency/resolver.d.ts.map +0 -1
  173. package/dependency/schemas.d.ts +0 -486
  174. package/dependency/schemas.d.ts.map +0 -1
  175. package/dependency/template-loader.d.ts +0 -204
  176. package/dependency/template-loader.d.ts.map +0 -1
  177. package/dependency/template-processor.d.ts +0 -118
  178. package/dependency/template-processor.d.ts.map +0 -1
  179. package/dependency/types.d.ts +0 -739
  180. package/dependency/types.d.ts.map +0 -1
  181. package/esm/base-template/index.mjs +0 -1359
  182. package/esm/build/index.mjs +0 -8601
  183. package/esm/bundler/index.mjs +0 -2895
  184. package/esm/dependency/index.mjs +0 -3068
  185. package/esm/handlebars/index.mjs +0 -587
  186. package/esm/registry/index.mjs +0 -6305
  187. package/esm/renderers/index.mjs +0 -1557
  188. package/esm/runtime/index.mjs +0 -5361
  189. package/esm/styles/index.mjs +0 -171
  190. package/esm/theme/index.mjs +0 -756
  191. package/esm/tool-template/index.mjs +0 -3652
  192. package/esm/validation/index.mjs +0 -542
  193. package/handlebars/expression-extractor.d.ts +0 -147
  194. package/handlebars/expression-extractor.d.ts.map +0 -1
  195. package/handlebars/helpers.d.ts +0 -339
  196. package/handlebars/helpers.d.ts.map +0 -1
  197. package/handlebars/index.d.ts +0 -195
  198. package/handlebars/index.d.ts.map +0 -1
  199. package/handlebars/index.js +0 -659
  200. package/preview/claude-preview.d.ts +0 -67
  201. package/preview/claude-preview.d.ts.map +0 -1
  202. package/preview/generic-preview.d.ts +0 -66
  203. package/preview/generic-preview.d.ts.map +0 -1
  204. package/preview/index.d.ts +0 -36
  205. package/preview/index.d.ts.map +0 -1
  206. package/preview/openai-preview.d.ts +0 -70
  207. package/preview/openai-preview.d.ts.map +0 -1
  208. package/preview/types.d.ts +0 -199
  209. package/preview/types.d.ts.map +0 -1
  210. package/registry/index.d.ts +0 -46
  211. package/registry/index.d.ts.map +0 -1
  212. package/registry/index.js +0 -6342
  213. package/registry/render-template.d.ts +0 -91
  214. package/registry/render-template.d.ts.map +0 -1
  215. package/registry/tool-ui.registry.d.ts +0 -294
  216. package/registry/tool-ui.registry.d.ts.map +0 -1
  217. package/registry/uri-utils.d.ts +0 -56
  218. package/registry/uri-utils.d.ts.map +0 -1
  219. package/renderers/cache.d.ts +0 -145
  220. package/renderers/cache.d.ts.map +0 -1
  221. package/renderers/html.renderer.d.ts +0 -123
  222. package/renderers/html.renderer.d.ts.map +0 -1
  223. package/renderers/index.d.ts +0 -36
  224. package/renderers/index.d.ts.map +0 -1
  225. package/renderers/index.js +0 -1603
  226. package/renderers/mdx-client.renderer.d.ts +0 -124
  227. package/renderers/mdx-client.renderer.d.ts.map +0 -1
  228. package/renderers/registry.d.ts +0 -133
  229. package/renderers/registry.d.ts.map +0 -1
  230. package/renderers/types.d.ts +0 -343
  231. package/renderers/types.d.ts.map +0 -1
  232. package/renderers/utils/detect.d.ts +0 -107
  233. package/renderers/utils/detect.d.ts.map +0 -1
  234. package/renderers/utils/hash.d.ts +0 -40
  235. package/renderers/utils/hash.d.ts.map +0 -1
  236. package/renderers/utils/index.d.ts +0 -9
  237. package/renderers/utils/index.d.ts.map +0 -1
  238. package/renderers/utils/transpiler.d.ts +0 -70
  239. package/renderers/utils/transpiler.d.ts.map +0 -1
  240. package/runtime/adapters/html.adapter.d.ts +0 -59
  241. package/runtime/adapters/html.adapter.d.ts.map +0 -1
  242. package/runtime/adapters/index.d.ts +0 -26
  243. package/runtime/adapters/index.d.ts.map +0 -1
  244. package/runtime/adapters/mdx.adapter.d.ts +0 -73
  245. package/runtime/adapters/mdx.adapter.d.ts.map +0 -1
  246. package/runtime/adapters/types.d.ts +0 -95
  247. package/runtime/adapters/types.d.ts.map +0 -1
  248. package/runtime/csp.d.ts +0 -48
  249. package/runtime/csp.d.ts.map +0 -1
  250. package/runtime/index.d.ts +0 -17
  251. package/runtime/index.d.ts.map +0 -1
  252. package/runtime/index.js +0 -5432
  253. package/runtime/mcp-bridge.d.ts +0 -101
  254. package/runtime/mcp-bridge.d.ts.map +0 -1
  255. package/runtime/renderer-runtime.d.ts +0 -133
  256. package/runtime/renderer-runtime.d.ts.map +0 -1
  257. package/runtime/sanitizer.d.ts +0 -180
  258. package/runtime/sanitizer.d.ts.map +0 -1
  259. package/runtime/types.d.ts +0 -415
  260. package/runtime/types.d.ts.map +0 -1
  261. package/runtime/wrapper.d.ts +0 -421
  262. package/runtime/wrapper.d.ts.map +0 -1
  263. package/styles/index.d.ts +0 -8
  264. package/styles/index.d.ts.map +0 -1
  265. package/styles/index.js +0 -222
  266. package/styles/variants.d.ts +0 -51
  267. package/styles/variants.d.ts.map +0 -1
  268. package/theme/cdn.d.ts +0 -195
  269. package/theme/cdn.d.ts.map +0 -1
  270. package/theme/css-to-theme.d.ts +0 -64
  271. package/theme/css-to-theme.d.ts.map +0 -1
  272. package/theme/index.d.ts +0 -19
  273. package/theme/index.d.ts.map +0 -1
  274. package/theme/index.js +0 -814
  275. package/theme/platforms.d.ts +0 -102
  276. package/theme/platforms.d.ts.map +0 -1
  277. package/theme/presets/github-openai.d.ts +0 -50
  278. package/theme/presets/github-openai.d.ts.map +0 -1
  279. package/theme/presets/index.d.ts +0 -11
  280. package/theme/presets/index.d.ts.map +0 -1
  281. package/theme/theme.d.ts +0 -396
  282. package/theme/theme.d.ts.map +0 -1
  283. package/tool-template/builder.d.ts +0 -213
  284. package/tool-template/builder.d.ts.map +0 -1
  285. package/tool-template/index.d.ts +0 -16
  286. package/tool-template/index.d.ts.map +0 -1
  287. package/tool-template/index.js +0 -3690
  288. package/validation/error-box.d.ts +0 -56
  289. package/validation/error-box.d.ts.map +0 -1
  290. package/validation/index.d.ts +0 -13
  291. package/validation/index.d.ts.map +0 -1
  292. package/validation/index.js +0 -576
  293. package/validation/schema-paths.d.ts +0 -118
  294. package/validation/schema-paths.d.ts.map +0 -1
  295. package/validation/template-validator.d.ts +0 -143
  296. package/validation/template-validator.d.ts.map +0 -1
  297. package/validation/wrapper.d.ts +0 -97
  298. package/validation/wrapper.d.ts.map +0 -1
@@ -1,1359 +0,0 @@
1
- // libs/uipack/src/theme/cdn.ts
2
- var CDN = {
3
- /**
4
- * Tailwind CSS v4 Browser CDN
5
- * Generates styles on-the-fly with @theme support
6
- * @see https://tailwindcss.com/docs/installation/play-cdn
7
- */
8
- tailwind: "https://cdn.jsdelivr.net/npm/@tailwindcss/browser@4",
9
- /**
10
- * HTMX 2.x - High power tools for HTML
11
- * Enables AJAX, WebSockets, Server Sent Events directly in HTML
12
- * @see https://htmx.org
13
- */
14
- htmx: {
15
- url: "https://cdnjs.cloudflare.com/ajax/libs/htmx/2.0.7/htmx.min.js",
16
- integrity: "sha512-T6VLg/MJYMbLTmQ8VLvonbWg8VOvmDhXcOvHzCwo6ShdGuUU5SEcp1IAPXL4k9lVoMi8gRXl5K/S/zh43Y9rJA=="
17
- },
18
- /**
19
- * Alpine.js - Lightweight reactive framework
20
- * Used for more complex client-side interactions
21
- * @see https://alpinejs.dev
22
- */
23
- alpine: {
24
- url: "https://cdn.jsdelivr.net/npm/alpinejs@3.14.3/dist/cdn.min.js",
25
- integrity: "sha384-6zY8MFQJ/EqS1r4RJl+7j8rvZPuBWpT0RzWf+IFcKhxqUzQNmJzA1X1VEVZhYaEz"
26
- },
27
- /**
28
- * Google Fonts - Inter for modern UI typography
29
- */
30
- fonts: {
31
- preconnect: ["https://fonts.googleapis.com", "https://fonts.gstatic.com"],
32
- inter: "https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700&display=swap",
33
- mono: "https://fonts.googleapis.com/css2?family=JetBrains+Mono:wght@400;500;600&display=swap"
34
- },
35
- /**
36
- * Lucide Icons - Beautiful & consistent icons
37
- * @see https://lucide.dev
38
- */
39
- icons: {
40
- url: "https://cdn.jsdelivr.net/npm/lucide@0.294.0/dist/umd/lucide.min.js",
41
- integrity: "sha384-wpLmHb7v7V1LsEuTmPQ9tXqWZvTtRWWVqJuE+Yz6X0I6O2T6bHJVeXH1lVWqF4qE"
42
- }
43
- };
44
- var scriptCache = /* @__PURE__ */ new Map();
45
- function getCachedScript(url) {
46
- return scriptCache.get(url);
47
- }
48
- function buildScriptTag(url, integrity, options = {}) {
49
- const attrs = [`src="${url}"`];
50
- if (integrity) {
51
- attrs.push(`integrity="${integrity}"`);
52
- attrs.push('crossorigin="anonymous"');
53
- }
54
- if (options.defer) attrs.push("defer");
55
- if (options.async) attrs.push("async");
56
- return `<script ${attrs.join(" ")}></script>`;
57
- }
58
- function buildInlineScriptTag(content) {
59
- return `<script>${content}</script>`;
60
- }
61
- function buildFontPreconnectFromTheme(theme) {
62
- const preconnect = theme.cdn?.fonts?.preconnect ?? CDN.fonts.preconnect;
63
- return preconnect.map((url, i) => `<link rel="preconnect" href="${url}"${i > 0 ? " crossorigin" : ""}>`).join("\n ");
64
- }
65
- function buildFontStylesheetsFromTheme(theme) {
66
- const stylesheets = theme.cdn?.fonts?.stylesheets ?? [CDN.fonts.inter];
67
- return stylesheets.map((url) => `<link href="${url}" rel="stylesheet">`).join("\n ");
68
- }
69
- function buildCdnScriptsFromTheme(theme, options = {}) {
70
- const { tailwind = true, htmx = true, alpine = false, icons = false, inline = false } = options;
71
- const scripts = [];
72
- const tailwindUrl = theme.cdn?.scripts?.tailwind ?? CDN.tailwind;
73
- const htmxConfig = theme.cdn?.scripts?.htmx ?? CDN.htmx;
74
- const alpineConfig = theme.cdn?.scripts?.alpine ?? CDN.alpine;
75
- const iconsConfig = theme.cdn?.icons?.script ?? CDN.icons;
76
- if (inline) {
77
- if (tailwind) {
78
- const cached = getCachedScript(tailwindUrl);
79
- if (cached) {
80
- scripts.push(buildInlineScriptTag(cached));
81
- } else {
82
- console.warn(
83
- "[frontmcp/ui] Inline mode requested but Tailwind script not cached. Call fetchAndCacheScriptsFromTheme() first."
84
- );
85
- }
86
- }
87
- if (htmx) {
88
- const cached = getCachedScript(htmxConfig.url);
89
- if (cached) {
90
- scripts.push(buildInlineScriptTag(cached));
91
- } else {
92
- console.warn(
93
- "[frontmcp/ui] Inline mode requested but HTMX script not cached. Call fetchAndCacheScriptsFromTheme() first."
94
- );
95
- }
96
- }
97
- if (alpine) {
98
- const cached = getCachedScript(alpineConfig.url);
99
- if (cached) {
100
- scripts.push(buildInlineScriptTag(cached));
101
- } else {
102
- console.warn(
103
- "[frontmcp/ui] Inline mode requested but Alpine.js script not cached. Call fetchAndCacheScriptsFromTheme() first."
104
- );
105
- }
106
- }
107
- if (icons) {
108
- const cached = getCachedScript(iconsConfig.url);
109
- if (cached) {
110
- scripts.push(buildInlineScriptTag(cached));
111
- } else {
112
- console.warn(
113
- "[frontmcp/ui] Inline mode requested but icons script not cached. Call fetchAndCacheScriptsFromTheme() first."
114
- );
115
- }
116
- }
117
- } else {
118
- if (tailwind) {
119
- scripts.push(buildScriptTag(tailwindUrl));
120
- }
121
- if (htmx) {
122
- scripts.push(buildScriptTag(htmxConfig.url, htmxConfig.integrity));
123
- }
124
- if (alpine) {
125
- scripts.push(buildScriptTag(alpineConfig.url, alpineConfig.integrity, { defer: true }));
126
- }
127
- if (icons) {
128
- scripts.push(buildScriptTag(iconsConfig.url, iconsConfig.integrity));
129
- }
130
- }
131
- return scripts.join("\n ");
132
- }
133
-
134
- // libs/uipack/src/theme/presets/github-openai.ts
135
- var GITHUB_OPENAI_THEME = {
136
- name: "github-openai",
137
- colors: {
138
- semantic: {
139
- // Primary: Near-black for main actions and branding
140
- primary: "#24292f",
141
- // Secondary: Medium gray for secondary elements
142
- secondary: "#57606a",
143
- // Accent: Blue for links, focus states, and highlights
144
- accent: "#0969da",
145
- // Status colors
146
- success: "#1a7f37",
147
- // GitHub green
148
- warning: "#9a6700",
149
- // Amber warning
150
- danger: "#cf222e",
151
- // GitHub red
152
- info: "#0969da"
153
- // Blue info
154
- },
155
- surface: {
156
- // Pure white background
157
- background: "#ffffff",
158
- // Light gray surface (GitHub code background style)
159
- surface: "#f6f8fa",
160
- // White elevated surfaces (modals, cards)
161
- elevated: "#ffffff",
162
- // Dark semi-transparent overlay
163
- overlay: "rgba(27, 31, 36, 0.5)"
164
- },
165
- text: {
166
- // Near-black for primary text
167
- primary: "#24292f",
168
- // Gray for secondary/muted text
169
- secondary: "#57606a",
170
- // Light gray for disabled text
171
- disabled: "#8c959f",
172
- // White for text on dark backgrounds
173
- inverse: "#ffffff",
174
- // Blue for links
175
- link: "#0969da"
176
- },
177
- border: {
178
- // Light gray border (GitHub style)
179
- default: "#d0d7de",
180
- // Medium gray on hover
181
- hover: "#8c959f",
182
- // Blue focus ring
183
- focus: "#0969da",
184
- // Subtle divider
185
- divider: "#d8dee4"
186
- }
187
- },
188
- typography: {
189
- families: {
190
- // System UI font stack (GitHub/Apple style)
191
- sans: '-apple-system, BlinkMacSystemFont, "Segoe UI", "Noto Sans", Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji"',
192
- // Monospace stack
193
- mono: 'ui-monospace, SFMono-Regular, SF Mono, Menlo, Consolas, "Liberation Mono", monospace'
194
- },
195
- sizes: {
196
- xs: "0.75rem",
197
- // 12px
198
- sm: "0.875rem",
199
- // 14px
200
- base: "1rem",
201
- // 16px
202
- lg: "1.125rem",
203
- // 18px
204
- xl: "1.25rem",
205
- // 20px
206
- "2xl": "1.5rem",
207
- // 24px
208
- "3xl": "1.875rem",
209
- // 30px
210
- "4xl": "2.25rem"
211
- // 36px
212
- },
213
- weights: {
214
- normal: "400",
215
- medium: "500",
216
- semibold: "600",
217
- bold: "700"
218
- }
219
- },
220
- radius: {
221
- none: "0",
222
- sm: "3px",
223
- // GitHub uses smaller radii
224
- md: "6px",
225
- lg: "8px",
226
- xl: "12px",
227
- "2xl": "16px",
228
- full: "9999px"
229
- },
230
- shadows: {
231
- // Subtle shadows with gray tones
232
- sm: "0 1px 0 rgba(27, 31, 36, 0.04)",
233
- md: "0 3px 6px rgba(140, 149, 159, 0.15)",
234
- lg: "0 8px 24px rgba(140, 149, 159, 0.2)",
235
- xl: "0 12px 28px rgba(140, 149, 159, 0.3)"
236
- },
237
- components: {
238
- button: {
239
- radius: "6px",
240
- paddingX: "16px",
241
- paddingY: "5px",
242
- fontSize: "14px",
243
- fontWeight: "500"
244
- },
245
- card: {
246
- radius: "6px",
247
- padding: "16px",
248
- shadow: "0 1px 0 rgba(27, 31, 36, 0.04)",
249
- borderWidth: "1px"
250
- },
251
- input: {
252
- radius: "6px",
253
- paddingX: "12px",
254
- paddingY: "5px",
255
- borderWidth: "1px",
256
- focusRingWidth: "3px"
257
- }
258
- },
259
- cdn: {
260
- fonts: {
261
- preconnect: ["https://fonts.googleapis.com", "https://fonts.gstatic.com"],
262
- stylesheets: [
263
- // System UI fonts don't need external stylesheets, but we include
264
- // Inter as an optional enhancement for consistent cross-platform rendering
265
- "https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700&display=swap"
266
- ]
267
- },
268
- icons: {
269
- script: {
270
- url: "https://cdn.jsdelivr.net/npm/lucide@0.294.0/dist/umd/lucide.min.js"
271
- }
272
- },
273
- scripts: {
274
- tailwind: "https://cdn.jsdelivr.net/npm/@tailwindcss/browser@4",
275
- htmx: {
276
- url: "https://cdnjs.cloudflare.com/ajax/libs/htmx/2.0.7/htmx.min.js",
277
- integrity: "sha512-T6VLg/MJYMbLTmQ8VLvonbWg8VOvmDhXcOvHzCwo6ShdGuUU5SEcp1IAPXL4k9lVoMi8gRXl5K/S/zh43Y9rJA=="
278
- },
279
- alpine: {
280
- url: "https://cdn.jsdelivr.net/npm/alpinejs@3.14.3/dist/cdn.min.js",
281
- integrity: "sha384-6zY8MFQJ/EqS1r4RJl+7j8rvZPuBWpT0RzWf+IFcKhxqUzQNmJzA1X1VEVZhYaEz"
282
- }
283
- }
284
- }
285
- };
286
- var DEFAULT_THEME = GITHUB_OPENAI_THEME;
287
-
288
- // libs/uipack/src/theme/theme.ts
289
- function emitColorScale(lines, name, scale) {
290
- for (const [shade, value] of Object.entries(scale)) {
291
- if (value) lines.push(`--color-${name}-${shade}: ${value};`);
292
- }
293
- }
294
- var OPACITY_VARIANTS = [10, 20, 30, 50, 70, 90];
295
- function emitColorWithOpacityVariants(lines, name, value) {
296
- lines.push(`--color-${name}: ${value};`);
297
- for (const opacity of OPACITY_VARIANTS) {
298
- lines.push(`--color-${name}-${opacity}: color-mix(in oklch, ${value} ${opacity}%, transparent);`);
299
- }
300
- }
301
- function emitBrandColorWithVariants(lines, name, value) {
302
- lines.push(`--color-${name}: ${value};`);
303
- lines.push(`--color-${name}-hover: color-mix(in oklch, ${value} 85%, black);`);
304
- for (const opacity of OPACITY_VARIANTS) {
305
- lines.push(`--color-${name}-${opacity}: color-mix(in oklch, ${value} ${opacity}%, transparent);`);
306
- }
307
- }
308
- function buildThemeCss(theme) {
309
- const lines = [];
310
- const semantic = theme.colors.semantic;
311
- if (typeof semantic.primary === "string") {
312
- emitBrandColorWithVariants(lines, "primary", semantic.primary);
313
- } else if (semantic.primary) {
314
- emitColorScale(lines, "primary", semantic.primary);
315
- }
316
- if (semantic.secondary) {
317
- if (typeof semantic.secondary === "string") {
318
- emitBrandColorWithVariants(lines, "secondary", semantic.secondary);
319
- } else {
320
- emitColorScale(lines, "secondary", semantic.secondary);
321
- }
322
- }
323
- if (semantic.accent) {
324
- if (typeof semantic.accent === "string") {
325
- lines.push(`--color-accent: ${semantic.accent};`);
326
- } else {
327
- emitColorScale(lines, "accent", semantic.accent);
328
- }
329
- }
330
- if (semantic.neutral) {
331
- if (typeof semantic.neutral === "string") {
332
- lines.push(`--color-neutral: ${semantic.neutral};`);
333
- } else {
334
- emitColorScale(lines, "neutral", semantic.neutral);
335
- }
336
- }
337
- if (semantic.success) emitColorWithOpacityVariants(lines, "success", semantic.success);
338
- if (semantic.warning) emitColorWithOpacityVariants(lines, "warning", semantic.warning);
339
- if (semantic.danger) emitColorWithOpacityVariants(lines, "danger", semantic.danger);
340
- if (semantic.info) emitColorWithOpacityVariants(lines, "info", semantic.info);
341
- const surface = theme.colors.surface;
342
- if (surface?.background) lines.push(`--color-background: ${surface.background};`);
343
- if (surface?.surface) lines.push(`--color-surface: ${surface.surface};`);
344
- if (surface?.elevated) lines.push(`--color-elevated: ${surface.elevated};`);
345
- if (surface?.overlay) lines.push(`--color-overlay: ${surface.overlay};`);
346
- const text = theme.colors.text;
347
- if (text?.primary) lines.push(`--color-text-primary: ${text.primary};`);
348
- if (text?.secondary) lines.push(`--color-text-secondary: ${text.secondary};`);
349
- if (text?.disabled) lines.push(`--color-text-disabled: ${text.disabled};`);
350
- if (text?.inverse) lines.push(`--color-text-inverse: ${text.inverse};`);
351
- if (text?.link) lines.push(`--color-text-link: ${text.link};`);
352
- const border = theme.colors.border;
353
- if (border?.default) lines.push(`--color-border: ${border.default};`);
354
- if (border?.hover) lines.push(`--color-border-hover: ${border.hover};`);
355
- if (border?.focus) lines.push(`--color-border-focus: ${border.focus};`);
356
- if (border?.divider) lines.push(`--color-divider: ${border.divider};`);
357
- if (theme.colors.custom) {
358
- for (const [key, value] of Object.entries(theme.colors.custom)) {
359
- lines.push(`--color-${key}: ${value};`);
360
- }
361
- }
362
- const typography = theme.typography;
363
- if (typography?.families?.sans) lines.push(`--font-sans: ${typography.families.sans};`);
364
- if (typography?.families?.serif) lines.push(`--font-serif: ${typography.families.serif};`);
365
- if (typography?.families?.mono) lines.push(`--font-mono: ${typography.families.mono};`);
366
- if (typography?.families?.display) lines.push(`--font-display: ${typography.families.display};`);
367
- const radius = theme.radius;
368
- if (radius?.none) lines.push(`--radius-none: ${radius.none};`);
369
- if (radius?.sm) lines.push(`--radius-sm: ${radius.sm};`);
370
- if (radius?.md) lines.push(`--radius-md: ${radius.md};`);
371
- if (radius?.lg) lines.push(`--radius-lg: ${radius.lg};`);
372
- if (radius?.xl) lines.push(`--radius-xl: ${radius.xl};`);
373
- if (radius?.["2xl"]) lines.push(`--radius-2xl: ${radius["2xl"]};`);
374
- if (radius?.full) lines.push(`--radius-full: ${radius.full};`);
375
- if (theme.customVars) {
376
- for (const [key, value] of Object.entries(theme.customVars)) {
377
- lines.push(`${key}: ${value};`);
378
- }
379
- }
380
- return lines.join("\n ");
381
- }
382
- function buildStyleBlock(theme) {
383
- const themeCss = buildThemeCss(theme);
384
- const customCss = theme.customCss || "";
385
- return `<style type="text/tailwindcss">
386
- @theme {
387
- ${themeCss}
388
- }
389
- ${customCss}
390
- </style>`;
391
- }
392
-
393
- // libs/uipack/src/base-template/theme-styles.ts
394
- function renderThemeStyles(options = {}) {
395
- const {
396
- theme = DEFAULT_THEME,
397
- tailwind = true,
398
- htmx = false,
399
- alpine = false,
400
- fonts = true,
401
- inline = false
402
- } = options;
403
- const parts = [];
404
- if (fonts && !inline) {
405
- parts.push(buildFontPreconnectFromTheme(theme));
406
- }
407
- if (fonts && !inline) {
408
- parts.push(buildFontStylesheetsFromTheme(theme));
409
- }
410
- const scriptOptions = {
411
- tailwind,
412
- htmx,
413
- alpine,
414
- inline
415
- };
416
- parts.push(buildCdnScriptsFromTheme(theme, scriptOptions));
417
- parts.push(buildStyleBlock(theme));
418
- return parts.filter(Boolean).join("\n ");
419
- }
420
- function renderMinimalThemeStyles(theme = DEFAULT_THEME) {
421
- return renderThemeStyles({
422
- theme,
423
- tailwind: true,
424
- htmx: false,
425
- alpine: false,
426
- fonts: false,
427
- inline: false
428
- });
429
- }
430
- function renderThemeCssOnly(theme = DEFAULT_THEME) {
431
- return buildStyleBlock(theme);
432
- }
433
-
434
- // libs/uipack/src/base-template/polyfills.ts
435
- function renderMcpSessionPolyfill(mcpSession) {
436
- const fallbackCode = mcpSession ? `
437
- return {
438
- mcpUrl: '${escapeJs(mcpSession.mcpUrl)}',
439
- sessionId: '${escapeJs(mcpSession.sessionId)}'
440
- };` : `
441
- return null;`;
442
- return `<script>
443
- (function() {
444
- 'use strict';
445
-
446
- window.__frontmcp = window.__frontmcp || {};
447
-
448
- /**
449
- * Auto-detect MCP session from page context.
450
- * Tries multiple sources in order of priority.
451
- */
452
- window.__frontmcp.detectMcpSession = function() {
453
- // 1. Check if explicitly set by developer
454
- if (window.__frontmcp.mcpSession) {
455
- return window.__frontmcp.mcpSession;
456
- }
457
-
458
- // 2. Check meta tags
459
- var mcpUrlMeta = document.querySelector('meta[name="mcp-url"]');
460
- var sessionIdMeta = document.querySelector('meta[name="mcp-session-id"]');
461
- if (mcpUrlMeta && sessionIdMeta) {
462
- return {
463
- mcpUrl: mcpUrlMeta.getAttribute('content'),
464
- sessionId: sessionIdMeta.getAttribute('content')
465
- };
466
- }
467
-
468
- // 3. Check URL query parameters (for ngrok/iframe scenarios)
469
- try {
470
- var params = new URLSearchParams(window.location.search);
471
- var mcpUrl = params.get('mcpUrl');
472
- var sessionId = params.get('sessionId');
473
- if (mcpUrl && sessionId) {
474
- return { mcpUrl: mcpUrl, sessionId: sessionId };
475
- }
476
- } catch (e) {
477
- // URLSearchParams may not be available in all environments
478
- }
479
-
480
- // 4. Use explicit fallback if provided at render time
481
- ${fallbackCode}
482
- };
483
-
484
- /**
485
- * Enhanced callTool that uses HTTP fallback when MCP bridge is unavailable.
486
- * This wraps window.mcpBridge.callTool with HTTP fallback support.
487
- */
488
- window.__frontmcp.callTool = async function(toolName, args) {
489
- // Priority 1: Direct OpenAI SDK call (most reliable in OpenAI iframe)
490
- if (typeof window !== 'undefined' && window.openai && typeof window.openai.callTool === 'function') {
491
- return window.openai.callTool(toolName, args);
492
- }
493
-
494
- // Priority 2: If MCP bridge has callTool, use it
495
- if (window.mcpBridge && typeof window.mcpBridge.callTool === 'function') {
496
- try {
497
- return await window.mcpBridge.callTool(toolName, args);
498
- } catch (e) {
499
- // If MCP bridge fails, fall through to HTTP fallback
500
- console.warn('MCP bridge callTool failed, trying HTTP fallback:', e.message);
501
- }
502
- }
503
-
504
- // Priority 3: HTTP fallback using detected session
505
- var session = window.__frontmcp.detectMcpSession();
506
- if (!session) {
507
- throw new Error(
508
- 'MCP session not available. Set window.__frontmcp.mcpSession, ' +
509
- 'add meta tags, or provide URL parameters.'
510
- );
511
- }
512
-
513
- var response = await fetch(session.mcpUrl, {
514
- method: 'POST',
515
- headers: {
516
- 'Content-Type': 'application/json',
517
- 'X-Session-Id': session.sessionId
518
- },
519
- body: JSON.stringify({
520
- jsonrpc: '2.0',
521
- method: 'tools/call',
522
- params: { name: toolName, arguments: args || {} },
523
- id: Date.now()
524
- })
525
- });
526
-
527
- if (!response.ok) {
528
- throw new Error('HTTP ' + response.status + ': ' + response.statusText);
529
- }
530
-
531
- var result = await response.json();
532
- if (result.error) {
533
- throw new Error(result.error.message || 'Tool call failed');
534
- }
535
- return result.result;
536
- };
537
-
538
- /**
539
- * Get tool output from all possible sources.
540
- * Priority: pre-rendered HTML from metadata > structured data > raw output
541
- *
542
- * Note: Returns undefined for null values from OpenAI (which is its initial state)
543
- * to allow proper loading state handling in the bridge.
544
- */
545
- window.__frontmcp.getToolOutput = function() {
546
- // 1. Check for pre-rendered HTML in OpenAI toolResponseMetadata (highest priority)
547
- // This is set when the tool has a UI template (React, HTML, etc.)
548
- if (window.openai && window.openai.toolResponseMetadata) {
549
- var html = window.openai.toolResponseMetadata['ui/html'];
550
- if (html && typeof html === 'string') {
551
- return html; // Return HTML string directly
552
- }
553
- }
554
-
555
- // 2. Check for pre-rendered HTML in MCP response metadata (for MCP Inspector, etc.)
556
- if (window.__mcpResponseMeta) {
557
- var mcpHtml = window.__mcpResponseMeta['ui/html'];
558
- if (mcpHtml && typeof mcpHtml === 'string') {
559
- return mcpHtml; // Return HTML string directly
560
- }
561
- }
562
-
563
- // 3. OpenAI injects structured data into window.openai.toolOutput
564
- // Skip null - that's OpenAI's initial state before real data is injected
565
- if (window.openai && window.openai.toolOutput !== undefined && window.openai.toolOutput !== null) {
566
- return window.openai.toolOutput;
567
- }
568
-
569
- // 4. MCP Bridge stores in window.__mcpToolOutput (skip null)
570
- if (window.__mcpToolOutput !== undefined && window.__mcpToolOutput !== null) {
571
- return window.__mcpToolOutput;
572
- }
573
-
574
- // 5. Explicit injection
575
- if (window.__frontmcp.toolOutput !== undefined) {
576
- return window.__frontmcp.toolOutput;
577
- }
578
-
579
- return undefined;
580
- };
581
- })();
582
- </script>`;
583
- }
584
- function escapeJs(str) {
585
- return str.replace(/\\/g, "\\\\").replace(/'/g, "\\'").replace(/"/g, '\\"').replace(/`/g, "\\`").replace(/\n/g, "\\n").replace(/\r/g, "\\r").replace(/<\/script/gi, "<\\/script").replace(/\u2028/g, "\\u2028").replace(/\u2029/g, "\\u2029");
586
- }
587
-
588
- // libs/uipack/src/base-template/bridge.ts
589
- function renderBridgeScript() {
590
- return `<script>
591
- (function() {
592
- 'use strict';
593
-
594
- // ============================================
595
- // Debug Mode
596
- // ============================================
597
-
598
- var DEBUG = window.location.search.indexOf('frontmcp_debug=1') > -1 ||
599
- window.localStorage.getItem('frontmcp_debug') === '1';
600
- var debugLog = [];
601
-
602
- function log(level, message, data) {
603
- var entry = {
604
- ts: new Date().toISOString(),
605
- level: level,
606
- message: message,
607
- data: data
608
- };
609
- debugLog.push(entry);
610
- if (DEBUG) {
611
- console[level === 'error' ? 'error' : level === 'warn' ? 'warn' : 'log'](
612
- '[frontmcp] ' + message,
613
- data || ''
614
- );
615
- }
616
- }
617
-
618
- // ============================================
619
- // Bridge State Management
620
- // ============================================
621
-
622
- var state = { data: null, loading: true, error: null };
623
- var subscribers = [];
624
- var stateVersion = 0; // For React useSyncExternalStore stability
625
-
626
- function notify() {
627
- stateVersion++;
628
- for (var i = 0; i < subscribers.length; i++) {
629
- try { subscribers[i](); } catch(e) {
630
- log('error', 'Subscriber error:', { error: e.message, stack: e.stack });
631
- }
632
- }
633
- // Dispatch custom event for HTMX
634
- if (typeof CustomEvent !== 'undefined') {
635
- document.dispatchEvent(new CustomEvent('frontmcp:change', { detail: state }));
636
- }
637
- }
638
-
639
- function setData(data) {
640
- // Skip if data hasn't actually changed (prevents unnecessary re-renders)
641
- if (!state.loading && state.data === data) return;
642
- log('info', 'setData called', {
643
- dataType: typeof data,
644
- dataLength: typeof data === 'string' ? data.length : 'N/A',
645
- isNull: data === null,
646
- isUndefined: data === undefined
647
- });
648
- state = { data: data, loading: false, error: null };
649
- notify();
650
- }
651
-
652
- function setError(error) {
653
- log('error', 'setError called', { error: error });
654
- state = { data: null, loading: false, error: error };
655
- notify();
656
- }
657
-
658
- function reset() {
659
- log('info', 'reset called');
660
- state = { data: null, loading: true, error: null };
661
- notify();
662
- }
663
-
664
- // ============================================
665
- // Data Validation
666
- // ============================================
667
-
668
- function validateToolOutput(data, source) {
669
- var errors = [];
670
-
671
- if (data === undefined) {
672
- errors.push('toolOutput is undefined');
673
- } else if (data === null) {
674
- // null is valid initial state from OpenAI
675
- log('info', 'toolOutput is null (OpenAI initial state)', { source: source });
676
- return { valid: false, errors: ['initial_state'], isInitial: true };
677
- }
678
-
679
- if (typeof data === 'object' && data !== null) {
680
- if (data.isError === true) {
681
- errors.push('Tool returned error: ' + JSON.stringify(data));
682
- }
683
- // Check for MCP error structure
684
- if (data.error && data.error.code) {
685
- errors.push('MCP error: code=' + data.error.code + ', message=' + (data.error.message || 'unknown'));
686
- }
687
- }
688
-
689
- if (errors.length > 0) {
690
- log('warn', 'Validation issues', { source: source, errors: errors });
691
- }
692
-
693
- return { valid: errors.length === 0, errors: errors, isInitial: false };
694
- }
695
-
696
- function validateMetadata(meta, source) {
697
- var errors = [];
698
-
699
- if (!meta) {
700
- errors.push('toolResponseMetadata is null/undefined');
701
- return { valid: false, errors: errors };
702
- }
703
-
704
- if (typeof meta !== 'object') {
705
- errors.push('toolResponseMetadata is not an object: ' + typeof meta);
706
- return { valid: false, errors: errors };
707
- }
708
-
709
- var html = meta['ui/html'];
710
- if (html !== undefined) {
711
- if (typeof html !== 'string') {
712
- errors.push('ui/html is not a string: ' + typeof html);
713
- } else if (html.length === 0) {
714
- errors.push('ui/html is empty string');
715
- } else if (html.indexOf('validation-error') > -1) {
716
- errors.push('ui/html contains validation error');
717
- log('warn', 'HTML contains validation error', { htmlPreview: html.substring(0, 500) });
718
- }
719
- }
720
-
721
- if (errors.length > 0) {
722
- log('warn', 'Metadata validation issues', { source: source, errors: errors, keys: Object.keys(meta) });
723
- }
724
-
725
- return { valid: errors.length === 0, errors: errors };
726
- }
727
-
728
- // ============================================
729
- // OpenAI Platform Interceptors
730
- // ============================================
731
-
732
- var openaiIntercepted = false;
733
-
734
- function checkOpenAIData(openai) {
735
- if (!openai) {
736
- log('warn', 'checkOpenAIData: openai object is null');
737
- return false;
738
- }
739
-
740
- log('info', 'checkOpenAIData called', {
741
- hasToolOutput: openai.toolOutput !== undefined,
742
- hasToolResponseMetadata: openai.toolResponseMetadata !== undefined,
743
- toolOutputType: typeof openai.toolOutput,
744
- metadataKeys: openai.toolResponseMetadata ? Object.keys(openai.toolResponseMetadata) : []
745
- });
746
-
747
- // Priority 1: Pre-rendered HTML from toolResponseMetadata
748
- if (openai.toolResponseMetadata) {
749
- var metaValidation = validateMetadata(openai.toolResponseMetadata, 'openai.toolResponseMetadata');
750
- var html = openai.toolResponseMetadata['ui/html'];
751
- if (html && typeof html === 'string') {
752
- log('info', 'Using ui/html from metadata', { htmlLength: html.length });
753
- setData(html);
754
- return true;
755
- }
756
- }
757
-
758
- // Priority 2: Tool output (skip null - that's OpenAI's initial state)
759
- if (openai.toolOutput !== undefined && openai.toolOutput !== null) {
760
- var outputValidation = validateToolOutput(openai.toolOutput, 'openai.toolOutput');
761
- if (!outputValidation.isInitial) {
762
- log('info', 'Using toolOutput', {
763
- valid: outputValidation.valid,
764
- errors: outputValidation.errors
765
- });
766
- setData(openai.toolOutput);
767
- return true;
768
- }
769
- }
770
-
771
- log('info', 'checkOpenAIData: no data found yet');
772
- return false;
773
- }
774
-
775
- function installPropertyInterceptor(obj, prop, onSet) {
776
- var current = obj[prop];
777
- var descriptor = Object.getOwnPropertyDescriptor(obj, prop);
778
-
779
- // Don't re-intercept
780
- if (descriptor && descriptor.get && descriptor.set) return;
781
-
782
- Object.defineProperty(obj, prop, {
783
- get: function() { return current; },
784
- set: function(val) {
785
- current = val;
786
- onSet(val);
787
- },
788
- enumerable: true,
789
- configurable: true
790
- });
791
- }
792
-
793
- function installOpenAIInterceptors(openai) {
794
- if (!openai || openaiIntercepted) return;
795
- openaiIntercepted = true;
796
-
797
- // Check existing data first
798
- if (checkOpenAIData(openai)) return;
799
-
800
- // Intercept toolOutput
801
- installPropertyInterceptor(openai, 'toolOutput', function(val) {
802
- checkOpenAIData(openai);
803
- });
804
-
805
- // Intercept toolResponseMetadata
806
- installPropertyInterceptor(openai, 'toolResponseMetadata', function(val) {
807
- checkOpenAIData(openai);
808
- });
809
- }
810
-
811
- // Install interceptor on window.openai
812
- if (typeof window.openai !== 'undefined') {
813
- installOpenAIInterceptors(window.openai);
814
- } else {
815
- // OpenAI object doesn't exist yet - wait for it
816
- var pendingOpenai = undefined;
817
- Object.defineProperty(window, 'openai', {
818
- get: function() { return pendingOpenai; },
819
- set: function(val) {
820
- pendingOpenai = val;
821
- if (val) installOpenAIInterceptors(val);
822
- },
823
- enumerable: true,
824
- configurable: true
825
- });
826
- }
827
-
828
- // ============================================
829
- // MCP/Generic Platform Support
830
- // ============================================
831
-
832
- // Check for existing __frontmcp.toolOutput
833
- if (window.__frontmcp && window.__frontmcp.toolOutput !== undefined) {
834
- setData(window.__frontmcp.toolOutput);
835
- }
836
-
837
- // Check for __mcpToolOutput
838
- if (window.__mcpToolOutput !== undefined && window.__mcpToolOutput !== null) {
839
- setData(window.__mcpToolOutput);
840
- }
841
-
842
- // Check for __mcpResponseMeta
843
- if (window.__mcpResponseMeta && window.__mcpResponseMeta['ui/html']) {
844
- setData(window.__mcpResponseMeta['ui/html']);
845
- }
846
-
847
- // ============================================
848
- // Bridge API
849
- // ============================================
850
-
851
- window.__frontmcp = window.__frontmcp || {};
852
-
853
- window.__frontmcp.bridge = {
854
- getState: function() { return state; },
855
-
856
- // React useSyncExternalStore compatible
857
- getSnapshot: function() { return state; },
858
- getServerSnapshot: function() { return { data: null, loading: true, error: null }; },
859
-
860
- subscribe: function(callback) {
861
- subscribers.push(callback);
862
- return function() {
863
- var idx = subscribers.indexOf(callback);
864
- if (idx > -1) subscribers.splice(idx, 1);
865
- };
866
- },
867
-
868
- hasData: function() { return !state.loading && state.data !== null; },
869
-
870
- // Manual control (for testing/custom injection)
871
- setData: setData,
872
- setError: setError,
873
- reset: reset,
874
-
875
- // Debug API
876
- debug: {
877
- getLogs: function() { return debugLog.slice(); },
878
- getLastErrors: function() {
879
- return debugLog.filter(function(e) { return e.level === 'error' || e.level === 'warn'; });
880
- },
881
- enableDebug: function() {
882
- DEBUG = true;
883
- window.localStorage.setItem('frontmcp_debug', '1');
884
- console.log('[frontmcp] Debug mode enabled. Reload page to see all logs.');
885
- },
886
- disableDebug: function() {
887
- DEBUG = false;
888
- window.localStorage.removeItem('frontmcp_debug');
889
- },
890
- isDebugEnabled: function() { return DEBUG; },
891
- getStateHistory: function() {
892
- return {
893
- current: state,
894
- version: stateVersion,
895
- subscriberCount: subscribers.length,
896
- openaiIntercepted: openaiIntercepted,
897
- pollCount: pollCount
898
- };
899
- },
900
- dumpAll: function() {
901
- console.group('[frontmcp] Debug Dump');
902
- console.log('State:', state);
903
- console.log('Version:', stateVersion);
904
- console.log('Subscribers:', subscribers.length);
905
- console.log('OpenAI Intercepted:', openaiIntercepted);
906
- console.log('Poll Count:', pollCount);
907
- console.log('Debug Logs:', debugLog);
908
- console.groupEnd();
909
- return {
910
- state: state,
911
- logs: debugLog,
912
- version: stateVersion,
913
- openaiIntercepted: openaiIntercepted
914
- };
915
- }
916
- }
917
- };
918
-
919
- // ============================================
920
- // Fallback Polling (for platforms that don't trigger setters)
921
- // ============================================
922
-
923
- var pollCount = 0;
924
- var maxPolls = 30; // ~3 seconds with exponential backoff
925
-
926
- function pollForData() {
927
- if (state.data !== null || pollCount >= maxPolls) return;
928
- pollCount++;
929
-
930
- // Check all sources
931
- if (window.openai) {
932
- if (checkOpenAIData(window.openai)) return;
933
- }
934
-
935
- // Use getToolOutput if available
936
- if (window.__frontmcp.getToolOutput) {
937
- var data = window.__frontmcp.getToolOutput();
938
- if (data !== undefined && data !== null) {
939
- setData(data);
940
- return;
941
- }
942
- }
943
-
944
- // Exponential backoff: 50ms, 75ms, 112ms, ...
945
- var delay = Math.min(50 * Math.pow(1.5, Math.min(pollCount, 10)), 500);
946
- setTimeout(pollForData, delay);
947
- }
948
-
949
- // Start polling after a brief delay (give interceptors time to fire)
950
- setTimeout(pollForData, 50);
951
- })();
952
- </script>`;
953
- }
954
- var BRIDGE_TYPES = `
955
- interface BridgeState<T = unknown> {
956
- data: T | null;
957
- loading: boolean;
958
- error: string | null;
959
- }
960
-
961
- interface PlatformBridge<T = unknown> {
962
- getState(): BridgeState<T>;
963
- subscribe(callback: () => void): () => void;
964
- getSnapshot(): BridgeState<T>;
965
- getServerSnapshot(): BridgeState<T>;
966
- hasData(): boolean;
967
- setData(data: T): void;
968
- setError(error: string): void;
969
- reset(): void;
970
- }
971
-
972
- declare global {
973
- interface Window {
974
- __frontmcp: {
975
- bridge: PlatformBridge;
976
- // ... other __frontmcp methods
977
- };
978
- }
979
- }
980
- `;
981
-
982
- // libs/uipack/src/base-template/default-base-template.ts
983
- function escapeAttr(str) {
984
- return str.replace(/&/g, "&amp;").replace(/"/g, "&quot;").replace(/'/g, "&#39;").replace(/</g, "&lt;").replace(/>/g, "&gt;");
985
- }
986
- function createDefaultBaseTemplate(options) {
987
- const {
988
- toolName,
989
- theme = DEFAULT_THEME,
990
- mcpSession,
991
- htmx = false,
992
- alpine = false,
993
- fonts = true,
994
- inline = false,
995
- headContent = "",
996
- bodyClass = "bg-transparent font-sans antialiased",
997
- containerClass = "p-4"
998
- } = options;
999
- const themeStylesOptions = {
1000
- theme,
1001
- tailwind: true,
1002
- htmx,
1003
- alpine,
1004
- fonts,
1005
- inline
1006
- };
1007
- const themeStyles = renderThemeStyles(themeStylesOptions);
1008
- const polyfills = renderMcpSessionPolyfill(mcpSession);
1009
- const bridge = renderBridgeScript();
1010
- return `<!DOCTYPE html>
1011
- <html lang="en">
1012
- <head>
1013
- <meta charset="utf-8">
1014
- <meta name="viewport" content="width=device-width, initial-scale=1">
1015
- <meta name="color-scheme" content="light dark">
1016
- <title>${escapeAttr(toolName)} Widget</title>
1017
- ${themeStyles}
1018
- ${polyfills}
1019
- ${bridge}
1020
- ${headContent}
1021
- </head>
1022
- <body class="${escapeAttr(bodyClass)}">
1023
- <div id="widget-root" class="${escapeAttr(containerClass)}"></div>
1024
-
1025
- <script>
1026
- (function() {
1027
- 'use strict';
1028
-
1029
- var root = document.getElementById('widget-root');
1030
-
1031
- /**
1032
- * Render the widget based on bridge state.
1033
- * Uses the reactive bridge to automatically re-render when data changes.
1034
- */
1035
- function render() {
1036
- var state = window.__frontmcp.bridge.getState();
1037
-
1038
- // Loading state
1039
- if (state.loading) {
1040
- root.innerHTML = renderLoading();
1041
- return;
1042
- }
1043
-
1044
- // Error state
1045
- if (state.error) {
1046
- root.innerHTML = renderError(state.error);
1047
- return;
1048
- }
1049
-
1050
- // No data state
1051
- if (state.data === null) {
1052
- root.innerHTML = renderEmpty();
1053
- return;
1054
- }
1055
-
1056
- // Render data
1057
- try {
1058
- // Check for custom renderer provided by developer
1059
- if (window.__frontmcp && typeof window.__frontmcp.renderContent === 'function') {
1060
- root.innerHTML = window.__frontmcp.renderContent(state.data);
1061
- return;
1062
- }
1063
-
1064
- // Fall back to default renderer
1065
- root.innerHTML = defaultRenderer(state.data);
1066
- } catch (e) {
1067
- console.error('[frontmcp] Error rendering widget:', e);
1068
- root.innerHTML = renderError(e.message || 'Render error');
1069
- }
1070
- }
1071
-
1072
- /**
1073
- * Default renderer for tool output.
1074
- * Handles both pre-rendered HTML strings and raw JSON data.
1075
- */
1076
- function defaultRenderer(data) {
1077
- // Check if data is pre-rendered HTML (server-side rendered widget)
1078
- if (typeof data === 'string' && data.trim().startsWith('<')) {
1079
- // Direct HTML injection - content was already rendered/sanitized server-side
1080
- return data;
1081
- }
1082
-
1083
- // Check for special wrapper with HTML content
1084
- if (data && typeof data === 'object' && data.__html) {
1085
- return data.__html;
1086
- }
1087
-
1088
- // Fallback: JSON renderer for raw data
1089
- var json = JSON.stringify(data, null, 2);
1090
- return '<pre class="p-4 bg-surface rounded-md overflow-auto text-sm font-mono text-text-primary border border-border">' +
1091
- escapeHtml(json) +
1092
- '</pre>';
1093
- }
1094
-
1095
- /**
1096
- * Render loading state with animated spinner.
1097
- */
1098
- function renderLoading() {
1099
- return '<div class="flex items-center justify-center p-8">' +
1100
- '<div class="flex flex-col items-center gap-3">' +
1101
- '<svg class="animate-spin h-8 w-8 text-blue-500" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24">' +
1102
- '<circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>' +
1103
- '<path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>' +
1104
- '</svg>' +
1105
- '<p class="text-text-secondary text-sm">Loading...</p>' +
1106
- '</div>' +
1107
- '</div>';
1108
- }
1109
-
1110
- /**
1111
- * Render error message.
1112
- */
1113
- function renderError(message) {
1114
- return '<div class="p-4 bg-red-50 border border-red-200 rounded-md">' +
1115
- '<p class="text-red-600 text-sm">Error: ' + escapeHtml(message) + '</p>' +
1116
- '</div>';
1117
- }
1118
-
1119
- /**
1120
- * Render empty state (no data available).
1121
- */
1122
- function renderEmpty() {
1123
- return '<div class="p-4 text-text-secondary text-sm text-center">' +
1124
- 'No data available' +
1125
- '</div>';
1126
- }
1127
-
1128
- /**
1129
- * Escape HTML special characters.
1130
- */
1131
- function escapeHtml(str) {
1132
- if (typeof str !== 'string') return str;
1133
- return str
1134
- .replace(/&/g, '&amp;')
1135
- .replace(/</g, '&lt;')
1136
- .replace(/>/g, '&gt;')
1137
- .replace(/"/g, '&quot;');
1138
- }
1139
-
1140
- /**
1141
- * Create debug overlay panel.
1142
- * Shows bridge state, logs, and OpenAI data status.
1143
- */
1144
- function createDebugOverlay() {
1145
- var overlay = document.createElement('div');
1146
- overlay.id = 'frontmcp-debug-overlay';
1147
- overlay.style.cssText = 'position:fixed;bottom:10px;right:10px;width:400px;max-height:60vh;' +
1148
- 'background:rgba(0,0,0,0.9);color:#0f0;font-family:monospace;font-size:11px;' +
1149
- 'padding:10px;border-radius:4px;z-index:99999;overflow:auto;display:none;';
1150
-
1151
- var toggle = document.createElement('button');
1152
- toggle.id = 'frontmcp-debug-toggle';
1153
- toggle.textContent = 'Debug';
1154
- toggle.style.cssText = 'position:fixed;bottom:10px;right:10px;padding:4px 8px;' +
1155
- 'background:#333;color:#0f0;font-family:monospace;font-size:10px;border:1px solid #0f0;' +
1156
- 'border-radius:4px;z-index:100000;cursor:pointer;';
1157
-
1158
- toggle.onclick = function() {
1159
- var o = document.getElementById('frontmcp-debug-overlay');
1160
- if (o.style.display === 'none') {
1161
- o.style.display = 'block';
1162
- toggle.style.right = '420px';
1163
- updateDebugOverlay();
1164
- } else {
1165
- o.style.display = 'none';
1166
- toggle.style.right = '10px';
1167
- }
1168
- };
1169
-
1170
- document.body.appendChild(overlay);
1171
- document.body.appendChild(toggle);
1172
-
1173
- // Auto-update every second
1174
- setInterval(updateDebugOverlay, 1000);
1175
- }
1176
-
1177
- function updateDebugOverlay() {
1178
- var overlay = document.getElementById('frontmcp-debug-overlay');
1179
- if (!overlay || overlay.style.display === 'none') return;
1180
-
1181
- var state = window.__frontmcp.bridge.getState();
1182
- var debug = window.__frontmcp.bridge.debug;
1183
- var logs = debug.getLogs();
1184
- var errors = debug.getLastErrors();
1185
- var stateHistory = debug.getStateHistory();
1186
-
1187
- var html = '<div style="margin-bottom:10px;border-bottom:1px solid #333;padding-bottom:5px;">' +
1188
- '<strong>FrontMCP Debug Panel</strong>' +
1189
- '<span style="float:right;color:#666;">v' + stateHistory.version + '</span>' +
1190
- '</div>';
1191
-
1192
- // Bridge State
1193
- html += '<div style="margin-bottom:10px;">' +
1194
- '<div style="color:#ff0;">Bridge State:</div>' +
1195
- '<div>Loading: ' + state.loading + '</div>' +
1196
- '<div>Has Data: ' + (state.data !== null) + '</div>' +
1197
- '<div>Data Type: ' + (state.data === null ? 'null' : typeof state.data) + '</div>' +
1198
- (typeof state.data === 'string' ? '<div>Data Length: ' + state.data.length + '</div>' : '') +
1199
- '<div>Error: ' + (state.error || 'none') + '</div>' +
1200
- '</div>';
1201
-
1202
- // OpenAI Status
1203
- html += '<div style="margin-bottom:10px;">' +
1204
- '<div style="color:#ff0;">OpenAI Status:</div>' +
1205
- '<div>Intercepted: ' + stateHistory.openaiIntercepted + '</div>' +
1206
- '<div>Poll Count: ' + stateHistory.pollCount + '</div>';
1207
-
1208
- if (window.openai) {
1209
- html += '<div>window.openai: exists</div>' +
1210
- '<div>toolOutput: ' + (window.openai.toolOutput !== undefined ? 'present' : 'undefined') + '</div>' +
1211
- '<div>toolResponseMetadata: ' + (window.openai.toolResponseMetadata !== undefined ? 'present' : 'undefined') + '</div>';
1212
- if (window.openai.toolResponseMetadata) {
1213
- html += '<div>meta keys: ' + Object.keys(window.openai.toolResponseMetadata).join(', ') + '</div>';
1214
- }
1215
- } else {
1216
- html += '<div>window.openai: undefined</div>';
1217
- }
1218
- html += '</div>';
1219
-
1220
- // Errors
1221
- if (errors.length > 0) {
1222
- html += '<div style="margin-bottom:10px;">' +
1223
- '<div style="color:#f00;">Errors (' + errors.length + '):</div>';
1224
- errors.slice(-5).forEach(function(e) {
1225
- html += '<div style="color:#f88;font-size:10px;">[' + e.level + '] ' + escapeHtml(e.message) + '</div>';
1226
- });
1227
- html += '</div>';
1228
- }
1229
-
1230
- // Recent Logs
1231
- html += '<div>' +
1232
- '<div style="color:#ff0;">Recent Logs (' + logs.length + '):</div>' +
1233
- '<div style="max-height:150px;overflow:auto;">';
1234
- logs.slice(-10).forEach(function(e) {
1235
- var color = e.level === 'error' ? '#f00' : e.level === 'warn' ? '#ff0' : '#0f0';
1236
- html += '<div style="color:' + color + ';font-size:10px;">' +
1237
- '[' + e.ts.split('T')[1].split('.')[0] + '] ' + escapeHtml(e.message) +
1238
- '</div>';
1239
- });
1240
- html += '</div></div>';
1241
-
1242
- // Actions
1243
- html += '<div style="margin-top:10px;border-top:1px solid #333;padding-top:5px;">' +
1244
- '<button onclick="console.log(window.__frontmcp.bridge.debug.dumpAll())" ' +
1245
- 'style="font-size:10px;padding:2px 6px;margin-right:5px;">Dump to Console</button>' +
1246
- '<button onclick="window.__frontmcp.bridge.reset()" ' +
1247
- 'style="font-size:10px;padding:2px 6px;">Reset Bridge</button>' +
1248
- '</div>';
1249
-
1250
- overlay.innerHTML = html;
1251
- }
1252
-
1253
- /**
1254
- * Initialize the widget with reactive rendering.
1255
- */
1256
- function init() {
1257
- // Subscribe to bridge state changes for reactive re-rendering
1258
- window.__frontmcp.bridge.subscribe(render);
1259
-
1260
- // Initial render
1261
- render();
1262
-
1263
- // Global click handler for SSR-compatible tool calls
1264
- // Handles buttons/elements with data-tool-call attribute
1265
- document.addEventListener('click', function(e) {
1266
- var target = e.target;
1267
- // Walk up the DOM to find element with data-tool-call (handles clicks on child elements)
1268
- while (target && target !== document.body) {
1269
- if (target.hasAttribute && target.hasAttribute('data-tool-call')) {
1270
- var toolName = target.getAttribute('data-tool-call');
1271
- var argsStr = target.getAttribute('data-tool-args');
1272
- var args = {};
1273
-
1274
- if (argsStr) {
1275
- try {
1276
- args = JSON.parse(argsStr);
1277
- } catch (parseErr) {
1278
- console.error('[frontmcp] Failed to parse data-tool-args:', parseErr);
1279
- }
1280
- }
1281
-
1282
- // Prevent default behavior and stop propagation
1283
- e.preventDefault();
1284
- e.stopPropagation();
1285
-
1286
- // Show loading state on the button
1287
- var originalContent = target.innerHTML;
1288
- var loadingContent = '<span style="display:inline-flex;align-items:center;">' +
1289
- '<span style="animation:spin 1s linear infinite;margin-right:4px;">\u23F3</span>' +
1290
- 'Loading...' +
1291
- '</span>';
1292
- target.innerHTML = loadingContent;
1293
- target.disabled = true;
1294
-
1295
- // Call the tool
1296
- if (window.__frontmcp && typeof window.__frontmcp.callTool === 'function') {
1297
- window.__frontmcp.callTool(toolName, args)
1298
- .then(function(result) {
1299
- console.log('[frontmcp] Tool call success:', toolName, result);
1300
- // Restore button state on success
1301
- target.innerHTML = originalContent;
1302
- target.disabled = false;
1303
- })
1304
- .catch(function(err) {
1305
- console.error('[frontmcp] Tool call failed:', toolName, err);
1306
- // Restore button state on error
1307
- target.innerHTML = originalContent;
1308
- target.disabled = false;
1309
- });
1310
- } else {
1311
- console.error('[frontmcp] callTool not available');
1312
- target.innerHTML = originalContent;
1313
- target.disabled = false;
1314
- }
1315
-
1316
- return;
1317
- }
1318
- target = target.parentElement;
1319
- }
1320
- });
1321
-
1322
- // Create debug overlay if debug mode is enabled
1323
- var DEBUG = window.location.search.indexOf('frontmcp_debug=1') > -1 ||
1324
- window.localStorage.getItem('frontmcp_debug') === '1';
1325
- if (DEBUG) {
1326
- createDebugOverlay();
1327
- }
1328
- }
1329
-
1330
- // Initialize when DOM is ready
1331
- if (document.readyState === 'loading') {
1332
- document.addEventListener('DOMContentLoaded', init);
1333
- } else {
1334
- init();
1335
- }
1336
- })();
1337
- </script>
1338
- </body>
1339
- </html>`;
1340
- }
1341
- function createMinimalBaseTemplate(toolName, mcpSession) {
1342
- return createDefaultBaseTemplate({
1343
- toolName,
1344
- mcpSession,
1345
- fonts: false,
1346
- htmx: false,
1347
- alpine: false
1348
- });
1349
- }
1350
- export {
1351
- BRIDGE_TYPES,
1352
- createDefaultBaseTemplate,
1353
- createMinimalBaseTemplate,
1354
- renderBridgeScript,
1355
- renderMcpSessionPolyfill,
1356
- renderMinimalThemeStyles,
1357
- renderThemeCssOnly,
1358
- renderThemeStyles
1359
- };