@seed-ship/mcp-ui-solid 1.0.42 → 1.1.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.
package/README.md CHANGED
@@ -1,54 +1,304 @@
1
- # @mcp-ui/solid
1
+ # @seed-ship/mcp-ui-solid
2
2
 
3
- SolidJS components for rendering MCP-generated UI resources.
3
+ SolidJS components for rendering MCP-generated UI resources. Part of the MCP UI ecosystem.
4
+
5
+ [![npm version](https://img.shields.io/npm/v/@seed-ship/mcp-ui-solid.svg)](https://www.npmjs.com/package/@seed-ship/mcp-ui-solid)
6
+ [![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)
7
+
8
+ ## Overview
9
+
10
+ `@seed-ship/mcp-ui-solid` provides a complete rendering solution for MCP (Model Context Protocol) generated UIs. It enables AI/LLM systems to generate structured, interactive dashboards that are rendered with SolidJS.
11
+
12
+ **Key Use Cases:**
13
+ - Render AI-generated dashboards and reports
14
+ - Display streaming UI components progressively
15
+ - Build interactive data visualizations from MCP resources
4
16
 
5
17
  ## Installation
6
18
 
7
19
  ```bash
8
- pnpm add @mcp-ui/solid
20
+ pnpm add @seed-ship/mcp-ui-solid
21
+ # or
22
+ npm install @seed-ship/mcp-ui-solid
9
23
  ```
10
24
 
11
- ## Usage
25
+ **Peer Dependencies:**
26
+ - `solid-js` ^1.9.0
27
+
28
+ ## Quick Start
12
29
 
13
30
  ```tsx
14
- import { UIResourceRenderer, StreamingUIRenderer } from '@mcp-ui/solid'
15
- import { useStreamingUI } from '@mcp-ui/solid/hooks'
31
+ import { UIResourceRenderer, StreamingUIRenderer } from '@seed-ship/mcp-ui-solid'
16
32
 
17
- // Static rendering
33
+ // Static rendering of a pre-built layout
18
34
  function Dashboard() {
19
35
  const layout = {
20
36
  id: 'dashboard-1',
21
37
  type: 'composite',
22
- components: [/* ... */]
38
+ components: [
39
+ {
40
+ type: 'metric',
41
+ id: 'revenue',
42
+ title: 'Total Revenue',
43
+ value: '$125,430',
44
+ trend: { direction: 'up', value: '+12%' },
45
+ position: { x: 0, y: 0, width: 4, height: 1 }
46
+ },
47
+ {
48
+ type: 'chart',
49
+ id: 'sales-chart',
50
+ chartType: 'line',
51
+ data: { /* chart data */ },
52
+ position: { x: 0, y: 1, width: 8, height: 2 }
53
+ }
54
+ ]
23
55
  }
24
56
 
25
57
  return <UIResourceRenderer content={layout} />
26
58
  }
27
59
 
28
- // Streaming rendering
60
+ // Streaming rendering from an MCP server
29
61
  function StreamingDashboard() {
30
62
  return (
31
63
  <StreamingUIRenderer
32
- query="Show me revenue trends"
33
- spaceIds={['space-1']}
34
- onComplete={(metadata) => console.log('Done!', metadata)}
64
+ query="Show me quarterly revenue trends"
65
+ spaceIds={['analytics-space']}
66
+ onComplete={(metadata) => console.log('Render complete', metadata)}
67
+ onError={(error) => console.error('Render failed', error)}
35
68
  />
36
69
  )
37
70
  }
38
71
  ```
39
72
 
40
- ## Features
73
+ ## Architecture
74
+
75
+ ```
76
+ ┌─────────────────────────────────────────────────────────────┐
77
+ │ Your SolidJS App │
78
+ ├─────────────────────────────────────────────────────────────┤
79
+ │ UIResourceRenderer │ StreamingUIRenderer │
80
+ │ (static layouts) │ (progressive SSE streaming) │
81
+ ├─────────────────────────────────────────────────────────────┤
82
+ │ Component Registry │
83
+ │ ┌─────────┬─────────┬────────┬──────────┬───────────────┐ │
84
+ │ │ Chart │ Table │ Metric │ Text │ Image/Link... │ │
85
+ │ │Renderer │Renderer │Renderer│Renderer │ Renderers │ │
86
+ │ └─────────┴─────────┴────────┴──────────┴───────────────┘ │
87
+ ├─────────────────────────────────────────────────────────────┤
88
+ │ Validation │ Error Boundaries │ Grid Layout (12-col) │
89
+ └─────────────────────────────────────────────────────────────┘
90
+ ```
91
+
92
+ ## Components
93
+
94
+ ### Core Renderers
95
+
96
+ | Component | Description | Key Props |
97
+ |-----------|-------------|-----------|
98
+ | `UIResourceRenderer` | Static layout rendering | `content`, `className` |
99
+ | `StreamingUIRenderer` | Progressive SSE streaming | `query`, `spaceIds`, `onComplete`, `onError` |
100
+ | `GenerativeUIErrorBoundary` | Error isolation with retry | `fallback`, `onError` |
101
+
102
+ ### Data Display Renderers
103
+
104
+ | Component | Type | Description |
105
+ |-----------|------|-------------|
106
+ | `ChartRenderer` | `chart` | Line, bar, pie, area charts (Chart.js) |
107
+ | `TableRenderer` | `table` | Data tables with sorting, markdown links |
108
+ | `MetricRenderer` | `metric` | KPI cards with trends and sparklines |
109
+ | `TextRenderer` | `text` | Markdown-enabled text blocks |
110
+ | `ImageRenderer` | `image` | Responsive images with lazy loading |
111
+ | `LinkRenderer` | `link` | External links with security attributes |
112
+ | `IframeRenderer` | `iframe` | Secure iframe embedding (sandboxed) |
113
+
114
+ ### Interactive Renderers
115
+
116
+ | Component | Type | Description |
117
+ |-----------|------|-------------|
118
+ | `ActionRenderer` | `action` | Interactive buttons with callbacks |
119
+ | `ArtifactRenderer` | `artifact` | File download/preview |
120
+ | `CarouselRenderer` | `carousel` | Image/content carousel |
121
+ | `FooterRenderer` | `footer` | Metadata and attribution display |
122
+
123
+ ## Exports
124
+
125
+ ### Main Export
126
+
127
+ ```typescript
128
+ import {
129
+ UIResourceRenderer,
130
+ StreamingUIRenderer,
131
+ GenerativeUIErrorBoundary,
132
+ // Individual renderers
133
+ ChartRenderer,
134
+ TableRenderer,
135
+ MetricRenderer,
136
+ TextRenderer,
137
+ ActionRenderer,
138
+ // ...
139
+ } from '@seed-ship/mcp-ui-solid'
140
+ ```
141
+
142
+ ### Hooks
143
+
144
+ ```typescript
145
+ import { useStreamingUI } from '@seed-ship/mcp-ui-solid/hooks'
146
+
147
+ const { components, isComplete, error, metadata } = useStreamingUI({
148
+ query: 'Show revenue data',
149
+ spaceIds: ['space-1']
150
+ })
151
+ ```
152
+
153
+ ### Validation (SSR-Safe)
154
+
155
+ ```typescript
156
+ // Safe to import on server - no browser APIs
157
+ import { validateUIResource, validateLayout } from '@seed-ship/mcp-ui-solid/validation'
158
+
159
+ const result = validateUIResource(resource)
160
+ if (!result.valid) {
161
+ console.error(result.errors)
162
+ }
163
+ ```
164
+
165
+ ### Types Only (SSR-Safe)
166
+
167
+ ```typescript
168
+ // Type-only imports - no runtime code
169
+ import type { UIResource, UIComponent, GridPosition } from '@seed-ship/mcp-ui-solid/types-only'
170
+ ```
171
+
172
+ ## SSR Compatibility
173
+
174
+ This package is fully SSR-compatible with SolidStart, Astro, and other SSR frameworks.
175
+
176
+ ### For SolidStart Users
177
+
178
+ Add `conditions` to your `app.config.ts` for optimal SSR behavior:
179
+
180
+ ```typescript
181
+ // app.config.ts
182
+ import { defineConfig } from '@solidjs/start/config'
183
+
184
+ export default defineConfig({
185
+ vite: {
186
+ resolve: {
187
+ conditions: ['solid', 'development', 'browser']
188
+ }
189
+ }
190
+ })
191
+ ```
192
+
193
+ ### Why This Matters
194
+
195
+ The `"solid"` condition enables Vite to use source exports, which are compiled in the same context as your app. This prevents:
196
+ - Hydration mismatches between server and client
197
+ - Module resolution conflicts with `solid-js/web`
198
+ - SSR crashes from client-only APIs
199
+
200
+ ### SSR-Safe Imports
201
+
202
+ For server-side code, use the dedicated sub-exports:
203
+
204
+ ```typescript
205
+ // Server-side file (.server.ts)
206
+ import type { UIResource } from '@seed-ship/mcp-ui-solid/types-only'
207
+ import { validateUIResource } from '@seed-ship/mcp-ui-solid/validation'
208
+
209
+ // These imports don't trigger browser APIs
210
+ ```
211
+
212
+ ## Grid System
213
+
214
+ Components use a 12-column responsive grid:
215
+
216
+ ```typescript
217
+ interface GridPosition {
218
+ x: number // Column start (0-11)
219
+ y: number // Row start
220
+ width: number // Columns span (1-12)
221
+ height: number // Rows span
222
+ }
223
+
224
+ // Example: Full-width header
225
+ { x: 0, y: 0, width: 12, height: 1 }
226
+
227
+ // Example: Two columns
228
+ { x: 0, y: 1, width: 6, height: 2 } // Left half
229
+ { x: 6, y: 1, width: 6, height: 2 } // Right half
230
+ ```
231
+
232
+ ## Advanced Usage
233
+
234
+ ### Custom Component Registry
235
+
236
+ ```typescript
237
+ import { registerComponent } from '@seed-ship/mcp-ui-solid'
238
+
239
+ // Register a custom renderer
240
+ registerComponent('custom-widget', (props) => {
241
+ return <div class="custom-widget">{props.content}</div>
242
+ })
243
+ ```
244
+
245
+ ### Error Handling
246
+
247
+ ```tsx
248
+ import { GenerativeUIErrorBoundary } from '@seed-ship/mcp-ui-solid'
249
+
250
+ <GenerativeUIErrorBoundary
251
+ fallback={(error, reset) => (
252
+ <div>
253
+ <p>Something went wrong: {error.message}</p>
254
+ <button onClick={reset}>Try Again</button>
255
+ </div>
256
+ )}
257
+ onError={(error) => {
258
+ // Log to error tracking service
259
+ Sentry.captureException(error)
260
+ }}
261
+ >
262
+ <UIResourceRenderer content={layout} />
263
+ </GenerativeUIErrorBoundary>
264
+ ```
265
+
266
+ ### Streaming with Custom SSE Endpoint
267
+
268
+ ```tsx
269
+ import { useStreamingUI } from '@seed-ship/mcp-ui-solid/hooks'
270
+
271
+ function CustomStreaming() {
272
+ const { components, isComplete } = useStreamingUI({
273
+ endpoint: '/api/custom-mcp/stream',
274
+ query: 'Generate report',
275
+ headers: {
276
+ 'Authorization': 'Bearer token'
277
+ }
278
+ })
279
+
280
+ return (
281
+ <div>
282
+ {components().map(comp => (
283
+ <UIResourceRenderer content={comp} />
284
+ ))}
285
+ </div>
286
+ )
287
+ }
288
+ ```
289
+
290
+ ## Related Packages
291
+
292
+ | Package | Description |
293
+ |---------|-------------|
294
+ | [`@seed-ship/mcp-ui-spec`](../mcp-ui-spec) | JSON schemas and Zod validators |
295
+ | [`@seed-ship/mcp-ui-cli`](../mcp-ui-cli) | CLI for validation and type generation |
41
296
 
42
- - 📊 **12-Column Grid Layout**: Responsive Bootstrap-like grid system
43
- - ⚡ **Progressive Streaming**: Components appear as they're generated
44
- - 🎨 **Smooth Animations**: Fade-in effects and skeleton loading states
45
- - 🔒 **Type Safety**: Full TypeScript support
46
- - ✅ **Validation**: Built-in component and layout validation
47
- - 🛡️ **Error Boundaries**: Graceful error handling
297
+ ## Versioning
48
298
 
49
- ## Documentation
299
+ This package follows [Semantic Versioning](https://semver.org/). See [CHANGELOG.md](./CHANGELOG.md) for release notes.
50
300
 
51
- See the [full documentation](../../docs/features/generative-ui/) for more details.
301
+ **Current Version:** 1.0.43
52
302
 
53
303
  ## License
54
304
 
@@ -84,6 +84,22 @@ function renderCellValue(value) {
84
84
  if (value === null || value === void 0) {
85
85
  return "-";
86
86
  }
87
+ if (typeof value === "object" && value !== null) {
88
+ if (value.url) {
89
+ const label = value.name || value.label || value.title || value.url;
90
+ const sanitizedLabel = purify_es.sanitize(String(label));
91
+ const sanitizedUrl = purify_es.sanitize(String(value.url));
92
+ return `<a href="${sanitizedUrl}" target="_blank" rel="noopener noreferrer" class="text-blue-600 dark:text-blue-400 hover:underline">${sanitizedLabel}</a>`;
93
+ }
94
+ if (value.name || value.label || value.title) {
95
+ return purify_es.sanitize(String(value.name || value.label || value.title));
96
+ }
97
+ try {
98
+ return JSON.stringify(value);
99
+ } catch {
100
+ return "-";
101
+ }
102
+ }
87
103
  let strValue = String(value);
88
104
  strValue = strValue.replace(/\s*[–-]\s*undefined\s*$/gi, "");
89
105
  strValue = strValue.replace(/^undefined\s*[–-]\s*/gi, "");
@@ -1 +1 @@
1
- {"version":3,"file":"UIResourceRenderer.cjs","sources":["../../src/components/UIResourceRenderer.tsx"],"sourcesContent":["/**\n * UI Resource Renderer Component\n * Phase 0: Foundation with iframe sandbox and composite grid support\n */\n\nimport DOMPurify from 'dompurify'\nimport { Component, createSignal, Show, For, createMemo, createEffect } from 'solid-js'\nimport { isServer } from 'solid-js/web'\nimport type { UIComponent, UILayout, RendererError, ComponentType } from '../types'\nimport { validateComponent, DEFAULT_RESOURCE_LIMITS } from '../services/validation'\nimport { GenerativeUIErrorBoundary } from './GenerativeUIErrorBoundary'\nimport { marked } from 'marked'\n\n/**\n * Props for UIResourceRenderer\n */\nexport interface UIResourceRendererProps {\n /**\n * Single component or full layout to render\n */\n content: UIComponent | UILayout\n\n /**\n * Lazy loading (default: true)\n */\n lazyLoad?: boolean\n\n /**\n * Error callback\n */\n onError?: (error: RendererError) => void\n\n /**\n * Custom CSS class\n */\n class?: string\n}\n\n/**\n * Render a single chart component in a sandboxed iframe\n */\nfunction ChartRenderer(props: {\n component: UIComponent\n onError?: (error: RendererError) => void\n}) {\n const [iframeUrl, setIframeUrl] = createSignal<string>()\n const [isLoading, setIsLoading] = createSignal(true)\n const [error, setError] = createSignal<string>()\n\n // Use createEffect instead of onMount for SSR compatibility\n // createEffect runs after hydration on client-side\n createEffect(() => {\n const chartParams = props.component.params as any\n if (!chartParams) return\n\n // Build Quickchart URL\n const chartConfig = {\n type: chartParams.type,\n data: chartParams.data,\n options: {\n ...chartParams.options,\n responsive: true,\n maintainAspectRatio: false,\n },\n }\n\n // Encode chart configuration for Quickchart API\n const configStr = encodeURIComponent(JSON.stringify(chartConfig))\n const url = `https://quickchart.io/chart?c=${configStr}&width=500&height=300&devicePixelRatio=2`\n\n // Validate domain (should always pass for quickchart.io)\n setIframeUrl(url)\n setIsLoading(false)\n })\n\n return (\n <div class=\"relative w-full h-full min-h-[300px] bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 overflow-hidden\">\n <Show when={isLoading()}>\n <div class=\"absolute inset-0 flex items-center justify-center\">\n <div class=\"animate-spin rounded-full h-8 w-8 border-b-2 border-blue-600\" />\n </div>\n </Show>\n\n <Show when={error()}>\n <div class=\"absolute inset-0 flex items-center justify-center p-4\">\n <div class=\"text-center\">\n <p class=\"text-red-600 dark:text-red-400 text-sm font-medium\">Chart Error</p>\n <p class=\"text-gray-600 dark:text-gray-400 text-xs mt-1\">{error()}</p>\n </div>\n </div>\n </Show>\n\n <Show when={iframeUrl() && !error()}>\n <div class=\"w-full h-full p-4\">\n <Show when={(props.component.params as any).title}>\n <h3 class=\"text-sm font-semibold text-gray-900 dark:text-white mb-3\">\n {(props.component.params as any).title}\n </h3>\n </Show>\n <div class=\"w-full h-full\">\n <img\n src={iframeUrl()}\n alt=\"Chart visualization\"\n class=\"w-full h-auto max-h-[300px] object-contain\"\n onError={() => {\n setError('Failed to load chart')\n props.onError?.({\n type: 'render',\n message: 'Chart rendering failed',\n componentId: props.component.id,\n })\n }}\n />\n </div>\n </div>\n </Show>\n </div>\n )\n}\n\n/**\n * Smart cell value renderer that handles markdown links and other formats\n */\nfunction renderCellValue(value: any): string {\n // Handle null/undefined\n if (value === null || value === undefined) {\n return '-'\n }\n\n // Convert to string\n let strValue = String(value)\n\n // Clean up \"undefined\" patterns from backend data\n // Pattern 1: \"Text – undefined\" or \"Text - undefined\" → \"Text\"\n strValue = strValue.replace(/\\s*[–-]\\s*undefined\\s*$/gi, '')\n // Pattern 2: \"undefined – Text\" or \"undefined - Text\" → \"Text\"\n strValue = strValue.replace(/^undefined\\s*[–-]\\s*/gi, '')\n // Pattern 3: standalone \"undefined\" → \"-\"\n if (strValue.trim().toLowerCase() === 'undefined') {\n return '-'\n }\n // Pattern 4: empty string after cleanup → \"-\"\n if (strValue.trim() === '') {\n return '-'\n }\n\n // Detect and convert markdown links: [text](url)\n const markdownLinkRegex = /\\[([^\\]]+)\\]\\(([^)]+)\\)/g\n if (markdownLinkRegex.test(strValue)) {\n // Replace all markdown links with HTML links\n const htmlValue = strValue.replace(\n /\\[([^\\]]+)\\]\\(([^)]+)\\)/g,\n '<a href=\"$2\" target=\"_blank\" rel=\"noopener noreferrer\" class=\"text-blue-600 dark:text-blue-400 hover:underline\">$1</a>'\n )\n return DOMPurify.sanitize(htmlValue, { ADD_ATTR: ['target', 'rel'] })\n }\n\n // Check if value contains markdown formatting (bold, italic, code, etc.)\n const hasMarkdown = /[*_`\\[\\]#]/.test(strValue)\n if (hasMarkdown) {\n // Parse with marked and sanitize\n const parsed = marked.parse(strValue, { async: false }) as string\n return DOMPurify.sanitize(parsed, { ADD_ATTR: ['target', 'rel'] })\n }\n\n // Plain text\n return strValue\n}\n\n/**\n * Render a table component\n */\nfunction TableRenderer(props: {\n component: UIComponent\n onError?: (error: RendererError) => void\n}) {\n const tableParams = props.component.params as any\n\n return (\n <div class=\"w-full h-full bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 overflow-hidden\">\n <div class=\"p-4\">\n <Show when={tableParams.title}>\n <h3 class=\"text-sm font-semibold text-gray-900 dark:text-white mb-3\">\n {tableParams.title}\n </h3>\n </Show>\n\n <div class=\"overflow-x-auto\">\n <table class=\"min-w-full divide-y divide-gray-200 dark:divide-gray-700 border-separate border-spacing-0\">\n <thead class=\"bg-gray-50 dark:bg-gray-900/50\">\n <tr>\n <For each={tableParams.columns}>\n {(column: any) => (\n <th\n scope=\"col\"\n class=\"px-6 py-3 text-left text-xs font-semibold text-gray-500 dark:text-gray-400 uppercase tracking-wider border-b border-gray-200 dark:border-gray-700 first:pl-6 last:pr-6\"\n style={column.width ? { width: column.width } : {}}\n >\n {column.label}\n </th>\n )}\n </For>\n </tr>\n </thead>\n <tbody class=\"bg-white dark:bg-gray-800 divide-y divide-gray-200 dark:divide-gray-700\">\n <For each={tableParams.rows.slice(0, DEFAULT_RESOURCE_LIMITS.maxTableRows)}>\n {(row: any, i) => (\n <tr class={`hover:bg-gray-50 dark:hover:bg-gray-700/50 transition-colors ${i() % 2 === 0 ? 'bg-white dark:bg-gray-800' : 'bg-gray-50/30 dark:bg-gray-800/50'}`}>\n <For each={tableParams.columns}>\n {(column: any) => (\n <td class=\"px-6 py-4 text-sm text-gray-700 dark:text-gray-200 whitespace-normal break-words leading-relaxed first:pl-6 last:pr-6\">\n <div innerHTML={renderCellValue(row[column.key])} />\n </td>\n )}\n </For>\n </tr>\n )}\n </For>\n </tbody>\n </table>\n </div>\n\n <Show when={tableParams.pagination}>\n <div class=\"mt-3 flex items-center justify-between text-xs text-gray-500 dark:text-gray-400\">\n <span>\n Showing {tableParams.pagination.currentPage * tableParams.pagination.pageSize + 1} -{' '}\n {Math.min(\n (tableParams.pagination.currentPage + 1) * tableParams.pagination.pageSize,\n tableParams.pagination.totalRows\n )}{' '}\n of {tableParams.pagination.totalRows}\n </span>\n </div>\n </Show>\n </div>\n </div>\n )\n}\n\n/**\n * Render a metric card component\n */\nfunction MetricRenderer(props: { component: UIComponent }) {\n const metricParams = props.component.params as any\n\n return (\n <div class=\"w-full h-full bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 p-4\">\n <div class=\"flex flex-col h-full justify-between\">\n <div>\n <p class=\"text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wide\">\n {metricParams.title}\n </p>\n <div class=\"mt-2 flex items-baseline\">\n <p class=\"text-2xl font-semibold text-gray-900 dark:text-white\">{metricParams.value}</p>\n <Show when={metricParams.unit}>\n <span class=\"ml-2 text-sm font-medium text-gray-500 dark:text-gray-400\">\n {metricParams.unit}\n </span>\n </Show>\n </div>\n </div>\n\n <Show when={metricParams.trend}>\n <div class=\"mt-3 flex items-center\">\n <span\n class={`text-sm font-medium ${metricParams.trend.direction === 'up'\n ? 'text-green-600 dark:text-green-400'\n : metricParams.trend.direction === 'down'\n ? 'text-red-600 dark:text-red-400'\n : 'text-gray-600 dark:text-gray-400'\n }`}\n >\n {metricParams.trend.direction === 'up'\n ? '�'\n : metricParams.trend.direction === 'down'\n ? '�'\n : '�'}{' '}\n {Math.abs(metricParams.trend.value)}%\n </span>\n </div>\n </Show>\n\n <Show when={metricParams.subtitle}>\n <p class=\"mt-2 text-xs text-gray-500 dark:text-gray-400\">{metricParams.subtitle}</p>\n </Show>\n </div>\n </div>\n )\n}\n\n/**\n * Extract image data from markdown image link format\n * Pattern: [![alt](image-url)](link-url)\\n*Photo by Author*\n */\nfunction extractImageFromMarkdown(content: string): { alt: string; imageUrl: string; linkUrl: string; credit: string } | null {\n // Pattern: [![alt text](image-url)](link-url) followed by optional credit line\n const imagePattern = /\\[!\\[([^\\]]*)\\]\\(([^)]+)\\)\\]\\(([^)]+)\\)\\s*\\*([^*]+)\\*/\n const match = content.match(imagePattern)\n\n if (match) {\n return {\n alt: match[1] || 'Image',\n imageUrl: match[2],\n linkUrl: match[3],\n credit: match[4].trim()\n }\n }\n\n return null\n}\n\n/**\n * Render a text component (with optional markdown)\n */\nfunction TextRenderer(props: { component: UIComponent }) {\n const textParams = props.component.params as any\n\n // Check if this is an image markdown that should be rendered as image component\n const imageData = createMemo(() => {\n if (textParams.markdown && textParams.content) {\n return extractImageFromMarkdown(textParams.content)\n }\n return null\n })\n\n // Convert markdown to HTML if markdown flag is true (and not an image component)\n const htmlContent = createMemo(() => {\n if (textParams.markdown && !imageData()) {\n return marked.parse(textParams.content, { async: false }) as string\n }\n return textParams.content\n })\n\n // Render as image component if we extracted image data\n return (\n <Show\n when={imageData()}\n fallback={\n <div class=\"w-full h-full bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 p-4\">\n <div\n class={`prose prose-sm dark:prose-invert max-w-none ${textParams.className || ''}`}\n innerHTML={htmlContent()}\n />\n </div>\n }\n >\n {(data) => (\n <div class=\"w-full h-full bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 overflow-hidden flex flex-col\">\n <div class=\"flex-1 flex items-center justify-center p-4 bg-gray-50 dark:bg-gray-900 min-h-[200px]\">\n <a href={data().linkUrl} target=\"_blank\" rel=\"noopener noreferrer\" class=\"cursor-zoom-in\">\n <img\n src={data().imageUrl}\n alt={data().alt}\n class=\"max-w-full max-h-[400px] object-contain rounded shadow-sm hover:opacity-90 transition-opacity\"\n loading=\"lazy\"\n />\n </a>\n </div>\n <div class=\"p-3 border-t border-gray-200 dark:border-gray-700 bg-white dark:bg-gray-800\">\n <p class=\"text-sm text-gray-600 dark:text-gray-400 text-center italic\">{data().credit}</p>\n </div>\n </div>\n )}\n </Show>\n )\n}\n\n/**\n * Render an iframe component\n */\nfunction IframeRenderer(props: { component: UIComponent }) {\n const params = props.component.params as any\n return (\n <div class=\"w-full h-full bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 overflow-hidden flex flex-col\">\n <Show when={params.title}>\n <div class=\"px-4 py-2 border-b border-gray-200 dark:border-gray-700 bg-gray-50 dark:bg-gray-900\">\n <h3 class=\"text-sm font-semibold text-gray-900 dark:text-white\">{params.title}</h3>\n </div>\n </Show>\n <iframe\n src={params.url}\n title={params.title || 'Embedded content'}\n class=\"w-full border-0 flex-1\"\n style={`height: ${params.height || '400px'}; min-height: 300px;`}\n sandbox=\"allow-scripts allow-same-origin allow-popups allow-forms\"\n loading=\"lazy\"\n />\n </div>\n )\n}\n\n/**\n * Render an image component\n */\nfunction ImageRenderer(props: { component: UIComponent }) {\n const params = props.component.params as any\n\n return (\n <div class=\"w-full h-full bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 overflow-hidden flex flex-col\">\n <div class=\"flex-1 flex items-center justify-center p-4 bg-gray-50 dark:bg-gray-900 min-h-[200px]\">\n <a href={params.url} target=\"_blank\" rel=\"noopener noreferrer\" class=\"cursor-zoom-in\">\n <img\n src={params.url}\n alt={params.alt || 'Image'}\n class=\"max-w-full max-h-[500px] object-contain rounded shadow-sm hover:opacity-95 transition-opacity\"\n loading=\"lazy\"\n />\n </a>\n </div>\n <Show when={params.caption}>\n <div class=\"p-3 border-t border-gray-200 dark:border-gray-700 bg-white dark:bg-gray-800\">\n <p class=\"text-sm text-gray-600 dark:text-gray-400 text-center\">{params.caption}</p>\n </div>\n </Show>\n </div>\n )\n}\n\n/**\n * Render a link component\n */\n/**\n * Render a link component\n */\nfunction LinkRenderer(props: { component: UIComponent }) {\n const params = props.component.params as any\n\n return (\n <a\n href={params.url}\n target=\"_blank\"\n rel=\"noopener noreferrer\"\n class=\"flex items-center gap-3 p-4 bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 hover:bg-gray-50 dark:hover:bg-gray-700 transition-colors group h-full\"\n onClick={(e) => e.stopPropagation()}\n >\n <div class=\"p-2 bg-blue-50 dark:bg-blue-900/30 rounded-full text-blue-600 dark:text-blue-400 group-hover:bg-blue-100 dark:group-hover:bg-blue-900/50 shrink-0 transition-colors\">\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n class=\"w-5 h-5\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n stroke=\"currentColor\"\n stroke-width=\"2\"\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n >\n <path d=\"M10 13a5 5 0 0 0 7.54.54l3-3a5 5 0 0 0-7.07-7.07l-1.72 1.71\" />\n <path d=\"M14 11a5 5 0 0 0-7.54-.54l-3 3a5 5 0 0 0 7.07 7.07l1.71-1.71\" />\n </svg>\n </div>\n <div class=\"flex-1 min-w-0\">\n <h4 class=\"text-sm font-medium text-gray-900 dark:text-white truncate group-hover:text-blue-600 dark:group-hover:text-blue-400 transition-colors\">\n {params.label || params.url}\n </h4>\n <Show when={params.description}>\n <p class=\"text-xs text-gray-500 dark:text-gray-400 truncate\">{params.description}</p>\n </Show>\n </div>\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n class=\"w-4 h-4 text-gray-400 group-hover:text-gray-600 dark:group-hover:text-gray-300 shrink-0 transition-colors\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n stroke=\"currentColor\"\n stroke-width=\"2\"\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n >\n <path d=\"M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6\" />\n <polyline points=\"15 3 21 3 21 9\" />\n <line x1=\"10\" y1=\"14\" x2=\"21\" y2=\"3\" />\n </svg>\n </a>\n )\n}\n\n/**\n * Render a single component with error boundary\n */\nfunction ComponentRenderer(props: {\n component: UIComponent\n onError?: (error: RendererError) => void\n}) {\n // Validate component before rendering\n const validation = validateComponent(props.component)\n if (!validation.valid) {\n props.onError?.({\n type: 'validation',\n message: 'Component validation failed',\n componentId: props.component.id,\n details: validation.errors,\n })\n\n return (\n <div class=\"w-full h-full bg-red-50 dark:bg-red-900/20 border border-red-200 dark:border-red-800 rounded-lg p-4\">\n <p class=\"text-sm font-medium text-red-900 dark:text-red-100\">Validation Error</p>\n <p class=\"text-xs text-red-700 dark:text-red-300 mt-1\">\n {validation.errors?.[0]?.message || 'Unknown validation error'}\n </p>\n </div>\n )\n }\n\n // Render based on component type with enhanced error boundary\n return (\n <GenerativeUIErrorBoundary\n componentId={props.component.id}\n componentType={props.component.type}\n onError={props.onError}\n allowRetry={true}\n >\n <Show when={props.component.type === 'chart'}>\n <ChartRenderer component={props.component} onError={props.onError} />\n </Show>\n <Show when={props.component.type === 'table'}>\n <TableRenderer component={props.component} onError={props.onError} />\n </Show>\n <Show when={props.component.type === 'metric'}>\n <MetricRenderer component={props.component} />\n </Show>\n <Show when={props.component.type === 'text'}>\n <TextRenderer component={props.component} />\n </Show>\n <Show when={props.component.type === 'iframe'}>\n <IframeRenderer component={props.component} />\n </Show>\n <Show when={props.component.type === 'image'}>\n <ImageRenderer component={props.component} />\n </Show>\n <Show when={props.component.type === 'link'}>\n <LinkRenderer component={props.component} />\n </Show>\n <Show when={props.component.type === 'action'}>\n <ActionRenderer component={props.component} />\n </Show>\n </GenerativeUIErrorBoundary>\n )\n}\n\n/**\n * Render an action component (button or link)\n */\nfunction ActionRenderer(props: { component: UIComponent }) {\n const params = props.component.params as any\n let dispatchAction: ((toolName: string, toolParams: any) => void) | null = null\n\n // Initialize CustomEvent dispatcher only on client-side\n // Use createEffect instead of onMount for SSR compatibility\n createEffect(() => {\n if (typeof window !== 'undefined') {\n dispatchAction = (toolName: string, toolParams: any) => {\n const event = new CustomEvent('mcp-action', {\n detail: {\n toolName,\n params: toolParams,\n },\n bubbles: true,\n })\n window.dispatchEvent(event)\n }\n }\n })\n\n // Handle click to execute tool via window event\n const handleClick = (e: MouseEvent) => {\n if (params.action === 'tool-call' && params.toolName) {\n e.preventDefault()\n // SSR-safe: Only call if dispatcher was initialized client-side\n dispatchAction?.(params.toolName, params.params || {})\n }\n }\n\n if (params.type === 'link' || params.action === 'link') {\n return (\n <a\n href={params.url || '#'}\n target={params.url ? '_blank' : undefined}\n rel=\"noopener noreferrer\"\n class={`inline-flex items-center gap-2 px-4 py-2 rounded-md text-sm font-medium transition-colors\n ${params.variant === 'primary' ? 'bg-blue-600 text-white hover:bg-blue-700' :\n params.variant === 'outline' ? 'border border-gray-300 text-gray-700 hover:bg-gray-50 dark:border-gray-600 dark:text-gray-300 dark:hover:bg-gray-800' :\n 'text-blue-600 hover:text-blue-800 dark:text-blue-400 dark:hover:text-blue-300'}`}\n onClick={handleClick}\n >\n <Show when={params.icon}>\n <span>{params.icon}</span>\n </Show>\n {params.label}\n </a>\n )\n }\n\n return (\n <button\n type={params.action === 'submit' ? 'submit' : 'button'}\n disabled={params.disabled}\n class={`inline-flex items-center gap-2 px-4 py-2 rounded-md text-sm font-medium transition-colors focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500\n ${params.variant === 'primary' ? 'bg-blue-600 text-white hover:bg-blue-700 shadow-sm' :\n params.variant === 'secondary' ? 'bg-gray-100 text-gray-900 hover:bg-gray-200 dark:bg-gray-700 dark:text-white dark:hover:bg-gray-600' :\n params.variant === 'outline' ? 'border border-gray-300 text-gray-700 hover:bg-gray-50 dark:border-gray-600 dark:text-gray-300 dark:hover:bg-gray-800' :\n params.variant === 'danger' ? 'bg-red-600 text-white hover:bg-red-700' :\n 'bg-transparent text-gray-700 hover:bg-gray-100 dark:text-gray-300 dark:hover:bg-gray-800'}\n ${params.disabled ? 'opacity-50 cursor-not-allowed' : ''}\n ${params.size === 'sm' ? 'px-3 py-1.5 text-xs' : params.size === 'lg' ? 'px-6 py-3 text-base' : ''}`}\n onClick={handleClick}\n >\n <Show when={params.icon}>\n <span>{params.icon}</span>\n </Show>\n {params.label}\n </button>\n )\n}\n\n/**\n * Main UIResourceRenderer component\n */\nexport const UIResourceRenderer: Component<UIResourceRendererProps> = (props) => {\n const layout = () => {\n // Check if content is a UIComponent (non-composite) vs UILayout (composite)\n if ('type' in props.content && (props.content as any).type !== 'composite') {\n return {\n id: 'single-component',\n components: [props.content as UIComponent],\n grid: {\n columns: 12,\n gap: '1rem',\n },\n } as UILayout\n }\n return props.content as UILayout\n }\n\n // Convert grid styles to CSS string\n const gridContainerStyle = () => {\n const layoutData = layout()\n return `grid-template-columns: repeat(${layoutData.grid.columns}, 1fr); gap: ${layoutData.grid.gap}`\n }\n\n // Convert component grid styles to CSS string\n const getGridStyleString = (component: UIComponent) => {\n // Defensive check for position field - default to full width\n if (!component.position) {\n return 'grid-column: 1 / span 12; grid-row: auto'\n }\n const { colStart, colSpan, rowStart, rowSpan = 1 } = component.position\n return `grid-column: ${colStart} / span ${colSpan}; grid-row: ${rowStart ? `${rowStart} / span ${rowSpan}` : 'auto'}`\n }\n\n const layoutData = layout()\n\n return (\n <div class={`w-full ${props.class || ''}`}>\n <div class=\"grid gap-4\" style={gridContainerStyle()}>\n <For each={layoutData.components}>\n {(component) => (\n <div style={getGridStyleString(component)}>\n <ComponentRenderer component={component} onError={props.onError} />\n </div>\n )}\n </For>\n </div>\n </div>\n )\n}\n"],"names":["ChartRenderer","props","iframeUrl","setIframeUrl","createSignal","isLoading","setIsLoading","error","setError","createEffect","chartParams","component","params","chartConfig","type","data","options","responsive","maintainAspectRatio","configStr","encodeURIComponent","JSON","stringify","url","_el$","_$getNextElement","_tmpl$5","_el$11","firstChild","_el$12","_co$2","_$getNextMarker","nextSibling","_el$13","_el$14","_co$3","_el$15","_el$16","_co$4","_$insert","_$createComponent","Show","when","children","_tmpl$","_el$3","_tmpl$2","_el$4","_el$5","_el$6","_$memo","_el$7","_tmpl$4","_el$1","_el$10","_co$","_el$9","_el$0","title","_el$8","_tmpl$3","addEventListener","onError","message","componentId","id","_$effect","_$setAttribute","renderCellValue","value","undefined","strValue","String","replace","trim","toLowerCase","markdownLinkRegex","test","htmlValue","DOMPurify","sanitize","ADD_ATTR","hasMarkdown","parsed","marked","parse","async","TableRenderer","tableParams","_el$17","_tmpl$7","_el$18","_el$38","_el$39","_co$8","_el$20","_el$21","_el$22","_el$23","_el$24","_el$40","_el$41","_co$9","_el$19","For","each","columns","column","_el$42","_tmpl$8","label","_$p","_$style","width","rows","slice","DEFAULT_RESOURCE_LIMITS","maxTableRows","row","i","_el$43","_tmpl$9","_el$44","_tmpl$0","_el$45","_$setProperty","key","_$className","pagination","_el$25","_tmpl$6","_el$26","_el$27","_el$32","_el$33","_co$5","_el$28","_el$34","_el$35","_co$6","_el$30","_el$36","_el$37","_co$7","currentPage","pageSize","Math","min","totalRows","MetricRenderer","metricParams","_el$46","_tmpl$12","_el$47","_el$48","_el$49","_el$50","_el$51","_el$53","_el$54","_co$0","_el$64","_el$65","_co$11","_el$66","_el$67","_co$12","unit","_el$52","_tmpl$1","trend","_el$55","_tmpl$10","_el$56","_el$59","_el$60","_co$1","_el$57","_el$61","_el$62","_co$10","_c$","direction","abs","subtitle","_el$63","_tmpl$11","extractImageFromMarkdown","content","imagePattern","match","alt","imageUrl","linkUrl","credit","TextRenderer","textParams","imageData","createMemo","markdown","htmlContent","fallback","_el$68","_tmpl$13","_el$69","_p$","_v$","className","_v$2","e","t","_el$70","_tmpl$14","_el$71","_el$72","_el$73","_el$74","_el$75","_v$3","_v$4","_v$5","a","IframeRenderer","_el$76","_tmpl$16","_el$80","_el$81","_co$13","_el$79","_el$77","_tmpl$15","_el$78","_v$6","_v$7","_v$8","height","ImageRenderer","_el$82","_tmpl$18","_el$83","_el$84","_el$85","_el$88","_el$89","_co$14","caption","_el$86","_tmpl$17","_el$87","_v$9","_v$0","_v$1","LinkRenderer","_el$90","_tmpl$20","_el$91","_el$92","_el$93","_el$95","_el$96","_co$15","$$click","stopPropagation","description","_el$94","_tmpl$19","_$runHydrationEvents","ComponentRenderer","validation","validateComponent","valid","details","errors","_el$97","_tmpl$21","_el$98","_el$99","GenerativeUIErrorBoundary","componentType","allowRetry","ActionRenderer","dispatchAction","window","toolName","toolParams","event","CustomEvent","detail","bubbles","dispatchEvent","handleClick","action","preventDefault","_el$100","_tmpl$23","_el$102","_el$103","_co$16","_el$104","_el$105","_co$17","icon","_el$101","_tmpl$22","_v$10","_v$11","_v$12","variant","_el$106","_tmpl$24","_el$108","_el$109","_co$18","_el$110","_el$111","_co$19","_el$107","_v$13","_v$14","disabled","_v$15","size","UIResourceRenderer","layout","components","grid","gap","gridContainerStyle","layoutData","getGridStyleString","position","colStart","colSpan","rowStart","rowSpan","_el$112","_tmpl$25","_el$113","_el$114","_tmpl$26","_v$16","class","_v$17","_$delegateEvents"],"mappings":"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"}
1
+ {"version":3,"file":"UIResourceRenderer.cjs","sources":["../../src/components/UIResourceRenderer.tsx"],"sourcesContent":["/**\n * UI Resource Renderer Component\n * Phase 0: Foundation with iframe sandbox and composite grid support\n */\n\nimport DOMPurify from 'dompurify'\nimport { Component, createSignal, Show, For, createMemo, createEffect } from 'solid-js'\nimport { isServer } from 'solid-js/web'\nimport type { UIComponent, UILayout, RendererError, ComponentType } from '../types'\nimport { validateComponent, DEFAULT_RESOURCE_LIMITS } from '../services/validation'\nimport { GenerativeUIErrorBoundary } from './GenerativeUIErrorBoundary'\nimport { marked } from 'marked'\n\n/**\n * Props for UIResourceRenderer\n */\nexport interface UIResourceRendererProps {\n /**\n * Single component or full layout to render\n */\n content: UIComponent | UILayout\n\n /**\n * Lazy loading (default: true)\n */\n lazyLoad?: boolean\n\n /**\n * Error callback\n */\n onError?: (error: RendererError) => void\n\n /**\n * Custom CSS class\n */\n class?: string\n}\n\n/**\n * Render a single chart component in a sandboxed iframe\n */\nfunction ChartRenderer(props: {\n component: UIComponent\n onError?: (error: RendererError) => void\n}) {\n const [iframeUrl, setIframeUrl] = createSignal<string>()\n const [isLoading, setIsLoading] = createSignal(true)\n const [error, setError] = createSignal<string>()\n\n // Use createEffect instead of onMount for SSR compatibility\n // createEffect runs after hydration on client-side\n createEffect(() => {\n const chartParams = props.component.params as any\n if (!chartParams) return\n\n // Build Quickchart URL\n const chartConfig = {\n type: chartParams.type,\n data: chartParams.data,\n options: {\n ...chartParams.options,\n responsive: true,\n maintainAspectRatio: false,\n },\n }\n\n // Encode chart configuration for Quickchart API\n const configStr = encodeURIComponent(JSON.stringify(chartConfig))\n const url = `https://quickchart.io/chart?c=${configStr}&width=500&height=300&devicePixelRatio=2`\n\n // Validate domain (should always pass for quickchart.io)\n setIframeUrl(url)\n setIsLoading(false)\n })\n\n return (\n <div class=\"relative w-full h-full min-h-[300px] bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 overflow-hidden\">\n <Show when={isLoading()}>\n <div class=\"absolute inset-0 flex items-center justify-center\">\n <div class=\"animate-spin rounded-full h-8 w-8 border-b-2 border-blue-600\" />\n </div>\n </Show>\n\n <Show when={error()}>\n <div class=\"absolute inset-0 flex items-center justify-center p-4\">\n <div class=\"text-center\">\n <p class=\"text-red-600 dark:text-red-400 text-sm font-medium\">Chart Error</p>\n <p class=\"text-gray-600 dark:text-gray-400 text-xs mt-1\">{error()}</p>\n </div>\n </div>\n </Show>\n\n <Show when={iframeUrl() && !error()}>\n <div class=\"w-full h-full p-4\">\n <Show when={(props.component.params as any).title}>\n <h3 class=\"text-sm font-semibold text-gray-900 dark:text-white mb-3\">\n {(props.component.params as any).title}\n </h3>\n </Show>\n <div class=\"w-full h-full\">\n <img\n src={iframeUrl()}\n alt=\"Chart visualization\"\n class=\"w-full h-auto max-h-[300px] object-contain\"\n onError={() => {\n setError('Failed to load chart')\n props.onError?.({\n type: 'render',\n message: 'Chart rendering failed',\n componentId: props.component.id,\n })\n }}\n />\n </div>\n </div>\n </Show>\n </div>\n )\n}\n\n/**\n * Smart cell value renderer that handles markdown links and other formats\n */\nfunction renderCellValue(value: any): string {\n // Handle null/undefined\n if (value === null || value === undefined) {\n return '-'\n }\n\n // Handle object with url/name properties (common source/link format from LLM)\n if (typeof value === 'object' && value !== null) {\n // Check for link-like objects: { url: \"...\", name/label/title: \"...\" }\n if (value.url) {\n const label = value.name || value.label || value.title || value.url\n const sanitizedLabel = DOMPurify.sanitize(String(label))\n const sanitizedUrl = DOMPurify.sanitize(String(value.url))\n return `<a href=\"${sanitizedUrl}\" target=\"_blank\" rel=\"noopener noreferrer\" class=\"text-blue-600 dark:text-blue-400 hover:underline\">${sanitizedLabel}</a>`\n }\n // Fallback: extract meaningful text from object properties\n if (value.name || value.label || value.title) {\n return DOMPurify.sanitize(String(value.name || value.label || value.title))\n }\n // Last resort: JSON stringify for debugging (better than [object Object])\n try {\n return JSON.stringify(value)\n } catch {\n return '-'\n }\n }\n\n // Convert to string\n let strValue = String(value)\n\n // Clean up \"undefined\" patterns from backend data\n // Pattern 1: \"Text – undefined\" or \"Text - undefined\" → \"Text\"\n strValue = strValue.replace(/\\s*[–-]\\s*undefined\\s*$/gi, '')\n // Pattern 2: \"undefined – Text\" or \"undefined - Text\" → \"Text\"\n strValue = strValue.replace(/^undefined\\s*[–-]\\s*/gi, '')\n // Pattern 3: standalone \"undefined\" → \"-\"\n if (strValue.trim().toLowerCase() === 'undefined') {\n return '-'\n }\n // Pattern 4: empty string after cleanup → \"-\"\n if (strValue.trim() === '') {\n return '-'\n }\n\n // Detect and convert markdown links: [text](url)\n const markdownLinkRegex = /\\[([^\\]]+)\\]\\(([^)]+)\\)/g\n if (markdownLinkRegex.test(strValue)) {\n // Replace all markdown links with HTML links\n const htmlValue = strValue.replace(\n /\\[([^\\]]+)\\]\\(([^)]+)\\)/g,\n '<a href=\"$2\" target=\"_blank\" rel=\"noopener noreferrer\" class=\"text-blue-600 dark:text-blue-400 hover:underline\">$1</a>'\n )\n return DOMPurify.sanitize(htmlValue, { ADD_ATTR: ['target', 'rel'] })\n }\n\n // Check if value contains markdown formatting (bold, italic, code, etc.)\n const hasMarkdown = /[*_`\\[\\]#]/.test(strValue)\n if (hasMarkdown) {\n // Parse with marked and sanitize\n const parsed = marked.parse(strValue, { async: false }) as string\n return DOMPurify.sanitize(parsed, { ADD_ATTR: ['target', 'rel'] })\n }\n\n // Plain text\n return strValue\n}\n\n/**\n * Render a table component\n */\nfunction TableRenderer(props: {\n component: UIComponent\n onError?: (error: RendererError) => void\n}) {\n const tableParams = props.component.params as any\n\n return (\n <div class=\"w-full h-full bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 overflow-hidden\">\n <div class=\"p-4\">\n <Show when={tableParams.title}>\n <h3 class=\"text-sm font-semibold text-gray-900 dark:text-white mb-3\">\n {tableParams.title}\n </h3>\n </Show>\n\n <div class=\"overflow-x-auto\">\n <table class=\"min-w-full divide-y divide-gray-200 dark:divide-gray-700 border-separate border-spacing-0\">\n <thead class=\"bg-gray-50 dark:bg-gray-900/50\">\n <tr>\n <For each={tableParams.columns}>\n {(column: any) => (\n <th\n scope=\"col\"\n class=\"px-6 py-3 text-left text-xs font-semibold text-gray-500 dark:text-gray-400 uppercase tracking-wider border-b border-gray-200 dark:border-gray-700 first:pl-6 last:pr-6\"\n style={column.width ? { width: column.width } : {}}\n >\n {column.label}\n </th>\n )}\n </For>\n </tr>\n </thead>\n <tbody class=\"bg-white dark:bg-gray-800 divide-y divide-gray-200 dark:divide-gray-700\">\n <For each={tableParams.rows.slice(0, DEFAULT_RESOURCE_LIMITS.maxTableRows)}>\n {(row: any, i) => (\n <tr class={`hover:bg-gray-50 dark:hover:bg-gray-700/50 transition-colors ${i() % 2 === 0 ? 'bg-white dark:bg-gray-800' : 'bg-gray-50/30 dark:bg-gray-800/50'}`}>\n <For each={tableParams.columns}>\n {(column: any) => (\n <td class=\"px-6 py-4 text-sm text-gray-700 dark:text-gray-200 whitespace-normal break-words leading-relaxed first:pl-6 last:pr-6\">\n <div innerHTML={renderCellValue(row[column.key])} />\n </td>\n )}\n </For>\n </tr>\n )}\n </For>\n </tbody>\n </table>\n </div>\n\n <Show when={tableParams.pagination}>\n <div class=\"mt-3 flex items-center justify-between text-xs text-gray-500 dark:text-gray-400\">\n <span>\n Showing {tableParams.pagination.currentPage * tableParams.pagination.pageSize + 1} -{' '}\n {Math.min(\n (tableParams.pagination.currentPage + 1) * tableParams.pagination.pageSize,\n tableParams.pagination.totalRows\n )}{' '}\n of {tableParams.pagination.totalRows}\n </span>\n </div>\n </Show>\n </div>\n </div>\n )\n}\n\n/**\n * Render a metric card component\n */\nfunction MetricRenderer(props: { component: UIComponent }) {\n const metricParams = props.component.params as any\n\n return (\n <div class=\"w-full h-full bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 p-4\">\n <div class=\"flex flex-col h-full justify-between\">\n <div>\n <p class=\"text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wide\">\n {metricParams.title}\n </p>\n <div class=\"mt-2 flex items-baseline\">\n <p class=\"text-2xl font-semibold text-gray-900 dark:text-white\">{metricParams.value}</p>\n <Show when={metricParams.unit}>\n <span class=\"ml-2 text-sm font-medium text-gray-500 dark:text-gray-400\">\n {metricParams.unit}\n </span>\n </Show>\n </div>\n </div>\n\n <Show when={metricParams.trend}>\n <div class=\"mt-3 flex items-center\">\n <span\n class={`text-sm font-medium ${metricParams.trend.direction === 'up'\n ? 'text-green-600 dark:text-green-400'\n : metricParams.trend.direction === 'down'\n ? 'text-red-600 dark:text-red-400'\n : 'text-gray-600 dark:text-gray-400'\n }`}\n >\n {metricParams.trend.direction === 'up'\n ? '�'\n : metricParams.trend.direction === 'down'\n ? '�'\n : '�'}{' '}\n {Math.abs(metricParams.trend.value)}%\n </span>\n </div>\n </Show>\n\n <Show when={metricParams.subtitle}>\n <p class=\"mt-2 text-xs text-gray-500 dark:text-gray-400\">{metricParams.subtitle}</p>\n </Show>\n </div>\n </div>\n )\n}\n\n/**\n * Extract image data from markdown image link format\n * Pattern: [![alt](image-url)](link-url)\\n*Photo by Author*\n */\nfunction extractImageFromMarkdown(content: string): { alt: string; imageUrl: string; linkUrl: string; credit: string } | null {\n // Pattern: [![alt text](image-url)](link-url) followed by optional credit line\n const imagePattern = /\\[!\\[([^\\]]*)\\]\\(([^)]+)\\)\\]\\(([^)]+)\\)\\s*\\*([^*]+)\\*/\n const match = content.match(imagePattern)\n\n if (match) {\n return {\n alt: match[1] || 'Image',\n imageUrl: match[2],\n linkUrl: match[3],\n credit: match[4].trim()\n }\n }\n\n return null\n}\n\n/**\n * Render a text component (with optional markdown)\n */\nfunction TextRenderer(props: { component: UIComponent }) {\n const textParams = props.component.params as any\n\n // Check if this is an image markdown that should be rendered as image component\n const imageData = createMemo(() => {\n if (textParams.markdown && textParams.content) {\n return extractImageFromMarkdown(textParams.content)\n }\n return null\n })\n\n // Convert markdown to HTML if markdown flag is true (and not an image component)\n const htmlContent = createMemo(() => {\n if (textParams.markdown && !imageData()) {\n return marked.parse(textParams.content, { async: false }) as string\n }\n return textParams.content\n })\n\n // Render as image component if we extracted image data\n return (\n <Show\n when={imageData()}\n fallback={\n <div class=\"w-full h-full bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 p-4\">\n <div\n class={`prose prose-sm dark:prose-invert max-w-none ${textParams.className || ''}`}\n innerHTML={htmlContent()}\n />\n </div>\n }\n >\n {(data) => (\n <div class=\"w-full h-full bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 overflow-hidden flex flex-col\">\n <div class=\"flex-1 flex items-center justify-center p-4 bg-gray-50 dark:bg-gray-900 min-h-[200px]\">\n <a href={data().linkUrl} target=\"_blank\" rel=\"noopener noreferrer\" class=\"cursor-zoom-in\">\n <img\n src={data().imageUrl}\n alt={data().alt}\n class=\"max-w-full max-h-[400px] object-contain rounded shadow-sm hover:opacity-90 transition-opacity\"\n loading=\"lazy\"\n />\n </a>\n </div>\n <div class=\"p-3 border-t border-gray-200 dark:border-gray-700 bg-white dark:bg-gray-800\">\n <p class=\"text-sm text-gray-600 dark:text-gray-400 text-center italic\">{data().credit}</p>\n </div>\n </div>\n )}\n </Show>\n )\n}\n\n/**\n * Render an iframe component\n */\nfunction IframeRenderer(props: { component: UIComponent }) {\n const params = props.component.params as any\n return (\n <div class=\"w-full h-full bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 overflow-hidden flex flex-col\">\n <Show when={params.title}>\n <div class=\"px-4 py-2 border-b border-gray-200 dark:border-gray-700 bg-gray-50 dark:bg-gray-900\">\n <h3 class=\"text-sm font-semibold text-gray-900 dark:text-white\">{params.title}</h3>\n </div>\n </Show>\n <iframe\n src={params.url}\n title={params.title || 'Embedded content'}\n class=\"w-full border-0 flex-1\"\n style={`height: ${params.height || '400px'}; min-height: 300px;`}\n sandbox=\"allow-scripts allow-same-origin allow-popups allow-forms\"\n loading=\"lazy\"\n />\n </div>\n )\n}\n\n/**\n * Render an image component\n */\nfunction ImageRenderer(props: { component: UIComponent }) {\n const params = props.component.params as any\n\n return (\n <div class=\"w-full h-full bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 overflow-hidden flex flex-col\">\n <div class=\"flex-1 flex items-center justify-center p-4 bg-gray-50 dark:bg-gray-900 min-h-[200px]\">\n <a href={params.url} target=\"_blank\" rel=\"noopener noreferrer\" class=\"cursor-zoom-in\">\n <img\n src={params.url}\n alt={params.alt || 'Image'}\n class=\"max-w-full max-h-[500px] object-contain rounded shadow-sm hover:opacity-95 transition-opacity\"\n loading=\"lazy\"\n />\n </a>\n </div>\n <Show when={params.caption}>\n <div class=\"p-3 border-t border-gray-200 dark:border-gray-700 bg-white dark:bg-gray-800\">\n <p class=\"text-sm text-gray-600 dark:text-gray-400 text-center\">{params.caption}</p>\n </div>\n </Show>\n </div>\n )\n}\n\n/**\n * Render a link component\n */\n/**\n * Render a link component\n */\nfunction LinkRenderer(props: { component: UIComponent }) {\n const params = props.component.params as any\n\n return (\n <a\n href={params.url}\n target=\"_blank\"\n rel=\"noopener noreferrer\"\n class=\"flex items-center gap-3 p-4 bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700 hover:bg-gray-50 dark:hover:bg-gray-700 transition-colors group h-full\"\n onClick={(e) => e.stopPropagation()}\n >\n <div class=\"p-2 bg-blue-50 dark:bg-blue-900/30 rounded-full text-blue-600 dark:text-blue-400 group-hover:bg-blue-100 dark:group-hover:bg-blue-900/50 shrink-0 transition-colors\">\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n class=\"w-5 h-5\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n stroke=\"currentColor\"\n stroke-width=\"2\"\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n >\n <path d=\"M10 13a5 5 0 0 0 7.54.54l3-3a5 5 0 0 0-7.07-7.07l-1.72 1.71\" />\n <path d=\"M14 11a5 5 0 0 0-7.54-.54l-3 3a5 5 0 0 0 7.07 7.07l1.71-1.71\" />\n </svg>\n </div>\n <div class=\"flex-1 min-w-0\">\n <h4 class=\"text-sm font-medium text-gray-900 dark:text-white truncate group-hover:text-blue-600 dark:group-hover:text-blue-400 transition-colors\">\n {params.label || params.url}\n </h4>\n <Show when={params.description}>\n <p class=\"text-xs text-gray-500 dark:text-gray-400 truncate\">{params.description}</p>\n </Show>\n </div>\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n class=\"w-4 h-4 text-gray-400 group-hover:text-gray-600 dark:group-hover:text-gray-300 shrink-0 transition-colors\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n stroke=\"currentColor\"\n stroke-width=\"2\"\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n >\n <path d=\"M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6\" />\n <polyline points=\"15 3 21 3 21 9\" />\n <line x1=\"10\" y1=\"14\" x2=\"21\" y2=\"3\" />\n </svg>\n </a>\n )\n}\n\n/**\n * Render a single component with error boundary\n */\nfunction ComponentRenderer(props: {\n component: UIComponent\n onError?: (error: RendererError) => void\n}) {\n // Validate component before rendering\n const validation = validateComponent(props.component)\n if (!validation.valid) {\n props.onError?.({\n type: 'validation',\n message: 'Component validation failed',\n componentId: props.component.id,\n details: validation.errors,\n })\n\n return (\n <div class=\"w-full h-full bg-red-50 dark:bg-red-900/20 border border-red-200 dark:border-red-800 rounded-lg p-4\">\n <p class=\"text-sm font-medium text-red-900 dark:text-red-100\">Validation Error</p>\n <p class=\"text-xs text-red-700 dark:text-red-300 mt-1\">\n {validation.errors?.[0]?.message || 'Unknown validation error'}\n </p>\n </div>\n )\n }\n\n // Render based on component type with enhanced error boundary\n return (\n <GenerativeUIErrorBoundary\n componentId={props.component.id}\n componentType={props.component.type}\n onError={props.onError}\n allowRetry={true}\n >\n <Show when={props.component.type === 'chart'}>\n <ChartRenderer component={props.component} onError={props.onError} />\n </Show>\n <Show when={props.component.type === 'table'}>\n <TableRenderer component={props.component} onError={props.onError} />\n </Show>\n <Show when={props.component.type === 'metric'}>\n <MetricRenderer component={props.component} />\n </Show>\n <Show when={props.component.type === 'text'}>\n <TextRenderer component={props.component} />\n </Show>\n <Show when={props.component.type === 'iframe'}>\n <IframeRenderer component={props.component} />\n </Show>\n <Show when={props.component.type === 'image'}>\n <ImageRenderer component={props.component} />\n </Show>\n <Show when={props.component.type === 'link'}>\n <LinkRenderer component={props.component} />\n </Show>\n <Show when={props.component.type === 'action'}>\n <ActionRenderer component={props.component} />\n </Show>\n </GenerativeUIErrorBoundary>\n )\n}\n\n/**\n * Render an action component (button or link)\n */\nfunction ActionRenderer(props: { component: UIComponent }) {\n const params = props.component.params as any\n let dispatchAction: ((toolName: string, toolParams: any) => void) | null = null\n\n // Initialize CustomEvent dispatcher only on client-side\n // Use createEffect instead of onMount for SSR compatibility\n createEffect(() => {\n if (typeof window !== 'undefined') {\n dispatchAction = (toolName: string, toolParams: any) => {\n const event = new CustomEvent('mcp-action', {\n detail: {\n toolName,\n params: toolParams,\n },\n bubbles: true,\n })\n window.dispatchEvent(event)\n }\n }\n })\n\n // Handle click to execute tool via window event\n const handleClick = (e: MouseEvent) => {\n if (params.action === 'tool-call' && params.toolName) {\n e.preventDefault()\n // SSR-safe: Only call if dispatcher was initialized client-side\n dispatchAction?.(params.toolName, params.params || {})\n }\n }\n\n if (params.type === 'link' || params.action === 'link') {\n return (\n <a\n href={params.url || '#'}\n target={params.url ? '_blank' : undefined}\n rel=\"noopener noreferrer\"\n class={`inline-flex items-center gap-2 px-4 py-2 rounded-md text-sm font-medium transition-colors\n ${params.variant === 'primary' ? 'bg-blue-600 text-white hover:bg-blue-700' :\n params.variant === 'outline' ? 'border border-gray-300 text-gray-700 hover:bg-gray-50 dark:border-gray-600 dark:text-gray-300 dark:hover:bg-gray-800' :\n 'text-blue-600 hover:text-blue-800 dark:text-blue-400 dark:hover:text-blue-300'}`}\n onClick={handleClick}\n >\n <Show when={params.icon}>\n <span>{params.icon}</span>\n </Show>\n {params.label}\n </a>\n )\n }\n\n return (\n <button\n type={params.action === 'submit' ? 'submit' : 'button'}\n disabled={params.disabled}\n class={`inline-flex items-center gap-2 px-4 py-2 rounded-md text-sm font-medium transition-colors focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500\n ${params.variant === 'primary' ? 'bg-blue-600 text-white hover:bg-blue-700 shadow-sm' :\n params.variant === 'secondary' ? 'bg-gray-100 text-gray-900 hover:bg-gray-200 dark:bg-gray-700 dark:text-white dark:hover:bg-gray-600' :\n params.variant === 'outline' ? 'border border-gray-300 text-gray-700 hover:bg-gray-50 dark:border-gray-600 dark:text-gray-300 dark:hover:bg-gray-800' :\n params.variant === 'danger' ? 'bg-red-600 text-white hover:bg-red-700' :\n 'bg-transparent text-gray-700 hover:bg-gray-100 dark:text-gray-300 dark:hover:bg-gray-800'}\n ${params.disabled ? 'opacity-50 cursor-not-allowed' : ''}\n ${params.size === 'sm' ? 'px-3 py-1.5 text-xs' : params.size === 'lg' ? 'px-6 py-3 text-base' : ''}`}\n onClick={handleClick}\n >\n <Show when={params.icon}>\n <span>{params.icon}</span>\n </Show>\n {params.label}\n </button>\n )\n}\n\n/**\n * Main UIResourceRenderer component\n */\nexport const UIResourceRenderer: Component<UIResourceRendererProps> = (props) => {\n const layout = () => {\n // Check if content is a UIComponent (non-composite) vs UILayout (composite)\n if ('type' in props.content && (props.content as any).type !== 'composite') {\n return {\n id: 'single-component',\n components: [props.content as UIComponent],\n grid: {\n columns: 12,\n gap: '1rem',\n },\n } as UILayout\n }\n return props.content as UILayout\n }\n\n // Convert grid styles to CSS string\n const gridContainerStyle = () => {\n const layoutData = layout()\n return `grid-template-columns: repeat(${layoutData.grid.columns}, 1fr); gap: ${layoutData.grid.gap}`\n }\n\n // Convert component grid styles to CSS string\n const getGridStyleString = (component: UIComponent) => {\n // Defensive check for position field - default to full width\n if (!component.position) {\n return 'grid-column: 1 / span 12; grid-row: auto'\n }\n const { colStart, colSpan, rowStart, rowSpan = 1 } = component.position\n return `grid-column: ${colStart} / span ${colSpan}; grid-row: ${rowStart ? `${rowStart} / span ${rowSpan}` : 'auto'}`\n }\n\n const layoutData = layout()\n\n return (\n <div class={`w-full ${props.class || ''}`}>\n <div class=\"grid gap-4\" style={gridContainerStyle()}>\n <For each={layoutData.components}>\n {(component) => (\n <div style={getGridStyleString(component)}>\n <ComponentRenderer component={component} onError={props.onError} />\n </div>\n )}\n </For>\n </div>\n </div>\n )\n}\n"],"names":["ChartRenderer","props","iframeUrl","setIframeUrl","createSignal","isLoading","setIsLoading","error","setError","createEffect","chartParams","component","params","chartConfig","type","data","options","responsive","maintainAspectRatio","configStr","encodeURIComponent","JSON","stringify","url","_el$","_$getNextElement","_tmpl$5","_el$11","firstChild","_el$12","_co$2","_$getNextMarker","nextSibling","_el$13","_el$14","_co$3","_el$15","_el$16","_co$4","_$insert","_$createComponent","Show","when","children","_tmpl$","_el$3","_tmpl$2","_el$4","_el$5","_el$6","_$memo","_el$7","_tmpl$4","_el$1","_el$10","_co$","_el$9","_el$0","title","_el$8","_tmpl$3","addEventListener","onError","message","componentId","id","_$effect","_$setAttribute","renderCellValue","value","undefined","label","name","sanitizedLabel","DOMPurify","sanitize","String","sanitizedUrl","strValue","replace","trim","toLowerCase","markdownLinkRegex","test","htmlValue","ADD_ATTR","hasMarkdown","parsed","marked","parse","async","TableRenderer","tableParams","_el$17","_tmpl$7","_el$18","_el$38","_el$39","_co$8","_el$20","_el$21","_el$22","_el$23","_el$24","_el$40","_el$41","_co$9","_el$19","For","each","columns","column","_el$42","_tmpl$8","_$p","_$style","width","rows","slice","DEFAULT_RESOURCE_LIMITS","maxTableRows","row","i","_el$43","_tmpl$9","_el$44","_tmpl$0","_el$45","_$setProperty","key","_$className","pagination","_el$25","_tmpl$6","_el$26","_el$27","_el$32","_el$33","_co$5","_el$28","_el$34","_el$35","_co$6","_el$30","_el$36","_el$37","_co$7","currentPage","pageSize","Math","min","totalRows","MetricRenderer","metricParams","_el$46","_tmpl$12","_el$47","_el$48","_el$49","_el$50","_el$51","_el$53","_el$54","_co$0","_el$64","_el$65","_co$11","_el$66","_el$67","_co$12","unit","_el$52","_tmpl$1","trend","_el$55","_tmpl$10","_el$56","_el$59","_el$60","_co$1","_el$57","_el$61","_el$62","_co$10","_c$","direction","abs","subtitle","_el$63","_tmpl$11","extractImageFromMarkdown","content","imagePattern","match","alt","imageUrl","linkUrl","credit","TextRenderer","textParams","imageData","createMemo","markdown","htmlContent","fallback","_el$68","_tmpl$13","_el$69","_p$","_v$","className","_v$2","e","t","_el$70","_tmpl$14","_el$71","_el$72","_el$73","_el$74","_el$75","_v$3","_v$4","_v$5","a","IframeRenderer","_el$76","_tmpl$16","_el$80","_el$81","_co$13","_el$79","_el$77","_tmpl$15","_el$78","_v$6","_v$7","_v$8","height","ImageRenderer","_el$82","_tmpl$18","_el$83","_el$84","_el$85","_el$88","_el$89","_co$14","caption","_el$86","_tmpl$17","_el$87","_v$9","_v$0","_v$1","LinkRenderer","_el$90","_tmpl$20","_el$91","_el$92","_el$93","_el$95","_el$96","_co$15","$$click","stopPropagation","description","_el$94","_tmpl$19","_$runHydrationEvents","ComponentRenderer","validation","validateComponent","valid","details","errors","_el$97","_tmpl$21","_el$98","_el$99","GenerativeUIErrorBoundary","componentType","allowRetry","ActionRenderer","dispatchAction","window","toolName","toolParams","event","CustomEvent","detail","bubbles","dispatchEvent","handleClick","action","preventDefault","_el$100","_tmpl$23","_el$102","_el$103","_co$16","_el$104","_el$105","_co$17","icon","_el$101","_tmpl$22","_v$10","_v$11","_v$12","variant","_el$106","_tmpl$24","_el$108","_el$109","_co$18","_el$110","_el$111","_co$19","_el$107","_v$13","_v$14","disabled","_v$15","size","UIResourceRenderer","layout","components","grid","gap","gridContainerStyle","layoutData","getGridStyleString","position","colStart","colSpan","rowStart","rowSpan","_el$112","_tmpl$25","_el$113","_el$114","_tmpl$26","_v$16","class","_v$17","_$delegateEvents"],"mappings":";;;;;;;;;AAyCA,SAASA,cAAcC,OAGpB;AACD,QAAM,CAACC,WAAWC,YAAY,IAAIC,qBAAAA;AAClC,QAAM,CAACC,WAAWC,YAAY,IAAIF,QAAAA,aAAa,IAAI;AACnD,QAAM,CAACG,OAAOC,QAAQ,IAAIJ,qBAAAA;AAI1BK,UAAAA,aAAa,MAAM;AACjB,UAAMC,cAAcT,MAAMU,UAAUC;AACpC,QAAI,CAACF,YAAa;AAGlB,UAAMG,cAAc;AAAA,MAClBC,MAAMJ,YAAYI;AAAAA,MAClBC,MAAML,YAAYK;AAAAA,MAClBC,SAAS;AAAA,QACP,GAAGN,YAAYM;AAAAA,QACfC,YAAY;AAAA,QACZC,qBAAqB;AAAA,MAAA;AAAA,IACvB;AAIF,UAAMC,YAAYC,mBAAmBC,KAAKC,UAAUT,WAAW,CAAC;AAChE,UAAMU,MAAM,iCAAiCJ,SAAS;AAGtDhB,iBAAaoB,GAAG;AAChBjB,iBAAa,KAAK;AAAA,EACpB,CAAC;AAED,UAAA,MAAA;AAAA,QAAAkB,OAAAC,IAAAA,eAAAC,OAAA,GAAAC,SAAAH,KAAAI,YAAA,CAAAC,QAAAC,KAAA,IAAAC,IAAAA,cAAAJ,OAAAK,WAAA,GAAAC,SAAAJ,OAAAG,aAAA,CAAAE,QAAAC,KAAA,IAAAJ,IAAAA,cAAAE,OAAAD,WAAA,GAAAI,SAAAF,OAAAF,aAAA,CAAAK,QAAAC,KAAA,IAAAP,IAAAA,cAAAK,OAAAJ,WAAA;AAAAO,eAAAf,MAAAgB,IAAAA,gBAEKC,cAAI;AAAA,MAAA,IAACC,OAAI;AAAA,eAAErC,UAAAA;AAAAA,MAAW;AAAA,MAAA,IAAAsC,WAAA;AAAA,eAAAlB,IAAAA,eAAAmB,MAAA;AAAA,MAAA;AAAA,IAAA,CAAA,GAAAf,QAAAC,KAAA;AAAAS,eAAAf,MAAAgB,IAAAA,gBAMtBC,cAAI;AAAA,MAAA,IAACC,OAAI;AAAA,eAAEnC,MAAAA;AAAAA,MAAO;AAAA,MAAA,IAAAoC,WAAA;AAAA,YAAAE,QAAApB,IAAAA,eAAAqB,OAAA,GAAAC,QAAAF,MAAAjB,YAAAoB,QAAAD,MAAAnB,YAAAqB,QAAAD,MAAAhB;AAAAO,YAAAA,OAAAU,OAI6C1C,KAAK;AAAA,eAAAsC;AAAAA,MAAA;AAAA,IAAA,CAAA,GAAAX,QAAAC,KAAA;AAAAI,eAAAf,MAAAgB,IAAAA,gBAKpEC,cAAI;AAAA,MAAA,IAACC,OAAI;AAAA,eAAEQ,IAAAA,aAAAhD,WAAW,EAAA,KAAI,CAACK,MAAAA;AAAAA,MAAO;AAAA,MAAA,IAAAoC,WAAA;AAAA,YAAAQ,QAAA1B,IAAAA,eAAA2B,OAAA,GAAAC,QAAAF,MAAAvB,YAAA,CAAA0B,QAAAC,IAAA,IAAAxB,kBAAAsB,MAAArB,WAAA,GAAAwB,QAAAF,OAAAtB,aAAAyB,QAAAD,MAAA5B;AAAAW,mBAAAY,OAAAX,IAAAA,gBAE9BC,cAAI;AAAA,UAAA,IAACC,OAAI;AAAA,mBAAGzC,MAAMU,UAAUC,OAAe8C;AAAAA,UAAK;AAAA,UAAA,IAAAf,WAAA;AAAA,gBAAAgB,QAAAlC,IAAAA,eAAAmC,OAAA;AAAArB,gBAAAA,OAAAoB,OAAA,MAE3C1D,MAAMU,UAAUC,OAAe8C,KAAK;AAAA,mBAAAC;AAAAA,UAAA;AAAA,QAAA,CAAA,GAAAL,QAAAC,IAAA;AAAAE,cAAAI,iBAAA,SAQ7B,MAAM;;AACbrD,mBAAS,sBAAsB;AAC/BP,sBAAM6D,YAAN7D,+BAAgB;AAAA,YACda,MAAM;AAAA,YACNiD,SAAS;AAAA,YACTC,aAAa/D,MAAMU,UAAUsD;AAAAA,UAAAA;AAAAA,QAEjC,CAAC;AAAAC,YAAAA,aAAAC,IAAAA,aAAAV,OAAA,OAVIvD,UAAAA,CAAW,CAAA;AAAA,eAAAiD;AAAAA,MAAA;AAAA,IAAA,CAAA,GAAAd,QAAAC,KAAA;AAAA,WAAAd;AAAAA,EAAA,GAAA;AAiB9B;AAKA,SAAS4C,gBAAgBC,OAAoB;AAE3C,MAAIA,UAAU,QAAQA,UAAUC,QAAW;AACzC,WAAO;AAAA,EACT;AAGA,MAAI,OAAOD,UAAU,YAAYA,UAAU,MAAM;AAE/C,QAAIA,MAAM9C,KAAK;AACb,YAAMgD,QAAQF,MAAMG,QAAQH,MAAME,SAASF,MAAMX,SAASW,MAAM9C;AAChE,YAAMkD,iBAAiBC,UAAUC,SAASC,OAAOL,KAAK,CAAC;AACvD,YAAMM,eAAeH,UAAUC,SAASC,OAAOP,MAAM9C,GAAG,CAAC;AACzD,aAAO,YAAYsD,YAAY,wGAAwGJ,cAAc;AAAA,IACvJ;AAEA,QAAIJ,MAAMG,QAAQH,MAAME,SAASF,MAAMX,OAAO;AAC5C,aAAOgB,UAAUC,SAASC,OAAOP,MAAMG,QAAQH,MAAME,SAASF,MAAMX,KAAK,CAAC;AAAA,IAC5E;AAEA,QAAI;AACF,aAAOrC,KAAKC,UAAU+C,KAAK;AAAA,IAC7B,QAAQ;AACN,aAAO;AAAA,IACT;AAAA,EACF;AAGA,MAAIS,WAAWF,OAAOP,KAAK;AAI3BS,aAAWA,SAASC,QAAQ,6BAA6B,EAAE;AAE3DD,aAAWA,SAASC,QAAQ,0BAA0B,EAAE;AAExD,MAAID,SAASE,KAAAA,EAAOC,YAAAA,MAAkB,aAAa;AACjD,WAAO;AAAA,EACT;AAEA,MAAIH,SAASE,KAAAA,MAAW,IAAI;AAC1B,WAAO;AAAA,EACT;AAGA,QAAME,oBAAoB;AAC1B,MAAIA,kBAAkBC,KAAKL,QAAQ,GAAG;AAEpC,UAAMM,YAAYN,SAASC,QACzB,4BACA,wHACF;AACA,WAAOL,UAAUC,SAASS,WAAW;AAAA,MAAEC,UAAU,CAAC,UAAU,KAAK;AAAA,IAAA,CAAG;AAAA,EACtE;AAGA,QAAMC,cAAc,aAAaH,KAAKL,QAAQ;AAC9C,MAAIQ,aAAa;AAEf,UAAMC,SAASC,WAAAA,OAAOC,MAAMX,UAAU;AAAA,MAAEY,OAAO;AAAA,IAAA,CAAO;AACtD,WAAOhB,UAAUC,SAASY,QAAQ;AAAA,MAAEF,UAAU,CAAC,UAAU,KAAK;AAAA,IAAA,CAAG;AAAA,EACnE;AAGA,SAAOP;AACT;AAKA,SAASa,cAAc1F,OAGpB;AACD,QAAM2F,cAAc3F,MAAMU,UAAUC;AAEpC,UAAA,MAAA;AAAA,QAAAiF,SAAApE,IAAAA,eAAAqE,OAAA,GAAAC,SAAAF,OAAAjE,YAAAoE,SAAAD,OAAAnE,YAAA,CAAAqE,QAAAC,KAAA,IAAAnE,kBAAAiE,OAAAhE,WAAA,GAAAmE,SAAAF,OAAAjE,aAAAoE,SAAAD,OAAAvE,YAAAyE,SAAAD,OAAAxE,YAAA0E,SAAAD,OAAAzE,YAAA2E,SAAAF,OAAArE,aAAAwE,SAAAL,OAAAnE,aAAA,CAAAyE,QAAAC,KAAA,IAAA3E,IAAAA,cAAAyE,OAAAxE,WAAA;AAAAO,eAAAwD,QAAAvD,IAAAA,gBAGOC,cAAI;AAAA,MAAA,IAACC,OAAI;AAAA,eAAEkD,YAAYlC;AAAAA,MAAK;AAAA,MAAA,IAAAf,WAAA;AAAA,YAAAgE,SAAAlF,IAAAA,eAAAmC,OAAA;AAAArB,YAAAA,OAAAoE,QAAA,MAExBf,YAAYlC,KAAK;AAAA,eAAAiD;AAAAA,MAAA;AAAA,IAAA,CAAA,GAAAV,QAAAC,KAAA;AAAA3D,eAAA+D,QAAA9D,IAAAA,gBAQboE,aAAG;AAAA,MAAA,IAACC,OAAI;AAAA,eAAEjB,YAAYkB;AAAAA,MAAO;AAAA,MAAAnE,UAC3BA,CAACoE,YAAW,MAAA;AAAA,YAAAC,SAAAvF,IAAAA,eAAAwF,OAAA;AAAA1E,YAAAA,OAAAyE,QAAA,MAMRD,OAAOxC,KAAK;AAAAL,YAAAA,OAAAgD,CAAAA,QAAAC,IAAAA,MAAAH,QAFND,OAAOK,QAAQ;AAAA,UAAEA,OAAOL,OAAOK;AAAAA,QAAAA,IAAU,CAAA,GAAEF,GAAA,CAAA;AAAA,eAAAF;AAAAA,MAAA,GAAA;AAAA,IAAA,CAIrD,CAAA;AAAAzE,eAAAgE,QAAA/D,IAAAA,gBAKJoE,aAAG;AAAA,MAAA,IAACC,OAAI;AAAA,eAAEjB,YAAYyB,KAAKC,MAAM,GAAGC,WAAAA,wBAAwBC,YAAY;AAAA,MAAC;AAAA,MAAA7E,UACvEA,CAAC8E,KAAUC,OAAC,MAAA;AAAA,YAAAC,SAAAlG,IAAAA,eAAAmG,OAAA;AAAArF,mBAAAoF,QAAAnF,IAAAA,gBAERoE,aAAG;AAAA,UAAA,IAACC,OAAI;AAAA,mBAAEjB,YAAYkB;AAAAA,UAAO;AAAA,UAAAnE,UAC3BA,CAACoE,YAAW,MAAA;AAAA,gBAAAc,SAAApG,IAAAA,eAAAqG,OAAA,GAAAC,SAAAF,OAAAjG;AAAAsC,uBAAA,MAAA8D,gBAAAD,QAAA,aAEO3D,gBAAgBqD,IAAIV,OAAOkB,GAAG,CAAC,CAAC,CAAA;AAAA,mBAAAJ;AAAAA,UAAA,GAAA;AAAA,QAAA,CAEnD,CAAA;AAAA3D,YAAAA,aAAAgE,IAAAA,UAAAP,QANM,gEAAgED,MAAM,MAAM,IAAI,8BAA8B,mCAAmC,EAAE,CAAA;AAAA,eAAAC;AAAAA,MAAA,GAAA;AAAA,IAAA,CAS/J,CAAA;AAAApF,eAAAwD,QAAAvD,IAAAA,gBAMRC,cAAI;AAAA,MAAA,IAACC,OAAI;AAAA,eAAEkD,YAAYuC;AAAAA,MAAU;AAAA,MAAA,IAAAxF,WAAA;AAAA,YAAAyF,SAAA3G,mBAAA4G,OAAA,GAAAC,SAAAF,OAAAxG,YAAA2G,SAAAD,OAAA1G,YAAA4G,SAAAD,OAAAvG,aAAA,CAAAyG,QAAAC,KAAA,IAAA3G,IAAAA,cAAAyG,OAAAxG,WAAA,GAAA2G,SAAAF,OAAAzG,aAAA4G,SAAAD,OAAA3G,aAAA,CAAA6G,QAAAC,KAAA,IAAA/G,kBAAA6G,OAAA5G,WAAA,GAAA+G,SAAAF,OAAA7G,aAAAgH,SAAAD,OAAA/G,aAAA,CAAAiH,QAAAC,KAAA,IAAAnH,IAAAA,cAAAiH,OAAAhH,WAAA;AAAAO,YAAAA,OAAA+F,QAAA,MAGnB1C,YAAYuC,WAAWgB,cAAcvD,YAAYuC,WAAWiB,WAAW,GAACX,QAAAC,KAAA;AAAAnG,YAAAA,OAAA+F,QAAA,MAChFe,KAAKC,KACH1D,YAAYuC,WAAWgB,cAAc,KAAKvD,YAAYuC,WAAWiB,UAClExD,YAAYuC,WAAWoB,SACzB,GAACV,QAAAC,KAAA;AAAAvG,YAAAA,OAAA+F,QAAA,MACG1C,YAAYuC,WAAWoB,WAASN,QAAAC,KAAA;AAAA,eAAAd;AAAAA,MAAA;AAAA,IAAA,CAAA,GAAA3B,QAAAC,KAAA;AAAA,WAAAb;AAAAA,EAAA,GAAA;AAOlD;AAKA,SAAS2D,eAAevJ,OAAmC;AACzD,QAAMwJ,eAAexJ,MAAMU,UAAUC;AAErC,UAAA,MAAA;AAAA,QAAA8I,SAAAjI,IAAAA,eAAAkI,QAAA,GAAAC,SAAAF,OAAA9H,YAAAiI,SAAAD,OAAAhI,YAAAkI,SAAAD,OAAAjI,YAAAmI,SAAAD,OAAA9H,aAAAgI,SAAAD,OAAAnI,YAAAqI,SAAAD,OAAAhI,aAAA,CAAAkI,QAAAC,KAAA,IAAApI,IAAAA,cAAAkI,OAAAjI,WAAA,GAAAoI,SAAAP,OAAA7H,aAAA,CAAAqI,QAAAC,MAAA,IAAAvI,IAAAA,cAAAqI,OAAApI,WAAA,GAAAuI,SAAAF,OAAArI,aAAA,CAAAwI,QAAAC,MAAA,IAAA1I,kBAAAwI,OAAAvI,WAAA;AAAAO,QAAAA,OAAAuH,QAAA,MAKWL,aAAa/F,KAAK;AAAAnB,QAAAA,OAAAyH,QAAA,MAG8CP,aAAapF,KAAK;AAAA9B,eAAAwH,QAAAvH,IAAAA,gBAClFC,cAAI;AAAA,MAAA,IAACC,OAAI;AAAA,eAAE+G,aAAaiB;AAAAA,MAAI;AAAA,MAAA,IAAA/H,WAAA;AAAA,YAAAgI,SAAAlJ,IAAAA,eAAAmJ,OAAA;AAAArI,YAAAA,OAAAoI,QAAA,MAExBlB,aAAaiB,IAAI;AAAA,eAAAC;AAAAA,MAAA;AAAA,IAAA,CAAA,GAAAT,QAAAC,KAAA;AAAA5H,eAAAqH,QAAApH,IAAAA,gBAMzBC,cAAI;AAAA,MAAA,IAACC,OAAI;AAAA,eAAE+G,aAAaoB;AAAAA,MAAK;AAAA,MAAA,IAAAlI,WAAA;AAAA,YAAAmI,SAAArJ,IAAAA,eAAAsJ,QAAA,GAAAC,SAAAF,OAAAlJ,YAAAqJ,SAAAD,OAAApJ,YAAA,CAAAsJ,QAAAC,KAAA,IAAApJ,IAAAA,cAAAkJ,OAAAjJ,WAAA,GAAAoJ,SAAAF,OAAAlJ,aAAAqJ,SAAAD,OAAApJ,aAAA,CAAAsJ,QAAAC,MAAA,IAAAxJ,IAAAA,cAAAsJ,OAAArJ,WAAA;AAAAsJ,eAAAtJ;AAAAO,YAAAA,OAAAyI,SAAA,MAAA;AAAA,cAAAQ,MAAAtI,IAAAA,KAAA,MAUvBuG,aAAaoB,MAAMY,cAAc,IAAI;AAAA,iBAAA,MAArCD,QACG,MACA/B,aAAaoB,MAAMY,cAAc,SAC/B,MACA;AAAA,QAAG,GAAA,GAAAP,QAAAC,KAAA;AAAA5I,mBAAAyI,QAAA,MACR3B,KAAKqC,IAAIjC,aAAaoB,MAAMxG,KAAK,GAACiH,QAAAC,MAAA;AAAArH,YAAAA,OAAA,MAAAgE,IAAAA,UAAA8C,QAZ5B,uBAAuBvB,aAAaoB,MAAMY,cAAc,OAC3D,uCACAhC,aAAaoB,MAAMY,cAAc,SAC/B,mCACA,kCAAkC,EACpC,CAAA;AAAA,eAAAX;AAAAA,MAAA;AAAA,IAAA,CAAA,GAAAT,QAAAC,MAAA;AAAA/H,eAAAqH,QAAApH,IAAAA,gBAYTC,cAAI;AAAA,MAAA,IAACC,OAAI;AAAA,eAAE+G,aAAakC;AAAAA,MAAQ;AAAA,MAAA,IAAAhJ,WAAA;AAAA,YAAAiJ,SAAAnK,IAAAA,eAAAoK,QAAA;AAAAtJ,YAAAA,OAAAqJ,QAAA,MAC2BnC,aAAakC,QAAQ;AAAA,eAAAC;AAAAA,MAAA;AAAA,IAAA,CAAA,GAAApB,QAAAC,MAAA;AAAA,WAAAf;AAAAA,EAAA,GAAA;AAKzF;AAMA,SAASoC,yBAAyBC,SAA4F;AAE5H,QAAMC,eAAe;AACrB,QAAMC,QAAQF,QAAQE,MAAMD,YAAY;AAExC,MAAIC,OAAO;AACT,WAAO;AAAA,MACLC,KAAKD,MAAM,CAAC,KAAK;AAAA,MACjBE,UAAUF,MAAM,CAAC;AAAA,MACjBG,SAASH,MAAM,CAAC;AAAA,MAChBI,QAAQJ,MAAM,CAAC,EAAEjH,KAAAA;AAAAA,IAAK;AAAA,EAE1B;AAEA,SAAO;AACT;AAKA,SAASsH,aAAarM,OAAmC;AACvD,QAAMsM,aAAatM,MAAMU,UAAUC;AAGnC,QAAM4L,YAAYC,QAAAA,WAAW,MAAM;AACjC,QAAIF,WAAWG,YAAYH,WAAWR,SAAS;AAC7C,aAAOD,yBAAyBS,WAAWR,OAAO;AAAA,IACpD;AACA,WAAO;AAAA,EACT,CAAC;AAGD,QAAMY,cAAcF,QAAAA,WAAW,MAAM;AACnC,QAAIF,WAAWG,YAAY,CAACF,aAAa;AACvC,aAAOhH,kBAAOC,MAAM8G,WAAWR,SAAS;AAAA,QAAErG,OAAO;AAAA,MAAA,CAAO;AAAA,IAC1D;AACA,WAAO6G,WAAWR;AAAAA,EACpB,CAAC;AAGD,SAAAvJ,IAAAA,gBACGC,QAAAA,MAAI;AAAA,IAAA,IACHC,OAAI;AAAA,aAAE8J,UAAAA;AAAAA,IAAW;AAAA,IAAA,IACjBI,WAAQ;AAAA,cAAA,MAAA;AAAA,YAAAC,SAAApL,IAAAA,eAAAqL,QAAA,GAAAC,SAAAF,OAAAjL;AAAAsC,YAAAA,OAAA8I,CAAAA,QAAA;AAAA,cAAAC,MAGK,+CAA+CV,WAAWW,aAAa,EAAE,IAAEC,OACvER,YAAAA;AAAaM,kBAAAD,IAAAI,KAAAlF,IAAAA,UAAA6E,QAAAC,IAAAI,IAAAH,GAAA;AAAAE,mBAAAH,IAAAK,KAAArF,IAAAA,YAAA+E,QAAA,aAAAC,IAAAK,IAAAF,IAAA;AAAA,iBAAAH;AAAAA,QAAA,GAAA;AAAA,UAAAI,GAAA9I;AAAAA,UAAA+I,GAAA/I;AAAAA,QAAAA,CAAA;AAAA,eAAAuI;AAAAA,MAAA,GAAA;AAAA,IAAA;AAAA,IAAAlK,UAK5B5B,WAAI,MAAA;AAAA,UAAAuM,SAAA7L,IAAAA,eAAA8L,QAAA,GAAAC,SAAAF,OAAA1L,YAAA6L,SAAAD,OAAA5L,YAAA8L,SAAAD,OAAA7L,YAAA+L,SAAAH,OAAAxL,aAAA4L,SAAAD,OAAA/L;AAAAW,UAAAA,OAAAqL,QAAA,MAawE7M,KAAAA,EAAOsL,MAAM;AAAAnI,UAAAA,OAAA8I,CAAAA,QAAA;AAAA,YAAAa,OAV5E9M,OAAOqL,SAAO0B,OAEd/M,OAAOoL,UAAQ4B,OACfhN,KAAAA,EAAOmL;AAAG2B,iBAAAb,IAAAI,KAAAjJ,IAAAA,aAAAsJ,QAAA,QAAAT,IAAAI,IAAAS,IAAA;AAAAC,iBAAAd,IAAAK,KAAAlJ,IAAAA,aAAAuJ,QAAA,OAAAV,IAAAK,IAAAS,IAAA;AAAAC,iBAAAf,IAAAgB,KAAA7J,IAAAA,aAAAuJ,QAAA,OAAAV,IAAAgB,IAAAD,IAAA;AAAA,eAAAf;AAAAA,MAAA,GAAA;AAAA,QAAAI,GAAA9I;AAAAA,QAAA+I,GAAA/I;AAAAA,QAAA0J,GAAA1J;AAAAA,MAAAA,CAAA;AAAA,aAAAgJ;AAAAA,IAAA,GAAA;AAAA,EAAA,CAUxB;AAGP;AAKA,SAASW,eAAehO,OAAmC;AACzD,QAAMW,SAASX,MAAMU,UAAUC;AAC/B,UAAA,MAAA;AAAA,QAAAsN,SAAAzM,IAAAA,eAAA0M,QAAA,GAAAC,SAAAF,OAAAtM,YAAA,CAAAyM,QAAAC,MAAA,IAAAvM,IAAAA,cAAAqM,OAAApM,WAAA,GAAAuM,SAAAF,OAAArM;AAAAO,eAAA2L,QAAA1L,IAAAA,gBAEKC,cAAI;AAAA,MAAA,IAACC,OAAI;AAAA,eAAE9B,OAAO8C;AAAAA,MAAK;AAAA,MAAA,IAAAf,WAAA;AAAA,YAAA6L,SAAA/M,IAAAA,eAAAgN,QAAA,GAAAC,SAAAF,OAAA5M;AAAAW,YAAAA,OAAAmM,QAAA,MAE6C9N,OAAO8C,KAAK;AAAA,eAAA8K;AAAAA,MAAA;AAAA,IAAA,CAAA,GAAAH,QAAAC,MAAA;AAAApK,QAAAA,OAAA8I,CAAAA,QAAA;AAAA,UAAA2B,OAI1E/N,OAAOW,KAAGqN,OACRhO,OAAO8C,SAAS,oBAAkBmL,OAElC,WAAWjO,OAAOkO,UAAU,OAAO;AAAsBH,eAAA3B,IAAAI,KAAAjJ,IAAAA,aAAAoK,QAAA,OAAAvB,IAAAI,IAAAuB,IAAA;AAAAC,eAAA5B,IAAAK,KAAAlJ,IAAAA,aAAAoK,QAAA,SAAAvB,IAAAK,IAAAuB,IAAA;AAAA5B,UAAAgB,IAAA7G,IAAAA,MAAAoH,QAAAM,MAAA7B,IAAAgB,CAAA;AAAA,aAAAhB;AAAAA,IAAA,GAAA;AAAA,MAAAI,GAAA9I;AAAAA,MAAA+I,GAAA/I;AAAAA,MAAA0J,GAAA1J;AAAAA,IAAAA,CAAA;AAAA,WAAA4J;AAAAA,EAAA,GAAA;AAMxE;AAKA,SAASa,cAAc9O,OAAmC;AACxD,QAAMW,SAASX,MAAMU,UAAUC;AAE/B,UAAA,MAAA;AAAA,QAAAoO,SAAAvN,mBAAAwN,QAAA,GAAAC,SAAAF,OAAApN,YAAAuN,SAAAD,OAAAtN,YAAAwN,SAAAD,OAAAvN,YAAAyN,SAAAH,OAAAlN,aAAA,CAAAsN,QAAAC,MAAA,IAAAxN,IAAAA,cAAAsN,OAAArN,WAAA;AAAAO,eAAAyM,QAAAxM,IAAAA,gBAYKC,cAAI;AAAA,MAAA,IAACC,OAAI;AAAA,eAAE9B,OAAO4O;AAAAA,MAAO;AAAA,MAAA,IAAA7M,WAAA;AAAA,YAAA8M,SAAAhO,IAAAA,eAAAiO,QAAA,GAAAC,SAAAF,OAAA7N;AAAAW,YAAAA,OAAAoN,QAAA,MAE2C/O,OAAO4O,OAAO;AAAA,eAAAC;AAAAA,MAAA;AAAA,IAAA,CAAA,GAAAH,QAAAC,MAAA;AAAArL,QAAAA,OAAA8I,CAAAA,QAAA;AAAA,UAAA4C,OAXxEhP,OAAOW,KAAGsO,OAEVjP,OAAOW,KAAGuO,OACVlP,OAAOsL,OAAO;AAAO0D,eAAA5C,IAAAI,KAAAjJ,IAAAA,aAAAgL,QAAA,QAAAnC,IAAAI,IAAAwC,IAAA;AAAAC,eAAA7C,IAAAK,KAAAlJ,IAAAA,aAAAiL,QAAA,OAAApC,IAAAK,IAAAwC,IAAA;AAAAC,eAAA9C,IAAAgB,KAAA7J,IAAAA,aAAAiL,QAAA,OAAApC,IAAAgB,IAAA8B,IAAA;AAAA,aAAA9C;AAAAA,IAAA,GAAA;AAAA,MAAAI,GAAA9I;AAAAA,MAAA+I,GAAA/I;AAAAA,MAAA0J,GAAA1J;AAAAA,IAAAA,CAAA;AAAA,WAAA0K;AAAAA,EAAA,GAAA;AAatC;AAQA,SAASe,aAAa9P,OAAmC;AACvD,QAAMW,SAASX,MAAMU,UAAUC;AAE/B,UAAA,MAAA;AAAA,QAAAoP,SAAAvO,mBAAAwO,QAAA,GAAAC,SAAAF,OAAApO,YAAAuO,SAAAD,OAAAlO,aAAAoO,SAAAD,OAAAvO,YAAAyO,SAAAD,OAAApO,aAAA,CAAAsO,QAAAC,MAAA,IAAAxO,IAAAA,cAAAsO,OAAArO,WAAA;AAAAgO,WAAAQ,UAMcpD,CAAAA,MAAMA,EAAEqD,gBAAAA;AAAiBlO,QAAAA,OAAA6N,QAAA,MAmB9BxP,OAAO2D,SAAS3D,OAAOW,GAAG;AAAAgB,eAAA4N,QAAA3N,IAAAA,gBAE5BC,cAAI;AAAA,MAAA,IAACC,OAAI;AAAA,eAAE9B,OAAO8P;AAAAA,MAAW;AAAA,MAAA,IAAA/N,WAAA;AAAA,YAAAgO,SAAAlP,IAAAA,eAAAmP,QAAA;AAAArO,YAAAA,OAAAoO,QAAA,MACkC/P,OAAO8P,WAAW;AAAA,eAAAC;AAAAA,MAAA;AAAA,IAAA,CAAA,GAAAL,QAAAC,MAAA;AAAArM,QAAAA,aAAAC,IAAAA,aAAA6L,QAAA,QA1B9EpP,OAAOW,GAAG,CAAA;AAAAsP,2BAAAA;AAAA,WAAAb;AAAAA,EAAA,GAAA;AA6CtB;AAKA,SAASc,kBAAkB7Q,OAGxB;;AAED,QAAM8Q,eAAaC,WAAAA,kBAAkB/Q,MAAMU,SAAS;AACpD,MAAI,CAACoQ,aAAWE,OAAO;AACrBhR,gBAAM6D,YAAN7D,+BAAgB;AAAA,MACda,MAAM;AAAA,MACNiD,SAAS;AAAA,MACTC,aAAa/D,MAAMU,UAAUsD;AAAAA,MAC7BiN,SAASH,aAAWI;AAAAA,IAAAA;AAGtB,YAAA,MAAA;AAAA,UAAAC,SAAA3P,mBAAA4P,QAAA,GAAAC,SAAAF,OAAAxP,YAAA2P,SAAAD,OAAAtP;AAAAO,UAAAA,OAAAgP,QAAA;;AAIOR,uBAAAA,MAAAA,aAAWI,WAAXJ,gBAAAA,IAAoB,OAApBA,mBAAwBhN,YAAW;AAAA,OAA0B;AAAA,aAAAqN;AAAAA,IAAA,GAAA;AAAA,EAItE;AAGA,SAAA5O,IAAAA,gBACGgP,0BAAAA,2BAAyB;AAAA,IAAA,IACxBxN,cAAW;AAAA,aAAE/D,MAAMU,UAAUsD;AAAAA,IAAE;AAAA,IAAA,IAC/BwN,gBAAa;AAAA,aAAExR,MAAMU,UAAUG;AAAAA,IAAI;AAAA,IAAA,IACnCgD,UAAO;AAAA,aAAE7D,MAAM6D;AAAAA,IAAO;AAAA,IACtB4N,YAAY;AAAA,IAAI,IAAA/O,WAAA;AAAA,aAAA,CAAAH,IAAAA,gBAEfC,cAAI;AAAA,QAAA,IAACC,OAAI;AAAA,iBAAEzC,MAAMU,UAAUG,SAAS;AAAA,QAAO;AAAA,QAAA,IAAA6B,WAAA;AAAA,iBAAAH,IAAAA,gBACzCxC,eAAa;AAAA,YAAA,IAACW,YAAS;AAAA,qBAAEV,MAAMU;AAAAA,YAAS;AAAA,YAAA,IAAEmD,UAAO;AAAA,qBAAE7D,MAAM6D;AAAAA,YAAO;AAAA,UAAA,CAAA;AAAA,QAAA;AAAA,MAAA,CAAA,GAAAtB,IAAAA,gBAElEC,cAAI;AAAA,QAAA,IAACC,OAAI;AAAA,iBAAEzC,MAAMU,UAAUG,SAAS;AAAA,QAAO;AAAA,QAAA,IAAA6B,WAAA;AAAA,iBAAAH,IAAAA,gBACzCmD,eAAa;AAAA,YAAA,IAAChF,YAAS;AAAA,qBAAEV,MAAMU;AAAAA,YAAS;AAAA,YAAA,IAAEmD,UAAO;AAAA,qBAAE7D,MAAM6D;AAAAA,YAAO;AAAA,UAAA,CAAA;AAAA,QAAA;AAAA,MAAA,CAAA,GAAAtB,IAAAA,gBAElEC,cAAI;AAAA,QAAA,IAACC,OAAI;AAAA,iBAAEzC,MAAMU,UAAUG,SAAS;AAAA,QAAQ;AAAA,QAAA,IAAA6B,WAAA;AAAA,iBAAAH,IAAAA,gBAC1CgH,gBAAc;AAAA,YAAA,IAAC7I,YAAS;AAAA,qBAAEV,MAAMU;AAAAA,YAAS;AAAA,UAAA,CAAA;AAAA,QAAA;AAAA,MAAA,CAAA,GAAA6B,IAAAA,gBAE3CC,cAAI;AAAA,QAAA,IAACC,OAAI;AAAA,iBAAEzC,MAAMU,UAAUG,SAAS;AAAA,QAAM;AAAA,QAAA,IAAA6B,WAAA;AAAA,iBAAAH,IAAAA,gBACxC8J,cAAY;AAAA,YAAA,IAAC3L,YAAS;AAAA,qBAAEV,MAAMU;AAAAA,YAAS;AAAA,UAAA,CAAA;AAAA,QAAA;AAAA,MAAA,CAAA,GAAA6B,IAAAA,gBAEzCC,cAAI;AAAA,QAAA,IAACC,OAAI;AAAA,iBAAEzC,MAAMU,UAAUG,SAAS;AAAA,QAAQ;AAAA,QAAA,IAAA6B,WAAA;AAAA,iBAAAH,IAAAA,gBAC1CyL,gBAAc;AAAA,YAAA,IAACtN,YAAS;AAAA,qBAAEV,MAAMU;AAAAA,YAAS;AAAA,UAAA,CAAA;AAAA,QAAA;AAAA,MAAA,CAAA,GAAA6B,IAAAA,gBAE3CC,cAAI;AAAA,QAAA,IAACC,OAAI;AAAA,iBAAEzC,MAAMU,UAAUG,SAAS;AAAA,QAAO;AAAA,QAAA,IAAA6B,WAAA;AAAA,iBAAAH,IAAAA,gBACzCuM,eAAa;AAAA,YAAA,IAACpO,YAAS;AAAA,qBAAEV,MAAMU;AAAAA,YAAS;AAAA,UAAA,CAAA;AAAA,QAAA;AAAA,MAAA,CAAA,GAAA6B,IAAAA,gBAE1CC,cAAI;AAAA,QAAA,IAACC,OAAI;AAAA,iBAAEzC,MAAMU,UAAUG,SAAS;AAAA,QAAM;AAAA,QAAA,IAAA6B,WAAA;AAAA,iBAAAH,IAAAA,gBACxCuN,cAAY;AAAA,YAAA,IAACpP,YAAS;AAAA,qBAAEV,MAAMU;AAAAA,YAAS;AAAA,UAAA,CAAA;AAAA,QAAA;AAAA,MAAA,CAAA,GAAA6B,IAAAA,gBAEzCC,cAAI;AAAA,QAAA,IAACC,OAAI;AAAA,iBAAEzC,MAAMU,UAAUG,SAAS;AAAA,QAAQ;AAAA,QAAA,IAAA6B,WAAA;AAAA,iBAAAH,IAAAA,gBAC1CmP,gBAAc;AAAA,YAAA,IAAChR,YAAS;AAAA,qBAAEV,MAAMU;AAAAA,YAAS;AAAA,UAAA,CAAA;AAAA,QAAA;AAAA,MAAA,CAAA,CAAA;AAAA,IAAA;AAAA,EAAA,CAAA;AAIlD;AAKA,SAASgR,eAAe1R,OAAmC;AACzD,QAAMW,SAASX,MAAMU,UAAUC;AAC/B,MAAIgR,iBAAuE;AAI3EnR,UAAAA,aAAa,MAAM;AACjB,QAAI,OAAOoR,WAAW,aAAa;AACjCD,uBAAiBA,CAACE,UAAkBC,eAAoB;AACtD,cAAMC,QAAQ,IAAIC,YAAY,cAAc;AAAA,UAC1CC,QAAQ;AAAA,YACNJ;AAAAA,YACAlR,QAAQmR;AAAAA,UAAAA;AAAAA,UAEVI,SAAS;AAAA,QAAA,CACV;AACDN,eAAOO,cAAcJ,KAAK;AAAA,MAC5B;AAAA,IACF;AAAA,EACF,CAAC;AAGD,QAAMK,cAAcA,CAACjF,MAAkB;AACrC,QAAIxM,OAAO0R,WAAW,eAAe1R,OAAOkR,UAAU;AACpD1E,QAAEmF,eAAAA;AAEFX,uDAAiBhR,OAAOkR,UAAUlR,OAAOA,UAAU,CAAA;AAAA,IACrD;AAAA,EACF;AAEA,MAAIA,OAAOE,SAAS,UAAUF,OAAO0R,WAAW,QAAQ;AACtD,YAAA,MAAA;AAAA,UAAAE,UAAA/Q,IAAAA,eAAAgR,QAAA,GAAAC,UAAAF,QAAA5Q,YAAA,CAAA+Q,SAAAC,MAAA,IAAA7Q,IAAAA,cAAA2Q,QAAA1Q,WAAA,GAAA6Q,UAAAF,QAAA3Q,aAAA,CAAA8Q,SAAAC,MAAA,IAAAhR,IAAAA,cAAA8Q,QAAA7Q,WAAA;AAAAwQ,cAAAhC,UASa6B;AAAW9P,iBAAAiQ,SAAAhQ,IAAAA,gBAEnBC,cAAI;AAAA,QAAA,IAACC,OAAI;AAAA,iBAAE9B,OAAOoS;AAAAA,QAAI;AAAA,QAAA,IAAArQ,WAAA;AAAA,cAAAsQ,UAAAxR,IAAAA,eAAAyR,QAAA;AAAA3Q,cAAAA,OAAA0Q,SAAA,MACdrS,OAAOoS,IAAI;AAAA,iBAAAC;AAAAA,QAAA;AAAA,MAAA,CAAA,GAAAN,SAAAC,MAAA;AAAArQ,UAAAA,OAAAiQ,SAAA,MAEnB5R,OAAO2D,OAAKuO,SAAAC,MAAA;AAAA7O,UAAAA,OAAA8I,CAAAA,QAAA;AAAA,YAAAmG,QAZPvS,OAAOW,OAAO,KAAG6R,QACfxS,OAAOW,MAAM,WAAW+C,QAAS+O,QAElC;AAAA,YACHzS,OAAO0S,YAAY,YAAY,6CAC/B1S,OAAO0S,YAAY,YAAY,yHAC7B,+EAA+E;AAAEH,kBAAAnG,IAAAI,KAAAjJ,IAAAA,aAAAqO,SAAA,QAAAxF,IAAAI,IAAA+F,KAAA;AAAAC,kBAAApG,IAAAK,KAAAlJ,IAAAA,aAAAqO,SAAA,UAAAxF,IAAAK,IAAA+F,KAAA;AAAAC,kBAAArG,IAAAgB,KAAA9F,IAAAA,UAAAsK,SAAAxF,IAAAgB,IAAAqF,KAAA;AAAA,eAAArG;AAAAA,MAAA,GAAA;AAAA,QAAAI,GAAA9I;AAAAA,QAAA+I,GAAA/I;AAAAA,QAAA0J,GAAA1J;AAAAA,MAAAA,CAAA;AAAAuM,6BAAAA;AAAA,aAAA2B;AAAAA,IAAA,GAAA;AAAA,EAS7F;AAEA,UAAA,MAAA;AAAA,QAAAe,UAAA9R,IAAAA,eAAA+R,QAAA,GAAAC,UAAAF,QAAA3R,YAAA,CAAA8R,SAAAC,MAAA,IAAA5R,IAAAA,cAAA0R,QAAAzR,WAAA,GAAA4R,UAAAF,QAAA1R,aAAA,CAAA6R,SAAAC,MAAA,IAAA/R,IAAAA,cAAA6R,QAAA5R,WAAA;AAAAuR,YAAA/C,UAYa6B;AAAW9P,eAAAgR,SAAA/Q,IAAAA,gBAEnBC,cAAI;AAAA,MAAA,IAACC,OAAI;AAAA,eAAE9B,OAAOoS;AAAAA,MAAI;AAAA,MAAA,IAAArQ,WAAA;AAAA,YAAAoR,UAAAtS,IAAAA,eAAAyR,QAAA;AAAA3Q,YAAAA,OAAAwR,SAAA,MACdnT,OAAOoS,IAAI;AAAA,eAAAe;AAAAA,MAAA;AAAA,IAAA,CAAA,GAAAL,SAAAC,MAAA;AAAApR,QAAAA,OAAAgR,SAAA,MAEnB3S,OAAO2D,OAAKsP,SAAAC,MAAA;AAAA5P,QAAAA,OAAA8I,CAAAA,QAAA;AAAA,UAAAgH,QAfPpT,OAAO0R,WAAW,WAAW,WAAW,UAAQ2B,QAC5CrT,OAAOsT,UAAQC,QAClB;AAAA,UACHvT,OAAO0S,YAAY,YAAY,uDAC/B1S,OAAO0S,YAAY,cAAc,wGAC/B1S,OAAO0S,YAAY,YAAY,yHAC7B1S,OAAO0S,YAAY,WAAW,2CAC5B,0FAA0F;AAAA,UAChG1S,OAAOsT,WAAW,kCAAkC,EAAE;AAAA,UACtDtT,OAAOwT,SAAS,OAAO,wBAAwBxT,OAAOwT,SAAS,OAAO,wBAAwB,EAAE;AAAEJ,gBAAAhH,IAAAI,KAAAjJ,IAAAA,aAAAoP,SAAA,QAAAvG,IAAAI,IAAA4G,KAAA;AAAAC,gBAAAjH,IAAAK,KAAArF,IAAAA,YAAAuL,SAAA,YAAAvG,IAAAK,IAAA4G,KAAA;AAAAE,gBAAAnH,IAAAgB,KAAA9F,IAAAA,UAAAqL,SAAAvG,IAAAgB,IAAAmG,KAAA;AAAA,aAAAnH;AAAAA,IAAA,GAAA;AAAA,MAAAI,GAAA9I;AAAAA,MAAA+I,GAAA/I;AAAAA,MAAA0J,GAAA1J;AAAAA,IAAAA,CAAA;AAAAuM,2BAAAA;AAAA,WAAA0C;AAAAA,EAAA,GAAA;AAS5G;AAKO,MAAMc,qBAA0DpU,CAAAA,UAAU;AAC/E,QAAMqU,SAASA,MAAM;AAEnB,QAAI,UAAUrU,MAAM8L,WAAY9L,MAAM8L,QAAgBjL,SAAS,aAAa;AAC1E,aAAO;AAAA,QACLmD,IAAI;AAAA,QACJsQ,YAAY,CAACtU,MAAM8L,OAAsB;AAAA,QACzCyI,MAAM;AAAA,UACJ1N,SAAS;AAAA,UACT2N,KAAK;AAAA,QAAA;AAAA,MACP;AAAA,IAEJ;AACA,WAAOxU,MAAM8L;AAAAA,EACf;AAGA,QAAM2I,qBAAqBA,MAAM;AAC/B,UAAMC,cAAaL,OAAAA;AACnB,WAAO,iCAAiCK,YAAWH,KAAK1N,OAAO,gBAAgB6N,YAAWH,KAAKC,GAAG;AAAA,EACpG;AAGA,QAAMG,qBAAqBA,CAACjU,cAA2B;AAErD,QAAI,CAACA,UAAUkU,UAAU;AACvB,aAAO;AAAA,IACT;AACA,UAAM;AAAA,MAAEC;AAAAA,MAAUC;AAAAA,MAASC;AAAAA,MAAUC,UAAU;AAAA,IAAA,IAAMtU,UAAUkU;AAC/D,WAAO,gBAAgBC,QAAQ,WAAWC,OAAO,eAAeC,WAAW,GAAGA,QAAQ,WAAWC,OAAO,KAAK,MAAM;AAAA,EACrH;AAEA,QAAMN,aAAaL,OAAAA;AAEnB,UAAA,MAAA;AAAA,QAAAY,UAAAzT,IAAAA,eAAA0T,QAAA,GAAAC,UAAAF,QAAAtT;AAAAW,eAAA6S,SAAA5S,IAAAA,gBAGOoE,aAAG;AAAA,MAAA,IAACC,OAAI;AAAA,eAAE8N,WAAWJ;AAAAA,MAAU;AAAA,MAAA5R,UAC5BhC,gBAAS,MAAA;AAAA,YAAA0U,UAAA5T,IAAAA,eAAA6T,QAAA;AAAA/S,mBAAA8S,SAAA7S,IAAAA,gBAENsO,mBAAiB;AAAA,UAACnQ;AAAAA,UAAoB,IAAEmD,UAAO;AAAA,mBAAE7D,MAAM6D;AAAAA,UAAO;AAAA,QAAA,CAAA,CAAA;AAAAI,mBAAAgD,SAAAC,UAAAkO,SADrDT,mBAAmBjU,SAAS,GAACuG,GAAA,CAAA;AAAA,eAAAmO;AAAAA,MAAA,GAAA;AAAA,IAAA,CAG1C,CAAA;AAAAnR,QAAAA,OAAA8I,CAAAA,QAAA;AAAA,UAAAuI,QAPK,UAAUtV,MAAMuV,SAAS,EAAE,IAAEC,QACRf,mBAAAA;AAAoBa,gBAAAvI,IAAAI,KAAAlF,IAAAA,UAAAgN,SAAAlI,IAAAI,IAAAmI,KAAA;AAAAvI,UAAAK,IAAAlG,IAAAA,MAAAiO,SAAAK,OAAAzI,IAAAK,CAAA;AAAA,aAAAL;AAAAA,IAAA,GAAA;AAAA,MAAAI,GAAA9I;AAAAA,MAAA+I,GAAA/I;AAAAA,IAAAA,CAAA;AAAA,WAAA4Q;AAAAA,EAAA,GAAA;AAWzD;AAACQ,IAAAA,eAAA,CAAA,OAAA,CAAA;;"}
@@ -1 +1 @@
1
- {"version":3,"file":"UIResourceRenderer.d.ts","sourceRoot":"","sources":["../../src/components/UIResourceRenderer.tsx"],"names":[],"mappings":"AAAA;;;GAGG;AAGH,OAAO,EAAE,SAAS,EAAqD,MAAM,UAAU,CAAA;AAEvF,OAAO,KAAK,EAAE,WAAW,EAAE,QAAQ,EAAE,aAAa,EAAiB,MAAM,UAAU,CAAA;AAKnF;;GAEG;AACH,MAAM,WAAW,uBAAuB;IACtC;;OAEG;IACH,OAAO,EAAE,WAAW,GAAG,QAAQ,CAAA;IAE/B;;OAEG;IACH,QAAQ,CAAC,EAAE,OAAO,CAAA;IAElB;;OAEG;IACH,OAAO,CAAC,EAAE,CAAC,KAAK,EAAE,aAAa,KAAK,IAAI,CAAA;IAExC;;OAEG;IACH,KAAK,CAAC,EAAE,MAAM,CAAA;CACf;AAkkBD;;GAEG;AACH,eAAO,MAAM,kBAAkB,EAAE,SAAS,CAAC,uBAAuB,CA+CjE,CAAA"}
1
+ {"version":3,"file":"UIResourceRenderer.d.ts","sourceRoot":"","sources":["../../src/components/UIResourceRenderer.tsx"],"names":[],"mappings":"AAAA;;;GAGG;AAGH,OAAO,EAAE,SAAS,EAAqD,MAAM,UAAU,CAAA;AAEvF,OAAO,KAAK,EAAE,WAAW,EAAE,QAAQ,EAAE,aAAa,EAAiB,MAAM,UAAU,CAAA;AAKnF;;GAEG;AACH,MAAM,WAAW,uBAAuB;IACtC;;OAEG;IACH,OAAO,EAAE,WAAW,GAAG,QAAQ,CAAA;IAE/B;;OAEG;IACH,QAAQ,CAAC,EAAE,OAAO,CAAA;IAElB;;OAEG;IACH,OAAO,CAAC,EAAE,CAAC,KAAK,EAAE,aAAa,KAAK,IAAI,CAAA;IAExC;;OAEG;IACH,KAAK,CAAC,EAAE,MAAM,CAAA;CACf;AAulBD;;GAEG;AACH,eAAO,MAAM,kBAAkB,EAAE,SAAS,CAAC,uBAAuB,CA+CjE,CAAA"}
@@ -82,6 +82,22 @@ function renderCellValue(value) {
82
82
  if (value === null || value === void 0) {
83
83
  return "-";
84
84
  }
85
+ if (typeof value === "object" && value !== null) {
86
+ if (value.url) {
87
+ const label = value.name || value.label || value.title || value.url;
88
+ const sanitizedLabel = purify.sanitize(String(label));
89
+ const sanitizedUrl = purify.sanitize(String(value.url));
90
+ return `<a href="${sanitizedUrl}" target="_blank" rel="noopener noreferrer" class="text-blue-600 dark:text-blue-400 hover:underline">${sanitizedLabel}</a>`;
91
+ }
92
+ if (value.name || value.label || value.title) {
93
+ return purify.sanitize(String(value.name || value.label || value.title));
94
+ }
95
+ try {
96
+ return JSON.stringify(value);
97
+ } catch {
98
+ return "-";
99
+ }
100
+ }
85
101
  let strValue = String(value);
86
102
  strValue = strValue.replace(/\s*[–-]\s*undefined\s*$/gi, "");
87
103
  strValue = strValue.replace(/^undefined\s*[–-]\s*/gi, "");