@frontmcp/ui 0.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +201 -0
- package/README.md +438 -0
- package/package.json +147 -0
- package/src/adapters/index.d.ts +10 -0
- package/src/adapters/index.js +18 -0
- package/src/adapters/index.js.map +1 -0
- package/src/adapters/platform-meta.d.ts +165 -0
- package/src/adapters/platform-meta.js +310 -0
- package/src/adapters/platform-meta.js.map +1 -0
- package/src/base-template/bridge.d.ts +89 -0
- package/src/base-template/bridge.js +452 -0
- package/src/base-template/bridge.js.map +1 -0
- package/src/base-template/default-base-template.d.ts +91 -0
- package/src/base-template/default-base-template.js +435 -0
- package/src/base-template/default-base-template.js.map +1 -0
- package/src/base-template/index.d.ts +14 -0
- package/src/base-template/index.js +30 -0
- package/src/base-template/index.js.map +1 -0
- package/src/base-template/polyfills.d.ts +30 -0
- package/src/base-template/polyfills.js +190 -0
- package/src/base-template/polyfills.js.map +1 -0
- package/src/base-template/theme-styles.d.ts +73 -0
- package/src/base-template/theme-styles.js +95 -0
- package/src/base-template/theme-styles.js.map +1 -0
- package/src/bridge/adapters/base-adapter.d.ts +103 -0
- package/src/bridge/adapters/base-adapter.js +314 -0
- package/src/bridge/adapters/base-adapter.js.map +1 -0
- package/src/bridge/adapters/claude.adapter.d.ts +66 -0
- package/src/bridge/adapters/claude.adapter.js +145 -0
- package/src/bridge/adapters/claude.adapter.js.map +1 -0
- package/src/bridge/adapters/ext-apps.adapter.d.ts +142 -0
- package/src/bridge/adapters/ext-apps.adapter.js +416 -0
- package/src/bridge/adapters/ext-apps.adapter.js.map +1 -0
- package/src/bridge/adapters/gemini.adapter.d.ts +63 -0
- package/src/bridge/adapters/gemini.adapter.js +160 -0
- package/src/bridge/adapters/gemini.adapter.js.map +1 -0
- package/src/bridge/adapters/generic.adapter.d.ts +55 -0
- package/src/bridge/adapters/generic.adapter.js +108 -0
- package/src/bridge/adapters/generic.adapter.js.map +1 -0
- package/src/bridge/adapters/index.d.ts +25 -0
- package/src/bridge/adapters/index.js +65 -0
- package/src/bridge/adapters/index.js.map +1 -0
- package/src/bridge/adapters/openai.adapter.d.ts +64 -0
- package/src/bridge/adapters/openai.adapter.js +194 -0
- package/src/bridge/adapters/openai.adapter.js.map +1 -0
- package/src/bridge/core/adapter-registry.d.ts +121 -0
- package/src/bridge/core/adapter-registry.js +271 -0
- package/src/bridge/core/adapter-registry.js.map +1 -0
- package/src/bridge/core/bridge-factory.d.ts +198 -0
- package/src/bridge/core/bridge-factory.js +428 -0
- package/src/bridge/core/bridge-factory.js.map +1 -0
- package/src/bridge/core/index.d.ts +9 -0
- package/src/bridge/core/index.js +22 -0
- package/src/bridge/core/index.js.map +1 -0
- package/src/bridge/index.d.ts +61 -0
- package/src/bridge/index.js +94 -0
- package/src/bridge/index.js.map +1 -0
- package/src/bridge/runtime/iife-generator.d.ts +61 -0
- package/src/bridge/runtime/iife-generator.js +940 -0
- package/src/bridge/runtime/iife-generator.js.map +1 -0
- package/src/bridge/runtime/index.d.ts +8 -0
- package/src/bridge/runtime/index.js +16 -0
- package/src/bridge/runtime/index.js.map +1 -0
- package/src/bridge/types.d.ts +385 -0
- package/src/bridge/types.js +11 -0
- package/src/bridge/types.js.map +1 -0
- package/src/build/cdn-resources.d.ts +140 -0
- package/src/build/cdn-resources.js +314 -0
- package/src/build/cdn-resources.js.map +1 -0
- package/src/build/index.d.ts +294 -0
- package/src/build/index.js +325 -0
- package/src/build/index.js.map +1 -0
- package/src/build/widget-manifest.d.ts +212 -0
- package/src/build/widget-manifest.js +652 -0
- package/src/build/widget-manifest.js.map +1 -0
- package/src/bundler/bundler.d.ts +110 -0
- package/src/bundler/bundler.js +432 -0
- package/src/bundler/bundler.js.map +1 -0
- package/src/bundler/cache.d.ts +172 -0
- package/src/bundler/cache.js +250 -0
- package/src/bundler/cache.js.map +1 -0
- package/src/bundler/index.d.ts +41 -0
- package/src/bundler/index.js +73 -0
- package/src/bundler/index.js.map +1 -0
- package/src/bundler/sandbox/enclave-adapter.d.ts +120 -0
- package/src/bundler/sandbox/enclave-adapter.js +339 -0
- package/src/bundler/sandbox/enclave-adapter.js.map +1 -0
- package/src/bundler/sandbox/executor.d.ts +13 -0
- package/src/bundler/sandbox/executor.js +22 -0
- package/src/bundler/sandbox/executor.js.map +1 -0
- package/src/bundler/sandbox/policy.d.ts +61 -0
- package/src/bundler/sandbox/policy.js +238 -0
- package/src/bundler/sandbox/policy.js.map +1 -0
- package/src/bundler/types.d.ts +347 -0
- package/src/bundler/types.js +132 -0
- package/src/bundler/types.js.map +1 -0
- package/src/components/alert.d.ts +71 -0
- package/src/components/alert.js +189 -0
- package/src/components/alert.js.map +1 -0
- package/src/components/alert.schema.d.ts +114 -0
- package/src/components/alert.schema.js +105 -0
- package/src/components/alert.schema.js.map +1 -0
- package/src/components/avatar.d.ts +76 -0
- package/src/components/avatar.js +176 -0
- package/src/components/avatar.js.map +1 -0
- package/src/components/avatar.schema.d.ts +169 -0
- package/src/components/avatar.schema.js +103 -0
- package/src/components/avatar.schema.js.map +1 -0
- package/src/components/badge.d.ts +70 -0
- package/src/components/badge.js +149 -0
- package/src/components/badge.js.map +1 -0
- package/src/components/badge.schema.d.ts +109 -0
- package/src/components/badge.schema.js +96 -0
- package/src/components/badge.schema.js.map +1 -0
- package/src/components/button.d.ts +111 -0
- package/src/components/button.js +336 -0
- package/src/components/button.js.map +1 -0
- package/src/components/button.schema.d.ts +148 -0
- package/src/components/button.schema.js +121 -0
- package/src/components/button.schema.js.map +1 -0
- package/src/components/card.d.ts +60 -0
- package/src/components/card.js +117 -0
- package/src/components/card.js.map +1 -0
- package/src/components/card.schema.d.ts +113 -0
- package/src/components/card.schema.js +98 -0
- package/src/components/card.schema.js.map +1 -0
- package/src/components/form.d.ts +239 -0
- package/src/components/form.js +420 -0
- package/src/components/form.js.map +1 -0
- package/src/components/form.schema.d.ts +441 -0
- package/src/components/form.schema.js +406 -0
- package/src/components/form.schema.js.map +1 -0
- package/src/components/index.d.ts +29 -0
- package/src/components/index.js +98 -0
- package/src/components/index.js.map +1 -0
- package/src/components/list.d.ts +127 -0
- package/src/components/list.js +279 -0
- package/src/components/list.js.map +1 -0
- package/src/components/list.schema.d.ts +134 -0
- package/src/components/list.schema.js +168 -0
- package/src/components/list.schema.js.map +1 -0
- package/src/components/modal.d.ts +111 -0
- package/src/components/modal.js +260 -0
- package/src/components/modal.js.map +1 -0
- package/src/components/modal.schema.d.ts +186 -0
- package/src/components/modal.schema.js +167 -0
- package/src/components/modal.schema.js.map +1 -0
- package/src/components/table.d.ts +105 -0
- package/src/components/table.js +283 -0
- package/src/components/table.js.map +1 -0
- package/src/components/table.schema.d.ts +159 -0
- package/src/components/table.schema.js +173 -0
- package/src/components/table.schema.js.map +1 -0
- package/src/handlebars/helpers.d.ts +348 -0
- package/src/handlebars/helpers.js +605 -0
- package/src/handlebars/helpers.js.map +1 -0
- package/src/handlebars/index.d.ts +193 -0
- package/src/handlebars/index.js +350 -0
- package/src/handlebars/index.js.map +1 -0
- package/src/index.d.ts +50 -0
- package/src/index.js +192 -0
- package/src/index.js.map +1 -0
- package/src/layouts/base.d.ts +88 -0
- package/src/layouts/base.js +227 -0
- package/src/layouts/base.js.map +1 -0
- package/src/layouts/index.d.ts +7 -0
- package/src/layouts/index.js +25 -0
- package/src/layouts/index.js.map +1 -0
- package/src/layouts/presets.d.ts +133 -0
- package/src/layouts/presets.js +277 -0
- package/src/layouts/presets.js.map +1 -0
- package/src/pages/consent.d.ts +116 -0
- package/src/pages/consent.js +218 -0
- package/src/pages/consent.js.map +1 -0
- package/src/pages/error.d.ts +100 -0
- package/src/pages/error.js +263 -0
- package/src/pages/error.js.map +1 -0
- package/src/pages/index.d.ts +8 -0
- package/src/pages/index.js +27 -0
- package/src/pages/index.js.map +1 -0
- package/src/react/Alert.d.ts +101 -0
- package/src/react/Alert.js +51 -0
- package/src/react/Alert.js.map +1 -0
- package/src/react/Badge.d.ts +100 -0
- package/src/react/Badge.js +55 -0
- package/src/react/Badge.js.map +1 -0
- package/src/react/Button.d.ts +108 -0
- package/src/react/Button.js +52 -0
- package/src/react/Button.js.map +1 -0
- package/src/react/Card.d.ts +103 -0
- package/src/react/Card.js +55 -0
- package/src/react/Card.js.map +1 -0
- package/src/react/hooks/context.d.ts +178 -0
- package/src/react/hooks/context.js +287 -0
- package/src/react/hooks/context.js.map +1 -0
- package/src/react/hooks/index.d.ts +41 -0
- package/src/react/hooks/index.js +61 -0
- package/src/react/hooks/index.js.map +1 -0
- package/src/react/hooks/tools.d.ts +283 -0
- package/src/react/hooks/tools.js +465 -0
- package/src/react/hooks/tools.js.map +1 -0
- package/src/react/index.d.ts +80 -0
- package/src/react/index.js +113 -0
- package/src/react/index.js.map +1 -0
- package/src/react/types.d.ts +105 -0
- package/src/react/types.js +12 -0
- package/src/react/types.js.map +1 -0
- package/src/react/utils.d.ts +42 -0
- package/src/react/utils.js +99 -0
- package/src/react/utils.js.map +1 -0
- package/src/registry/index.d.ts +45 -0
- package/src/registry/index.js +67 -0
- package/src/registry/index.js.map +1 -0
- package/src/registry/render-template.d.ts +86 -0
- package/src/registry/render-template.js +239 -0
- package/src/registry/render-template.js.map +1 -0
- package/src/registry/tool-ui.registry.d.ts +260 -0
- package/src/registry/tool-ui.registry.js +438 -0
- package/src/registry/tool-ui.registry.js.map +1 -0
- package/src/registry/uri-utils.d.ts +55 -0
- package/src/registry/uri-utils.js +97 -0
- package/src/registry/uri-utils.js.map +1 -0
- package/src/render/index.d.ts +7 -0
- package/src/render/index.js +14 -0
- package/src/render/index.js.map +1 -0
- package/src/render/prerender.d.ts +56 -0
- package/src/render/prerender.js +98 -0
- package/src/render/prerender.js.map +1 -0
- package/src/renderers/cache.d.ts +144 -0
- package/src/renderers/cache.js +240 -0
- package/src/renderers/cache.js.map +1 -0
- package/src/renderers/html.renderer.d.ts +122 -0
- package/src/renderers/html.renderer.js +204 -0
- package/src/renderers/html.renderer.js.map +1 -0
- package/src/renderers/index.d.ts +35 -0
- package/src/renderers/index.js +70 -0
- package/src/renderers/index.js.map +1 -0
- package/src/renderers/mdx.renderer.d.ts +119 -0
- package/src/renderers/mdx.renderer.js +305 -0
- package/src/renderers/mdx.renderer.js.map +1 -0
- package/src/renderers/react.renderer.d.ts +95 -0
- package/src/renderers/react.renderer.js +260 -0
- package/src/renderers/react.renderer.js.map +1 -0
- package/src/renderers/registry.d.ts +133 -0
- package/src/renderers/registry.js +232 -0
- package/src/renderers/registry.js.map +1 -0
- package/src/renderers/types.d.ts +341 -0
- package/src/renderers/types.js +9 -0
- package/src/renderers/types.js.map +1 -0
- package/src/renderers/utils/detect.d.ts +106 -0
- package/src/renderers/utils/detect.js +267 -0
- package/src/renderers/utils/detect.js.map +1 -0
- package/src/renderers/utils/hash.d.ts +39 -0
- package/src/renderers/utils/hash.js +75 -0
- package/src/renderers/utils/hash.js.map +1 -0
- package/src/renderers/utils/index.d.ts +8 -0
- package/src/renderers/utils/index.js +28 -0
- package/src/renderers/utils/index.js.map +1 -0
- package/src/renderers/utils/transpiler.d.ts +88 -0
- package/src/renderers/utils/transpiler.js +215 -0
- package/src/renderers/utils/transpiler.js.map +1 -0
- package/src/runtime/adapters/html.adapter.d.ts +58 -0
- package/src/runtime/adapters/html.adapter.js +131 -0
- package/src/runtime/adapters/html.adapter.js.map +1 -0
- package/src/runtime/adapters/index.d.ts +25 -0
- package/src/runtime/adapters/index.js +54 -0
- package/src/runtime/adapters/index.js.map +1 -0
- package/src/runtime/adapters/mdx.adapter.d.ts +72 -0
- package/src/runtime/adapters/mdx.adapter.js +241 -0
- package/src/runtime/adapters/mdx.adapter.js.map +1 -0
- package/src/runtime/adapters/react.adapter.d.ts +69 -0
- package/src/runtime/adapters/react.adapter.js +245 -0
- package/src/runtime/adapters/react.adapter.js.map +1 -0
- package/src/runtime/adapters/types.d.ts +94 -0
- package/src/runtime/adapters/types.js +11 -0
- package/src/runtime/adapters/types.js.map +1 -0
- package/src/runtime/csp.d.ts +37 -0
- package/src/runtime/csp.js +140 -0
- package/src/runtime/csp.js.map +1 -0
- package/src/runtime/index.d.ts +16 -0
- package/src/runtime/index.js +72 -0
- package/src/runtime/index.js.map +1 -0
- package/src/runtime/mcp-bridge.d.ts +100 -0
- package/src/runtime/mcp-bridge.js +581 -0
- package/src/runtime/mcp-bridge.js.map +1 -0
- package/src/runtime/renderer-runtime.d.ts +132 -0
- package/src/runtime/renderer-runtime.js +389 -0
- package/src/runtime/renderer-runtime.js.map +1 -0
- package/src/runtime/sanitizer.d.ts +171 -0
- package/src/runtime/sanitizer.js +318 -0
- package/src/runtime/sanitizer.js.map +1 -0
- package/src/runtime/types.d.ts +414 -0
- package/src/runtime/types.js +12 -0
- package/src/runtime/types.js.map +1 -0
- package/src/runtime/wrapper.d.ts +375 -0
- package/src/runtime/wrapper.js +1793 -0
- package/src/runtime/wrapper.js.map +1 -0
- package/src/styles/index.d.ts +7 -0
- package/src/styles/index.js +11 -0
- package/src/styles/index.js.map +1 -0
- package/src/styles/variants.d.ts +50 -0
- package/src/styles/variants.js +175 -0
- package/src/styles/variants.js.map +1 -0
- package/src/theme/cdn.d.ts +194 -0
- package/src/theme/cdn.js +375 -0
- package/src/theme/cdn.js.map +1 -0
- package/src/theme/index.d.ts +17 -0
- package/src/theme/index.js +57 -0
- package/src/theme/index.js.map +1 -0
- package/src/theme/platforms.d.ts +106 -0
- package/src/theme/platforms.js +161 -0
- package/src/theme/platforms.js.map +1 -0
- package/src/theme/presets/github-openai.d.ts +49 -0
- package/src/theme/presets/github-openai.js +189 -0
- package/src/theme/presets/github-openai.js.map +1 -0
- package/src/theme/presets/index.d.ts +10 -0
- package/src/theme/presets/index.js +17 -0
- package/src/theme/presets/index.js.map +1 -0
- package/src/theme/theme.d.ts +395 -0
- package/src/theme/theme.js +332 -0
- package/src/theme/theme.js.map +1 -0
- package/src/tool-template/builder.d.ts +212 -0
- package/src/tool-template/builder.js +397 -0
- package/src/tool-template/builder.js.map +1 -0
- package/src/tool-template/index.d.ts +15 -0
- package/src/tool-template/index.js +38 -0
- package/src/tool-template/index.js.map +1 -0
- package/src/types/index.d.ts +13 -0
- package/src/types/index.js +26 -0
- package/src/types/index.js.map +1 -0
- package/src/types/ui-config.d.ts +357 -0
- package/src/types/ui-config.js +12 -0
- package/src/types/ui-config.js.map +1 -0
- package/src/types/ui-runtime.d.ts +965 -0
- package/src/types/ui-runtime.js +117 -0
- package/src/types/ui-runtime.js.map +1 -0
- package/src/validation/error-box.d.ts +55 -0
- package/src/validation/error-box.js +75 -0
- package/src/validation/error-box.js.map +1 -0
- package/src/validation/index.d.ts +12 -0
- package/src/validation/index.js +21 -0
- package/src/validation/index.js.map +1 -0
- package/src/validation/wrapper.d.ts +96 -0
- package/src/validation/wrapper.js +117 -0
- package/src/validation/wrapper.js.map +1 -0
- package/src/web-components/core/attribute-parser.d.ts +85 -0
- package/src/web-components/core/attribute-parser.js +189 -0
- package/src/web-components/core/attribute-parser.js.map +1 -0
- package/src/web-components/core/base-element.d.ts +197 -0
- package/src/web-components/core/base-element.js +289 -0
- package/src/web-components/core/base-element.js.map +1 -0
- package/src/web-components/core/index.d.ts +8 -0
- package/src/web-components/core/index.js +18 -0
- package/src/web-components/core/index.js.map +1 -0
- package/src/web-components/elements/fmcp-alert.d.ts +45 -0
- package/src/web-components/elements/fmcp-alert.js +93 -0
- package/src/web-components/elements/fmcp-alert.js.map +1 -0
- package/src/web-components/elements/fmcp-badge.d.ts +46 -0
- package/src/web-components/elements/fmcp-badge.js +99 -0
- package/src/web-components/elements/fmcp-badge.js.map +1 -0
- package/src/web-components/elements/fmcp-button.d.ts +124 -0
- package/src/web-components/elements/fmcp-button.js +233 -0
- package/src/web-components/elements/fmcp-button.js.map +1 -0
- package/src/web-components/elements/fmcp-card.d.ts +52 -0
- package/src/web-components/elements/fmcp-card.js +115 -0
- package/src/web-components/elements/fmcp-card.js.map +1 -0
- package/src/web-components/elements/fmcp-input.d.ts +95 -0
- package/src/web-components/elements/fmcp-input.js +248 -0
- package/src/web-components/elements/fmcp-input.js.map +1 -0
- package/src/web-components/elements/fmcp-select.d.ts +99 -0
- package/src/web-components/elements/fmcp-select.js +243 -0
- package/src/web-components/elements/fmcp-select.js.map +1 -0
- package/src/web-components/elements/index.d.ts +12 -0
- package/src/web-components/elements/index.js +34 -0
- package/src/web-components/elements/index.js.map +1 -0
- package/src/web-components/index.d.ts +49 -0
- package/src/web-components/index.js +75 -0
- package/src/web-components/index.js.map +1 -0
- package/src/web-components/register.d.ts +56 -0
- package/src/web-components/register.js +80 -0
- package/src/web-components/register.js.map +1 -0
- package/src/web-components/types.d.ts +121 -0
- package/src/web-components/types.js +25 -0
- package/src/web-components/types.js.map +1 -0
- package/src/widgets/index.d.ts +7 -0
- package/src/widgets/index.js +24 -0
- package/src/widgets/index.js.map +1 -0
- package/src/widgets/progress.d.ts +132 -0
- package/src/widgets/progress.js +303 -0
- package/src/widgets/progress.js.map +1 -0
- package/src/widgets/resource.d.ts +162 -0
- package/src/widgets/resource.js +340 -0
- package/src/widgets/resource.js.map +1 -0
|
@@ -0,0 +1,340 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* OpenAI App SDK Resource Widgets
|
|
4
|
+
*
|
|
5
|
+
* Components for displaying resources in OpenAI's App SDK format.
|
|
6
|
+
* These widgets are designed to work with OpenAI Canvas and similar interfaces.
|
|
7
|
+
*/
|
|
8
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
9
|
+
exports.resourceWidget = resourceWidget;
|
|
10
|
+
exports.resourceList = resourceList;
|
|
11
|
+
exports.resourceItem = resourceItem;
|
|
12
|
+
exports.codePreview = codePreview;
|
|
13
|
+
exports.imagePreview = imagePreview;
|
|
14
|
+
const base_1 = require("../layouts/base");
|
|
15
|
+
const card_1 = require("../components/card");
|
|
16
|
+
const badge_1 = require("../components/badge");
|
|
17
|
+
const button_1 = require("../components/button");
|
|
18
|
+
// ============================================
|
|
19
|
+
// Resource Icons
|
|
20
|
+
// ============================================
|
|
21
|
+
const resourceIcons = {
|
|
22
|
+
document: `<svg class="w-8 h-8" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
23
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
|
|
24
|
+
</svg>`,
|
|
25
|
+
image: `<svg class="w-8 h-8" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
26
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 16l4.586-4.586a2 2 0 012.828 0L16 16m-2-2l1.586-1.586a2 2 0 012.828 0L20 14m-6-6h.01M6 20h12a2 2 0 002-2V6a2 2 0 00-2-2H6a2 2 0 00-2 2v12a2 2 0 002 2z"/>
|
|
27
|
+
</svg>`,
|
|
28
|
+
code: `<svg class="w-8 h-8" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
29
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10 20l4-16m4 4l4 4-4 4M6 16l-4-4 4-4"/>
|
|
30
|
+
</svg>`,
|
|
31
|
+
data: `<svg class="w-8 h-8" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
32
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 7v10c0 2.21 3.582 4 8 4s8-1.79 8-4V7M4 7c0 2.21 3.582 4 8 4s8-1.79 8-4M4 7c0-2.21 3.582-4 8-4s8 1.79 8 4"/>
|
|
33
|
+
</svg>`,
|
|
34
|
+
file: `<svg class="w-8 h-8" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
35
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M7 21h10a2 2 0 002-2V9.414a1 1 0 00-.293-.707l-5.414-5.414A1 1 0 0012.586 3H7a2 2 0 00-2 2v14a2 2 0 002 2z"/>
|
|
36
|
+
</svg>`,
|
|
37
|
+
link: `<svg class="w-8 h-8" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
38
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M13.828 10.172a4 4 0 00-5.656 0l-4 4a4 4 0 105.656 5.656l1.102-1.101m-.758-4.899a4 4 0 005.656 0l4-4a4 4 0 00-5.656-5.656l-1.1 1.1"/>
|
|
39
|
+
</svg>`,
|
|
40
|
+
user: `<svg class="w-8 h-8" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
41
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M16 7a4 4 0 11-8 0 4 4 0 018 0zM12 14a7 7 0 00-7 7h14a7 7 0 00-7-7z"/>
|
|
42
|
+
</svg>`,
|
|
43
|
+
event: `<svg class="w-8 h-8" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
44
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z"/>
|
|
45
|
+
</svg>`,
|
|
46
|
+
message: `<svg class="w-8 h-8" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
47
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 10h.01M12 10h.01M16 10h.01M9 16H5a2 2 0 01-2-2V6a2 2 0 012-2h14a2 2 0 012 2v8a2 2 0 01-2 2h-5l-5 5v-5z"/>
|
|
48
|
+
</svg>`,
|
|
49
|
+
task: `<svg class="w-8 h-8" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
50
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"/>
|
|
51
|
+
</svg>`,
|
|
52
|
+
custom: `<svg class="w-8 h-8" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
53
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M19 11H5m14 0a2 2 0 012 2v6a2 2 0 01-2 2H5a2 2 0 01-2-2v-6a2 2 0 012-2m14 0V9a2 2 0 00-2-2M5 11V9a2 2 0 012-2m0 0V5a2 2 0 012-2h6a2 2 0 012 2v2M7 7h10"/>
|
|
54
|
+
</svg>`,
|
|
55
|
+
};
|
|
56
|
+
// ============================================
|
|
57
|
+
// Utility Functions
|
|
58
|
+
// ============================================
|
|
59
|
+
/**
|
|
60
|
+
* Format file size
|
|
61
|
+
*/
|
|
62
|
+
function formatFileSize(bytes) {
|
|
63
|
+
if (bytes === 0)
|
|
64
|
+
return '0 B';
|
|
65
|
+
const k = 1024;
|
|
66
|
+
const sizes = ['B', 'KB', 'MB', 'GB', 'TB'];
|
|
67
|
+
const i = Math.floor(Math.log(bytes) / Math.log(k));
|
|
68
|
+
return `${parseFloat((bytes / Math.pow(k, i)).toFixed(1))} ${sizes[i]}`;
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Format date
|
|
72
|
+
*/
|
|
73
|
+
function formatDate(date) {
|
|
74
|
+
const d = typeof date === 'string' ? new Date(date) : date;
|
|
75
|
+
return d.toLocaleDateString('en-US', {
|
|
76
|
+
year: 'numeric',
|
|
77
|
+
month: 'short',
|
|
78
|
+
day: 'numeric',
|
|
79
|
+
});
|
|
80
|
+
}
|
|
81
|
+
// ============================================
|
|
82
|
+
// Resource Widget Builder
|
|
83
|
+
// ============================================
|
|
84
|
+
/**
|
|
85
|
+
* Build a resource widget
|
|
86
|
+
*/
|
|
87
|
+
function resourceWidget(options) {
|
|
88
|
+
const { type, title, description, icon, thumbnail, url, meta, status, actions = [], className = '', cardOptions = {}, } = options;
|
|
89
|
+
// Icon/Thumbnail
|
|
90
|
+
const iconHtml = thumbnail
|
|
91
|
+
? `<div class="w-16 h-16 rounded-lg overflow-hidden bg-gray-100 flex-shrink-0">
|
|
92
|
+
<img src="${(0, base_1.escapeHtml)(thumbnail)}" alt="${(0, base_1.escapeHtml)(title)}" class="w-full h-full object-cover">
|
|
93
|
+
</div>`
|
|
94
|
+
: `<div class="w-16 h-16 rounded-lg bg-gray-100 flex items-center justify-center flex-shrink-0 text-gray-400">
|
|
95
|
+
${icon || resourceIcons[type]}
|
|
96
|
+
</div>`;
|
|
97
|
+
// Status badge
|
|
98
|
+
const statusHtml = status ? (0, badge_1.badge)(status.label, { variant: status.variant, size: 'sm' }) : '';
|
|
99
|
+
// Metadata
|
|
100
|
+
const metaItems = [];
|
|
101
|
+
if (meta?.size) {
|
|
102
|
+
metaItems.push(formatFileSize(meta.size));
|
|
103
|
+
}
|
|
104
|
+
if (meta?.mimeType) {
|
|
105
|
+
metaItems.push(meta.mimeType);
|
|
106
|
+
}
|
|
107
|
+
if (meta?.updatedAt) {
|
|
108
|
+
metaItems.push(`Updated ${formatDate(meta.updatedAt)}`);
|
|
109
|
+
}
|
|
110
|
+
else if (meta?.createdAt) {
|
|
111
|
+
metaItems.push(`Created ${formatDate(meta.createdAt)}`);
|
|
112
|
+
}
|
|
113
|
+
if (meta?.author) {
|
|
114
|
+
metaItems.push(`by ${meta.author}`);
|
|
115
|
+
}
|
|
116
|
+
const metaHtml = metaItems.length > 0 ? `<div class="text-xs text-text-secondary mt-1">${metaItems.join(' • ')}</div>` : '';
|
|
117
|
+
// Tags
|
|
118
|
+
const tagsHtml = meta?.tags && meta.tags.length > 0
|
|
119
|
+
? `<div class="flex flex-wrap gap-1 mt-2">
|
|
120
|
+
${meta.tags.map((tag) => (0, badge_1.badge)(tag, { variant: 'default', size: 'sm' })).join('')}
|
|
121
|
+
</div>`
|
|
122
|
+
: '';
|
|
123
|
+
// Actions
|
|
124
|
+
const actionsHtml = actions.length > 0
|
|
125
|
+
? `<div class="flex gap-2 mt-4">
|
|
126
|
+
${actions
|
|
127
|
+
.map((action) => {
|
|
128
|
+
const variantMap = {
|
|
129
|
+
primary: 'primary',
|
|
130
|
+
secondary: 'secondary',
|
|
131
|
+
danger: 'danger',
|
|
132
|
+
ghost: 'ghost',
|
|
133
|
+
};
|
|
134
|
+
const variant = action.variant ? variantMap[action.variant] : 'ghost';
|
|
135
|
+
const htmxAttrs = [];
|
|
136
|
+
if (action.htmx) {
|
|
137
|
+
if (action.htmx.get)
|
|
138
|
+
htmxAttrs.push(`hx-get="${(0, base_1.escapeHtml)(action.htmx.get)}"`);
|
|
139
|
+
if (action.htmx.post)
|
|
140
|
+
htmxAttrs.push(`hx-post="${(0, base_1.escapeHtml)(action.htmx.post)}"`);
|
|
141
|
+
if (action.htmx.delete)
|
|
142
|
+
htmxAttrs.push(`hx-delete="${(0, base_1.escapeHtml)(action.htmx.delete)}"`);
|
|
143
|
+
if (action.htmx.target)
|
|
144
|
+
htmxAttrs.push(`hx-target="${(0, base_1.escapeHtml)(action.htmx.target)}"`);
|
|
145
|
+
if (action.htmx.swap)
|
|
146
|
+
htmxAttrs.push(`hx-swap="${(0, base_1.escapeHtml)(action.htmx.swap)}"`);
|
|
147
|
+
if (action.htmx.confirm)
|
|
148
|
+
htmxAttrs.push(`hx-confirm="${(0, base_1.escapeHtml)(action.htmx.confirm)}"`);
|
|
149
|
+
}
|
|
150
|
+
return (0, button_1.button)(action.label, {
|
|
151
|
+
variant: variant,
|
|
152
|
+
size: 'sm',
|
|
153
|
+
href: action.href,
|
|
154
|
+
disabled: action.disabled,
|
|
155
|
+
iconBefore: action.icon,
|
|
156
|
+
});
|
|
157
|
+
})
|
|
158
|
+
.join('')}
|
|
159
|
+
</div>`
|
|
160
|
+
: '';
|
|
161
|
+
// Content
|
|
162
|
+
const content = `
|
|
163
|
+
<div class="flex gap-4">
|
|
164
|
+
${iconHtml}
|
|
165
|
+
<div class="flex-1 min-w-0">
|
|
166
|
+
<div class="flex items-start justify-between gap-2">
|
|
167
|
+
<div class="min-w-0">
|
|
168
|
+
${url
|
|
169
|
+
? `<a href="${(0, base_1.escapeHtml)(url)}" class="font-semibold text-text-primary hover:text-primary truncate block">${(0, base_1.escapeHtml)(title)}</a>`
|
|
170
|
+
: `<h3 class="font-semibold text-text-primary truncate">${(0, base_1.escapeHtml)(title)}</h3>`}
|
|
171
|
+
${description
|
|
172
|
+
? `<p class="text-sm text-text-secondary mt-0.5 line-clamp-2">${(0, base_1.escapeHtml)(description)}</p>`
|
|
173
|
+
: ''}
|
|
174
|
+
${metaHtml}
|
|
175
|
+
${tagsHtml}
|
|
176
|
+
</div>
|
|
177
|
+
${statusHtml}
|
|
178
|
+
</div>
|
|
179
|
+
${actionsHtml}
|
|
180
|
+
</div>
|
|
181
|
+
</div>
|
|
182
|
+
`;
|
|
183
|
+
return (0, card_1.card)(content, {
|
|
184
|
+
variant: 'default',
|
|
185
|
+
size: 'md',
|
|
186
|
+
className: `resource-widget resource-${type} ${className}`,
|
|
187
|
+
...cardOptions,
|
|
188
|
+
});
|
|
189
|
+
}
|
|
190
|
+
/**
|
|
191
|
+
* Build a resource list widget
|
|
192
|
+
*/
|
|
193
|
+
function resourceList(options) {
|
|
194
|
+
const { resources, title, emptyMessage = 'No resources found', layout = 'list', columns = 2, className = '', showLoadMore = false, loadMoreUrl, } = options;
|
|
195
|
+
// Title
|
|
196
|
+
const titleHtml = title ? `<h2 class="text-lg font-semibold text-text-primary mb-4">${(0, base_1.escapeHtml)(title)}</h2>` : '';
|
|
197
|
+
// Empty state
|
|
198
|
+
if (resources.length === 0) {
|
|
199
|
+
return `<div class="${className}">
|
|
200
|
+
${titleHtml}
|
|
201
|
+
<div class="text-center py-12 text-text-secondary">
|
|
202
|
+
<svg class="w-12 h-12 mx-auto mb-4 text-gray-300" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
203
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M20 13V6a2 2 0 00-2-2H6a2 2 0 00-2 2v7m16 0v5a2 2 0 01-2 2H6a2 2 0 01-2-2v-5m16 0h-2.586a1 1 0 00-.707.293l-2.414 2.414a1 1 0 01-.707.293h-3.172a1 1 0 01-.707-.293l-2.414-2.414A1 1 0 006.586 13H4"/>
|
|
204
|
+
</svg>
|
|
205
|
+
<p>${(0, base_1.escapeHtml)(emptyMessage)}</p>
|
|
206
|
+
</div>
|
|
207
|
+
</div>`;
|
|
208
|
+
}
|
|
209
|
+
// Layout classes
|
|
210
|
+
const layoutClasses = layout === 'grid' ? `grid grid-cols-1 md:grid-cols-${columns} gap-4` : 'space-y-4';
|
|
211
|
+
// Resources
|
|
212
|
+
const resourcesHtml = resources.map((r) => resourceWidget(r)).join('\n');
|
|
213
|
+
// Load more button
|
|
214
|
+
const loadMoreHtml = showLoadMore && loadMoreUrl
|
|
215
|
+
? `<div class="text-center mt-6">
|
|
216
|
+
${(0, button_1.button)('Load More', {
|
|
217
|
+
variant: 'outline',
|
|
218
|
+
htmx: {
|
|
219
|
+
get: loadMoreUrl,
|
|
220
|
+
target: 'closest .resource-list',
|
|
221
|
+
swap: 'beforeend',
|
|
222
|
+
},
|
|
223
|
+
})}
|
|
224
|
+
</div>`
|
|
225
|
+
: '';
|
|
226
|
+
return `<div class="resource-list ${className}">
|
|
227
|
+
${titleHtml}
|
|
228
|
+
<div class="${layoutClasses}">
|
|
229
|
+
${resourcesHtml}
|
|
230
|
+
</div>
|
|
231
|
+
${loadMoreHtml}
|
|
232
|
+
</div>`;
|
|
233
|
+
}
|
|
234
|
+
// ============================================
|
|
235
|
+
// Compact Resource Item
|
|
236
|
+
// ============================================
|
|
237
|
+
/**
|
|
238
|
+
* Build a compact resource item (for inline display)
|
|
239
|
+
*/
|
|
240
|
+
function resourceItem(options) {
|
|
241
|
+
const { type, title, description, icon, url, meta, status } = options;
|
|
242
|
+
const iconHtml = `<div class="w-10 h-10 rounded-lg bg-gray-100 flex items-center justify-center flex-shrink-0 text-gray-400">
|
|
243
|
+
${icon || resourceIcons[type]}
|
|
244
|
+
</div>`;
|
|
245
|
+
const statusHtml = status ? (0, badge_1.badge)(status.label, { variant: status.variant, size: 'sm' }) : '';
|
|
246
|
+
const metaText = meta?.size ? formatFileSize(meta.size) : '';
|
|
247
|
+
const titleElement = url
|
|
248
|
+
? `<a href="${(0, base_1.escapeHtml)(url)}" class="font-medium text-text-primary hover:text-primary">${(0, base_1.escapeHtml)(title)}</a>`
|
|
249
|
+
: `<span class="font-medium text-text-primary">${(0, base_1.escapeHtml)(title)}</span>`;
|
|
250
|
+
return `<div class="flex items-center gap-3 p-3 rounded-lg hover:bg-gray-50 transition-colors">
|
|
251
|
+
${iconHtml}
|
|
252
|
+
<div class="flex-1 min-w-0">
|
|
253
|
+
<div class="flex items-center gap-2">
|
|
254
|
+
${titleElement}
|
|
255
|
+
${statusHtml}
|
|
256
|
+
</div>
|
|
257
|
+
${description || metaText
|
|
258
|
+
? `<p class="text-sm text-text-secondary truncate">${(0, base_1.escapeHtml)(description || metaText)}</p>`
|
|
259
|
+
: ''}
|
|
260
|
+
</div>
|
|
261
|
+
</div>`;
|
|
262
|
+
}
|
|
263
|
+
/**
|
|
264
|
+
* Build a code preview widget
|
|
265
|
+
*/
|
|
266
|
+
function codePreview(options) {
|
|
267
|
+
const { code, language = 'text', filename, lineNumbers = true, maxHeight = '400px', showCopy = true, className = '', } = options;
|
|
268
|
+
const lines = code.split('\n');
|
|
269
|
+
const lineNumbersHtml = lineNumbers
|
|
270
|
+
? `<div class="text-right select-none pr-4 text-gray-500">
|
|
271
|
+
${lines.map((_, i) => `<div>${i + 1}</div>`).join('')}
|
|
272
|
+
</div>`
|
|
273
|
+
: '';
|
|
274
|
+
const copyScript = showCopy
|
|
275
|
+
? `<script>
|
|
276
|
+
function copyCode(btn, code) {
|
|
277
|
+
navigator.clipboard.writeText(code).then(() => {
|
|
278
|
+
const original = btn.innerHTML;
|
|
279
|
+
btn.innerHTML = '<svg class="w-4 h-4 text-success" fill="none" stroke="currentColor" viewBox="0 0 24 24"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M5 13l4 4L19 7"/></svg>';
|
|
280
|
+
setTimeout(() => btn.innerHTML = original, 2000);
|
|
281
|
+
});
|
|
282
|
+
}
|
|
283
|
+
</script>`
|
|
284
|
+
: '';
|
|
285
|
+
const copyButton = showCopy
|
|
286
|
+
? `<button
|
|
287
|
+
type="button"
|
|
288
|
+
onclick="copyCode(this, ${(0, base_1.escapeHtml)(JSON.stringify(code))})"
|
|
289
|
+
class="p-1.5 rounded hover:bg-gray-700 transition-colors"
|
|
290
|
+
title="Copy code"
|
|
291
|
+
>
|
|
292
|
+
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
293
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 16H6a2 2 0 01-2-2V6a2 2 0 012-2h8a2 2 0 012 2v2m-6 12h8a2 2 0 002-2v-8a2 2 0 00-2-2h-8a2 2 0 00-2 2v8a2 2 0 002 2z"/>
|
|
294
|
+
</svg>
|
|
295
|
+
</button>`
|
|
296
|
+
: '';
|
|
297
|
+
return `<div class="code-preview rounded-lg overflow-hidden ${className}">
|
|
298
|
+
${filename || showCopy
|
|
299
|
+
? `
|
|
300
|
+
<div class="flex items-center justify-between px-4 py-2 bg-gray-800 border-b border-gray-700">
|
|
301
|
+
${filename ? `<span class="text-sm text-gray-300">${(0, base_1.escapeHtml)(filename)}</span>` : '<span></span>'}
|
|
302
|
+
<div class="flex items-center gap-2">
|
|
303
|
+
${language ? `<span class="text-xs text-gray-500">${(0, base_1.escapeHtml)(language)}</span>` : ''}
|
|
304
|
+
${copyButton}
|
|
305
|
+
</div>
|
|
306
|
+
</div>
|
|
307
|
+
`
|
|
308
|
+
: ''}
|
|
309
|
+
<div class="bg-gray-900 p-4 overflow-auto" style="max-height: ${maxHeight}">
|
|
310
|
+
<div class="flex text-sm font-mono">
|
|
311
|
+
${lineNumbersHtml}
|
|
312
|
+
<pre class="flex-1 text-gray-100"><code>${(0, base_1.escapeHtml)(code)}</code></pre>
|
|
313
|
+
</div>
|
|
314
|
+
</div>
|
|
315
|
+
${copyScript}
|
|
316
|
+
</div>`;
|
|
317
|
+
}
|
|
318
|
+
/**
|
|
319
|
+
* Build an image preview widget
|
|
320
|
+
*/
|
|
321
|
+
function imagePreview(options) {
|
|
322
|
+
const { src, alt, caption, maxHeight = '400px', clickable = true, className = '' } = options;
|
|
323
|
+
const imageHtml = `<img
|
|
324
|
+
src="${(0, base_1.escapeHtml)(src)}"
|
|
325
|
+
alt="${(0, base_1.escapeHtml)(alt)}"
|
|
326
|
+
class="max-w-full h-auto rounded-lg"
|
|
327
|
+
style="max-height: ${maxHeight}"
|
|
328
|
+
>`;
|
|
329
|
+
const captionHtml = caption
|
|
330
|
+
? `<p class="text-sm text-text-secondary mt-2 text-center">${(0, base_1.escapeHtml)(caption)}</p>`
|
|
331
|
+
: '';
|
|
332
|
+
const content = clickable
|
|
333
|
+
? `<a href="${(0, base_1.escapeHtml)(src)}" target="_blank" rel="noopener" class="block">${imageHtml}</a>`
|
|
334
|
+
: imageHtml;
|
|
335
|
+
return `<div class="image-preview ${className}">
|
|
336
|
+
${content}
|
|
337
|
+
${captionHtml}
|
|
338
|
+
</div>`;
|
|
339
|
+
}
|
|
340
|
+
//# sourceMappingURL=resource.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"resource.js","sourceRoot":"","sources":["../../../src/widgets/resource.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * OpenAI App SDK Resource Widgets\n *\n * Components for displaying resources in OpenAI's App SDK format.\n * These widgets are designed to work with OpenAI Canvas and similar interfaces.\n */\n\nimport { escapeHtml } from '../layouts/base';\nimport { card, type CardOptions } from '../components/card';\nimport { badge, type BadgeVariant } from '../components/badge';\nimport { button } from '../components/button';\n\n// ============================================\n// Resource Types\n// ============================================\n\n/**\n * Resource type identifiers\n */\nexport type ResourceType =\n | 'document'\n | 'image'\n | 'code'\n | 'data'\n | 'file'\n | 'link'\n | 'user'\n | 'event'\n | 'message'\n | 'task'\n | 'custom';\n\n/**\n * Resource metadata\n */\nexport interface ResourceMeta {\n /** Creation timestamp */\n createdAt?: string | Date;\n /** Last modified timestamp */\n updatedAt?: string | Date;\n /** Author/creator */\n author?: string;\n /** File size (bytes) */\n size?: number;\n /** MIME type */\n mimeType?: string;\n /** Tags */\n tags?: string[];\n /** Custom metadata */\n [key: string]: unknown;\n}\n\n/**\n * Resource action\n */\nexport interface ResourceAction {\n /** Action label */\n label: string;\n /** Action icon */\n icon?: string;\n /** Action URL */\n href?: string;\n /** HTMX attributes */\n htmx?: {\n get?: string;\n post?: string;\n delete?: string;\n target?: string;\n swap?: string;\n confirm?: string;\n };\n /** Variant */\n variant?: 'primary' | 'secondary' | 'danger' | 'ghost';\n /** Disabled */\n disabled?: boolean;\n}\n\n/**\n * Base resource options\n */\nexport interface ResourceOptions {\n /** Resource type */\n type: ResourceType;\n /** Resource title */\n title: string;\n /** Resource description */\n description?: string;\n /** Resource icon */\n icon?: string;\n /** Resource thumbnail URL */\n thumbnail?: string;\n /** Resource URL */\n url?: string;\n /** Resource metadata */\n meta?: ResourceMeta;\n /** Resource status */\n status?: {\n label: string;\n variant: BadgeVariant;\n };\n /** Available actions */\n actions?: ResourceAction[];\n /** Additional CSS classes */\n className?: string;\n /** Card options */\n cardOptions?: Partial<CardOptions>;\n}\n\n// ============================================\n// Resource Icons\n// ============================================\n\nconst resourceIcons: Record<ResourceType, string> = {\n document: `<svg class=\"w-8 h-8\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z\"/>\n </svg>`,\n image: `<svg class=\"w-8 h-8\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M4 16l4.586-4.586a2 2 0 012.828 0L16 16m-2-2l1.586-1.586a2 2 0 012.828 0L20 14m-6-6h.01M6 20h12a2 2 0 002-2V6a2 2 0 00-2-2H6a2 2 0 00-2 2v12a2 2 0 002 2z\"/>\n </svg>`,\n code: `<svg class=\"w-8 h-8\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M10 20l4-16m4 4l4 4-4 4M6 16l-4-4 4-4\"/>\n </svg>`,\n data: `<svg class=\"w-8 h-8\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M4 7v10c0 2.21 3.582 4 8 4s8-1.79 8-4V7M4 7c0 2.21 3.582 4 8 4s8-1.79 8-4M4 7c0-2.21 3.582-4 8-4s8 1.79 8 4\"/>\n </svg>`,\n file: `<svg class=\"w-8 h-8\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M7 21h10a2 2 0 002-2V9.414a1 1 0 00-.293-.707l-5.414-5.414A1 1 0 0012.586 3H7a2 2 0 00-2 2v14a2 2 0 002 2z\"/>\n </svg>`,\n link: `<svg class=\"w-8 h-8\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M13.828 10.172a4 4 0 00-5.656 0l-4 4a4 4 0 105.656 5.656l1.102-1.101m-.758-4.899a4 4 0 005.656 0l4-4a4 4 0 00-5.656-5.656l-1.1 1.1\"/>\n </svg>`,\n user: `<svg class=\"w-8 h-8\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M16 7a4 4 0 11-8 0 4 4 0 018 0zM12 14a7 7 0 00-7 7h14a7 7 0 00-7-7z\"/>\n </svg>`,\n event: `<svg class=\"w-8 h-8\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z\"/>\n </svg>`,\n message: `<svg class=\"w-8 h-8\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M8 10h.01M12 10h.01M16 10h.01M9 16H5a2 2 0 01-2-2V6a2 2 0 012-2h14a2 2 0 012 2v8a2 2 0 01-2 2h-5l-5 5v-5z\"/>\n </svg>`,\n task: `<svg class=\"w-8 h-8\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4\"/>\n </svg>`,\n custom: `<svg class=\"w-8 h-8\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M19 11H5m14 0a2 2 0 012 2v6a2 2 0 01-2 2H5a2 2 0 01-2-2v-6a2 2 0 012-2m14 0V9a2 2 0 00-2-2M5 11V9a2 2 0 012-2m0 0V5a2 2 0 012-2h6a2 2 0 012 2v2M7 7h10\"/>\n </svg>`,\n};\n\n// ============================================\n// Utility Functions\n// ============================================\n\n/**\n * Format file size\n */\nfunction formatFileSize(bytes: number): string {\n if (bytes === 0) return '0 B';\n const k = 1024;\n const sizes = ['B', 'KB', 'MB', 'GB', 'TB'];\n const i = Math.floor(Math.log(bytes) / Math.log(k));\n return `${parseFloat((bytes / Math.pow(k, i)).toFixed(1))} ${sizes[i]}`;\n}\n\n/**\n * Format date\n */\nfunction formatDate(date: string | Date): string {\n const d = typeof date === 'string' ? new Date(date) : date;\n return d.toLocaleDateString('en-US', {\n year: 'numeric',\n month: 'short',\n day: 'numeric',\n });\n}\n\n// ============================================\n// Resource Widget Builder\n// ============================================\n\n/**\n * Build a resource widget\n */\nexport function resourceWidget(options: ResourceOptions): string {\n const {\n type,\n title,\n description,\n icon,\n thumbnail,\n url,\n meta,\n status,\n actions = [],\n className = '',\n cardOptions = {},\n } = options;\n\n // Icon/Thumbnail\n const iconHtml = thumbnail\n ? `<div class=\"w-16 h-16 rounded-lg overflow-hidden bg-gray-100 flex-shrink-0\">\n <img src=\"${escapeHtml(thumbnail)}\" alt=\"${escapeHtml(title)}\" class=\"w-full h-full object-cover\">\n </div>`\n : `<div class=\"w-16 h-16 rounded-lg bg-gray-100 flex items-center justify-center flex-shrink-0 text-gray-400\">\n ${icon || resourceIcons[type]}\n </div>`;\n\n // Status badge\n const statusHtml = status ? badge(status.label, { variant: status.variant, size: 'sm' }) : '';\n\n // Metadata\n const metaItems: string[] = [];\n if (meta?.size) {\n metaItems.push(formatFileSize(meta.size));\n }\n if (meta?.mimeType) {\n metaItems.push(meta.mimeType);\n }\n if (meta?.updatedAt) {\n metaItems.push(`Updated ${formatDate(meta.updatedAt)}`);\n } else if (meta?.createdAt) {\n metaItems.push(`Created ${formatDate(meta.createdAt)}`);\n }\n if (meta?.author) {\n metaItems.push(`by ${meta.author}`);\n }\n\n const metaHtml =\n metaItems.length > 0 ? `<div class=\"text-xs text-text-secondary mt-1\">${metaItems.join(' • ')}</div>` : '';\n\n // Tags\n const tagsHtml =\n meta?.tags && meta.tags.length > 0\n ? `<div class=\"flex flex-wrap gap-1 mt-2\">\n ${meta.tags.map((tag) => badge(tag, { variant: 'default', size: 'sm' })).join('')}\n </div>`\n : '';\n\n // Actions\n const actionsHtml =\n actions.length > 0\n ? `<div class=\"flex gap-2 mt-4\">\n ${actions\n .map((action) => {\n const variantMap: Record<string, string> = {\n primary: 'primary',\n secondary: 'secondary',\n danger: 'danger',\n ghost: 'ghost',\n };\n const variant = action.variant ? variantMap[action.variant] : 'ghost';\n\n const htmxAttrs: string[] = [];\n if (action.htmx) {\n if (action.htmx.get) htmxAttrs.push(`hx-get=\"${escapeHtml(action.htmx.get)}\"`);\n if (action.htmx.post) htmxAttrs.push(`hx-post=\"${escapeHtml(action.htmx.post)}\"`);\n if (action.htmx.delete) htmxAttrs.push(`hx-delete=\"${escapeHtml(action.htmx.delete)}\"`);\n if (action.htmx.target) htmxAttrs.push(`hx-target=\"${escapeHtml(action.htmx.target)}\"`);\n if (action.htmx.swap) htmxAttrs.push(`hx-swap=\"${escapeHtml(action.htmx.swap)}\"`);\n if (action.htmx.confirm) htmxAttrs.push(`hx-confirm=\"${escapeHtml(action.htmx.confirm)}\"`);\n }\n\n return button(action.label, {\n variant: variant as 'primary' | 'secondary' | 'danger' | 'ghost',\n size: 'sm',\n href: action.href,\n disabled: action.disabled,\n iconBefore: action.icon,\n });\n })\n .join('')}\n </div>`\n : '';\n\n // Content\n const content = `\n <div class=\"flex gap-4\">\n ${iconHtml}\n <div class=\"flex-1 min-w-0\">\n <div class=\"flex items-start justify-between gap-2\">\n <div class=\"min-w-0\">\n ${\n url\n ? `<a href=\"${escapeHtml(\n url,\n )}\" class=\"font-semibold text-text-primary hover:text-primary truncate block\">${escapeHtml(\n title,\n )}</a>`\n : `<h3 class=\"font-semibold text-text-primary truncate\">${escapeHtml(title)}</h3>`\n }\n ${\n description\n ? `<p class=\"text-sm text-text-secondary mt-0.5 line-clamp-2\">${escapeHtml(description)}</p>`\n : ''\n }\n ${metaHtml}\n ${tagsHtml}\n </div>\n ${statusHtml}\n </div>\n ${actionsHtml}\n </div>\n </div>\n `;\n\n return card(content, {\n variant: 'default',\n size: 'md',\n className: `resource-widget resource-${type} ${className}`,\n ...cardOptions,\n });\n}\n\n// ============================================\n// Resource List Widget\n// ============================================\n\n/**\n * Resource list options\n */\nexport interface ResourceListOptions {\n /** Resources to display */\n resources: ResourceOptions[];\n /** List title */\n title?: string;\n /** Empty state message */\n emptyMessage?: string;\n /** Grid or list layout */\n layout?: 'list' | 'grid';\n /** Grid columns */\n columns?: 1 | 2 | 3 | 4;\n /** Additional CSS classes */\n className?: string;\n /** Show load more button */\n showLoadMore?: boolean;\n /** Load more URL */\n loadMoreUrl?: string;\n}\n\n/**\n * Build a resource list widget\n */\nexport function resourceList(options: ResourceListOptions): string {\n const {\n resources,\n title,\n emptyMessage = 'No resources found',\n layout = 'list',\n columns = 2,\n className = '',\n showLoadMore = false,\n loadMoreUrl,\n } = options;\n\n // Title\n const titleHtml = title ? `<h2 class=\"text-lg font-semibold text-text-primary mb-4\">${escapeHtml(title)}</h2>` : '';\n\n // Empty state\n if (resources.length === 0) {\n return `<div class=\"${className}\">\n ${titleHtml}\n <div class=\"text-center py-12 text-text-secondary\">\n <svg class=\"w-12 h-12 mx-auto mb-4 text-gray-300\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M20 13V6a2 2 0 00-2-2H6a2 2 0 00-2 2v7m16 0v5a2 2 0 01-2 2H6a2 2 0 01-2-2v-5m16 0h-2.586a1 1 0 00-.707.293l-2.414 2.414a1 1 0 01-.707.293h-3.172a1 1 0 01-.707-.293l-2.414-2.414A1 1 0 006.586 13H4\"/>\n </svg>\n <p>${escapeHtml(emptyMessage)}</p>\n </div>\n </div>`;\n }\n\n // Layout classes\n const layoutClasses = layout === 'grid' ? `grid grid-cols-1 md:grid-cols-${columns} gap-4` : 'space-y-4';\n\n // Resources\n const resourcesHtml = resources.map((r) => resourceWidget(r)).join('\\n');\n\n // Load more button\n const loadMoreHtml =\n showLoadMore && loadMoreUrl\n ? `<div class=\"text-center mt-6\">\n ${button('Load More', {\n variant: 'outline',\n htmx: {\n get: loadMoreUrl,\n target: 'closest .resource-list',\n swap: 'beforeend',\n },\n })}\n </div>`\n : '';\n\n return `<div class=\"resource-list ${className}\">\n ${titleHtml}\n <div class=\"${layoutClasses}\">\n ${resourcesHtml}\n </div>\n ${loadMoreHtml}\n </div>`;\n}\n\n// ============================================\n// Compact Resource Item\n// ============================================\n\n/**\n * Build a compact resource item (for inline display)\n */\nexport function resourceItem(options: Omit<ResourceOptions, 'cardOptions'>): string {\n const { type, title, description, icon, url, meta, status } = options;\n\n const iconHtml = `<div class=\"w-10 h-10 rounded-lg bg-gray-100 flex items-center justify-center flex-shrink-0 text-gray-400\">\n ${icon || resourceIcons[type]}\n </div>`;\n\n const statusHtml = status ? badge(status.label, { variant: status.variant, size: 'sm' }) : '';\n\n const metaText = meta?.size ? formatFileSize(meta.size) : '';\n\n const titleElement = url\n ? `<a href=\"${escapeHtml(url)}\" class=\"font-medium text-text-primary hover:text-primary\">${escapeHtml(title)}</a>`\n : `<span class=\"font-medium text-text-primary\">${escapeHtml(title)}</span>`;\n\n return `<div class=\"flex items-center gap-3 p-3 rounded-lg hover:bg-gray-50 transition-colors\">\n ${iconHtml}\n <div class=\"flex-1 min-w-0\">\n <div class=\"flex items-center gap-2\">\n ${titleElement}\n ${statusHtml}\n </div>\n ${\n description || metaText\n ? `<p class=\"text-sm text-text-secondary truncate\">${escapeHtml(description || metaText)}</p>`\n : ''\n }\n </div>\n </div>`;\n}\n\n// ============================================\n// Preview Widget\n// ============================================\n\n/**\n * Code preview options\n */\nexport interface CodePreviewOptions {\n /** Code content */\n code: string;\n /** Language */\n language?: string;\n /** Filename */\n filename?: string;\n /** Show line numbers */\n lineNumbers?: boolean;\n /** Max height */\n maxHeight?: string;\n /** Copy button */\n showCopy?: boolean;\n /** Additional CSS classes */\n className?: string;\n}\n\n/**\n * Build a code preview widget\n */\nexport function codePreview(options: CodePreviewOptions): string {\n const {\n code,\n language = 'text',\n filename,\n lineNumbers = true,\n maxHeight = '400px',\n showCopy = true,\n className = '',\n } = options;\n\n const lines = code.split('\\n');\n\n const lineNumbersHtml = lineNumbers\n ? `<div class=\"text-right select-none pr-4 text-gray-500\">\n ${lines.map((_, i) => `<div>${i + 1}</div>`).join('')}\n </div>`\n : '';\n\n const copyScript = showCopy\n ? `<script>\n function copyCode(btn, code) {\n navigator.clipboard.writeText(code).then(() => {\n const original = btn.innerHTML;\n btn.innerHTML = '<svg class=\"w-4 h-4 text-success\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\"><path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M5 13l4 4L19 7\"/></svg>';\n setTimeout(() => btn.innerHTML = original, 2000);\n });\n }\n </script>`\n : '';\n\n const copyButton = showCopy\n ? `<button\n type=\"button\"\n onclick=\"copyCode(this, ${escapeHtml(JSON.stringify(code))})\"\n class=\"p-1.5 rounded hover:bg-gray-700 transition-colors\"\n title=\"Copy code\"\n >\n <svg class=\"w-4 h-4\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M8 16H6a2 2 0 01-2-2V6a2 2 0 012-2h8a2 2 0 012 2v2m-6 12h8a2 2 0 002-2v-8a2 2 0 00-2-2h-8a2 2 0 00-2 2v8a2 2 0 002 2z\"/>\n </svg>\n </button>`\n : '';\n\n return `<div class=\"code-preview rounded-lg overflow-hidden ${className}\">\n ${\n filename || showCopy\n ? `\n <div class=\"flex items-center justify-between px-4 py-2 bg-gray-800 border-b border-gray-700\">\n ${filename ? `<span class=\"text-sm text-gray-300\">${escapeHtml(filename)}</span>` : '<span></span>'}\n <div class=\"flex items-center gap-2\">\n ${language ? `<span class=\"text-xs text-gray-500\">${escapeHtml(language)}</span>` : ''}\n ${copyButton}\n </div>\n </div>\n `\n : ''\n }\n <div class=\"bg-gray-900 p-4 overflow-auto\" style=\"max-height: ${maxHeight}\">\n <div class=\"flex text-sm font-mono\">\n ${lineNumbersHtml}\n <pre class=\"flex-1 text-gray-100\"><code>${escapeHtml(code)}</code></pre>\n </div>\n </div>\n ${copyScript}\n </div>`;\n}\n\n/**\n * Image preview options\n */\nexport interface ImagePreviewOptions {\n /** Image URL */\n src: string;\n /** Alt text */\n alt: string;\n /** Caption */\n caption?: string;\n /** Max height */\n maxHeight?: string;\n /** Clickable (opens in new tab) */\n clickable?: boolean;\n /** Additional CSS classes */\n className?: string;\n}\n\n/**\n * Build an image preview widget\n */\nexport function imagePreview(options: ImagePreviewOptions): string {\n const { src, alt, caption, maxHeight = '400px', clickable = true, className = '' } = options;\n\n const imageHtml = `<img\n src=\"${escapeHtml(src)}\"\n alt=\"${escapeHtml(alt)}\"\n class=\"max-w-full h-auto rounded-lg\"\n style=\"max-height: ${maxHeight}\"\n >`;\n\n const captionHtml = caption\n ? `<p class=\"text-sm text-text-secondary mt-2 text-center\">${escapeHtml(caption)}</p>`\n : '';\n\n const content = clickable\n ? `<a href=\"${escapeHtml(src)}\" target=\"_blank\" rel=\"noopener\" class=\"block\">${imageHtml}</a>`\n : imageHtml;\n\n return `<div class=\"image-preview ${className}\">\n ${content}\n ${captionHtml}\n </div>`;\n}\n"]}
|