@cuadra-ai/uikit 0.4.21 → 0.4.23
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/dist/index.cjs +13 -13
- package/dist/index.cjs.map +1 -1
- package/dist/index.mjs +994 -987
- package/dist/index.mjs.map +1 -1
- package/dist/widget/cuadra-uikit.umd.js +20 -20
- package/dist/widget/cuadra-uikit.umd.js.map +1 -1
- package/package.json +1 -1
package/dist/index.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.cjs","sources":["../src/utils/autoInjectStyles.ts","../node_modules/lucide-react/dist/esm/shared/src/utils.js","../node_modules/lucide-react/dist/esm/defaultAttributes.js","../node_modules/lucide-react/dist/esm/Icon.js","../node_modules/lucide-react/dist/esm/createLucideIcon.js","../node_modules/lucide-react/dist/esm/icons/arrow-up.js","../node_modules/lucide-react/dist/esm/icons/brain.js","../node_modules/lucide-react/dist/esm/icons/check.js","../node_modules/lucide-react/dist/esm/icons/chevron-down.js","../node_modules/lucide-react/dist/esm/icons/circle-alert.js","../node_modules/lucide-react/dist/esm/icons/copy.js","../node_modules/lucide-react/dist/esm/icons/ellipsis-vertical.js","../node_modules/lucide-react/dist/esm/icons/eye.js","../node_modules/lucide-react/dist/esm/icons/file-spreadsheet.js","../node_modules/lucide-react/dist/esm/icons/file-text.js","../node_modules/lucide-react/dist/esm/icons/file-type.js","../node_modules/lucide-react/dist/esm/icons/loader-circle.js","../node_modules/lucide-react/dist/esm/icons/message-square.js","../node_modules/lucide-react/dist/esm/icons/paperclip.js","../node_modules/lucide-react/dist/esm/icons/pencil.js","../node_modules/lucide-react/dist/esm/icons/plus.js","../node_modules/lucide-react/dist/esm/icons/trash-2.js","../node_modules/lucide-react/dist/esm/icons/upload.js","../node_modules/lucide-react/dist/esm/icons/x.js","../src/utils/sanitize.ts","../src/lib/cuadraChatClient.ts","../src/adapters/messageConverter.ts","../node_modules/assistant-stream/dist/utils/promiseWithResolvers.js","../node_modules/assistant-stream/dist/core/utils/stream/merge.js","../node_modules/assistant-stream/dist/core/modules/text.js","../node_modules/assistant-stream/dist/core/modules/tool-call.js","../node_modules/assistant-stream/dist/core/utils/Counter.js","../node_modules/assistant-stream/dist/core/utils/stream/path-utils.js","../node_modules/nanoid/non-secure/index.js","../node_modules/assistant-stream/dist/core/utils/generateId.js","../node_modules/assistant-stream/dist/core/modules/assistant-stream.js","../src/adapters/threadListAdapter.tsx","../src/adapters/streamingMetadataStore.ts","../src/adapters/chatModelAdapter.ts","../src/adapters/attachmentAdapter.ts","../src/adapters/attachmentErrorStore.ts","../src/components/CuadraRuntimeProvider.tsx","../node_modules/ccount/index.js","../node_modules/micromark-util-character/index.js","../node_modules/mdast-util-find-and-replace/node_modules/escape-string-regexp/index.js","../node_modules/unist-util-is/lib/index.js","../node_modules/unist-util-visit-parents/lib/index.js","../node_modules/mdast-util-find-and-replace/lib/index.js","../node_modules/mdast-util-gfm-autolink-literal/lib/index.js","../node_modules/micromark-util-normalize-identifier/index.js","../node_modules/mdast-util-gfm-footnote/lib/index.js","../node_modules/mdast-util-gfm-strikethrough/lib/index.js","../node_modules/markdown-table/index.js","../node_modules/mdast-util-to-markdown/lib/handle/blockquote.js","../node_modules/mdast-util-to-markdown/lib/util/pattern-in-scope.js","../node_modules/mdast-util-to-markdown/lib/handle/break.js","../node_modules/longest-streak/index.js","../node_modules/mdast-util-to-markdown/lib/util/format-code-as-indented.js","../node_modules/mdast-util-to-markdown/lib/util/check-fence.js","../node_modules/mdast-util-to-markdown/lib/handle/code.js","../node_modules/mdast-util-to-markdown/lib/util/check-quote.js","../node_modules/mdast-util-to-markdown/lib/handle/definition.js","../node_modules/mdast-util-to-markdown/lib/util/check-emphasis.js","../node_modules/mdast-util-to-markdown/lib/util/encode-character-reference.js","../node_modules/micromark-util-classify-character/index.js","../node_modules/mdast-util-to-markdown/lib/util/encode-info.js","../node_modules/mdast-util-to-markdown/lib/handle/emphasis.js","../node_modules/unist-util-visit/lib/index.js","../node_modules/mdast-util-to-string/lib/index.js","../node_modules/mdast-util-to-markdown/lib/util/format-heading-as-setext.js","../node_modules/mdast-util-to-markdown/lib/handle/heading.js","../node_modules/mdast-util-to-markdown/lib/handle/html.js","../node_modules/mdast-util-to-markdown/lib/handle/image.js","../node_modules/mdast-util-to-markdown/lib/handle/image-reference.js","../node_modules/mdast-util-to-markdown/lib/handle/inline-code.js","../node_modules/mdast-util-to-markdown/lib/util/format-link-as-autolink.js","../node_modules/mdast-util-to-markdown/lib/handle/link.js","../node_modules/mdast-util-to-markdown/lib/handle/link-reference.js","../node_modules/mdast-util-to-markdown/lib/util/check-bullet.js","../node_modules/mdast-util-to-markdown/lib/util/check-bullet-other.js","../node_modules/mdast-util-to-markdown/lib/util/check-bullet-ordered.js","../node_modules/mdast-util-to-markdown/lib/util/check-rule.js","../node_modules/mdast-util-to-markdown/lib/handle/list.js","../node_modules/mdast-util-to-markdown/lib/util/check-list-item-indent.js","../node_modules/mdast-util-to-markdown/lib/handle/list-item.js","../node_modules/mdast-util-to-markdown/lib/handle/paragraph.js","../node_modules/mdast-util-phrasing/lib/index.js","../node_modules/mdast-util-to-markdown/lib/handle/root.js","../node_modules/mdast-util-to-markdown/lib/util/check-strong.js","../node_modules/mdast-util-to-markdown/lib/handle/strong.js","../node_modules/mdast-util-to-markdown/lib/handle/text.js","../node_modules/mdast-util-to-markdown/lib/util/check-rule-repetition.js","../node_modules/mdast-util-to-markdown/lib/handle/thematic-break.js","../node_modules/mdast-util-to-markdown/lib/handle/index.js","../node_modules/mdast-util-gfm-table/lib/index.js","../node_modules/mdast-util-gfm-task-list-item/lib/index.js","../node_modules/mdast-util-gfm/lib/index.js","../node_modules/micromark-util-chunked/index.js","../node_modules/micromark-util-combine-extensions/index.js","../node_modules/micromark-extension-gfm-autolink-literal/lib/syntax.js","../node_modules/micromark-util-resolve-all/index.js","../node_modules/micromark-factory-space/index.js","../node_modules/micromark-core-commonmark/lib/blank-line.js","../node_modules/micromark-extension-gfm-footnote/lib/syntax.js","../node_modules/micromark-extension-gfm-strikethrough/lib/syntax.js","../node_modules/micromark-extension-gfm-table/lib/edit-map.js","../node_modules/micromark-extension-gfm-table/lib/infer.js","../node_modules/micromark-extension-gfm-table/lib/syntax.js","../node_modules/micromark-extension-gfm-task-list-item/lib/syntax.js","../node_modules/micromark-extension-gfm/index.js","../node_modules/remark-gfm/lib/index.js","../src/utils/cn.ts","../src/components/MarkdownText.tsx","../src/components/ui/badge.tsx","../src/components/SourceCitations.tsx","../src/components/SimpleThread.tsx","../src/widget/components/TexturedCard.tsx","../src/widget/context/CuadraWidgetContext.tsx","../src/utils/useInjectStyles.ts","../src/components/WidgetContent.tsx","../src/utils/applyTheme.ts","../src/components/ErrorBoundary.tsx","../src/components/CuadraChatContext.tsx","../src/types/props.ts","../src/components/CuadraChat.tsx"],"sourcesContent":["/**\n * Automatically inject UIKit styles when the package is imported\n * This ensures styles are loaded without requiring clients to manually import them\n * \n * The CSS import in index.ts should handle most cases, but this provides a runtime\n * fallback that injects a link tag to load the CSS file if needed.\n */\n\nlet stylesInjected = false;\nlet injectionAttempted = false;\n\n/**\n * Automatically inject styles into the DOM\n * This runs once when the module is first imported\n */\nfunction autoInjectStyles(): void {\n // Only attempt injection once\n if (injectionAttempted) {\n return;\n }\n injectionAttempted = true;\n\n // Only inject if we're in a browser environment\n if (typeof document === 'undefined' || typeof window === 'undefined') {\n return;\n }\n\n // Check if styles are already loaded (via manual import or previous injection)\n const existingLink = document.querySelector('link[data-cuadra-uikit-styles]');\n const existingStyle = document.querySelector('style[data-cuadra-uikit-styles]');\n \n if (existingLink || existingStyle) {\n stylesInjected = true;\n return;\n }\n\n // Check if CSS classes are already available (styles might be loaded via CSS import)\n // This is a quick check to see if Tailwind classes work\n try {\n const testEl = document.createElement('div');\n testEl.className = 'cuadra-flex';\n testEl.style.visibility = 'hidden';\n testEl.style.position = 'absolute';\n testEl.style.pointerEvents = 'none';\n document.body.appendChild(testEl);\n const stylesLoaded = window.getComputedStyle(testEl).display === 'flex';\n document.body.removeChild(testEl);\n\n if (stylesLoaded) {\n stylesInjected = true;\n return;\n }\n } catch (_error) {\n // If check fails, proceed to inject styles\n }\n\n // Inject styles as a fallback\n // The CSS import in index.ts should handle most bundlers, but this ensures\n // styles work even if the CSS import isn't processed correctly\n injectStylesLink();\n}\n\n/**\n * Inject a link tag to load the CSS file\n * This is a fallback for cases where CSS import doesn't work\n */\nfunction injectStylesLink(): void {\n if (stylesInjected) {\n return;\n }\n\n const link = document.createElement('link');\n link.setAttribute('data-cuadra-uikit-styles', 'true');\n link.rel = 'stylesheet';\n link.type = 'text/css';\n \n // Try to determine the correct path to the CSS file\n // In most bundlers, we can use import.meta.url to get the current module's URL\n try {\n // Try relative path from the built JS file\n // This works when the package is installed via npm\n const currentUrl = new URL(import.meta.url);\n const cssUrl = new URL('../dist/uikit.css', currentUrl);\n link.href = cssUrl.href;\n } catch (_error) {\n // Fallback to a common path\n link.href = '/node_modules/@cuadra-ai/uikit/dist/uikit.css';\n }\n\n // Handle load success\n link.onload = () => {\n stylesInjected = true;\n };\n\n // Handle load error - try alternative paths\n link.onerror = () => {\n // Try alternative paths\n const alternatives = [\n '/node_modules/@cuadra-ai/uikit/dist/uikit.css',\n 'https://unpkg.com/@cuadra-ai/uikit@latest/dist/uikit.css',\n ];\n\n let attemptIndex = 0;\n const tryNext = () => {\n if (attemptIndex < alternatives.length) {\n link.href = alternatives[attemptIndex++];\n link.onerror = tryNext;\n link.onload = () => {\n stylesInjected = true;\n };\n }\n };\n\n tryNext();\n };\n\n document.head.appendChild(link);\n}\n\n// Auto-inject styles when this module is imported\n// This runs immediately when the package is imported\n// We use a small delay to ensure DOM is ready\nif (typeof document !== 'undefined' && typeof window !== 'undefined') {\n if (document.readyState === 'loading') {\n document.addEventListener('DOMContentLoaded', autoInjectStyles);\n } else {\n // DOM is already ready, inject immediately\n autoInjectStyles();\n }\n}\n\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nconst toKebabCase = (string) => string.replace(/([a-z0-9])([A-Z])/g, \"$1-$2\").toLowerCase();\nconst toCamelCase = (string) => string.replace(\n /^([A-Z])|[\\s-_]+(\\w)/g,\n (match, p1, p2) => p2 ? p2.toUpperCase() : p1.toLowerCase()\n);\nconst toPascalCase = (string) => {\n const camelCase = toCamelCase(string);\n return camelCase.charAt(0).toUpperCase() + camelCase.slice(1);\n};\nconst mergeClasses = (...classes) => classes.filter((className, index, array) => {\n return Boolean(className) && className.trim() !== \"\" && array.indexOf(className) === index;\n}).join(\" \").trim();\nconst hasA11yProp = (props) => {\n for (const prop in props) {\n if (prop.startsWith(\"aria-\") || prop === \"role\" || prop === \"title\") {\n return true;\n }\n }\n};\n\nexport { hasA11yProp, mergeClasses, toCamelCase, toKebabCase, toPascalCase };\n//# sourceMappingURL=utils.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nvar defaultAttributes = {\n xmlns: \"http://www.w3.org/2000/svg\",\n width: 24,\n height: 24,\n viewBox: \"0 0 24 24\",\n fill: \"none\",\n stroke: \"currentColor\",\n strokeWidth: 2,\n strokeLinecap: \"round\",\n strokeLinejoin: \"round\"\n};\n\nexport { defaultAttributes as default };\n//# sourceMappingURL=defaultAttributes.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport { forwardRef, createElement } from 'react';\nimport defaultAttributes from './defaultAttributes.js';\nimport { mergeClasses, hasA11yProp } from './shared/src/utils.js';\n\nconst Icon = forwardRef(\n ({\n color = \"currentColor\",\n size = 24,\n strokeWidth = 2,\n absoluteStrokeWidth,\n className = \"\",\n children,\n iconNode,\n ...rest\n }, ref) => createElement(\n \"svg\",\n {\n ref,\n ...defaultAttributes,\n width: size,\n height: size,\n stroke: color,\n strokeWidth: absoluteStrokeWidth ? Number(strokeWidth) * 24 / Number(size) : strokeWidth,\n className: mergeClasses(\"lucide\", className),\n ...!children && !hasA11yProp(rest) && { \"aria-hidden\": \"true\" },\n ...rest\n },\n [\n ...iconNode.map(([tag, attrs]) => createElement(tag, attrs)),\n ...Array.isArray(children) ? children : [children]\n ]\n )\n);\n\nexport { Icon as default };\n//# sourceMappingURL=Icon.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport { forwardRef, createElement } from 'react';\nimport { mergeClasses, toKebabCase, toPascalCase } from './shared/src/utils.js';\nimport Icon from './Icon.js';\n\nconst createLucideIcon = (iconName, iconNode) => {\n const Component = forwardRef(\n ({ className, ...props }, ref) => createElement(Icon, {\n ref,\n iconNode,\n className: mergeClasses(\n `lucide-${toKebabCase(toPascalCase(iconName))}`,\n `lucide-${iconName}`,\n className\n ),\n ...props\n })\n );\n Component.displayName = toPascalCase(iconName);\n return Component;\n};\n\nexport { createLucideIcon as default };\n//# sourceMappingURL=createLucideIcon.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"path\", { d: \"m5 12 7-7 7 7\", key: \"hav0vg\" }],\n [\"path\", { d: \"M12 19V5\", key: \"x0mq9r\" }]\n];\nconst ArrowUp = createLucideIcon(\"arrow-up\", __iconNode);\n\nexport { __iconNode, ArrowUp as default };\n//# sourceMappingURL=arrow-up.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"path\", { d: \"M12 18V5\", key: \"adv99a\" }],\n [\"path\", { d: \"M15 13a4.17 4.17 0 0 1-3-4 4.17 4.17 0 0 1-3 4\", key: \"1e3is1\" }],\n [\"path\", { d: \"M17.598 6.5A3 3 0 1 0 12 5a3 3 0 1 0-5.598 1.5\", key: \"1gqd8o\" }],\n [\"path\", { d: \"M17.997 5.125a4 4 0 0 1 2.526 5.77\", key: \"iwvgf7\" }],\n [\"path\", { d: \"M18 18a4 4 0 0 0 2-7.464\", key: \"efp6ie\" }],\n [\"path\", { d: \"M19.967 17.483A4 4 0 1 1 12 18a4 4 0 1 1-7.967-.517\", key: \"1gq6am\" }],\n [\"path\", { d: \"M6 18a4 4 0 0 1-2-7.464\", key: \"k1g0md\" }],\n [\"path\", { d: \"M6.003 5.125a4 4 0 0 0-2.526 5.77\", key: \"q97ue3\" }]\n];\nconst Brain = createLucideIcon(\"brain\", __iconNode);\n\nexport { __iconNode, Brain as default };\n//# sourceMappingURL=brain.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [[\"path\", { d: \"M20 6 9 17l-5-5\", key: \"1gmf2c\" }]];\nconst Check = createLucideIcon(\"check\", __iconNode);\n\nexport { __iconNode, Check as default };\n//# sourceMappingURL=check.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [[\"path\", { d: \"m6 9 6 6 6-6\", key: \"qrunsl\" }]];\nconst ChevronDown = createLucideIcon(\"chevron-down\", __iconNode);\n\nexport { __iconNode, ChevronDown as default };\n//# sourceMappingURL=chevron-down.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"circle\", { cx: \"12\", cy: \"12\", r: \"10\", key: \"1mglay\" }],\n [\"line\", { x1: \"12\", x2: \"12\", y1: \"8\", y2: \"12\", key: \"1pkeuh\" }],\n [\"line\", { x1: \"12\", x2: \"12.01\", y1: \"16\", y2: \"16\", key: \"4dfq90\" }]\n];\nconst CircleAlert = createLucideIcon(\"circle-alert\", __iconNode);\n\nexport { __iconNode, CircleAlert as default };\n//# sourceMappingURL=circle-alert.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"rect\", { width: \"14\", height: \"14\", x: \"8\", y: \"8\", rx: \"2\", ry: \"2\", key: \"17jyea\" }],\n [\"path\", { d: \"M4 16c-1.1 0-2-.9-2-2V4c0-1.1.9-2 2-2h10c1.1 0 2 .9 2 2\", key: \"zix9uf\" }]\n];\nconst Copy = createLucideIcon(\"copy\", __iconNode);\n\nexport { __iconNode, Copy as default };\n//# sourceMappingURL=copy.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"circle\", { cx: \"12\", cy: \"12\", r: \"1\", key: \"41hilf\" }],\n [\"circle\", { cx: \"12\", cy: \"5\", r: \"1\", key: \"gxeob9\" }],\n [\"circle\", { cx: \"12\", cy: \"19\", r: \"1\", key: \"lyex9k\" }]\n];\nconst EllipsisVertical = createLucideIcon(\"ellipsis-vertical\", __iconNode);\n\nexport { __iconNode, EllipsisVertical as default };\n//# sourceMappingURL=ellipsis-vertical.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\n \"path\",\n {\n d: \"M2.062 12.348a1 1 0 0 1 0-.696 10.75 10.75 0 0 1 19.876 0 1 1 0 0 1 0 .696 10.75 10.75 0 0 1-19.876 0\",\n key: \"1nclc0\"\n }\n ],\n [\"circle\", { cx: \"12\", cy: \"12\", r: \"3\", key: \"1v7zrd\" }]\n];\nconst Eye = createLucideIcon(\"eye\", __iconNode);\n\nexport { __iconNode, Eye as default };\n//# sourceMappingURL=eye.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\n \"path\",\n {\n d: \"M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z\",\n key: \"1oefj6\"\n }\n ],\n [\"path\", { d: \"M14 2v5a1 1 0 0 0 1 1h5\", key: \"wfsgrz\" }],\n [\"path\", { d: \"M8 13h2\", key: \"yr2amv\" }],\n [\"path\", { d: \"M14 13h2\", key: \"un5t4a\" }],\n [\"path\", { d: \"M8 17h2\", key: \"2yhykz\" }],\n [\"path\", { d: \"M14 17h2\", key: \"10kma7\" }]\n];\nconst FileSpreadsheet = createLucideIcon(\"file-spreadsheet\", __iconNode);\n\nexport { __iconNode, FileSpreadsheet as default };\n//# sourceMappingURL=file-spreadsheet.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\n \"path\",\n {\n d: \"M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z\",\n key: \"1oefj6\"\n }\n ],\n [\"path\", { d: \"M14 2v5a1 1 0 0 0 1 1h5\", key: \"wfsgrz\" }],\n [\"path\", { d: \"M10 9H8\", key: \"b1mrlr\" }],\n [\"path\", { d: \"M16 13H8\", key: \"t4e002\" }],\n [\"path\", { d: \"M16 17H8\", key: \"z1uh3a\" }]\n];\nconst FileText = createLucideIcon(\"file-text\", __iconNode);\n\nexport { __iconNode, FileText as default };\n//# sourceMappingURL=file-text.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\n \"path\",\n {\n d: \"M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z\",\n key: \"1oefj6\"\n }\n ],\n [\"path\", { d: \"M14 2v5a1 1 0 0 0 1 1h5\", key: \"wfsgrz\" }],\n [\"path\", { d: \"M11 18h2\", key: \"12mj7e\" }],\n [\"path\", { d: \"M12 12v6\", key: \"3ahymv\" }],\n [\"path\", { d: \"M9 13v-.5a.5.5 0 0 1 .5-.5h5a.5.5 0 0 1 .5.5v.5\", key: \"qbrxap\" }]\n];\nconst FileType = createLucideIcon(\"file-type\", __iconNode);\n\nexport { __iconNode, FileType as default };\n//# sourceMappingURL=file-type.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [[\"path\", { d: \"M21 12a9 9 0 1 1-6.219-8.56\", key: \"13zald\" }]];\nconst LoaderCircle = createLucideIcon(\"loader-circle\", __iconNode);\n\nexport { __iconNode, LoaderCircle as default };\n//# sourceMappingURL=loader-circle.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\n \"path\",\n {\n d: \"M22 17a2 2 0 0 1-2 2H6.828a2 2 0 0 0-1.414.586l-2.202 2.202A.71.71 0 0 1 2 21.286V5a2 2 0 0 1 2-2h16a2 2 0 0 1 2 2z\",\n key: \"18887p\"\n }\n ]\n];\nconst MessageSquare = createLucideIcon(\"message-square\", __iconNode);\n\nexport { __iconNode, MessageSquare as default };\n//# sourceMappingURL=message-square.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\n \"path\",\n {\n d: \"m16 6-8.414 8.586a2 2 0 0 0 2.829 2.829l8.414-8.586a4 4 0 1 0-5.657-5.657l-8.379 8.551a6 6 0 1 0 8.485 8.485l8.379-8.551\",\n key: \"1miecu\"\n }\n ]\n];\nconst Paperclip = createLucideIcon(\"paperclip\", __iconNode);\n\nexport { __iconNode, Paperclip as default };\n//# sourceMappingURL=paperclip.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\n \"path\",\n {\n d: \"M21.174 6.812a1 1 0 0 0-3.986-3.987L3.842 16.174a2 2 0 0 0-.5.83l-1.321 4.352a.5.5 0 0 0 .623.622l4.353-1.32a2 2 0 0 0 .83-.497z\",\n key: \"1a8usu\"\n }\n ],\n [\"path\", { d: \"m15 5 4 4\", key: \"1mk7zo\" }]\n];\nconst Pencil = createLucideIcon(\"pencil\", __iconNode);\n\nexport { __iconNode, Pencil as default };\n//# sourceMappingURL=pencil.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"path\", { d: \"M5 12h14\", key: \"1ays0h\" }],\n [\"path\", { d: \"M12 5v14\", key: \"s699le\" }]\n];\nconst Plus = createLucideIcon(\"plus\", __iconNode);\n\nexport { __iconNode, Plus as default };\n//# sourceMappingURL=plus.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"path\", { d: \"M10 11v6\", key: \"nco0om\" }],\n [\"path\", { d: \"M14 11v6\", key: \"outv1u\" }],\n [\"path\", { d: \"M19 6v14a2 2 0 0 1-2 2H7a2 2 0 0 1-2-2V6\", key: \"miytrc\" }],\n [\"path\", { d: \"M3 6h18\", key: \"d0wm0j\" }],\n [\"path\", { d: \"M8 6V4a2 2 0 0 1 2-2h4a2 2 0 0 1 2 2v2\", key: \"e791ji\" }]\n];\nconst Trash2 = createLucideIcon(\"trash-2\", __iconNode);\n\nexport { __iconNode, Trash2 as default };\n//# sourceMappingURL=trash-2.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"path\", { d: \"M12 3v12\", key: \"1x0j5s\" }],\n [\"path\", { d: \"m17 8-5-5-5 5\", key: \"7q97r8\" }],\n [\"path\", { d: \"M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4\", key: \"ih7n3h\" }]\n];\nconst Upload = createLucideIcon(\"upload\", __iconNode);\n\nexport { __iconNode, Upload as default };\n//# sourceMappingURL=upload.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"path\", { d: \"M18 6 6 18\", key: \"1bl5f8\" }],\n [\"path\", { d: \"m6 6 12 12\", key: \"d8bk6v\" }]\n];\nconst X = createLucideIcon(\"x\", __iconNode);\n\nexport { __iconNode, X as default };\n//# sourceMappingURL=x.js.map\n","/**\n * Security utilities for input sanitization\n * \n * These utilities provide defense-in-depth protection against common\n * security issues. They should be used in addition to proper input\n * validation at the API level.\n */\n\n/**\n * Common prompt injection patterns to filter\n * These patterns are used by various LLM implementations\n */\nconst INJECTION_PATTERNS = [\n // Llama/Mistral instruction format\n /\\[INST\\]/gi,\n /\\[\\/INST\\]/gi,\n // Llama system prompt format\n /<<SYS>>/gi,\n /<<\\/SYS>>/gi,\n // Special token markers\n /<\\|.*?\\|>/g,\n // OpenAI-style system markers\n /<\\|im_start\\|>/gi,\n /<\\|im_end\\|>/gi,\n // Anthropic-style markers\n /\\[HUMAN\\]/gi,\n /\\[ASSISTANT\\]/gi,\n // Generic role injection attempts\n /^(system|user|assistant):\\s*/gim,\n];\n\n/**\n * Sanitize a system prompt to reduce prompt injection risks\n * \n * **Important:** This is defense-in-depth, not a complete solution.\n * Never use untrusted user input directly as a system prompt.\n * If you need dynamic prompts, use a server-side allow list.\n * \n * @param prompt - The system prompt to sanitize\n * @returns Sanitized prompt with common injection patterns removed\n * \n * @example\n * ```ts\n * const userInput = \"Hello [INST] ignore previous [/INST]\";\n * const safe = sanitizeSystemPrompt(userInput);\n * // Result: \"Hello ignore previous \"\n * ```\n */\nexport function sanitizeSystemPrompt(prompt: string): string {\n if (!prompt || typeof prompt !== 'string') {\n return '';\n }\n\n let sanitized = prompt;\n\n // Remove injection patterns\n for (const pattern of INJECTION_PATTERNS) {\n sanitized = sanitized.replace(pattern, '');\n }\n\n // Normalize whitespace (collapse multiple spaces/newlines)\n sanitized = sanitized\n .replace(/\\n{3,}/g, '\\n\\n')\n .replace(/ {2,}/g, ' ')\n .trim();\n\n return sanitized;\n}\n\n/**\n * Validate that a URL uses a safe protocol (http or https)\n * \n * Prevents javascript: protocol XSS attacks and other unsafe schemes.\n * \n * @param url - The URL to validate\n * @returns true if the URL uses http or https protocol\n * \n * @example\n * ```ts\n * isValidUrl('https://example.com'); // true\n * isValidUrl('javascript:alert(1)'); // false\n * isValidUrl('data:text/html,...'); // false\n * ```\n */\nexport function isValidUrl(url: string): boolean {\n if (!url || typeof url !== 'string') {\n return false;\n }\n\n try {\n const parsed = new URL(url);\n return ['http:', 'https:'].includes(parsed.protocol);\n } catch {\n return false;\n }\n}\n\n/**\n * Validation result for URL validation\n */\nexport interface UrlValidationResult {\n /** Whether the URL is valid */\n valid: boolean;\n /** Error message if invalid */\n error?: string;\n /** Parsed URL if valid */\n parsed?: URL;\n}\n\n/**\n * Validate a base URL for API configuration\n * \n * Ensures the URL is properly formatted and uses a secure protocol.\n * Returns detailed validation result for error messaging.\n * \n * @param url - The base URL to validate\n * @returns Validation result with error details\n * \n * @example\n * ```ts\n * const result = validateBaseUrl('https://api.example.com');\n * if (!result.valid) {\n * console.error(result.error);\n * }\n * ```\n */\nexport function validateBaseUrl(url: string): UrlValidationResult {\n if (!url) {\n return { valid: false, error: 'URL is required' };\n }\n\n if (typeof url !== 'string') {\n return { valid: false, error: 'URL must be a string' };\n }\n\n // Allow relative URLs for proxy mode\n if (url.startsWith('/') || url.startsWith('./')) {\n return { valid: true };\n }\n\n try {\n const parsed = new URL(url);\n\n if (!['http:', 'https:'].includes(parsed.protocol)) {\n return { \n valid: false, \n error: `Invalid protocol \"${parsed.protocol}\". Only http and https are allowed.` \n };\n }\n\n // Warn about non-HTTPS in production contexts\n if (parsed.protocol === 'http:' && !isLocalhostUrl(url)) {\n return {\n valid: true,\n parsed,\n error: 'Warning: Using HTTP for non-localhost URL. Consider using HTTPS.',\n };\n }\n\n return { valid: true, parsed };\n } catch {\n return { valid: false, error: 'Invalid URL format' };\n }\n}\n\n/**\n * Check if a URL is a localhost URL\n */\nfunction isLocalhostUrl(url: string): boolean {\n try {\n const parsed = new URL(url);\n return ['localhost', '127.0.0.1', '[::1]'].includes(parsed.hostname);\n } catch {\n return false;\n }\n}\n\n/**\n * Sanitize a string for safe display (escape HTML entities)\n * \n * Note: React already escapes content in JSX, so this is mainly\n * useful when constructing strings for non-React contexts.\n * \n * @param text - The text to sanitize\n * @returns Text with HTML entities escaped\n */\nexport function escapeHtml(text: string): string {\n if (!text || typeof text !== 'string') {\n return '';\n }\n\n const htmlEntities: Record<string, string> = {\n '&': '&',\n '<': '<',\n '>': '>',\n '\"': '"',\n \"'\": ''',\n '/': '/',\n };\n\n return text.replace(/[&<>\"'/]/g, (char) => htmlEntities[char] || char);\n}\n\n/**\n * Generate a cryptographically secure UUID v4\n * \n * Uses crypto.randomUUID when available, falls back to\n * crypto.getRandomValues for broader compatibility.\n * \n * @returns A UUID v4 string\n */\nexport function generateSecureUUID(): string {\n // Prefer native randomUUID (modern browsers, Node 19+)\n if (typeof crypto !== 'undefined' && crypto.randomUUID) {\n return crypto.randomUUID();\n }\n\n // Fallback using crypto.getRandomValues (still cryptographically secure)\n if (typeof crypto !== 'undefined' && crypto.getRandomValues) {\n const bytes = new Uint8Array(16);\n crypto.getRandomValues(bytes);\n\n // Set version (4) and variant (10xx) bits per RFC 4122\n bytes[6] = (bytes[6] & 0x0f) | 0x40;\n bytes[8] = (bytes[8] & 0x3f) | 0x80;\n\n const hex = Array.from(bytes, (b) => b.toString(16).padStart(2, '0')).join('');\n return `${hex.slice(0, 8)}-${hex.slice(8, 12)}-${hex.slice(12, 16)}-${hex.slice(16, 20)}-${hex.slice(20)}`;\n }\n\n // Last resort fallback (not cryptographically secure)\n // This should rarely be reached in modern environments\n if (typeof process !== 'undefined' && process.env?.NODE_ENV === 'development') {\n // eslint-disable-next-line no-console\n console.warn('[CuadraUIKit] Using non-cryptographic fallback for UUID generation');\n }\n\n return `${Date.now().toString(36)}-${Math.random().toString(36).slice(2)}-${Math.random().toString(36).slice(2)}`;\n}\n","import type {\n ChatOut,\n ChatRequest,\n ChatSummaryPage,\n ChatUpdate,\n GenerateTitleRequest,\n GenerateTitleResponse,\n ListChatsParams,\n ListModelsParams,\n ModelOut,\n ModelPage,\n Source,\n} from '../types/cuadra';\nimport { generateSecureUUID } from '../utils/sanitize';\n\n/**\n * SSE Chunk types for streaming responses\n * Supports current API format and AI SDK UI Message Stream Protocol\n */\nexport interface SSEChunk {\n // Common fields\n type?: string;\n id?: string; // Chat/message ID or block ID\n \n // Current API format fields\n delta?: string; // Delta content (incremental text)\n reasoning?: string; // Accumulated reasoning content\n sources?: Source[] | null; // RAG sources (sent in first chunk)\n toolCalls?: unknown[] | null; // Tool calls (future)\n finished?: boolean; // Whether the stream is finished\n usage?: {\n inputTokens: number;\n outputTokens: number;\n totalTokens: number;\n cost?: number | null;\n };\n \n // Legacy format fields\n content?: string;\n chatId?: string;\n message?: {\n role: string;\n content: string;\n };\n done?: boolean;\n \n // AI SDK Protocol: Source document event\n sourceId?: string;\n mediaType?: string;\n title?: string; // Filename for source-document events\n \n // AI SDK Protocol: Message metadata\n messageId?: string;\n \n // AI SDK Protocol: Finish event\n finishReason?: string;\n \n // AI SDK Protocol: Error event\n errorText?: string;\n}\n\n/**\n * CuadraChatClient - API client for Cuadra AI REST API\n * Handles authentication, HTTP requests, and SSE streaming\n */\nexport class CuadraChatClient {\n private baseUrl: string;\n private sessionToken: string | null = null;\n private isProxyMode: boolean = false;\n\n constructor(baseUrl: string, sessionToken?: string, isProxyMode: boolean = false) {\n this.baseUrl = baseUrl.replace(/\\/$/, ''); // Remove trailing slash\n this.isProxyMode = isProxyMode;\n if (sessionToken) {\n this.sessionToken = sessionToken;\n }\n }\n\n /**\n * Set or update the session token\n */\n setSessionToken(token: string | null): void {\n this.sessionToken = token;\n }\n\n /**\n * Create or continue a chat with streaming support\n * Returns a ReadableStream for SSE parsing\n */\n private getUrl(endpoint: string): string {\n if (this.isProxyMode) {\n // Proxy mode: remove /v1 prefix\n const path = `${this.baseUrl}${endpoint.replace('/v1', '')}`;\n // If baseUrl is relative, return as-is; otherwise construct full URL\n if (path.startsWith('/') || path.startsWith('./')) {\n return path;\n }\n return path;\n }\n return `${this.baseUrl}${endpoint}`;\n }\n\n async createOrContinueChat(\n request: ChatRequest,\n abortSignal?: AbortSignal,\n ): Promise<ReadableStream<Uint8Array>> {\n const url = this.getUrl('/v1/chats');\n \n // Generate idempotency key for create operation\n const idempotencyKey = this.generateIdempotencyKey();\n \n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n 'Accept': 'text/event-stream', // Tell server we expect SSE streaming\n 'Accept-Encoding': 'identity', // Disable compression to prevent buffering\n 'Cache-Control': 'no-cache', // Prevent caching of streaming response\n 'X-Accel-Buffering': 'no', // Disable nginx buffering if behind reverse proxy\n 'Idempotency-Key': idempotencyKey,\n };\n\n if (this.sessionToken) {\n headers['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n // Ensure stream is true for streaming\n const body: ChatRequest = {\n ...request,\n stream: true,\n };\n\n const response = await fetch(url, {\n method: 'POST',\n headers,\n body: JSON.stringify(body),\n signal: abortSignal,\n cache: 'no-store',\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n if (!response.body) {\n throw new Error('Response body is null');\n }\n\n return response.body;\n }\n\n /**\n * Get a specific chat by ID\n */\n async getChat(chatId: string): Promise<ChatOut> {\n const url = this.getUrl(`/v1/chats/${chatId}`);\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n };\n\n if (this.sessionToken) {\n headers['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url, {\n method: 'GET',\n headers,\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n return response.json();\n }\n\n /**\n * List chats with pagination\n */\n async listChats(params?: ListChatsParams): Promise<ChatSummaryPage> {\n const baseUrl = this.getUrl('/v1/chats');\n // Handle relative URLs (for proxy mode)\n const url = baseUrl.startsWith('/') || baseUrl.startsWith('./')\n ? new URL(baseUrl, window.location.origin)\n : new URL(baseUrl);\n \n if (params?.limit) {\n url.searchParams.set('limit', params.limit.toString());\n }\n if (params?.cursor) {\n url.searchParams.set('cursor', params.cursor);\n }\n if (params?.['expand[]']) {\n params['expand[]'].forEach((expand) => {\n url.searchParams.append('expand[]', expand);\n });\n }\n\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n };\n\n if (this.sessionToken) {\n headers['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url.toString(), {\n method: 'GET',\n headers,\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n return response.json();\n }\n\n /**\n * Generate a cryptographically secure UUID v4 idempotency key\n * Uses the shared secure UUID generator from sanitize utils\n */\n private generateIdempotencyKey(): string {\n return generateSecureUUID();\n }\n\n /**\n * Update a chat\n */\n async updateChat(chatId: string, update: ChatUpdate): Promise<ChatOut> {\n const endpoint = `/v1/chats/${chatId}`;\n const baseUrl = this.getUrl(endpoint);\n \n // Handle relative URLs (for proxy mode)\n const url = baseUrl.startsWith('/') || baseUrl.startsWith('./')\n ? new URL(baseUrl, window.location.origin)\n : new URL(baseUrl);\n \n // Generate idempotency key for update operation\n const idempotencyKey = this.generateIdempotencyKey();\n \n const headers: HeadersInit = {\n 'Content-Type': 'application/json',\n 'Idempotency-Key': idempotencyKey,\n };\n\n if (this.sessionToken) {\n (headers as Record<string, string>)['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url.toString(), {\n method: 'PATCH',\n headers,\n body: JSON.stringify(update),\n });\n\n if (!response.ok) {\n const errorText = await response.text();\n let errorData: { detail?: string; message?: string } = {};\n try {\n errorData = JSON.parse(errorText) as { detail?: string; message?: string };\n } catch {\n errorData = { detail: errorText };\n }\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n // Handle empty responses (e.g., 204 No Content)\n if (response.status === 204) {\n return {} as ChatOut;\n }\n\n const text = await response.text();\n if (!text) {\n return {} as ChatOut;\n }\n\n try {\n return JSON.parse(text);\n } catch {\n return {} as ChatOut;\n }\n }\n\n /**\n * Delete a chat\n */\n async deleteChat(chatId: string): Promise<void> {\n const endpoint = `/v1/chats/${chatId}`;\n const baseUrl = this.getUrl(endpoint);\n \n // Handle relative URLs (for proxy mode)\n const url = baseUrl.startsWith('/') || baseUrl.startsWith('./')\n ? new URL(baseUrl, window.location.origin)\n : new URL(baseUrl);\n \n // Generate idempotency key for delete operation\n const idempotencyKey = this.generateIdempotencyKey();\n \n const headers: HeadersInit = {\n 'Content-Type': 'application/json',\n 'Idempotency-Key': idempotencyKey,\n };\n\n if (this.sessionToken) {\n (headers as Record<string, string>)['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url.toString(), {\n method: 'DELETE',\n headers,\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n }\n\n /**\n * Generate an AI-powered title for a chat.\n * Uses a fast model on the backend to analyze the conversation.\n */\n async generateChatTitle(\n chatId: string,\n request?: GenerateTitleRequest,\n ): Promise<GenerateTitleResponse> {\n const endpoint = `/v1/chats/${chatId}/title`;\n const baseUrl = this.getUrl(endpoint);\n\n const url = baseUrl.startsWith('/') || baseUrl.startsWith('./')\n ? new URL(baseUrl, window.location.origin)\n : new URL(baseUrl);\n\n const idempotencyKey = this.generateIdempotencyKey();\n\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n 'Idempotency-Key': idempotencyKey,\n };\n\n if (this.sessionToken) {\n headers['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url.toString(), {\n method: 'POST',\n headers,\n body: request ? JSON.stringify(request) : undefined,\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n return response.json();\n }\n\n /**\n * List available models\n */\n async listModels(params?: ListModelsParams): Promise<ModelPage> {\n const baseUrl = this.getUrl('/v1/models');\n // Handle relative URLs (for proxy mode)\n const url = baseUrl.startsWith('/') || baseUrl.startsWith('./')\n ? new URL(baseUrl, window.location.origin)\n : new URL(baseUrl);\n \n if (params?.limit) {\n url.searchParams.set('limit', params.limit.toString());\n }\n if (params?.cursor) {\n url.searchParams.set('cursor', params.cursor);\n }\n if (params?.['expand[]']) {\n params['expand[]'].forEach((expand) => {\n url.searchParams.append('expand[]', expand);\n });\n }\n\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n };\n\n if (this.sessionToken) {\n headers['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url.toString(), {\n method: 'GET',\n headers,\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n return response.json();\n }\n\n /**\n * Get a specific model by ID\n */\n async getModel(modelId: string): Promise<ModelOut> {\n const url = this.getUrl(`/v1/models/${modelId}`);\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n };\n\n if (this.sessionToken) {\n headers['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url, {\n method: 'GET',\n headers,\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n return response.json();\n }\n\n /**\n * Upload a file to Cuadra API\n * @param file - The file to upload\n * @param resourceType - Optional resource type (e.g., 'chat')\n * @param resourceId - Optional resource ID to associate with\n * @returns Promise resolving to file metadata\n */\n async uploadFile(\n file: File,\n resourceType?: string,\n resourceId?: string,\n ): Promise<{ id: string; name: string; [key: string]: unknown }> {\n const url = this.getUrl('/v1/files');\n \n // Generate idempotency key\n const idempotencyKey = this.generateIdempotencyKey();\n \n const formData = new FormData();\n formData.append('file', file);\n \n if (resourceType) {\n formData.append('resource_type', resourceType);\n }\n if (resourceId) {\n formData.append('resource_id', resourceId);\n }\n\n const headers: Record<string, string> = {\n 'Idempotency-Key': idempotencyKey,\n };\n\n if (this.sessionToken) {\n headers['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url, {\n method: 'POST',\n headers,\n body: formData,\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n return response.json();\n }\n}\n\n/**\n * Parse SSE stream into chunks\n */\nexport async function* parseSSEStream(\n stream: ReadableStream<Uint8Array>,\n abortSignal?: AbortSignal,\n): AsyncGenerator<SSEChunk> {\n const reader = stream.getReader();\n const decoder = new TextDecoder();\n let buffer = '';\n\n try {\n while (true) {\n if (abortSignal?.aborted) {\n reader.cancel();\n break;\n }\n\n const { done, value } = await reader.read();\n if (done) break;\n\n buffer += decoder.decode(value, { stream: true });\n const lines = buffer.split('\\n\\n');\n buffer = lines.pop() || '';\n\n for (const line of lines) {\n if (line.startsWith('data: ')) {\n const data = line.slice(6);\n if (data.trim() === '[DONE]') {\n yield { done: true };\n continue;\n }\n try {\n const parsed = JSON.parse(data);\n yield parsed;\n } catch {\n // Skip invalid JSON\n }\n }\n }\n }\n } finally {\n reader.releaseLock();\n }\n}\n\n","import type { ThreadMessage } from '@assistant-ui/react';\nimport type { MessageCreate } from '../types/cuadra';\n\n/**\n * Convert assistant-ui ThreadMessage to Cuadra API MessageCreate format\n * Note: Attachments are included in the message but may need to be uploaded separately\n * depending on API requirements\n */\nexport function convertToCuadraMessages(\n messages: readonly ThreadMessage[],\n): MessageCreate[] {\n return messages.map((msg) => {\n // Extract text content from assistant-ui message format\n let content = '';\n\n if (typeof msg.content === 'string') {\n content = msg.content;\n } else if (Array.isArray(msg.content)) {\n // Find text content in content array\n const textContent = msg.content.find((c) => c.type === 'text');\n if (\n textContent &&\n typeof textContent === 'object' &&\n 'text' in textContent\n ) {\n content = textContent.text as string;\n }\n }\n\n // Note: Attachments are available in msg.attachments\n // For now, we'll include them in the message structure\n // The actual file upload to Cuadra API should be handled separately\n // if the API requires file uploads before message creation\n \n return {\n role: msg.role as 'system' | 'user' | 'assistant',\n content,\n // Attachments will be handled separately - files are available in msg.attachments\n };\n });\n}\n\n/**\n * Convert Cuadra API MessageOut to assistant-ui ThreadMessage format\n */\nexport function convertFromCuadraMessage(\n msg: {\n role: string;\n content: string;\n id: string;\n createdAt: string;\n },\n): ThreadMessage {\n return {\n id: msg.id,\n role: msg.role as 'user' | 'assistant' | 'system',\n content: [\n {\n type: 'text' as const,\n text: msg.content,\n },\n ],\n createdAt: new Date(msg.createdAt),\n metadata: {\n custom: {},\n },\n ...(msg.role === 'assistant' && {\n status: {\n type: 'complete' as const,\n reason: 'stop' as const,\n },\n metadata: {\n unstable_state: null,\n unstable_annotations: [],\n unstable_data: [],\n steps: [],\n custom: {},\n },\n }),\n ...(msg.role === 'user' && {\n attachments: [],\n }),\n } as ThreadMessage;\n}\n\n","export const promiseWithResolvers = function () {\n let resolve;\n let reject;\n const promise = new Promise((res, rej) => {\n resolve = res;\n reject = rej;\n });\n if (!resolve || !reject)\n throw new Error(\"Failed to create promise\");\n return { promise, resolve, reject };\n};\n//# sourceMappingURL=promiseWithResolvers.js.map","import { promiseWithResolvers } from \"../../../utils/promiseWithResolvers.js\";\nexport const createMergeStream = () => {\n const list = [];\n let sealed = false;\n let controller;\n let currentPull;\n const handlePull = (item) => {\n if (!item.promise) {\n // TODO for most streams, we can directly pipeTo to avoid the microTask queue\n // add an option to eagerly pipe the stream to the merge stream\n // ideally, using assistant-stream w sync run method + piping to a sync WritableStream runs in the same microtask\n // this is useful because we often use AssistantStreams internally as a serialization utility, e. g. AssistantTransformStream\n // idea: avoid reader.read() by instead using a WritableStream & if (!hasPendingPull) await waitForPull()?\n item.promise = item.reader\n .read()\n .then(({ done, value }) => {\n item.promise = undefined;\n if (done) {\n list.splice(list.indexOf(item), 1);\n if (sealed && list.length === 0) {\n controller.close();\n }\n }\n else {\n controller.enqueue(value);\n }\n currentPull?.resolve();\n currentPull = undefined;\n })\n .catch((e) => {\n console.error(e);\n list.forEach((item) => {\n item.reader.cancel();\n });\n list.length = 0;\n controller.error(e);\n currentPull?.reject(e);\n currentPull = undefined;\n });\n }\n };\n const readable = new ReadableStream({\n start(c) {\n controller = c;\n },\n pull() {\n currentPull = promiseWithResolvers();\n list.forEach((item) => {\n handlePull(item);\n });\n return currentPull.promise;\n },\n cancel() {\n list.forEach((item) => {\n item.reader.cancel();\n });\n list.length = 0;\n },\n });\n return {\n readable,\n isSealed() {\n return sealed;\n },\n seal() {\n sealed = true;\n if (list.length === 0)\n controller.close();\n },\n addStream(stream) {\n if (sealed)\n throw new Error(\"Cannot add streams after the run callback has settled.\");\n const item = { reader: stream.getReader() };\n list.push(item);\n handlePull(item);\n },\n enqueue(chunk) {\n this.addStream(new ReadableStream({\n start(c) {\n c.enqueue(chunk);\n c.close();\n },\n }));\n },\n };\n};\n// TODO\n// export class SpanContainerMerger {\n// public get isSealed() {\n// return this.mergeStream.isSealed();\n// }\n// public get readable() {\n// return this.mergeStream.readable;\n// }\n// private subAllocator = new Counter();\n// private mergeStream = createMergeStream();\n// constructor() {\n// // id 0 is auto allocated\n// this.subAllocator.up();\n// }\n// add(stream: ReadableStream<AssistantStreamChunk>) {\n// this.mergeStream.addStream(\n// stream.pipeThrough(new SpanParentEncoder(this.subAllocator)),\n// );\n// }\n// enqueue(chunk: AssistantStreamChunk & { parentId: 0 }) {\n// this.mergeStream.addStream(\n// new ReadableStream({\n// start(c) {\n// c.enqueue(chunk);\n// c.close();\n// },\n// }),\n// );\n// }\n// seal() {\n// this.mergeStream.seal();\n// }\n// }\n// export class SpanContainerSplitter {\n// public writable;\n// private isSealed = false;\n// private writers = new Map<\n// number,\n// WritableStreamDefaultWriter<AssistantStreamChunk>\n// >();\n// private closeTasks: Promise<void>[] = [];\n// private allocator = new Counter();\n// private subAllocator = new Counter();\n// constructor() {\n// // id 0 is auto-allocated\n// this.allocator.up();\n// this.writable = new WritableStream({\n// write: (chunk) => {\n// const { type, parentId } = chunk;\n// const writer = this.writers.get(parentId);\n// if (writer === undefined) throw new Error(\"Parent id not found\");\n// writer.write(chunk);\n// if (type === \"span\") {\n// // allocate a new span id\n// this.writers.set(this.allocator.up(), writer);\n// }\n// if (type === \"finish\") {\n// this.writers.delete(parentId);\n// writer.close();\n// if (this.writers.size === 0) {\n// const closeTask = this.writable.close();\n// this.closeTasks.push(closeTask);\n// closeTask.then(() => {\n// this.closeTasks.splice(this.closeTasks.indexOf(closeTask), 1);\n// });\n// }\n// }\n// },\n// close: async () => {\n// if (this.writers.size > 0) throw new Error(\"Not all writers closed\");\n// // await and throw on any errors\n// await Promise.all(this.closeTasks);\n// },\n// });\n// }\n// add(stream: WritableStream<AssistantStreamChunk>) {\n// if (this.isSealed) throw new Error(\"Cannot add streams after sealing\");\n// const decoder = new SpanParentDecoder(this.subAllocator);\n// decoder.readable.pipeTo(stream);\n// this.writers.set(this.allocator.up(), decoder.writable.getWriter());\n// }\n// seal() {\n// this.isSealed = true;\n// if (this.writers.size === 0) this.writable.close();\n// }\n// }\n//# sourceMappingURL=merge.js.map","class TextStreamControllerImpl {\n _controller;\n _isClosed = false;\n constructor(controller) {\n this._controller = controller;\n }\n append(textDelta) {\n this._controller.enqueue({\n type: \"text-delta\",\n path: [],\n textDelta,\n });\n return this;\n }\n close() {\n if (this._isClosed)\n return;\n this._isClosed = true;\n this._controller.enqueue({\n type: \"part-finish\",\n path: [],\n });\n this._controller.close();\n }\n}\nexport const createTextStream = (readable) => {\n return new ReadableStream({\n start(c) {\n return readable.start?.(new TextStreamControllerImpl(c));\n },\n pull(c) {\n return readable.pull?.(new TextStreamControllerImpl(c));\n },\n cancel(c) {\n return readable.cancel?.(c);\n },\n });\n};\nexport const createTextStreamController = () => {\n let controller;\n const stream = createTextStream({\n start(c) {\n controller = c;\n },\n });\n return [stream, controller];\n};\n//# sourceMappingURL=text.js.map","import { createTextStream } from \"./text.js\";\nclass ToolCallStreamControllerImpl {\n _controller;\n _isClosed = false;\n _mergeTask;\n constructor(_controller) {\n this._controller = _controller;\n const stream = createTextStream({\n start: (c) => {\n this._argsTextController = c;\n },\n });\n let hasArgsText = false;\n this._mergeTask = stream.pipeTo(new WritableStream({\n write: (chunk) => {\n switch (chunk.type) {\n case \"text-delta\":\n hasArgsText = true;\n this._controller.enqueue(chunk);\n break;\n case \"part-finish\":\n if (!hasArgsText) {\n // if no argsText was provided, assume empty object\n this._controller.enqueue({\n type: \"text-delta\",\n textDelta: \"{}\",\n path: [],\n });\n }\n this._controller.enqueue({\n type: \"tool-call-args-text-finish\",\n path: [],\n });\n break;\n default:\n throw new Error(`Unexpected chunk type: ${chunk.type}`);\n }\n },\n }));\n }\n get argsText() {\n return this._argsTextController;\n }\n _argsTextController;\n async setResponse(response) {\n this._argsTextController.close();\n await Promise.resolve(); // flush microtask queue\n // TODO switch argsTextController to be something that doesn'#t require this\n this._controller.enqueue({\n type: \"result\",\n path: [],\n ...(response.artifact !== undefined\n ? { artifact: response.artifact }\n : {}),\n result: response.result,\n isError: response.isError ?? false,\n });\n }\n async close() {\n if (this._isClosed)\n return;\n this._isClosed = true;\n this._argsTextController.close();\n await this._mergeTask;\n this._controller.enqueue({\n type: \"part-finish\",\n path: [],\n });\n this._controller.close();\n }\n}\nexport const createToolCallStream = (readable) => {\n return new ReadableStream({\n start(c) {\n return readable.start?.(new ToolCallStreamControllerImpl(c));\n },\n pull(c) {\n return readable.pull?.(new ToolCallStreamControllerImpl(c));\n },\n cancel(c) {\n return readable.cancel?.(c);\n },\n });\n};\nexport const createToolCallStreamController = () => {\n let controller;\n const stream = createToolCallStream({\n start(c) {\n controller = c;\n },\n });\n return [stream, controller];\n};\n//# sourceMappingURL=tool-call.js.map","export class Counter {\n value = -1;\n up() {\n return ++this.value;\n }\n}\n//# sourceMappingURL=Counter.js.map","import { Counter } from \"../Counter.js\";\nexport class PathAppendEncoder extends TransformStream {\n constructor(idx) {\n super({\n transform(chunk, controller) {\n controller.enqueue({\n ...chunk,\n path: [idx, ...chunk.path],\n });\n },\n });\n }\n}\nexport class PathAppendDecoder extends TransformStream {\n constructor(idx) {\n super({\n transform(chunk, controller) {\n const { path: [idx2, ...path], } = chunk;\n if (idx !== idx2)\n throw new Error(`Path mismatch: expected ${idx}, got ${idx2}`);\n controller.enqueue({\n ...chunk,\n path,\n });\n },\n });\n }\n}\nexport class PathMergeEncoder extends TransformStream {\n constructor(counter) {\n const innerCounter = new Counter();\n const mapping = new Map();\n super({\n transform(chunk, controller) {\n if (chunk.type === \"part-start\" && chunk.path.length === 0) {\n mapping.set(innerCounter.up(), counter.up());\n }\n const [idx, ...path] = chunk.path;\n if (idx === undefined) {\n controller.enqueue(chunk);\n return;\n }\n const mappedIdx = mapping.get(idx);\n if (mappedIdx === undefined)\n throw new Error(\"Path not found\");\n controller.enqueue({\n ...chunk,\n path: [mappedIdx, ...path],\n });\n },\n });\n }\n}\n//# sourceMappingURL=path-utils.js.map","/* @ts-self-types=\"./index.d.ts\" */\nlet urlAlphabet =\n 'useandom-26T198340PX75pxJACKVERYMINDBUSHWOLF_GQZbfghjklqvwyzrict'\nexport let customAlphabet = (alphabet, defaultSize = 21) => {\n return (size = defaultSize) => {\n let id = ''\n let i = size | 0\n while (i--) {\n id += alphabet[(Math.random() * alphabet.length) | 0]\n }\n return id\n }\n}\nexport let nanoid = (size = 21) => {\n let id = ''\n let i = size | 0\n while (i--) {\n id += urlAlphabet[(Math.random() * 64) | 0]\n }\n return id\n}\n","import { customAlphabet } from \"nanoid/non-secure\";\nexport const generateId = customAlphabet(\"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\", 7);\n//# sourceMappingURL=generateId.js.map","import { AssistantStream } from \"../AssistantStream.js\";\nimport { createMergeStream } from \"../utils/stream/merge.js\";\nimport { createTextStreamController } from \"./text.js\";\nimport { createToolCallStreamController, } from \"./tool-call.js\";\nimport { Counter } from \"../utils/Counter.js\";\nimport { PathAppendEncoder, PathMergeEncoder, } from \"../utils/stream/path-utils.js\";\nimport { DataStreamEncoder } from \"../serialization/data-stream/DataStream.js\";\nimport { generateId } from \"../utils/generateId.js\";\nimport { promiseWithResolvers } from \"../../utils/promiseWithResolvers.js\";\nclass AssistantStreamControllerImpl {\n _state;\n _parentId;\n constructor(state) {\n this._state = state || {\n merger: createMergeStream(),\n contentCounter: new Counter(),\n };\n }\n get __internal_isClosed() {\n return this._state.merger.isSealed();\n }\n __internal_getReadable() {\n return this._state.merger.readable;\n }\n __internal_subscribeToClose(callback) {\n this._state.closeSubscriber = callback;\n }\n _addPart(part, stream) {\n if (this._state.append) {\n this._state.append.controller.close();\n this._state.append = undefined;\n }\n this.enqueue({\n type: \"part-start\",\n part,\n path: [],\n });\n this._state.merger.addStream(stream.pipeThrough(new PathAppendEncoder(this._state.contentCounter.value)));\n }\n merge(stream) {\n this._state.merger.addStream(stream.pipeThrough(new PathMergeEncoder(this._state.contentCounter)));\n }\n appendText(textDelta) {\n if (this._state.append?.kind !== \"text\") {\n this._state.append = {\n kind: \"text\",\n controller: this.addTextPart(),\n };\n }\n this._state.append.controller.append(textDelta);\n }\n appendReasoning(textDelta) {\n if (this._state.append?.kind !== \"reasoning\") {\n this._state.append = {\n kind: \"reasoning\",\n controller: this.addReasoningPart(),\n };\n }\n this._state.append.controller.append(textDelta);\n }\n addTextPart() {\n const [stream, controller] = createTextStreamController();\n this._addPart({ type: \"text\" }, stream);\n return controller;\n }\n addReasoningPart() {\n const [stream, controller] = createTextStreamController();\n this._addPart({ type: \"reasoning\" }, stream);\n return controller;\n }\n addToolCallPart(options) {\n const opt = typeof options === \"string\" ? { toolName: options } : options;\n const toolName = opt.toolName;\n const toolCallId = opt.toolCallId ?? generateId();\n const [stream, controller] = createToolCallStreamController();\n this._addPart({\n type: \"tool-call\",\n toolName,\n toolCallId,\n ...(this._parentId && { parentId: this._parentId }),\n }, stream);\n if (opt.argsText !== undefined) {\n controller.argsText.append(opt.argsText);\n controller.argsText.close();\n }\n if (opt.args !== undefined) {\n controller.argsText.append(JSON.stringify(opt.args));\n controller.argsText.close();\n }\n if (opt.response !== undefined) {\n controller.setResponse(opt.response);\n }\n return controller;\n }\n appendSource(options) {\n this._addPart({ ...options, ...(this._parentId && { parentId: this._parentId }) }, new ReadableStream({\n start(controller) {\n controller.enqueue({\n type: \"part-finish\",\n path: [],\n });\n controller.close();\n },\n }));\n }\n appendFile(options) {\n this._addPart(options, new ReadableStream({\n start(controller) {\n controller.enqueue({\n type: \"part-finish\",\n path: [],\n });\n controller.close();\n },\n }));\n }\n enqueue(chunk) {\n this._state.merger.enqueue(chunk);\n if (chunk.type === \"part-start\" && chunk.path.length === 0) {\n this._state.contentCounter.up();\n }\n }\n withParentId(parentId) {\n const controller = new AssistantStreamControllerImpl(this._state);\n controller._parentId = parentId;\n return controller;\n }\n close() {\n this._state.append?.controller?.close();\n this._state.merger.seal();\n this._state.closeSubscriber?.();\n }\n}\nexport function createAssistantStream(callback) {\n const controller = new AssistantStreamControllerImpl();\n const runTask = async () => {\n try {\n await callback(controller);\n }\n catch (e) {\n if (!controller.__internal_isClosed) {\n controller.enqueue({\n type: \"error\",\n path: [],\n error: String(e),\n });\n }\n throw e;\n }\n finally {\n if (!controller.__internal_isClosed) {\n controller.close();\n }\n }\n };\n runTask();\n return controller.__internal_getReadable();\n}\nexport function createAssistantStreamController() {\n const { resolve, promise } = promiseWithResolvers();\n let controller;\n const stream = createAssistantStream((c) => {\n controller = c;\n controller.__internal_subscribeToClose(resolve);\n return promise;\n });\n return [stream, controller];\n}\nexport function createAssistantStreamResponse(callback) {\n return AssistantStream.toResponse(createAssistantStream(callback), new DataStreamEncoder());\n}\n//# sourceMappingURL=assistant-stream.js.map","import {\n type unstable_RemoteThreadListAdapter as RemoteThreadListAdapter,\n RuntimeAdapterProvider,\n type ThreadHistoryAdapter,\n type ThreadMessage,\n useThreadListItem,\n} from '@assistant-ui/react';\nimport { createAssistantStream } from 'assistant-stream';\nimport React from 'react';\nimport { CuadraChatClient } from '../lib/cuadraChatClient';\nimport { convertFromCuadraMessage } from './messageConverter';\n\n/**\n * Store mapping of local thread IDs to server chat IDs\n * This is used to update remoteId when server chat ID is received\n */\nexport const localToRemoteIdMap = new Map<string, string>();\n\n/**\n * Store mapping of message IDs to their chat (remote) IDs.\n * Populated during history loading so the chat model adapter\n * can resolve which chat a conversation belongs to.\n */\nexport const messageIdToChatIdMap = new Map<string, string>();\n\n/**\n * The remoteId of the thread that is currently active (rendering).\n * Set by the unstable_Provider so the chat model adapter can map\n * server chat IDs back to local thread IDs without a reverse lookup.\n */\nexport let activeThreadRemoteId: string | undefined;\n\n/**\n * Resolve a remoteId (which may be a __LOCALID_*) to the actual server chat ID.\n * Uses multiple resolution strategies in order of reliability.\n */\nfunction resolveServerId(\n remoteId: string,\n messages?: readonly ThreadMessage[],\n): string {\n // 1. Direct lookup in localToRemoteIdMap\n const mapped = localToRemoteIdMap.get(remoteId);\n if (mapped) return mapped;\n\n // 2. Scan messages for a registered chatId (populated by the chat model\n // adapter when the SSE stream returns a chat ID)\n if (messages) {\n for (const msg of messages) {\n const chatId = messageIdToChatIdMap.get(msg.id);\n if (chatId) {\n // Back-fill so rename/delete/fetch also benefit\n localToRemoteIdMap.set(remoteId, chatId);\n return chatId;\n }\n }\n }\n\n // 3. Fall back to the remoteId as-is (valid for pre-existing chats\n // whose remoteId is already the server chat ID)\n return remoteId;\n}\n\nexport function createThreadListAdapter(\n client: CuadraChatClient,\n onChatIdReceived?: (localThreadId: string, serverChatId: string) => void,\n): RemoteThreadListAdapter & {\n updateRemoteId: (localThreadId: string, serverChatId: string) => void;\n} {\n return {\n async list() {\n const all = [];\n const limit = 50;\n let cursor: string | undefined = undefined;\n let safety = 0;\n\n // Fetch all chats with pagination\n do {\n const response = await client.listChats({ limit, cursor });\n const items = response.items || [];\n\n if (Array.isArray(items) && items.length > 0) {\n all.push(...items);\n }\n\n cursor = response.nextCursor || undefined;\n safety++;\n } while (cursor && safety < 50);\n\n // Map to assistant-ui thread format\n // Filter out deleted chats and chats without IDs\n const threads = all\n .map((chat) => {\n const id = chat.id;\n // Skip if no ID or if deleted\n if (!id || chat.deletedAt) return null;\n\n return {\n status: 'regular' as const,\n remoteId: id,\n title: chat.title || 'New Chat',\n createdAt: chat.createdAt ? new Date(chat.createdAt) : new Date(),\n updatedAt: chat.updatedAt ? new Date(chat.updatedAt) : new Date(),\n };\n })\n .filter((t) => t !== null) as Array<{\n status: 'regular' | 'archived';\n remoteId: string;\n title?: string;\n createdAt: Date;\n updatedAt: Date;\n }>;\n\n return { threads };\n },\n\n async initialize(threadId: string) {\n // For LocalRuntime, threadId is already generated locally (e.g., __LOCALID_...)\n // Check if we have a server chat ID for this local thread ID\n const serverChatId = localToRemoteIdMap.get(threadId);\n const remoteId = serverChatId || threadId;\n\n return { remoteId, externalId: undefined };\n },\n\n async rename(remoteId: string, newTitle: string) {\n const actualId = resolveServerId(remoteId);\n await client.updateChat(actualId, { title: newTitle });\n },\n\n async archive(_remoteId: string) {\n // Cuadra API uses soft delete, which we can treat as archive\n // If API adds explicit archive support, implement here\n },\n\n async unarchive(_remoteId: string) {\n // If Cuadra supports unarchive, implement here\n // Currently not supported\n },\n\n async delete(remoteId: string) {\n const actualId = resolveServerId(remoteId);\n await client.deleteChat(actualId);\n },\n\n async generateTitle(remoteId: string, _messages: readonly ThreadMessage[]) {\n const actualId = resolveServerId(remoteId, _messages);\n\n // Call the backend title generation endpoint.\n // The API uses a fast model to analyze the conversation and returns a title.\n let title: string | undefined;\n try {\n const result = await client.generateChatTitle(actualId);\n title = result.title;\n } catch (error) {\n // eslint-disable-next-line no-console -- Intentional: surface title-generation failures for debugging\n console.error('[generateTitle] API call failed, falling back to first message', error);\n // Fallback: extract from first user message if the endpoint fails\n const firstUserMsg = _messages.find((m) => m.role === 'user');\n const rawText = firstUserMsg?.content\n ?.filter((p): p is { type: 'text'; text: string } => p.type === 'text')\n .map((p) => p.text)\n .join(' ')\n .trim();\n title = rawText\n ? rawText.length > 60\n ? rawText.slice(0, 57) + '...'\n : rawText\n : undefined;\n }\n\n return createAssistantStream((controller) => {\n const text = controller.addTextPart();\n if (title) {\n text.append(title);\n }\n text.close();\n controller.close();\n });\n },\n\n async fetch(threadId: string) {\n try {\n const chat = await client.getChat(threadId);\n const id = chat.id || threadId;\n\n return {\n status: 'regular' as const,\n remoteId: id,\n title: chat.title || 'Chat',\n createdAt: chat.createdAt ? new Date(chat.createdAt) : new Date(),\n updatedAt: chat.updatedAt ? new Date(chat.updatedAt) : new Date(),\n };\n } catch {\n // Return default if fetch fails\n return {\n status: 'regular' as const,\n remoteId: threadId,\n createdAt: new Date(),\n updatedAt: new Date(),\n };\n }\n },\n\n // Add a method to update the remoteId mapping when chat ID is received\n updateRemoteId(localThreadId: string, serverChatId: string) {\n localToRemoteIdMap.set(localThreadId, serverChatId);\n onChatIdReceived?.(localThreadId, serverChatId);\n },\n\n // Provider component that adds thread-specific history adapter\n unstable_Provider: ({ children }: { children?: React.ReactNode }) => {\n const threadListItem = useThreadListItem();\n const remoteId = threadListItem?.remoteId;\n const isMain = threadListItem?.isMain;\n\n // Keep activeThreadRemoteId in sync so the chat model adapter\n // can look it up when a server chat ID arrives during streaming.\n // Only the active (main) thread sets the global — multiple Provider\n // instances are mounted simultaneously (one per thread).\n // We intentionally allow clearing to `undefined` when a new thread\n // is created (remoteId is undefined for local-only threads).\n React.useEffect(() => {\n if (isMain) {\n activeThreadRemoteId = remoteId;\n }\n }, [remoteId, isMain]);\n\n // Create thread-specific history adapter\n const history = React.useMemo<ThreadHistoryAdapter>(\n () => ({\n async load() {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n if (!remoteId) return { messages: [] } as any;\n\n try {\n // Fetch chat with messages expanded\n const chat = await client.getChat(remoteId);\n \n if (!chat.messages || chat.messages.length === 0) {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n return { messages: [] } as any;\n }\n\n // Convert API messages to ExportedMessageRepository format\n // Each item needs { message, parentId } where parentId links to the previous message\n const converted = chat.messages.map((msg) => convertFromCuadraMessage(msg));\n const messages = converted.map((message, idx) => ({\n message,\n parentId: idx > 0 ? converted[idx - 1]!.id : null,\n }));\n\n // Register every message ID → chatId so the model adapter\n // can resolve which chat this thread belongs to\n for (const msg of converted) {\n messageIdToChatIdMap.set(msg.id, remoteId);\n }\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n return { messages } as any;\n } catch (_error) {\n // Failed to load messages for thread\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n return { messages: [] } as any;\n }\n },\n\n async append(_message) {\n // Messages are automatically persisted by the API when sent\n // This method is called by the runtime but we don't need to do anything\n // since messages are already saved via the chat API\n },\n }),\n [remoteId, client],\n );\n\n const adapters = React.useMemo(() => ({ history }), [history]);\n\n return (\n <RuntimeAdapterProvider adapters={adapters}>\n {children}\n </RuntimeAdapterProvider>\n );\n },\n };\n}\n\n","import type { Source } from '../types/cuadra';\n\n/**\n * Metadata stored per message\n * Keyed by message/chat ID to persist across the conversation\n */\nexport interface MessageMetadata {\n /** RAG sources/citations used to generate the response */\n sources?: Source[];\n}\n\n/**\n * Store for per-message metadata (sources)\n * Keyed by message ID so each message retains its own sources\n * Reasoning is handled via native assistant-ui content parts\n */\nclass StreamingMetadataStore {\n /** Per-message metadata, keyed by message ID */\n private messageMetadata: Map<string, MessageMetadata> = new Map();\n /** Current streaming message ID */\n private currentMessageId: string | null = null;\n private listeners: Set<() => void> = new Set();\n\n /**\n * Set the current streaming message ID\n */\n setCurrentMessage(messageId: string): void {\n this.currentMessageId = messageId;\n }\n\n /**\n * Update sources for the current message\n */\n update(data: Partial<MessageMetadata>): void {\n if (!this.currentMessageId) return;\n \n const existing = this.messageMetadata.get(this.currentMessageId) || {};\n const newSources = data.sources;\n \n // Only update if sources actually changed\n if (newSources && newSources.length > 0) {\n this.messageMetadata.set(this.currentMessageId, {\n ...existing,\n sources: newSources,\n });\n this.notify();\n }\n }\n\n /**\n * Get metadata for the current streaming message\n */\n get(): MessageMetadata {\n if (!this.currentMessageId) return {};\n return this.messageMetadata.get(this.currentMessageId) || {};\n }\n\n /**\n * Get metadata for a specific message ID\n */\n getForMessage(messageId: string): MessageMetadata {\n return this.messageMetadata.get(messageId) || {};\n }\n\n /**\n * Get the current message ID\n */\n getCurrentMessageId(): string | null {\n return this.currentMessageId;\n }\n\n /**\n * Clear current message (but keep historical metadata)\n * Called when starting a new message stream\n */\n clear(): void {\n this.currentMessageId = null;\n // Don't clear messageMetadata - keep historical sources!\n }\n\n /**\n * Subscribe to changes\n */\n subscribe(listener: () => void): () => void {\n this.listeners.add(listener);\n return () => this.listeners.delete(listener);\n }\n\n private notify(): void {\n this.listeners.forEach(listener => listener());\n }\n}\n\n// Singleton instance\nexport const streamingMetadataStore = new StreamingMetadataStore();\n\n// Re-export the old type name for compatibility\nexport type StreamingMetadata = MessageMetadata;\n\n","import type { ChatModelAdapter, ChatModelRunOptions } from '@assistant-ui/react';\nimport { CuadraChatClient, parseSSEStream, type SSEChunk } from '../lib/cuadraChatClient';\nimport { convertToCuadraMessages } from './messageConverter';\nimport { activeThreadRemoteId, localToRemoteIdMap, messageIdToChatIdMap } from './threadListAdapter';\nimport { streamingMetadataStore } from './streamingMetadataStore';\nimport type { Source } from '../types/cuadra';\n\n// Re-export for convenience\nexport type { StreamingMetadata } from './streamingMetadataStore';\nexport { streamingMetadataStore } from './streamingMetadataStore';\n\n// Chat ID resolution uses a single strategy: messageIdToChatIdMap.\n// Thread identification uses unstable_threadId from run() options (v0.12+),\n// falling back to activeThreadRemoteId global for compatibility.\n\n/**\n * Queue for pre-made responses to intercept API calls\n * When a pre-made suggestion is clicked, the response is queued here\n * The adapter will return this response instead of calling the real API\n */\ninterface PreMadeResponse {\n question: string;\n response: string;\n streamingSpeed?: number;\n initialDelay?: number;\n}\n\nconst preMadeResponseQueue: PreMadeResponse[] = [];\n\n/**\n * Store for pre-made conversation history\n * These messages will be included in subsequent API calls for context\n */\ninterface PreMadeMessage {\n role: 'user' | 'assistant';\n content: string;\n}\n\nlet preMadeConversationHistory: PreMadeMessage[] = [];\n\n/**\n * Queue a pre-made response to be returned instead of calling the API\n * Call this before triggering the normal send flow\n * @param question - The user's question\n * @param response - The text to return as the assistant's response\n * @param streamingSpeed - Speed in ms between words (default: 50)\n * @param initialDelay - Delay in ms before streaming starts (default: 800)\n */\nexport function queuePreMadeResponse(question: string, response: string, streamingSpeed: number = 50, initialDelay: number = 800): void {\n preMadeResponseQueue.push({ question, response, streamingSpeed, initialDelay });\n}\n\n/**\n * Check if there's a pending pre-made response\n */\nexport function hasPendingPreMadeResponse(): boolean {\n return preMadeResponseQueue.length > 0;\n}\n\n/**\n * Clear all pending pre-made responses and conversation history\n */\nexport function clearPreMadeResponses(): void {\n preMadeResponseQueue.length = 0;\n preMadeConversationHistory = [];\n}\n\n/**\n * Get the pre-made conversation history for context\n */\nexport function getPreMadeConversationHistory(): PreMadeMessage[] {\n return preMadeConversationHistory;\n}\n\nexport interface ChatModelAdapterOptions {\n modelId?: string | null; // Optional - backend can resolve model ID in proxy mode\n /**\n * Lazy getter for modelId. When provided, called at request time instead of\n * using the static `modelId`. Useful to avoid recreating the adapter (and\n * rebuilding the runtime) every time the selected model changes.\n */\n getModelId?: () => string | null | undefined;\n systemPrompt?: string;\n ephemeral?: boolean;\n /** Enable reasoning/thinking output for supported models (e.g., o1, o3) */\n enableReasoning?: boolean;\n onChatCreated?: (chatId: string) => void;\n onUserMessage?: () => void; // Called when a user message is sent\n onThreadIdUpdate?: (localThreadId: string, serverChatId: string) => void; // Called when chat ID is received to update thread mapping\n /** Interceptor called before each API request. Return true to allow, false to block. */\n onBeforeRequest?: () => boolean | Promise<boolean>;\n /** \n * Mock response to show when onBeforeRequest returns false.\n * If set, instead of throwing an error, this message will be streamed as a response.\n * Can be a string or a function that returns a string (called each time a request is blocked).\n * Useful for showing helpful messages like \"Complete setup before chatting\".\n */\n mockResponseOnBlocked?: string | (() => string);\n}\n\n/**\n * Creates a ChatModelAdapter for LocalRuntime that connects to Cuadra API\n * Handles streaming responses and converts between assistant-ui and API formats\n */\nexport function createChatModelAdapter(\n client: CuadraChatClient,\n options: ChatModelAdapterOptions,\n): ChatModelAdapter {\n const { modelId: staticModelId, getModelId, systemPrompt, ephemeral, enableReasoning, onChatCreated, onUserMessage, onThreadIdUpdate, onBeforeRequest, mockResponseOnBlocked } = options;\n const resolveModelId = () => (getModelId ? getModelId() : staticModelId) ?? null;\n \n return {\n async *run(options: ChatModelRunOptions) {\n const { messages, abortSignal } = options;\n // unstable_threadId is the remoteId of the current thread, provided\n // directly by the runtime (available since @assistant-ui/react 0.12).\n // For brand-new threads it may be undefined at run start.\n const threadId = (options as { unstable_threadId?: string }).unstable_threadId;\n \n // Check if there's a pending pre-made response\n const pendingPreMade = preMadeResponseQueue.shift();\n if (pendingPreMade) {\n // Simulate the API response with streaming\n onUserMessage?.();\n \n // Initial \"thinking\" delay before streaming starts\n const initialDelay = pendingPreMade.initialDelay ?? 800;\n if (initialDelay > 0) {\n await new Promise(resolve => setTimeout(resolve, initialDelay));\n }\n \n // Check if aborted during delay\n if (abortSignal?.aborted) {\n throw new Error('Request aborted');\n }\n \n const words = pendingPreMade.response.split(' ');\n let accumulatedText = '';\n const speed = pendingPreMade.streamingSpeed || 50;\n \n // Generate a fake chat ID for ephemeral pre-made chats\n const fakeChatId = `premade-${Date.now()}`;\n \n // If ephemeral, we don't need a real chat ID\n // If not ephemeral, you might want to still create a real chat\n if (ephemeral) {\n onChatCreated?.(fakeChatId);\n }\n \n // Simulate streaming word by word\n for (let i = 0; i < words.length; i++) {\n if (abortSignal?.aborted) {\n throw new Error('Request aborted');\n }\n \n accumulatedText += (i === 0 ? '' : ' ') + words[i];\n \n yield {\n content: [{ type: 'text', text: accumulatedText }],\n };\n \n // Delay between words for streaming effect\n await new Promise(resolve => setTimeout(resolve, speed + Math.random() * (speed * 0.5)));\n }\n \n // Final yield\n yield {\n content: [{ type: 'text', text: accumulatedText }],\n };\n \n // Store the pre-made Q&A in conversation history for context in follow-up calls\n preMadeConversationHistory.push(\n { role: 'user', content: pendingPreMade.question },\n { role: 'assistant', content: pendingPreMade.response }\n );\n \n return;\n }\n \n // Resolve the server chat ID for this conversation.\n // The history loader registers every message ID → chatId in\n // messageIdToChatIdMap. We scan the current messages for a match.\n // For new threads (first message), no match is found and the server\n // creates a new chat.\n let serverChatId: string | undefined;\n\n for (const msg of messages) {\n const mapped = messageIdToChatIdMap.get(msg.id);\n if (mapped) {\n serverChatId = mapped;\n break;\n }\n }\n\n onUserMessage?.();\n\n // Convert assistant-ui ThreadMessage format to API MessageCreate format\n let messagesToSend = messages;\n \n // Find the last assistant message index\n let lastAssistantIndex = -1;\n for (let i = messages.length - 1; i >= 0; i--) {\n if (messages[i].role === 'assistant') {\n lastAssistantIndex = i;\n break;\n }\n }\n \n // If we have a chatId OR we have assistant messages (indicating we're continuing a conversation),\n // only send new messages (messages after the last assistant response)\n // This is because the backend already has the conversation history\n if (serverChatId || lastAssistantIndex >= 0) {\n // If we found an assistant message, only send messages after it (new user messages)\n if (lastAssistantIndex >= 0) {\n messagesToSend = messages.slice(lastAssistantIndex + 1);\n }\n // If we have chatId but no assistant message yet, keep all messages as fallback\n // This ensures we don't lose messages in edge cases\n }\n\n // If we have pre-made conversation history, prepend it to the API messages\n // This provides context for follow-up messages after a pre-made Q&A\n const apiMessages = convertToCuadraMessages(messagesToSend);\n \n if (preMadeConversationHistory.length > 0 && !serverChatId) {\n // Only include pre-made history for the first real API call (no chatId yet)\n // Convert pre-made history directly to API format and prepend\n const preMadeApiMessages = preMadeConversationHistory.map(msg => ({\n role: msg.role as 'user' | 'assistant',\n content: msg.content,\n }));\n \n // Prepend pre-made history to the API messages\n apiMessages.unshift(...preMadeApiMessages);\n \n // Clear the pre-made history after using it (it's now part of the real conversation)\n preMadeConversationHistory = [];\n }\n\n // Upload any file attachments from the last user message\n const fileIds: string[] = [];\n let lastUserMessage: (typeof messagesToSend)[number] | undefined;\n for (let i = messagesToSend.length - 1; i >= 0; i--) {\n if (messagesToSend[i]!.role === 'user') { lastUserMessage = messagesToSend[i]; break; }\n }\n if (lastUserMessage && 'attachments' in lastUserMessage) {\n const attachments = (lastUserMessage as { attachments?: Array<{ file?: File }> }).attachments;\n if (attachments && attachments.length > 0) {\n for (const attachment of attachments) {\n if (attachment.file) {\n try {\n const uploaded = await client.uploadFile(attachment.file);\n fileIds.push(uploaded.id);\n } catch (_uploadError) {\n // File upload is best-effort — don't block the message\n }\n }\n }\n }\n }\n\n // Create request\n // modelId is optional - backend can resolve it in proxy mode\n const request: {\n messages: typeof apiMessages;\n chatId: string | null;\n modelId?: string | null;\n systemPrompt?: string;\n ephemeral?: boolean;\n enableReasoning?: boolean;\n stream: boolean;\n fileIds?: string[];\n } = {\n messages: apiMessages,\n chatId: serverChatId || null, // Use server chatId if available\n systemPrompt,\n ephemeral,\n enableReasoning,\n stream: true,\n };\n \n // Only include modelId if provided (backend may resolve it)\n const currentModelId = resolveModelId();\n if (currentModelId && currentModelId.trim() !== '') {\n request.modelId = currentModelId;\n }\n\n // Include file IDs if any files were uploaded\n if (fileIds.length > 0) {\n request.fileIds = fileIds;\n }\n\n // Call interceptor before making the API request\n if (onBeforeRequest) {\n const shouldProceed = await onBeforeRequest();\n if (!shouldProceed) {\n // Request was blocked by interceptor\n // If mockResponseOnBlocked is set, show it instead of throwing\n if (mockResponseOnBlocked) {\n // Get the response text (call function if it's a function)\n const responseText = typeof mockResponseOnBlocked === 'function' \n ? mockResponseOnBlocked() \n : mockResponseOnBlocked;\n\n // Get the last user message to use as the \"question\"\n const userMessages = messages.filter(m => m.role === 'user');\n const lastUserMessage = userMessages.length > 0 ? userMessages[userMessages.length - 1] : undefined;\n let userQuestion = 'Message';\n if (lastUserMessage) {\n if (typeof lastUserMessage.content === 'string') {\n userQuestion = lastUserMessage.content;\n } else if (Array.isArray(lastUserMessage.content)) {\n const textPart = lastUserMessage.content.find(\n (c: { type: string; text?: string }) => c.type === 'text'\n ) as { type: string; text?: string } | undefined;\n userQuestion = textPart?.text || 'Message';\n }\n }\n \n // Queue the mock response and let the pre-made handler process it\n queuePreMadeResponse(userQuestion, responseText, 40, 400);\n \n // Process the queued response (same as the pre-made response flow above)\n const pendingMock = preMadeResponseQueue.shift();\n if (pendingMock) {\n onUserMessage?.();\n \n // Initial \"thinking\" delay before streaming starts\n const initialDelay = pendingMock.initialDelay ?? 400;\n if (initialDelay > 0) {\n await new Promise(resolve => setTimeout(resolve, initialDelay));\n }\n \n // Check if aborted during delay\n if (abortSignal?.aborted) {\n throw new Error('Request aborted');\n }\n \n const words = pendingMock.response.split(' ');\n let accumulatedText = '';\n const speed = pendingMock.streamingSpeed || 40;\n \n // Generate a fake chat ID for the mock response\n const fakeChatId = `mock-${Date.now()}`;\n if (ephemeral) {\n onChatCreated?.(fakeChatId);\n }\n \n // Simulate streaming word by word\n for (let i = 0; i < words.length; i++) {\n if (abortSignal?.aborted) {\n throw new Error('Request aborted');\n }\n \n accumulatedText += (i === 0 ? '' : ' ') + words[i];\n \n yield {\n content: [{ type: 'text', text: accumulatedText }],\n };\n \n // Delay between words for streaming effect\n await new Promise(resolve => setTimeout(resolve, speed + Math.random() * (speed * 0.3)));\n }\n \n // Final yield\n yield {\n content: [{ type: 'text', text: accumulatedText }],\n };\n \n return;\n }\n }\n \n throw new Error('Request blocked');\n }\n }\n\n // Stream from API\n const stream = await client.createOrContinueChat(request, abortSignal);\n\n let accumulatedText = '';\n let accumulatedReasoning = '';\n let chatId: string | undefined;\n const sources: Source[] = [];\n let lastSourcesCount = 0;\n let hasAddedReasoningToTextDelay = false; // Track if we've added delay between reasoning and text\n\n // Generate a unique ID for this message stream\n const streamMessageId = `msg_${Date.now()}_${Math.random().toString(36).substring(2, 9)}`;\n \n // Set up store for this new message (clears current but keeps history)\n streamingMetadataStore.clear();\n streamingMetadataStore.setCurrentMessage(streamMessageId);\n\n // Helper to update store only when sources change (reasoning now goes through content parts)\n const updateStoreIfNeeded = () => {\n const sourcesChanged = sources.length !== lastSourcesCount;\n \n if (sourcesChanged) {\n lastSourcesCount = sources.length;\n streamingMetadataStore.update({\n sources: sources.length > 0 ? [...sources] : undefined,\n });\n }\n };\n\n // Helper to create yield object with proper content parts\n // Uses native assistant-ui reasoning content parts\n const createYieldObject = () => {\n updateStoreIfNeeded();\n \n // Build content array with reasoning first (if any), then text\n const content: Array<{ type: 'reasoning' | 'text'; text: string }> = [];\n \n if (accumulatedReasoning) {\n content.push({ type: 'reasoning' as const, text: accumulatedReasoning });\n }\n \n if (accumulatedText) {\n content.push({ type: 'text' as const, text: accumulatedText });\n }\n \n // Fallback to empty text if nothing yet\n if (content.length === 0) {\n content.push({ type: 'text' as const, text: '' });\n }\n \n return { content };\n };\n\n // Parse SSE stream\n for await (const chunk of parseSSEStream(stream, abortSignal)) {\n // Check abort signal\n if (abortSignal?.aborted) {\n throw new Error('Request aborted');\n }\n\n // Handle chat ID from chunk\n const chunkId = chunk.id || chunk.chatId || chunk.messageId;\n if (chunkId && !chatId) {\n chatId = chunkId;\n // Prefer the threadId from run() options (stable, provided by runtime).\n // Fall back to activeThreadRemoteId which may have been updated by the\n // time the SSE chunk arrives (after HTTP roundtrip gives React time\n // to flush the useEffect that sets it).\n const threadIdForStorage = threadId || activeThreadRemoteId;\n \n onChatCreated?.(chunkId);\n \n // Register all current messages against this chat ID so\n // future calls to run() can resolve the chat\n for (const msg of messages) {\n messageIdToChatIdMap.set(msg.id, chunkId);\n }\n \n // Map the active local thread ID → server chat ID so that\n // rename / delete / generateTitle can resolve the real ID.\n if (threadIdForStorage && !localToRemoteIdMap.has(threadIdForStorage)) {\n localToRemoteIdMap.set(threadIdForStorage, chunkId);\n }\n\n // Notify the thread list adapter about the ID update\n if (onThreadIdUpdate && threadIdForStorage) {\n onThreadIdUpdate(threadIdForStorage, chunkId);\n }\n }\n\n // Handle sources array (sent in first chunk typically)\n const chunkSources = (chunk as SSEChunk & { sources?: Source[] | null }).sources;\n if (chunkSources && Array.isArray(chunkSources) && chunkSources.length > 0) {\n // Add new sources (avoid duplicates by sourceId)\n for (const source of chunkSources) {\n if (!sources.some(s => s.sourceId === source.sourceId)) {\n sources.push({\n sourceId: source.sourceId,\n filename: source.filename || 'Unknown',\n score: source.score ?? 1.0,\n chunkId: source.chunkId,\n datasetId: source.datasetId,\n });\n }\n }\n }\n\n // Handle reasoning field (string delta on each chunk)\n const chunkReasoning = (chunk as SSEChunk & { reasoning?: string }).reasoning;\n if (chunkReasoning && chunkReasoning.length > 0) {\n // Reasoning comes as delta - accumulate it\n accumulatedReasoning += chunkReasoning;\n // Yield immediately to show reasoning as it streams\n yield createYieldObject();\n }\n\n // Handle AI SDK Protocol events (for future compatibility)\n if (chunk.type) {\n switch (chunk.type) {\n case 'source-document':\n if (chunk.sourceId) {\n sources.push({\n sourceId: chunk.sourceId,\n filename: chunk.title || 'Unknown',\n score: 1.0,\n });\n }\n continue;\n \n case 'reasoning-start':\n // Reasoning now flows through content parts\n continue;\n \n case 'reasoning-delta':\n if (chunk.delta) {\n accumulatedReasoning += chunk.delta;\n }\n if (accumulatedText || accumulatedReasoning) {\n yield createYieldObject();\n }\n continue;\n \n case 'reasoning-end':\n // Reasoning now flows through content parts\n if (accumulatedText || accumulatedReasoning) {\n yield createYieldObject();\n }\n continue;\n \n case 'text-delta':\n if (chunk.delta) {\n // Add small delay between reasoning and first text chunk for better UX\n if (accumulatedReasoning && !hasAddedReasoningToTextDelay && !accumulatedText) {\n hasAddedReasoningToTextDelay = true;\n await new Promise(resolve => setTimeout(resolve, 400));\n }\n accumulatedText += chunk.delta;\n }\n if (accumulatedText) {\n yield createYieldObject();\n }\n continue;\n \n case 'error':\n throw new Error(chunk.errorText || 'Stream error');\n \n case 'text-start':\n case 'text-end':\n case 'start':\n case 'start-step':\n case 'finish-step':\n case 'finish':\n continue;\n }\n }\n\n // Handle current API format: {\"id\": \"...\", \"delta\": \"...\", \"reasoning\": \"\", \"sources\": [...], \"finished\": false}\n const delta = chunk.delta;\n if (delta !== undefined && delta !== null) {\n // Add small delay between reasoning and first text chunk for better UX\n if (accumulatedReasoning && !hasAddedReasoningToTextDelay && !accumulatedText) {\n hasAddedReasoningToTextDelay = true;\n await new Promise(resolve => setTimeout(resolve, 400));\n }\n accumulatedText += delta;\n }\n\n // Handle legacy formats for compatibility\n if (chunk.content) {\n accumulatedText += chunk.content;\n }\n\n // Handle message object format\n if (chunk.message) {\n accumulatedText += chunk.message.content || '';\n }\n\n // Check if finished\n if (chunk.finished) {\n // Stream is complete, yield final accumulated result\n if (accumulatedText || accumulatedReasoning || sources.length > 0) {\n yield createYieldObject();\n }\n break;\n }\n\n // Yield incremental updates\n if (accumulatedText || accumulatedReasoning || sources.length > 0) {\n yield createYieldObject();\n }\n }\n\n // Final yield with accumulated content\n if (accumulatedText || accumulatedReasoning || sources.length > 0) {\n yield createYieldObject();\n }\n },\n };\n}\n\n","import type {\n AttachmentAdapter,\n CompleteAttachment,\n PendingAttachment,\n} from '@assistant-ui/react';\n\n/**\n * Attachment validation configuration\n */\nexport interface AttachmentValidationConfig {\n /**\n * Maximum file size in bytes\n * @default 10 * 1024 * 1024 (10MB)\n */\n maxFileSizeBytes?: number;\n\n /**\n * Allowed MIME types (e.g., ['image/png', 'image/jpeg'])\n * If not provided, defaults to common document and image types\n */\n allowedMimeTypes?: string[];\n\n /**\n * Allowed file extensions (e.g., ['.pdf', '.docx'])\n * If not provided, defaults to common document and image extensions\n */\n allowedExtensions?: string[];\n\n /**\n * Maximum number of attachments per message\n * @default 5\n */\n maxAttachments?: number;\n\n /**\n * Whether to validate file content type matches extension\n * @default true\n */\n validateContentType?: boolean;\n}\n\n/**\n * Attachment validation error\n */\nexport class AttachmentValidationError extends Error {\n constructor(\n message: string,\n public readonly code: 'FILE_TOO_LARGE' | 'INVALID_TYPE' | 'TOO_MANY_FILES' | 'VALIDATION_FAILED' | 'VISION_NOT_SUPPORTED',\n public readonly file?: File\n ) {\n super(message);\n this.name = 'AttachmentValidationError';\n }\n}\n\n/**\n * Attachment adapter options\n */\nexport interface AttachmentAdapterOptions {\n /**\n * Validation configuration\n */\n validation?: AttachmentValidationConfig;\n\n /**\n * Whether the current model supports vision/image inputs.\n * When false, image attachments will be rejected with a clear error.\n * @default true\n */\n supportsVision?: boolean;\n\n /**\n * Callback when validation fails\n * Allows custom error handling (e.g., show toast notification)\n */\n onValidationError?: (error: AttachmentValidationError) => void;\n\n /**\n * Callback when attachment is added\n */\n onAttachmentAdded?: (attachment: PendingAttachment) => void;\n\n /**\n * Callback when attachment is removed\n */\n onAttachmentRemoved?: (attachmentId: string) => void;\n}\n\n/**\n * Default allowed MIME types\n */\nconst DEFAULT_ALLOWED_MIME_TYPES = [\n // Images\n 'image/jpeg',\n 'image/png',\n 'image/gif',\n 'image/webp',\n 'image/svg+xml',\n // Documents\n 'application/pdf',\n 'application/msword',\n 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',\n 'application/vnd.ms-excel',\n 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',\n // Text\n 'text/plain',\n 'text/markdown',\n 'text/csv',\n 'application/json',\n];\n\n/**\n * Default allowed file extensions\n */\nconst DEFAULT_ALLOWED_EXTENSIONS = [\n '.pdf',\n '.doc',\n '.docx',\n '.xls',\n '.xlsx',\n '.txt',\n '.md',\n '.csv',\n '.json',\n '.jpg',\n '.jpeg',\n '.png',\n '.gif',\n '.webp',\n '.svg',\n];\n\n/**\n * Default max file size (10MB)\n */\nconst DEFAULT_MAX_FILE_SIZE = 10 * 1024 * 1024;\n\n/**\n * Format bytes to human-readable size\n */\nfunction formatFileSize(bytes: number): string {\n if (bytes === 0) return '0 Bytes';\n \n const k = 1024;\n const sizes = ['Bytes', 'KB', 'MB', 'GB'];\n const i = Math.floor(Math.log(bytes) / Math.log(k));\n \n return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];\n}\n\n/**\n * Get file extension from filename\n */\nfunction getFileExtension(filename: string): string {\n const ext = filename.lastIndexOf('.') >= 0 \n ? filename.slice(filename.lastIndexOf('.')).toLowerCase() \n : '';\n return ext;\n}\n\n/**\n * Check if MIME type matches extension (basic validation)\n */\nfunction mimeMatchesExtension(mimeType: string, extension: string): boolean {\n const mimeToExtMap: Record<string, string[]> = {\n 'image/jpeg': ['.jpg', '.jpeg'],\n 'image/png': ['.png'],\n 'image/gif': ['.gif'],\n 'image/webp': ['.webp'],\n 'image/svg+xml': ['.svg'],\n 'application/pdf': ['.pdf'],\n 'text/plain': ['.txt'],\n 'text/markdown': ['.md'],\n 'text/csv': ['.csv'],\n 'application/json': ['.json'],\n };\n \n const validExtensions = mimeToExtMap[mimeType];\n if (!validExtensions) {\n // Unknown MIME type, allow it\n return true;\n }\n \n return validExtensions.includes(extension.toLowerCase());\n}\n\n/**\n * Generate a secure attachment ID\n */\nfunction generateAttachmentId(): string {\n // Use crypto.randomUUID if available\n if (typeof crypto !== 'undefined' && crypto.randomUUID) {\n return 'attachment-' + crypto.randomUUID();\n }\n \n // Fallback\n return 'attachment-' + Date.now() + '-' + Math.random().toString(36).slice(2, 11);\n}\n\n/**\n * Create an attachment adapter with enhanced validation\n */\nexport function createAttachmentAdapter(\n options: AttachmentAdapterOptions = {}\n): AttachmentAdapter {\n const {\n validation = {},\n supportsVision = true,\n onValidationError,\n onAttachmentAdded,\n onAttachmentRemoved,\n } = options;\n\n const {\n maxFileSizeBytes = DEFAULT_MAX_FILE_SIZE,\n allowedMimeTypes = DEFAULT_ALLOWED_MIME_TYPES,\n allowedExtensions = DEFAULT_ALLOWED_EXTENSIONS,\n validateContentType = true,\n } = validation;\n\n // Build accept string from allowed extensions\n const acceptString = allowedExtensions.join(',');\n\n /**\n * Validate a file before adding it as an attachment\n */\n function validateFile(file: File): void {\n // Check file size\n if (file.size > maxFileSizeBytes) {\n const error = new AttachmentValidationError(\n 'File \"' + file.name + '\" is too large (' + formatFileSize(file.size) + '). Maximum size is ' + formatFileSize(maxFileSizeBytes) + '.',\n 'FILE_TOO_LARGE',\n file\n );\n onValidationError?.(error);\n throw error;\n }\n\n // Check vision support for image files\n if (!supportsVision && file.type.startsWith('image/')) {\n const error = new AttachmentValidationError(\n 'The current model does not support image inputs. Please select a model with vision capabilities to attach images.',\n 'VISION_NOT_SUPPORTED',\n file\n );\n onValidationError?.(error);\n throw error;\n }\n\n // Check extension\n const extension = getFileExtension(file.name);\n if (extension && !allowedExtensions.some(ext => ext.toLowerCase() === extension)) {\n const error = new AttachmentValidationError(\n 'File type \"' + extension + '\" is not allowed. Allowed types: ' + allowedExtensions.join(', '),\n 'INVALID_TYPE',\n file\n );\n onValidationError?.(error);\n throw error;\n }\n\n // Check MIME type\n if (file.type && !allowedMimeTypes.some(mime => {\n // Support wildcards like 'image/*'\n if (mime.endsWith('/*')) {\n const prefix = mime.slice(0, -1);\n return file.type.startsWith(prefix);\n }\n return file.type === mime;\n })) {\n const error = new AttachmentValidationError(\n 'File type \"' + file.type + '\" is not allowed.',\n 'INVALID_TYPE',\n file\n );\n onValidationError?.(error);\n throw error;\n }\n\n // Validate MIME type matches extension\n if (validateContentType && file.type && extension) {\n if (!mimeMatchesExtension(file.type, extension)) {\n const error = new AttachmentValidationError(\n 'File extension \"' + extension + '\" does not match content type \"' + file.type + '\".',\n 'VALIDATION_FAILED',\n file\n );\n onValidationError?.(error);\n throw error;\n }\n }\n }\n\n return {\n accept: acceptString,\n\n async add({ file }): Promise<PendingAttachment> {\n // Validate the file\n validateFile(file);\n\n // Determine attachment type based on file\n let type: 'image' | 'document' | 'file' = 'file';\n\n if (file.type.startsWith('image/')) {\n type = 'image';\n } else if (\n file.type.includes('pdf') ||\n file.type.includes('document') ||\n file.type.includes('text') ||\n file.name.endsWith('.pdf') ||\n file.name.endsWith('.docx') ||\n file.name.endsWith('.txt') ||\n file.name.endsWith('.md') ||\n file.name.endsWith('.csv') ||\n file.name.endsWith('.json')\n ) {\n type = 'document';\n }\n\n const attachment: PendingAttachment = {\n id: generateAttachmentId(),\n type,\n name: file.name,\n contentType: file.type,\n file,\n status: { type: 'requires-action', reason: 'composer-send' },\n };\n\n onAttachmentAdded?.(attachment);\n\n return attachment;\n },\n\n async remove(attachment) {\n onAttachmentRemoved?.(attachment.id);\n // No other cleanup needed - files are handled by the browser\n },\n\n async send(attachment: PendingAttachment): Promise<CompleteAttachment> {\n // For images, create a data URL\n if (attachment.type === 'image' && attachment.file) {\n const dataURL = await new Promise<string>((resolve, reject) => {\n const reader = new FileReader();\n reader.onload = () => resolve(reader.result as string);\n reader.onerror = () => reject(new Error('Failed to read file'));\n reader.readAsDataURL(attachment.file!);\n });\n\n return {\n ...attachment,\n status: { type: 'complete' },\n content: [\n {\n type: 'image',\n image: dataURL,\n },\n ],\n };\n }\n\n // For other files, return as-is\n return {\n ...attachment,\n status: { type: 'complete' },\n content: [],\n };\n },\n };\n}\n","/**\n * Lightweight reactive store for surfacing attachment validation errors in the UI.\n *\n * The attachment adapter runs outside React's tree (inside @assistant-ui/react internals),\n * so we use a tiny pub/sub store — identical to the pattern used by streamingMetadataStore.\n */\n\ntype Listener = () => void;\n\nexport interface AttachmentError {\n message: string;\n code: string;\n timestamp: number;\n}\n\nclass AttachmentErrorStore {\n private current: AttachmentError | null = null;\n private listeners = new Set<Listener>();\n\n /** Push a new error (triggers subscribers). */\n push(message: string, code: string) {\n this.current = { message, code, timestamp: Date.now() };\n this.notify();\n }\n\n /** Clear the current error (triggers subscribers). */\n clear() {\n if (this.current) {\n this.current = null;\n this.notify();\n }\n }\n\n /** Get the current error (snapshot for useSyncExternalStore). */\n getSnapshot = (): AttachmentError | null => this.current;\n\n /** Subscribe (for useSyncExternalStore). Returns unsubscribe function. */\n subscribe = (listener: Listener): (() => void) => {\n this.listeners.add(listener);\n return () => this.listeners.delete(listener);\n };\n\n private notify() {\n this.listeners.forEach((l) => l());\n }\n}\n\nexport const attachmentErrorStore = new AttachmentErrorStore();\n","import React, { useCallback, useEffect, useMemo, useRef, useState } from 'react';\nimport {\n AssistantRuntimeProvider,\n type ThreadMessageLike,\n unstable_useRemoteThreadListRuntime,\n useLocalRuntime,\n} from '@assistant-ui/react';\nimport { CuadraChatClient } from '../lib/cuadraChatClient';\nimport { createChatModelAdapter } from '../adapters/chatModelAdapter';\nimport { createThreadListAdapter, messageIdToChatIdMap } from '../adapters/threadListAdapter';\nimport { convertFromCuadraMessage } from '../adapters/messageConverter';\nimport { type AttachmentValidationError, createAttachmentAdapter } from '../adapters/attachmentAdapter';\nimport { attachmentErrorStore } from '../adapters/attachmentErrorStore';\n\nexport interface CuadraRuntimeProviderProps {\n children: React.ReactNode;\n baseUrl: string;\n sessionToken?: string | null;\n isProxyMode?: boolean; // If true, uses proxy URL patterns (removes /v1 prefix)\n\n // Mode selection\n mode?: 'singleChat' | 'multiChat'; // Default: 'multiChat'\n\n // Model configuration\n modelMode?: 'selector' | 'fixed'; // Default: 'fixed'\n modelId?: string; // Required if modelMode === 'fixed'\n onModelChange?: (modelId: string) => void; // Called when model changes (selector mode)\n\n // Chat configuration\n ephemeral?: boolean; // Default: false - creates temporary chats that auto-delete\n systemPrompt?: string;\n /** Enable reasoning/thinking output for supported models (e.g., o1, o3) */\n enableReasoning?: boolean;\n\n // Thread configuration\n initialThreadId?: string; // Load existing thread (singleChat fetches messages; multiChat uses thread list)\n\n // Callbacks\n onError?: (error: Error) => void;\n onChatCreated?: (chatId: string) => void;\n onUserMessage?: () => void; // Called when a user message is sent\n onThreadIdUpdate?: (\n oldThreadId: string,\n newThreadId: string,\n ) => void; // Callback when thread ID is updated from local to server\n onChatsLoaded?: () => void; // Callback when chats/thread list is loaded\n /** Enable file attachments in the chat input */\n enableAttachments?: boolean;\n /** Whether the current model supports vision/image inputs. When false, image attachments are rejected. */\n supportsVision?: boolean;\n /** Interceptor called before each API request. Return true to allow, false to block. */\n onBeforeRequest?: () => boolean | Promise<boolean>;\n /** \n * Mock response to show when onBeforeRequest returns false.\n * If set, instead of throwing an error, this message will be streamed as a response.\n * Can be a string or a function that returns a string (called each time a request is blocked).\n * Useful for showing helpful messages like \"Complete setup before chatting\".\n */\n mockResponseOnBlocked?: string | (() => string);\n}\n\n/**\n * Main provider component that sets up assistant-ui runtime with Cuadra API\n * Supports single/multi chat modes, fixed/selector model modes, and ephemeral chats\n */\nexport function CuadraRuntimeProvider({\n children,\n baseUrl,\n sessionToken,\n isProxyMode = false,\n mode = 'multiChat',\n modelId,\n onModelChange: _onModelChange,\n ephemeral = false,\n systemPrompt,\n enableReasoning = false,\n initialThreadId,\n onError,\n onChatCreated,\n onUserMessage,\n onThreadIdUpdate,\n onChatsLoaded,\n enableAttachments = false,\n supportsVision = true,\n onBeforeRequest,\n mockResponseOnBlocked,\n}: CuadraRuntimeProviderProps) {\n // Create API client\n const client = useMemo(\n () => new CuadraChatClient(baseUrl, sessionToken || undefined, isProxyMode),\n [baseUrl, sessionToken, isProxyMode],\n );\n\n // Track selected model ID (syncs with prop)\n const [selectedModelId, setSelectedModelId] = useState<string | null>(modelId || null);\n\n // Sync selectedModelId with modelId prop\n useEffect(() => {\n if (modelId) {\n setSelectedModelId(modelId);\n }\n }, [modelId]);\n\n // Handle chat created callback\n const handleChatCreated = useCallback(\n (chatId: string) => {\n onChatCreated?.(chatId);\n },\n [onChatCreated],\n );\n\n // Handle thread ID update\n const handleThreadIdUpdate = useCallback(\n (oldId: string, newId: string) => {\n onThreadIdUpdate?.(oldId, newId);\n },\n [onThreadIdUpdate],\n );\n\n // For singleChat mode, create model adapter and use useLocalRuntime\n const modelAdapter = useMemo(() => {\n return createChatModelAdapter(client, {\n modelId: selectedModelId || undefined, // Pass undefined if not set (backend resolves it)\n systemPrompt,\n ephemeral,\n enableReasoning,\n onChatCreated: handleChatCreated,\n onUserMessage,\n onBeforeRequest,\n mockResponseOnBlocked,\n });\n }, [client, selectedModelId, systemPrompt, ephemeral, enableReasoning, handleChatCreated, onUserMessage, onBeforeRequest, mockResponseOnBlocked]);\n\n if (mode === 'singleChat') {\n return (\n <SingleChatProvider\n client={client}\n initialThreadId={initialThreadId}\n modelAdapter={modelAdapter}\n enableAttachments={enableAttachments}\n supportsVision={supportsVision}\n onChatCreated={handleChatCreated}\n >\n {children}\n </SingleChatProvider>\n );\n }\n\n // For multiChat mode, use RemoteThreadListRuntime\n // Pass adapter options instead of the adapter itself so we can add thread ID update callback\n return (\n <MultiChatProvider\n client={client}\n modelAdapterOptions={{\n modelId: selectedModelId || undefined,\n systemPrompt,\n ephemeral,\n enableReasoning,\n onChatCreated: handleChatCreated,\n onUserMessage,\n onBeforeRequest,\n mockResponseOnBlocked,\n }}\n initialThreadId={initialThreadId}\n onChatsLoaded={onChatsLoaded}\n onThreadIdUpdate={handleThreadIdUpdate}\n onError={onError}\n enableAttachments={enableAttachments}\n supportsVision={supportsVision}\n >\n {children}\n </MultiChatProvider>\n );\n}\n\n/**\n * Internal provider for singleChat mode using useLocalRuntime.\n * When initialThreadId is supplied, fetches the existing conversation\n * from the API and pre-populates the runtime with those messages.\n */\nfunction SingleChatProvider({\n client,\n initialThreadId,\n modelAdapter,\n enableAttachments,\n supportsVision = true,\n onChatCreated,\n children,\n}: {\n client: CuadraChatClient;\n initialThreadId?: string;\n modelAdapter: ReturnType<typeof createChatModelAdapter>;\n enableAttachments?: boolean;\n supportsVision?: boolean;\n onChatCreated?: (chatId: string) => void;\n children: React.ReactNode;\n}) {\n const [initialMessages, setInitialMessages] = useState<ThreadMessageLike[] | undefined>(undefined);\n const [isLoadingThread, setIsLoadingThread] = useState(!!initialThreadId);\n\n // Fetch existing thread messages when initialThreadId is provided\n useEffect(() => {\n if (!initialThreadId) {\n setIsLoadingThread(false);\n return;\n }\n\n let cancelled = false;\n\n async function loadThread() {\n try {\n const chat = await client.getChat(initialThreadId!);\n\n if (cancelled) return;\n\n if (chat.messages && chat.messages.length > 0) {\n const converted: ThreadMessageLike[] = chat.messages.map((msg) =>\n convertFromCuadraMessage(msg),\n );\n\n // Pre-register every message ID → chatId so the adapter's run()\n // resolves the server chat and appends to the existing thread.\n for (const msg of chat.messages) {\n messageIdToChatIdMap.set(msg.id, chat.id);\n }\n\n setInitialMessages(converted);\n }\n\n // Notify parent about the loaded chat\n onChatCreated?.(chat.id);\n } catch (_err) {\n // If fetching fails, fall back to a fresh chat silently\n } finally {\n if (!cancelled) {\n setIsLoadingThread(false);\n }\n }\n }\n\n loadThread();\n\n return () => {\n cancelled = true;\n };\n }, [client, initialThreadId, onChatCreated]);\n\n // While loading the thread, render nothing (the parent can show a loading state)\n if (isLoadingThread) {\n return null;\n }\n\n // Once loaded, render the runtime provider with pre-populated messages\n return (\n <SingleChatRuntime\n modelAdapter={modelAdapter}\n initialMessages={initialMessages}\n enableAttachments={enableAttachments}\n supportsVision={supportsVision}\n >\n {children}\n </SingleChatRuntime>\n );\n}\n\n/**\n * Thin wrapper that calls useLocalRuntime with the resolved initialMessages.\n * Separated so that SingleChatProvider can delay rendering until the fetch\n * completes, keeping hooks unconditional.\n */\nfunction SingleChatRuntime({\n modelAdapter,\n initialMessages,\n enableAttachments,\n supportsVision = true,\n children,\n}: {\n modelAdapter: ReturnType<typeof createChatModelAdapter>;\n initialMessages?: ThreadMessageLike[];\n enableAttachments?: boolean;\n supportsVision?: boolean;\n children: React.ReactNode;\n}) {\n // Create attachment adapter only if enabled\n const attachmentAdapter = useMemo(() => {\n return enableAttachments ? createAttachmentAdapter({\n supportsVision,\n onValidationError: (error: AttachmentValidationError) => {\n attachmentErrorStore.push(error.message, error.code);\n },\n }) : undefined;\n }, [enableAttachments, supportsVision]);\n \n const localRuntime = useLocalRuntime(modelAdapter, {\n initialMessages,\n adapters: {\n ...(attachmentAdapter && { attachments: attachmentAdapter }),\n },\n });\n\n return (\n <AssistantRuntimeProvider runtime={localRuntime}>\n {children}\n </AssistantRuntimeProvider>\n );\n}\n\n/**\n * Internal provider for multiChat mode using RemoteThreadListRuntime\n */\nfunction MultiChatProvider({\n client,\n modelAdapterOptions,\n initialThreadId: _initialThreadId,\n onChatsLoaded,\n onThreadIdUpdate,\n onError: _onError,\n enableAttachments,\n supportsVision,\n children,\n}: {\n client: CuadraChatClient;\n modelAdapterOptions: {\n modelId?: string | null;\n systemPrompt?: string;\n ephemeral?: boolean;\n enableReasoning?: boolean;\n onChatCreated?: (chatId: string) => void;\n onUserMessage?: () => void;\n onBeforeRequest?: () => boolean | Promise<boolean>;\n mockResponseOnBlocked?: string | (() => string);\n };\n initialThreadId?: string;\n onChatsLoaded?: () => void;\n onThreadIdUpdate?: (oldId: string, newId: string) => void;\n onError?: (error: Error) => void;\n enableAttachments?: boolean;\n supportsVision?: boolean;\n children: React.ReactNode;\n}) {\n // Create thread list adapter\n const threadListAdapter = useMemo(() => {\n return createThreadListAdapter(client, (localThreadId, serverChatId) => {\n onThreadIdUpdate?.(localThreadId, serverChatId);\n });\n }, [client, onThreadIdUpdate]);\n\n // Create a callback to update thread ID mapping when chat is created\n const handleThreadIdUpdateFromAdapter = useCallback(\n (localThreadId: string, serverChatId: string) => {\n // Update the adapter's remote ID mapping\n if (threadListAdapter.updateRemoteId) {\n threadListAdapter.updateRemoteId(localThreadId, serverChatId);\n }\n // Also call the prop callback\n onThreadIdUpdate?.(localThreadId, serverChatId);\n },\n [threadListAdapter, onThreadIdUpdate],\n );\n\n // Use a ref for onChatsLoaded so wrappedAdapter stays stable\n const onChatsLoadedRef = useRef(onChatsLoaded);\n onChatsLoadedRef.current = onChatsLoaded;\n\n // Wrap list method to call onChatsLoaded (stable — only depends on threadListAdapter)\n const wrappedAdapter = useMemo(() => {\n return {\n ...threadListAdapter,\n async list() {\n const result = await threadListAdapter.list();\n onChatsLoadedRef.current?.();\n return result;\n },\n };\n }, [threadListAdapter]);\n\n // Expose adapter globally for external access (e.g., rename/delete)\n useEffect(() => {\n (window as Window & { __cuadraThreadListAdapter?: typeof wrappedAdapter }).__cuadraThreadListAdapter =\n wrappedAdapter;\n (window as Window & { __cuadraThreadListRuntime?: unknown }).__cuadraThreadListRuntime = null;\n }, [wrappedAdapter]);\n\n // Keep modelId in a ref so changing it doesn't recreate the adapter\n // (which would rebuild the runtime and lose the current thread).\n // The adapter reads the ref lazily at request time via getModelId.\n const modelIdRef = useRef(modelAdapterOptions.modelId);\n modelIdRef.current = modelAdapterOptions.modelId;\n\n // Stable options that exclude modelId (modelId is read via ref)\n const stableAdapterOptions = useMemo(() => {\n const { modelId: _modelId, ...rest } = modelAdapterOptions;\n return rest;\n }, [\n modelAdapterOptions.systemPrompt,\n modelAdapterOptions.ephemeral,\n modelAdapterOptions.enableReasoning,\n modelAdapterOptions.onChatCreated,\n modelAdapterOptions.onUserMessage,\n modelAdapterOptions.onBeforeRequest,\n modelAdapterOptions.mockResponseOnBlocked,\n ]);\n\n // Create model adapter with thread ID update callback for multiChat mode\n const modelAdapter = useMemo(() => {\n return createChatModelAdapter(client, {\n ...stableAdapterOptions,\n getModelId: () => modelIdRef.current,\n onThreadIdUpdate: handleThreadIdUpdateFromAdapter,\n });\n }, [client, stableAdapterOptions, handleThreadIdUpdateFromAdapter]);\n\n // Create attachment adapter only if enabled\n const attachmentAdapter = useMemo(() => {\n return enableAttachments ? createAttachmentAdapter({\n supportsVision,\n onValidationError: (error: AttachmentValidationError) => {\n attachmentErrorStore.push(error.message, error.code);\n },\n }) : undefined;\n }, [enableAttachments, supportsVision]);\n \n // Create runtime hook that uses the model adapter\n const runtimeHook = useCallback(() => {\n return useLocalRuntime(modelAdapter, {\n adapters: {\n ...(attachmentAdapter && { attachments: attachmentAdapter }),\n },\n });\n }, [modelAdapter, attachmentAdapter]);\n\n // Use RemoteThreadListRuntime hook\n const runtime = unstable_useRemoteThreadListRuntime({\n adapter: wrappedAdapter,\n runtimeHook,\n });\n\n // Expose runtime globally for external refresh\n useEffect(() => {\n (window as Window & { __cuadraThreadListRuntime?: typeof runtime }).__cuadraThreadListRuntime =\n runtime;\n }, [runtime]);\n\n return (\n <AssistantRuntimeProvider runtime={runtime}>{children}</AssistantRuntimeProvider>\n );\n}\n","/**\n * Count how often a character (or substring) is used in a string.\n *\n * @param {string} value\n * Value to search in.\n * @param {string} character\n * Character (or substring) to look for.\n * @return {number}\n * Number of times `character` occurred in `value`.\n */\nexport function ccount(value, character) {\n const source = String(value)\n\n if (typeof character !== 'string') {\n throw new TypeError('Expected character')\n }\n\n let count = 0\n let index = source.indexOf(character)\n\n while (index !== -1) {\n count++\n index = source.indexOf(character, index + character.length)\n }\n\n return count\n}\n","/**\n * @import {Code} from 'micromark-util-types'\n */\n\n/**\n * Check whether the character code represents an ASCII alpha (`a` through `z`,\n * case insensitive).\n *\n * An **ASCII alpha** is an ASCII upper alpha or ASCII lower alpha.\n *\n * An **ASCII upper alpha** is a character in the inclusive range U+0041 (`A`)\n * to U+005A (`Z`).\n *\n * An **ASCII lower alpha** is a character in the inclusive range U+0061 (`a`)\n * to U+007A (`z`).\n *\n * @param code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport const asciiAlpha = regexCheck(/[A-Za-z]/);\n\n/**\n * Check whether the character code represents an ASCII alphanumeric (`a`\n * through `z`, case insensitive, or `0` through `9`).\n *\n * An **ASCII alphanumeric** is an ASCII digit (see `asciiDigit`) or ASCII alpha\n * (see `asciiAlpha`).\n *\n * @param code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport const asciiAlphanumeric = regexCheck(/[\\dA-Za-z]/);\n\n/**\n * Check whether the character code represents an ASCII atext.\n *\n * atext is an ASCII alphanumeric (see `asciiAlphanumeric`), or a character in\n * the inclusive ranges U+0023 NUMBER SIGN (`#`) to U+0027 APOSTROPHE (`'`),\n * U+002A ASTERISK (`*`), U+002B PLUS SIGN (`+`), U+002D DASH (`-`), U+002F\n * SLASH (`/`), U+003D EQUALS TO (`=`), U+003F QUESTION MARK (`?`), U+005E\n * CARET (`^`) to U+0060 GRAVE ACCENT (`` ` ``), or U+007B LEFT CURLY BRACE\n * (`{`) to U+007E TILDE (`~`).\n *\n * See:\n * **\\[RFC5322]**:\n * [Internet Message Format](https://tools.ietf.org/html/rfc5322).\n * P. Resnick.\n * IETF.\n *\n * @param code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport const asciiAtext = regexCheck(/[#-'*+\\--9=?A-Z^-~]/);\n\n/**\n * Check whether a character code is an ASCII control character.\n *\n * An **ASCII control** is a character in the inclusive range U+0000 NULL (NUL)\n * to U+001F (US), or U+007F (DEL).\n *\n * @param {Code} code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport function asciiControl(code) {\n return (\n // Special whitespace codes (which have negative values), C0 and Control\n // character DEL\n code !== null && (code < 32 || code === 127)\n );\n}\n\n/**\n * Check whether the character code represents an ASCII digit (`0` through `9`).\n *\n * An **ASCII digit** is a character in the inclusive range U+0030 (`0`) to\n * U+0039 (`9`).\n *\n * @param code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport const asciiDigit = regexCheck(/\\d/);\n\n/**\n * Check whether the character code represents an ASCII hex digit (`a` through\n * `f`, case insensitive, or `0` through `9`).\n *\n * An **ASCII hex digit** is an ASCII digit (see `asciiDigit`), ASCII upper hex\n * digit, or an ASCII lower hex digit.\n *\n * An **ASCII upper hex digit** is a character in the inclusive range U+0041\n * (`A`) to U+0046 (`F`).\n *\n * An **ASCII lower hex digit** is a character in the inclusive range U+0061\n * (`a`) to U+0066 (`f`).\n *\n * @param code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport const asciiHexDigit = regexCheck(/[\\dA-Fa-f]/);\n\n/**\n * Check whether the character code represents ASCII punctuation.\n *\n * An **ASCII punctuation** is a character in the inclusive ranges U+0021\n * EXCLAMATION MARK (`!`) to U+002F SLASH (`/`), U+003A COLON (`:`) to U+0040 AT\n * SIGN (`@`), U+005B LEFT SQUARE BRACKET (`[`) to U+0060 GRAVE ACCENT\n * (`` ` ``), or U+007B LEFT CURLY BRACE (`{`) to U+007E TILDE (`~`).\n *\n * @param code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport const asciiPunctuation = regexCheck(/[!-/:-@[-`{-~]/);\n\n/**\n * Check whether a character code is a markdown line ending.\n *\n * A **markdown line ending** is the virtual characters M-0003 CARRIAGE RETURN\n * LINE FEED (CRLF), M-0004 LINE FEED (LF) and M-0005 CARRIAGE RETURN (CR).\n *\n * In micromark, the actual character U+000A LINE FEED (LF) and U+000D CARRIAGE\n * RETURN (CR) are replaced by these virtual characters depending on whether\n * they occurred together.\n *\n * @param {Code} code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport function markdownLineEnding(code) {\n return code !== null && code < -2;\n}\n\n/**\n * Check whether a character code is a markdown line ending (see\n * `markdownLineEnding`) or markdown space (see `markdownSpace`).\n *\n * @param {Code} code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport function markdownLineEndingOrSpace(code) {\n return code !== null && (code < 0 || code === 32);\n}\n\n/**\n * Check whether a character code is a markdown space.\n *\n * A **markdown space** is the concrete character U+0020 SPACE (SP) and the\n * virtual characters M-0001 VIRTUAL SPACE (VS) and M-0002 HORIZONTAL TAB (HT).\n *\n * In micromark, the actual character U+0009 CHARACTER TABULATION (HT) is\n * replaced by one M-0002 HORIZONTAL TAB (HT) and between 0 and 3 M-0001 VIRTUAL\n * SPACE (VS) characters, depending on the column at which the tab occurred.\n *\n * @param {Code} code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport function markdownSpace(code) {\n return code === -2 || code === -1 || code === 32;\n}\n\n// Size note: removing ASCII from the regex and using `asciiPunctuation` here\n// In fact adds to the bundle size.\n/**\n * Check whether the character code represents Unicode punctuation.\n *\n * A **Unicode punctuation** is a character in the Unicode `Pc` (Punctuation,\n * Connector), `Pd` (Punctuation, Dash), `Pe` (Punctuation, Close), `Pf`\n * (Punctuation, Final quote), `Pi` (Punctuation, Initial quote), `Po`\n * (Punctuation, Other), or `Ps` (Punctuation, Open) categories, or an ASCII\n * punctuation (see `asciiPunctuation`).\n *\n * See:\n * **\\[UNICODE]**:\n * [The Unicode Standard](https://www.unicode.org/versions/).\n * Unicode Consortium.\n *\n * @param code\n * Code.\n * @returns\n * Whether it matches.\n */\nexport const unicodePunctuation = regexCheck(/\\p{P}|\\p{S}/u);\n\n/**\n * Check whether the character code represents Unicode whitespace.\n *\n * Note that this does handle micromark specific markdown whitespace characters.\n * See `markdownLineEndingOrSpace` to check that.\n *\n * A **Unicode whitespace** is a character in the Unicode `Zs` (Separator,\n * Space) category, or U+0009 CHARACTER TABULATION (HT), U+000A LINE FEED (LF),\n * U+000C (FF), or U+000D CARRIAGE RETURN (CR) (**\\[UNICODE]**).\n *\n * See:\n * **\\[UNICODE]**:\n * [The Unicode Standard](https://www.unicode.org/versions/).\n * Unicode Consortium.\n *\n * @param code\n * Code.\n * @returns\n * Whether it matches.\n */\nexport const unicodeWhitespace = regexCheck(/\\s/);\n\n/**\n * Create a code check from a regex.\n *\n * @param {RegExp} regex\n * Expression.\n * @returns {(code: Code) => boolean}\n * Check.\n */\nfunction regexCheck(regex) {\n return check;\n\n /**\n * Check whether a code matches the bound regex.\n *\n * @param {Code} code\n * Character code.\n * @returns {boolean}\n * Whether the character code matches the bound regex.\n */\n function check(code) {\n return code !== null && code > -1 && regex.test(String.fromCharCode(code));\n }\n}","export default function escapeStringRegexp(string) {\n\tif (typeof string !== 'string') {\n\t\tthrow new TypeError('Expected a string');\n\t}\n\n\t// Escape characters with special meaning either inside or outside character sets.\n\t// Use a simple backslash escape when it’s always valid, and a `\\xnn` escape when the simpler form would be disallowed by Unicode patterns’ stricter grammar.\n\treturn string\n\t\t.replace(/[|\\\\{}()[\\]^$+*?.]/g, '\\\\$&')\n\t\t.replace(/-/g, '\\\\x2d');\n}\n","/**\n * @import {Node, Parent} from 'unist'\n */\n\n/**\n * @template Fn\n * @template Fallback\n * @typedef {Fn extends (value: any) => value is infer Thing ? Thing : Fallback} Predicate\n */\n\n/**\n * @callback Check\n * Check that an arbitrary value is a node.\n * @param {unknown} this\n * The given context.\n * @param {unknown} [node]\n * Anything (typically a node).\n * @param {number | null | undefined} [index]\n * The node’s position in its parent.\n * @param {Parent | null | undefined} [parent]\n * The node’s parent.\n * @returns {boolean}\n * Whether this is a node and passes a test.\n *\n * @typedef {Record<string, unknown> | Node} Props\n * Object to check for equivalence.\n *\n * Note: `Node` is included as it is common but is not indexable.\n *\n * @typedef {Array<Props | TestFunction | string> | ReadonlyArray<Props | TestFunction | string> | Props | TestFunction | string | null | undefined} Test\n * Check for an arbitrary node.\n *\n * @callback TestFunction\n * Check if a node passes a test.\n * @param {unknown} this\n * The given context.\n * @param {Node} node\n * A node.\n * @param {number | undefined} [index]\n * The node’s position in its parent.\n * @param {Parent | undefined} [parent]\n * The node’s parent.\n * @returns {boolean | undefined | void}\n * Whether this node passes the test.\n *\n * Note: `void` is included until TS sees no return as `undefined`.\n */\n\n/**\n * Check if `node` is a `Node` and whether it passes the given test.\n *\n * @param {unknown} node\n * Thing to check, typically `Node`.\n * @param {Test} test\n * A check for a specific node.\n * @param {number | null | undefined} index\n * The node’s position in its parent.\n * @param {Parent | null | undefined} parent\n * The node’s parent.\n * @param {unknown} context\n * Context object (`this`) to pass to `test` functions.\n * @returns {boolean}\n * Whether `node` is a node and passes a test.\n */\nexport const is =\n // Note: overloads in JSDoc can’t yet use different `@template`s.\n /**\n * @type {(\n * (<Condition extends ReadonlyArray<string>>(node: unknown, test: Condition, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & {type: Condition[number]}) &\n * (<Condition extends Array<string>>(node: unknown, test: Condition, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & {type: Condition[number]}) &\n * (<Condition extends string>(node: unknown, test: Condition, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & {type: Condition}) &\n * (<Condition extends Props>(node: unknown, test: Condition, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & Condition) &\n * (<Condition extends TestFunction>(node: unknown, test: Condition, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & Predicate<Condition, Node>) &\n * ((node?: null | undefined) => false) &\n * ((node: unknown, test?: null | undefined, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node) &\n * ((node: unknown, test?: Test, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => boolean)\n * )}\n */\n (\n /**\n * @param {unknown} [node]\n * @param {Test} [test]\n * @param {number | null | undefined} [index]\n * @param {Parent | null | undefined} [parent]\n * @param {unknown} [context]\n * @returns {boolean}\n */\n // eslint-disable-next-line max-params\n function (node, test, index, parent, context) {\n const check = convert(test)\n\n if (\n index !== undefined &&\n index !== null &&\n (typeof index !== 'number' ||\n index < 0 ||\n index === Number.POSITIVE_INFINITY)\n ) {\n throw new Error('Expected positive finite index')\n }\n\n if (\n parent !== undefined &&\n parent !== null &&\n (!is(parent) || !parent.children)\n ) {\n throw new Error('Expected parent node')\n }\n\n if (\n (parent === undefined || parent === null) !==\n (index === undefined || index === null)\n ) {\n throw new Error('Expected both parent and index')\n }\n\n return looksLikeANode(node)\n ? check.call(context, node, index, parent)\n : false\n }\n )\n\n/**\n * Generate an assertion from a test.\n *\n * Useful if you’re going to test many nodes, for example when creating a\n * utility where something else passes a compatible test.\n *\n * The created function is a bit faster because it expects valid input only:\n * a `node`, `index`, and `parent`.\n *\n * @param {Test} test\n * * when nullish, checks if `node` is a `Node`.\n * * when `string`, works like passing `(node) => node.type === test`.\n * * when `function` checks if function passed the node is true.\n * * when `object`, checks that all keys in test are in node, and that they have (strictly) equal values.\n * * when `array`, checks if any one of the subtests pass.\n * @returns {Check}\n * An assertion.\n */\nexport const convert =\n // Note: overloads in JSDoc can’t yet use different `@template`s.\n /**\n * @type {(\n * (<Condition extends string>(test: Condition) => (node: unknown, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & {type: Condition}) &\n * (<Condition extends Props>(test: Condition) => (node: unknown, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & Condition) &\n * (<Condition extends TestFunction>(test: Condition) => (node: unknown, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & Predicate<Condition, Node>) &\n * ((test?: null | undefined) => (node?: unknown, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node) &\n * ((test?: Test) => Check)\n * )}\n */\n (\n /**\n * @param {Test} [test]\n * @returns {Check}\n */\n function (test) {\n if (test === null || test === undefined) {\n return ok\n }\n\n if (typeof test === 'function') {\n return castFactory(test)\n }\n\n if (typeof test === 'object') {\n return Array.isArray(test)\n ? anyFactory(test)\n : // Cast because `ReadonlyArray` goes into the above but `isArray`\n // narrows to `Array`.\n propertiesFactory(/** @type {Props} */ (test))\n }\n\n if (typeof test === 'string') {\n return typeFactory(test)\n }\n\n throw new Error('Expected function, string, or object as test')\n }\n )\n\n/**\n * @param {Array<Props | TestFunction | string>} tests\n * @returns {Check}\n */\nfunction anyFactory(tests) {\n /** @type {Array<Check>} */\n const checks = []\n let index = -1\n\n while (++index < tests.length) {\n checks[index] = convert(tests[index])\n }\n\n return castFactory(any)\n\n /**\n * @this {unknown}\n * @type {TestFunction}\n */\n function any(...parameters) {\n let index = -1\n\n while (++index < checks.length) {\n if (checks[index].apply(this, parameters)) return true\n }\n\n return false\n }\n}\n\n/**\n * Turn an object into a test for a node with a certain fields.\n *\n * @param {Props} check\n * @returns {Check}\n */\nfunction propertiesFactory(check) {\n const checkAsRecord = /** @type {Record<string, unknown>} */ (check)\n\n return castFactory(all)\n\n /**\n * @param {Node} node\n * @returns {boolean}\n */\n function all(node) {\n const nodeAsRecord = /** @type {Record<string, unknown>} */ (\n /** @type {unknown} */ (node)\n )\n\n /** @type {string} */\n let key\n\n for (key in check) {\n if (nodeAsRecord[key] !== checkAsRecord[key]) return false\n }\n\n return true\n }\n}\n\n/**\n * Turn a string into a test for a node with a certain type.\n *\n * @param {string} check\n * @returns {Check}\n */\nfunction typeFactory(check) {\n return castFactory(type)\n\n /**\n * @param {Node} node\n */\n function type(node) {\n return node && node.type === check\n }\n}\n\n/**\n * Turn a custom test into a test for a node that passes that test.\n *\n * @param {TestFunction} testFunction\n * @returns {Check}\n */\nfunction castFactory(testFunction) {\n return check\n\n /**\n * @this {unknown}\n * @type {Check}\n */\n function check(value, index, parent) {\n return Boolean(\n looksLikeANode(value) &&\n testFunction.call(\n this,\n value,\n typeof index === 'number' ? index : undefined,\n parent || undefined\n )\n )\n }\n}\n\nfunction ok() {\n return true\n}\n\n/**\n * @param {unknown} value\n * @returns {value is Node}\n */\nfunction looksLikeANode(value) {\n return value !== null && typeof value === 'object' && 'type' in value\n}\n","/**\n * @import {Node as UnistNode, Parent as UnistParent} from 'unist'\n */\n\n/**\n * @typedef {Exclude<import('unist-util-is').Test, undefined> | undefined} Test\n * Test from `unist-util-is`.\n *\n * Note: we have remove and add `undefined`, because otherwise when generating\n * automatic `.d.ts` files, TS tries to flatten paths from a local perspective,\n * which doesn’t work when publishing on npm.\n */\n\n/**\n * @typedef {(\n * Fn extends (value: any) => value is infer Thing\n * ? Thing\n * : Fallback\n * )} Predicate\n * Get the value of a type guard `Fn`.\n * @template Fn\n * Value; typically function that is a type guard (such as `(x): x is Y`).\n * @template Fallback\n * Value to yield if `Fn` is not a type guard.\n */\n\n/**\n * @typedef {(\n * Check extends null | undefined // No test.\n * ? Value\n * : Value extends {type: Check} // String (type) test.\n * ? Value\n * : Value extends Check // Partial test.\n * ? Value\n * : Check extends Function // Function test.\n * ? Predicate<Check, Value> extends Value\n * ? Predicate<Check, Value>\n * : never\n * : never // Some other test?\n * )} MatchesOne\n * Check whether a node matches a primitive check in the type system.\n * @template Value\n * Value; typically unist `Node`.\n * @template Check\n * Value; typically `unist-util-is`-compatible test, but not arrays.\n */\n\n/**\n * @typedef {(\n * Check extends ReadonlyArray<infer T>\n * ? MatchesOne<Value, T>\n * : Check extends Array<infer T>\n * ? MatchesOne<Value, T>\n * : MatchesOne<Value, Check>\n * )} Matches\n * Check whether a node matches a check in the type system.\n * @template Value\n * Value; typically unist `Node`.\n * @template Check\n * Value; typically `unist-util-is`-compatible test.\n */\n\n/**\n * @typedef {0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10} Uint\n * Number; capped reasonably.\n */\n\n/**\n * @typedef {I extends 0 ? 1 : I extends 1 ? 2 : I extends 2 ? 3 : I extends 3 ? 4 : I extends 4 ? 5 : I extends 5 ? 6 : I extends 6 ? 7 : I extends 7 ? 8 : I extends 8 ? 9 : 10} Increment\n * Increment a number in the type system.\n * @template {Uint} [I=0]\n * Index.\n */\n\n/**\n * @typedef {(\n * Node extends UnistParent\n * ? Node extends {children: Array<infer Children>}\n * ? Child extends Children ? Node : never\n * : never\n * : never\n * )} InternalParent\n * Collect nodes that can be parents of `Child`.\n * @template {UnistNode} Node\n * All node types in a tree.\n * @template {UnistNode} Child\n * Node to search for.\n */\n\n/**\n * @typedef {InternalParent<InclusiveDescendant<Tree>, Child>} Parent\n * Collect nodes in `Tree` that can be parents of `Child`.\n * @template {UnistNode} Tree\n * All node types in a tree.\n * @template {UnistNode} Child\n * Node to search for.\n */\n\n/**\n * @typedef {(\n * Depth extends Max\n * ? never\n * :\n * | InternalParent<Node, Child>\n * | InternalAncestor<Node, InternalParent<Node, Child>, Max, Increment<Depth>>\n * )} InternalAncestor\n * Collect nodes in `Tree` that can be ancestors of `Child`.\n * @template {UnistNode} Node\n * All node types in a tree.\n * @template {UnistNode} Child\n * Node to search for.\n * @template {Uint} [Max=10]\n * Max; searches up to this depth.\n * @template {Uint} [Depth=0]\n * Current depth.\n */\n\n/**\n * @typedef {InternalAncestor<InclusiveDescendant<Tree>, Child>} Ancestor\n * Collect nodes in `Tree` that can be ancestors of `Child`.\n * @template {UnistNode} Tree\n * All node types in a tree.\n * @template {UnistNode} Child\n * Node to search for.\n */\n\n/**\n * @typedef {(\n * Tree extends UnistParent\n * ? Depth extends Max\n * ? Tree\n * : Tree | InclusiveDescendant<Tree['children'][number], Max, Increment<Depth>>\n * : Tree\n * )} InclusiveDescendant\n * Collect all (inclusive) descendants of `Tree`.\n *\n * > 👉 **Note**: for performance reasons, this seems to be the fastest way to\n * > recurse without actually running into an infinite loop, which the\n * > previous version did.\n * >\n * > Practically, a max of `2` is typically enough assuming a `Root` is\n * > passed, but it doesn’t improve performance.\n * > It gets higher with `List > ListItem > Table > TableRow > TableCell`.\n * > Using up to `10` doesn’t hurt or help either.\n * @template {UnistNode} Tree\n * Tree type.\n * @template {Uint} [Max=10]\n * Max; searches up to this depth.\n * @template {Uint} [Depth=0]\n * Current depth.\n */\n\n/**\n * @typedef {'skip' | boolean} Action\n * Union of the action types.\n *\n * @typedef {number} Index\n * Move to the sibling at `index` next (after node itself is completely\n * traversed).\n *\n * Useful if mutating the tree, such as removing the node the visitor is\n * currently on, or any of its previous siblings.\n * Results less than 0 or greater than or equal to `children.length` stop\n * traversing the parent.\n *\n * @typedef {[(Action | null | undefined | void)?, (Index | null | undefined)?]} ActionTuple\n * List with one or two values, the first an action, the second an index.\n *\n * @typedef {Action | ActionTuple | Index | null | undefined | void} VisitorResult\n * Any value that can be returned from a visitor.\n */\n\n/**\n * @callback Visitor\n * Handle a node (matching `test`, if given).\n *\n * Visitors are free to transform `node`.\n * They can also transform the parent of node (the last of `ancestors`).\n *\n * Replacing `node` itself, if `SKIP` is not returned, still causes its\n * descendants to be walked (which is a bug).\n *\n * When adding or removing previous siblings of `node` (or next siblings, in\n * case of reverse), the `Visitor` should return a new `Index` to specify the\n * sibling to traverse after `node` is traversed.\n * Adding or removing next siblings of `node` (or previous siblings, in case\n * of reverse) is handled as expected without needing to return a new `Index`.\n *\n * Removing the children property of an ancestor still results in them being\n * traversed.\n * @param {Visited} node\n * Found node.\n * @param {Array<VisitedParents>} ancestors\n * Ancestors of `node`.\n * @returns {VisitorResult}\n * What to do next.\n *\n * An `Index` is treated as a tuple of `[CONTINUE, Index]`.\n * An `Action` is treated as a tuple of `[Action]`.\n *\n * Passing a tuple back only makes sense if the `Action` is `SKIP`.\n * When the `Action` is `EXIT`, that action can be returned.\n * When the `Action` is `CONTINUE`, `Index` can be returned.\n * @template {UnistNode} [Visited=UnistNode]\n * Visited node type.\n * @template {UnistParent} [VisitedParents=UnistParent]\n * Ancestor type.\n */\n\n/**\n * @typedef {Visitor<Matches<InclusiveDescendant<Tree>, Check>, Ancestor<Tree, Matches<InclusiveDescendant<Tree>, Check>>>} BuildVisitor\n * Build a typed `Visitor` function from a tree and a test.\n *\n * It will infer which values are passed as `node` and which as `parents`.\n * @template {UnistNode} [Tree=UnistNode]\n * Tree type.\n * @template {Test} [Check=Test]\n * Test type.\n */\n\nimport {convert} from 'unist-util-is'\nimport {color} from 'unist-util-visit-parents/do-not-use-color'\n\n/** @type {Readonly<ActionTuple>} */\nconst empty = []\n\n/**\n * Continue traversing as normal.\n */\nexport const CONTINUE = true\n\n/**\n * Stop traversing immediately.\n */\nexport const EXIT = false\n\n/**\n * Do not traverse this node’s children.\n */\nexport const SKIP = 'skip'\n\n/**\n * Visit nodes, with ancestral information.\n *\n * This algorithm performs *depth-first* *tree traversal* in *preorder*\n * (**NLR**) or if `reverse` is given, in *reverse preorder* (**NRL**).\n *\n * You can choose for which nodes `visitor` is called by passing a `test`.\n * For complex tests, you should test yourself in `visitor`, as it will be\n * faster and will have improved type information.\n *\n * Walking the tree is an intensive task.\n * Make use of the return values of the visitor when possible.\n * Instead of walking a tree multiple times, walk it once, use `unist-util-is`\n * to check if a node matches, and then perform different operations.\n *\n * You can change the tree.\n * See `Visitor` for more info.\n *\n * @overload\n * @param {Tree} tree\n * @param {Check} check\n * @param {BuildVisitor<Tree, Check>} visitor\n * @param {boolean | null | undefined} [reverse]\n * @returns {undefined}\n *\n * @overload\n * @param {Tree} tree\n * @param {BuildVisitor<Tree>} visitor\n * @param {boolean | null | undefined} [reverse]\n * @returns {undefined}\n *\n * @param {UnistNode} tree\n * Tree to traverse.\n * @param {Visitor | Test} test\n * `unist-util-is`-compatible test\n * @param {Visitor | boolean | null | undefined} [visitor]\n * Handle each node.\n * @param {boolean | null | undefined} [reverse]\n * Traverse in reverse preorder (NRL) instead of the default preorder (NLR).\n * @returns {undefined}\n * Nothing.\n *\n * @template {UnistNode} Tree\n * Node type.\n * @template {Test} Check\n * `unist-util-is`-compatible test.\n */\nexport function visitParents(tree, test, visitor, reverse) {\n /** @type {Test} */\n let check\n\n if (typeof test === 'function' && typeof visitor !== 'function') {\n reverse = visitor\n // @ts-expect-error no visitor given, so `visitor` is test.\n visitor = test\n } else {\n // @ts-expect-error visitor given, so `test` isn’t a visitor.\n check = test\n }\n\n const is = convert(check)\n const step = reverse ? -1 : 1\n\n factory(tree, undefined, [])()\n\n /**\n * @param {UnistNode} node\n * @param {number | undefined} index\n * @param {Array<UnistParent>} parents\n */\n function factory(node, index, parents) {\n const value = /** @type {Record<string, unknown>} */ (\n node && typeof node === 'object' ? node : {}\n )\n\n if (typeof value.type === 'string') {\n const name =\n // `hast`\n typeof value.tagName === 'string'\n ? value.tagName\n : // `xast`\n typeof value.name === 'string'\n ? value.name\n : undefined\n\n Object.defineProperty(visit, 'name', {\n value:\n 'node (' + color(node.type + (name ? '<' + name + '>' : '')) + ')'\n })\n }\n\n return visit\n\n function visit() {\n /** @type {Readonly<ActionTuple>} */\n let result = empty\n /** @type {Readonly<ActionTuple>} */\n let subresult\n /** @type {number} */\n let offset\n /** @type {Array<UnistParent>} */\n let grandparents\n\n if (!test || is(node, index, parents[parents.length - 1] || undefined)) {\n // @ts-expect-error: `visitor` is now a visitor.\n result = toResult(visitor(node, parents))\n\n if (result[0] === EXIT) {\n return result\n }\n }\n\n if ('children' in node && node.children) {\n const nodeAsParent = /** @type {UnistParent} */ (node)\n\n if (nodeAsParent.children && result[0] !== SKIP) {\n offset = (reverse ? nodeAsParent.children.length : -1) + step\n grandparents = parents.concat(nodeAsParent)\n\n while (offset > -1 && offset < nodeAsParent.children.length) {\n const child = nodeAsParent.children[offset]\n\n subresult = factory(child, offset, grandparents)()\n\n if (subresult[0] === EXIT) {\n return subresult\n }\n\n offset =\n typeof subresult[1] === 'number' ? subresult[1] : offset + step\n }\n }\n }\n\n return result\n }\n }\n}\n\n/**\n * Turn a return value into a clean result.\n *\n * @param {VisitorResult} value\n * Valid return values from visitors.\n * @returns {Readonly<ActionTuple>}\n * Clean result.\n */\nfunction toResult(value) {\n if (Array.isArray(value)) {\n return value\n }\n\n if (typeof value === 'number') {\n return [CONTINUE, value]\n }\n\n return value === null || value === undefined ? empty : [value]\n}\n","/**\n * @import {Nodes, Parents, PhrasingContent, Root, Text} from 'mdast'\n * @import {BuildVisitor, Test, VisitorResult} from 'unist-util-visit-parents'\n */\n\n/**\n * @typedef RegExpMatchObject\n * Info on the match.\n * @property {number} index\n * The index of the search at which the result was found.\n * @property {string} input\n * A copy of the search string in the text node.\n * @property {[...Array<Parents>, Text]} stack\n * All ancestors of the text node, where the last node is the text itself.\n *\n * @typedef {RegExp | string} Find\n * Pattern to find.\n *\n * Strings are escaped and then turned into global expressions.\n *\n * @typedef {Array<FindAndReplaceTuple>} FindAndReplaceList\n * Several find and replaces, in array form.\n *\n * @typedef {[Find, Replace?]} FindAndReplaceTuple\n * Find and replace in tuple form.\n *\n * @typedef {ReplaceFunction | string | null | undefined} Replace\n * Thing to replace with.\n *\n * @callback ReplaceFunction\n * Callback called when a search matches.\n * @param {...any} parameters\n * The parameters are the result of corresponding search expression:\n *\n * * `value` (`string`) — whole match\n * * `...capture` (`Array<string>`) — matches from regex capture groups\n * * `match` (`RegExpMatchObject`) — info on the match\n * @returns {Array<PhrasingContent> | PhrasingContent | string | false | null | undefined}\n * Thing to replace with.\n *\n * * when `null`, `undefined`, `''`, remove the match\n * * …or when `false`, do not replace at all\n * * …or when `string`, replace with a text node of that value\n * * …or when `Node` or `Array<Node>`, replace with those nodes\n *\n * @typedef {[RegExp, ReplaceFunction]} Pair\n * Normalized find and replace.\n *\n * @typedef {Array<Pair>} Pairs\n * All find and replaced.\n *\n * @typedef Options\n * Configuration.\n * @property {Test | null | undefined} [ignore]\n * Test for which nodes to ignore (optional).\n */\n\nimport escape from 'escape-string-regexp'\nimport {visitParents} from 'unist-util-visit-parents'\nimport {convert} from 'unist-util-is'\n\n/**\n * Find patterns in a tree and replace them.\n *\n * The algorithm searches the tree in *preorder* for complete values in `Text`\n * nodes.\n * Partial matches are not supported.\n *\n * @param {Nodes} tree\n * Tree to change.\n * @param {FindAndReplaceList | FindAndReplaceTuple} list\n * Patterns to find.\n * @param {Options | null | undefined} [options]\n * Configuration (when `find` is not `Find`).\n * @returns {undefined}\n * Nothing.\n */\nexport function findAndReplace(tree, list, options) {\n const settings = options || {}\n const ignored = convert(settings.ignore || [])\n const pairs = toPairs(list)\n let pairIndex = -1\n\n while (++pairIndex < pairs.length) {\n visitParents(tree, 'text', visitor)\n }\n\n /** @type {BuildVisitor<Root, 'text'>} */\n function visitor(node, parents) {\n let index = -1\n /** @type {Parents | undefined} */\n let grandparent\n\n while (++index < parents.length) {\n const parent = parents[index]\n /** @type {Array<Nodes> | undefined} */\n const siblings = grandparent ? grandparent.children : undefined\n\n if (\n ignored(\n parent,\n siblings ? siblings.indexOf(parent) : undefined,\n grandparent\n )\n ) {\n return\n }\n\n grandparent = parent\n }\n\n if (grandparent) {\n return handler(node, parents)\n }\n }\n\n /**\n * Handle a text node which is not in an ignored parent.\n *\n * @param {Text} node\n * Text node.\n * @param {Array<Parents>} parents\n * Parents.\n * @returns {VisitorResult}\n * Result.\n */\n function handler(node, parents) {\n const parent = parents[parents.length - 1]\n const find = pairs[pairIndex][0]\n const replace = pairs[pairIndex][1]\n let start = 0\n /** @type {Array<Nodes>} */\n const siblings = parent.children\n const index = siblings.indexOf(node)\n let change = false\n /** @type {Array<PhrasingContent>} */\n let nodes = []\n\n find.lastIndex = 0\n\n let match = find.exec(node.value)\n\n while (match) {\n const position = match.index\n /** @type {RegExpMatchObject} */\n const matchObject = {\n index: match.index,\n input: match.input,\n stack: [...parents, node]\n }\n let value = replace(...match, matchObject)\n\n if (typeof value === 'string') {\n value = value.length > 0 ? {type: 'text', value} : undefined\n }\n\n // It wasn’t a match after all.\n if (value === false) {\n // False acts as if there was no match.\n // So we need to reset `lastIndex`, which currently being at the end of\n // the current match, to the beginning.\n find.lastIndex = position + 1\n } else {\n if (start !== position) {\n nodes.push({\n type: 'text',\n value: node.value.slice(start, position)\n })\n }\n\n if (Array.isArray(value)) {\n nodes.push(...value)\n } else if (value) {\n nodes.push(value)\n }\n\n start = position + match[0].length\n change = true\n }\n\n if (!find.global) {\n break\n }\n\n match = find.exec(node.value)\n }\n\n if (change) {\n if (start < node.value.length) {\n nodes.push({type: 'text', value: node.value.slice(start)})\n }\n\n parent.children.splice(index, 1, ...nodes)\n } else {\n nodes = [node]\n }\n\n return index + nodes.length\n }\n}\n\n/**\n * Turn a tuple or a list of tuples into pairs.\n *\n * @param {FindAndReplaceList | FindAndReplaceTuple} tupleOrList\n * Schema.\n * @returns {Pairs}\n * Clean pairs.\n */\nfunction toPairs(tupleOrList) {\n /** @type {Pairs} */\n const result = []\n\n if (!Array.isArray(tupleOrList)) {\n throw new TypeError('Expected find and replace tuple or list of tuples')\n }\n\n /** @type {FindAndReplaceList} */\n // @ts-expect-error: correct.\n const list =\n !tupleOrList[0] || Array.isArray(tupleOrList[0])\n ? tupleOrList\n : [tupleOrList]\n\n let index = -1\n\n while (++index < list.length) {\n const tuple = list[index]\n result.push([toExpression(tuple[0]), toFunction(tuple[1])])\n }\n\n return result\n}\n\n/**\n * Turn a find into an expression.\n *\n * @param {Find} find\n * Find.\n * @returns {RegExp}\n * Expression.\n */\nfunction toExpression(find) {\n return typeof find === 'string' ? new RegExp(escape(find), 'g') : find\n}\n\n/**\n * Turn a replace into a function.\n *\n * @param {Replace} replace\n * Replace.\n * @returns {ReplaceFunction}\n * Function.\n */\nfunction toFunction(replace) {\n return typeof replace === 'function'\n ? replace\n : function () {\n return replace\n }\n}\n","/**\n * @import {RegExpMatchObject, ReplaceFunction} from 'mdast-util-find-and-replace'\n * @import {CompileContext, Extension as FromMarkdownExtension, Handle as FromMarkdownHandle, Transform as FromMarkdownTransform} from 'mdast-util-from-markdown'\n * @import {ConstructName, Options as ToMarkdownExtension} from 'mdast-util-to-markdown'\n * @import {Link, PhrasingContent} from 'mdast'\n */\n\nimport {ccount} from 'ccount'\nimport {ok as assert} from 'devlop'\nimport {unicodePunctuation, unicodeWhitespace} from 'micromark-util-character'\nimport {findAndReplace} from 'mdast-util-find-and-replace'\n\n/** @type {ConstructName} */\nconst inConstruct = 'phrasing'\n/** @type {Array<ConstructName>} */\nconst notInConstruct = ['autolink', 'link', 'image', 'label']\n\n/**\n * Create an extension for `mdast-util-from-markdown` to enable GFM autolink\n * literals in markdown.\n *\n * @returns {FromMarkdownExtension}\n * Extension for `mdast-util-to-markdown` to enable GFM autolink literals.\n */\nexport function gfmAutolinkLiteralFromMarkdown() {\n return {\n transforms: [transformGfmAutolinkLiterals],\n enter: {\n literalAutolink: enterLiteralAutolink,\n literalAutolinkEmail: enterLiteralAutolinkValue,\n literalAutolinkHttp: enterLiteralAutolinkValue,\n literalAutolinkWww: enterLiteralAutolinkValue\n },\n exit: {\n literalAutolink: exitLiteralAutolink,\n literalAutolinkEmail: exitLiteralAutolinkEmail,\n literalAutolinkHttp: exitLiteralAutolinkHttp,\n literalAutolinkWww: exitLiteralAutolinkWww\n }\n }\n}\n\n/**\n * Create an extension for `mdast-util-to-markdown` to enable GFM autolink\n * literals in markdown.\n *\n * @returns {ToMarkdownExtension}\n * Extension for `mdast-util-to-markdown` to enable GFM autolink literals.\n */\nexport function gfmAutolinkLiteralToMarkdown() {\n return {\n unsafe: [\n {\n character: '@',\n before: '[+\\\\-.\\\\w]',\n after: '[\\\\-.\\\\w]',\n inConstruct,\n notInConstruct\n },\n {\n character: '.',\n before: '[Ww]',\n after: '[\\\\-.\\\\w]',\n inConstruct,\n notInConstruct\n },\n {\n character: ':',\n before: '[ps]',\n after: '\\\\/',\n inConstruct,\n notInConstruct\n }\n ]\n }\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterLiteralAutolink(token) {\n this.enter({type: 'link', title: null, url: '', children: []}, token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterLiteralAutolinkValue(token) {\n this.config.enter.autolinkProtocol.call(this, token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitLiteralAutolinkHttp(token) {\n this.config.exit.autolinkProtocol.call(this, token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitLiteralAutolinkWww(token) {\n this.config.exit.data.call(this, token)\n const node = this.stack[this.stack.length - 1]\n assert(node.type === 'link')\n node.url = 'http://' + this.sliceSerialize(token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitLiteralAutolinkEmail(token) {\n this.config.exit.autolinkEmail.call(this, token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitLiteralAutolink(token) {\n this.exit(token)\n}\n\n/** @type {FromMarkdownTransform} */\nfunction transformGfmAutolinkLiterals(tree) {\n findAndReplace(\n tree,\n [\n [/(https?:\\/\\/|www(?=\\.))([-.\\w]+)([^ \\t\\r\\n]*)/gi, findUrl],\n [/(?<=^|\\s|\\p{P}|\\p{S})([-.\\w+]+)@([-\\w]+(?:\\.[-\\w]+)+)/gu, findEmail]\n ],\n {ignore: ['link', 'linkReference']}\n )\n}\n\n/**\n * @type {ReplaceFunction}\n * @param {string} _\n * @param {string} protocol\n * @param {string} domain\n * @param {string} path\n * @param {RegExpMatchObject} match\n * @returns {Array<PhrasingContent> | Link | false}\n */\n// eslint-disable-next-line max-params\nfunction findUrl(_, protocol, domain, path, match) {\n let prefix = ''\n\n // Not an expected previous character.\n if (!previous(match)) {\n return false\n }\n\n // Treat `www` as part of the domain.\n if (/^w/i.test(protocol)) {\n domain = protocol + domain\n protocol = ''\n prefix = 'http://'\n }\n\n if (!isCorrectDomain(domain)) {\n return false\n }\n\n const parts = splitUrl(domain + path)\n\n if (!parts[0]) return false\n\n /** @type {Link} */\n const result = {\n type: 'link',\n title: null,\n url: prefix + protocol + parts[0],\n children: [{type: 'text', value: protocol + parts[0]}]\n }\n\n if (parts[1]) {\n return [result, {type: 'text', value: parts[1]}]\n }\n\n return result\n}\n\n/**\n * @type {ReplaceFunction}\n * @param {string} _\n * @param {string} atext\n * @param {string} label\n * @param {RegExpMatchObject} match\n * @returns {Link | false}\n */\nfunction findEmail(_, atext, label, match) {\n if (\n // Not an expected previous character.\n !previous(match, true) ||\n // Label ends in not allowed character.\n /[-\\d_]$/.test(label)\n ) {\n return false\n }\n\n return {\n type: 'link',\n title: null,\n url: 'mailto:' + atext + '@' + label,\n children: [{type: 'text', value: atext + '@' + label}]\n }\n}\n\n/**\n * @param {string} domain\n * @returns {boolean}\n */\nfunction isCorrectDomain(domain) {\n const parts = domain.split('.')\n\n if (\n parts.length < 2 ||\n (parts[parts.length - 1] &&\n (/_/.test(parts[parts.length - 1]) ||\n !/[a-zA-Z\\d]/.test(parts[parts.length - 1]))) ||\n (parts[parts.length - 2] &&\n (/_/.test(parts[parts.length - 2]) ||\n !/[a-zA-Z\\d]/.test(parts[parts.length - 2])))\n ) {\n return false\n }\n\n return true\n}\n\n/**\n * @param {string} url\n * @returns {[string, string | undefined]}\n */\nfunction splitUrl(url) {\n const trailExec = /[!\"&'),.:;<>?\\]}]+$/.exec(url)\n\n if (!trailExec) {\n return [url, undefined]\n }\n\n url = url.slice(0, trailExec.index)\n\n let trail = trailExec[0]\n let closingParenIndex = trail.indexOf(')')\n const openingParens = ccount(url, '(')\n let closingParens = ccount(url, ')')\n\n while (closingParenIndex !== -1 && openingParens > closingParens) {\n url += trail.slice(0, closingParenIndex + 1)\n trail = trail.slice(closingParenIndex + 1)\n closingParenIndex = trail.indexOf(')')\n closingParens++\n }\n\n return [url, trail]\n}\n\n/**\n * @param {RegExpMatchObject} match\n * @param {boolean | null | undefined} [email=false]\n * @returns {boolean}\n */\nfunction previous(match, email) {\n const code = match.input.charCodeAt(match.index - 1)\n\n return (\n (match.index === 0 ||\n unicodeWhitespace(code) ||\n unicodePunctuation(code)) &&\n // If it’s an email, the previous character should not be a slash.\n (!email || code !== 47)\n )\n}\n","/**\n * Normalize an identifier (as found in references, definitions).\n *\n * Collapses markdown whitespace, trim, and then lower- and uppercase.\n *\n * Some characters are considered “uppercase”, such as U+03F4 (`ϴ`), but if their\n * lowercase counterpart (U+03B8 (`θ`)) is uppercased will result in a different\n * uppercase character (U+0398 (`Θ`)).\n * So, to get a canonical form, we perform both lower- and uppercase.\n *\n * Using uppercase last makes sure keys will never interact with default\n * prototypal values (such as `constructor`): nothing in the prototype of\n * `Object` is uppercase.\n *\n * @param {string} value\n * Identifier to normalize.\n * @returns {string}\n * Normalized identifier.\n */\nexport function normalizeIdentifier(value) {\n return value\n // Collapse markdown whitespace.\n .replace(/[\\t\\n\\r ]+/g, \" \")\n // Trim.\n .replace(/^ | $/g, '')\n // Some characters are considered “uppercase”, but if their lowercase\n // counterpart is uppercased will result in a different uppercase\n // character.\n // Hence, to get that form, we perform both lower- and uppercase.\n // Upper case makes sure keys will not interact with default prototypal\n // methods: no method is uppercase.\n .toLowerCase().toUpperCase();\n}","/**\n * @import {\n * CompileContext,\n * Extension as FromMarkdownExtension,\n * Handle as FromMarkdownHandle\n * } from 'mdast-util-from-markdown'\n * @import {ToMarkdownOptions} from 'mdast-util-gfm-footnote'\n * @import {\n * Handle as ToMarkdownHandle,\n * Map,\n * Options as ToMarkdownExtension\n * } from 'mdast-util-to-markdown'\n * @import {FootnoteDefinition, FootnoteReference} from 'mdast'\n */\n\nimport {ok as assert} from 'devlop'\nimport {normalizeIdentifier} from 'micromark-util-normalize-identifier'\n\nfootnoteReference.peek = footnoteReferencePeek\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterFootnoteCallString() {\n this.buffer()\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterFootnoteCall(token) {\n this.enter({type: 'footnoteReference', identifier: '', label: ''}, token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterFootnoteDefinitionLabelString() {\n this.buffer()\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterFootnoteDefinition(token) {\n this.enter(\n {type: 'footnoteDefinition', identifier: '', label: '', children: []},\n token\n )\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitFootnoteCallString(token) {\n const label = this.resume()\n const node = this.stack[this.stack.length - 1]\n assert(node.type === 'footnoteReference')\n node.identifier = normalizeIdentifier(\n this.sliceSerialize(token)\n ).toLowerCase()\n node.label = label\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitFootnoteCall(token) {\n this.exit(token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitFootnoteDefinitionLabelString(token) {\n const label = this.resume()\n const node = this.stack[this.stack.length - 1]\n assert(node.type === 'footnoteDefinition')\n node.identifier = normalizeIdentifier(\n this.sliceSerialize(token)\n ).toLowerCase()\n node.label = label\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitFootnoteDefinition(token) {\n this.exit(token)\n}\n\n/** @type {ToMarkdownHandle} */\nfunction footnoteReferencePeek() {\n return '['\n}\n\n/**\n * @type {ToMarkdownHandle}\n * @param {FootnoteReference} node\n */\nfunction footnoteReference(node, _, state, info) {\n const tracker = state.createTracker(info)\n let value = tracker.move('[^')\n const exit = state.enter('footnoteReference')\n const subexit = state.enter('reference')\n value += tracker.move(\n state.safe(state.associationId(node), {after: ']', before: value})\n )\n subexit()\n exit()\n value += tracker.move(']')\n return value\n}\n\n/**\n * Create an extension for `mdast-util-from-markdown` to enable GFM footnotes\n * in markdown.\n *\n * @returns {FromMarkdownExtension}\n * Extension for `mdast-util-from-markdown`.\n */\nexport function gfmFootnoteFromMarkdown() {\n return {\n enter: {\n gfmFootnoteCallString: enterFootnoteCallString,\n gfmFootnoteCall: enterFootnoteCall,\n gfmFootnoteDefinitionLabelString: enterFootnoteDefinitionLabelString,\n gfmFootnoteDefinition: enterFootnoteDefinition\n },\n exit: {\n gfmFootnoteCallString: exitFootnoteCallString,\n gfmFootnoteCall: exitFootnoteCall,\n gfmFootnoteDefinitionLabelString: exitFootnoteDefinitionLabelString,\n gfmFootnoteDefinition: exitFootnoteDefinition\n }\n }\n}\n\n/**\n * Create an extension for `mdast-util-to-markdown` to enable GFM footnotes\n * in markdown.\n *\n * @param {ToMarkdownOptions | null | undefined} [options]\n * Configuration (optional).\n * @returns {ToMarkdownExtension}\n * Extension for `mdast-util-to-markdown`.\n */\nexport function gfmFootnoteToMarkdown(options) {\n // To do: next major: change default.\n let firstLineBlank = false\n\n if (options && options.firstLineBlank) {\n firstLineBlank = true\n }\n\n return {\n handlers: {footnoteDefinition, footnoteReference},\n // This is on by default already.\n unsafe: [{character: '[', inConstruct: ['label', 'phrasing', 'reference']}]\n }\n\n /**\n * @type {ToMarkdownHandle}\n * @param {FootnoteDefinition} node\n */\n function footnoteDefinition(node, _, state, info) {\n const tracker = state.createTracker(info)\n let value = tracker.move('[^')\n const exit = state.enter('footnoteDefinition')\n const subexit = state.enter('label')\n value += tracker.move(\n state.safe(state.associationId(node), {before: value, after: ']'})\n )\n subexit()\n\n value += tracker.move(']:')\n\n if (node.children && node.children.length > 0) {\n tracker.shift(4)\n\n value += tracker.move(\n (firstLineBlank ? '\\n' : ' ') +\n state.indentLines(\n state.containerFlow(node, tracker.current()),\n firstLineBlank ? mapAll : mapExceptFirst\n )\n )\n }\n\n exit()\n\n return value\n }\n}\n\n/** @type {Map} */\nfunction mapExceptFirst(line, index, blank) {\n return index === 0 ? line : mapAll(line, index, blank)\n}\n\n/** @type {Map} */\nfunction mapAll(line, index, blank) {\n return (blank ? '' : ' ') + line\n}\n","/**\n * @typedef {import('mdast').Delete} Delete\n *\n * @typedef {import('mdast-util-from-markdown').CompileContext} CompileContext\n * @typedef {import('mdast-util-from-markdown').Extension} FromMarkdownExtension\n * @typedef {import('mdast-util-from-markdown').Handle} FromMarkdownHandle\n *\n * @typedef {import('mdast-util-to-markdown').ConstructName} ConstructName\n * @typedef {import('mdast-util-to-markdown').Handle} ToMarkdownHandle\n * @typedef {import('mdast-util-to-markdown').Options} ToMarkdownExtension\n */\n\n/**\n * List of constructs that occur in phrasing (paragraphs, headings), but cannot\n * contain strikethrough.\n * So they sort of cancel each other out.\n * Note: could use a better name.\n *\n * Note: keep in sync with: <https://github.com/syntax-tree/mdast-util-to-markdown/blob/8ce8dbf/lib/unsafe.js#L14>\n *\n * @type {Array<ConstructName>}\n */\nconst constructsWithoutStrikethrough = [\n 'autolink',\n 'destinationLiteral',\n 'destinationRaw',\n 'reference',\n 'titleQuote',\n 'titleApostrophe'\n]\n\nhandleDelete.peek = peekDelete\n\n/**\n * Create an extension for `mdast-util-from-markdown` to enable GFM\n * strikethrough in markdown.\n *\n * @returns {FromMarkdownExtension}\n * Extension for `mdast-util-from-markdown` to enable GFM strikethrough.\n */\nexport function gfmStrikethroughFromMarkdown() {\n return {\n canContainEols: ['delete'],\n enter: {strikethrough: enterStrikethrough},\n exit: {strikethrough: exitStrikethrough}\n }\n}\n\n/**\n * Create an extension for `mdast-util-to-markdown` to enable GFM\n * strikethrough in markdown.\n *\n * @returns {ToMarkdownExtension}\n * Extension for `mdast-util-to-markdown` to enable GFM strikethrough.\n */\nexport function gfmStrikethroughToMarkdown() {\n return {\n unsafe: [\n {\n character: '~',\n inConstruct: 'phrasing',\n notInConstruct: constructsWithoutStrikethrough\n }\n ],\n handlers: {delete: handleDelete}\n }\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterStrikethrough(token) {\n this.enter({type: 'delete', children: []}, token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitStrikethrough(token) {\n this.exit(token)\n}\n\n/**\n * @type {ToMarkdownHandle}\n * @param {Delete} node\n */\nfunction handleDelete(node, _, state, info) {\n const tracker = state.createTracker(info)\n const exit = state.enter('strikethrough')\n let value = tracker.move('~~')\n value += state.containerPhrasing(node, {\n ...tracker.current(),\n before: value,\n after: '~'\n })\n value += tracker.move('~~')\n exit()\n return value\n}\n\n/** @type {ToMarkdownHandle} */\nfunction peekDelete() {\n return '~'\n}\n","// To do: next major: remove.\n/**\n * @typedef {Options} MarkdownTableOptions\n * Configuration.\n */\n\n/**\n * @typedef Options\n * Configuration.\n * @property {boolean | null | undefined} [alignDelimiters=true]\n * Whether to align the delimiters (default: `true`);\n * they are aligned by default:\n *\n * ```markdown\n * | Alpha | B |\n * | ----- | ----- |\n * | C | Delta |\n * ```\n *\n * Pass `false` to make them staggered:\n *\n * ```markdown\n * | Alpha | B |\n * | - | - |\n * | C | Delta |\n * ```\n * @property {ReadonlyArray<string | null | undefined> | string | null | undefined} [align]\n * How to align columns (default: `''`);\n * one style for all columns or styles for their respective columns;\n * each style is either `'l'` (left), `'r'` (right), or `'c'` (center);\n * other values are treated as `''`, which doesn’t place the colon in the\n * alignment row but does align left;\n * *only the lowercased first character is used, so `Right` is fine.*\n * @property {boolean | null | undefined} [delimiterEnd=true]\n * Whether to end each row with the delimiter (default: `true`).\n *\n * > 👉 **Note**: please don’t use this: it could create fragile structures\n * > that aren’t understandable to some markdown parsers.\n *\n * When `true`, there are ending delimiters:\n *\n * ```markdown\n * | Alpha | B |\n * | ----- | ----- |\n * | C | Delta |\n * ```\n *\n * When `false`, there are no ending delimiters:\n *\n * ```markdown\n * | Alpha | B\n * | ----- | -----\n * | C | Delta\n * ```\n * @property {boolean | null | undefined} [delimiterStart=true]\n * Whether to begin each row with the delimiter (default: `true`).\n *\n * > 👉 **Note**: please don’t use this: it could create fragile structures\n * > that aren’t understandable to some markdown parsers.\n *\n * When `true`, there are starting delimiters:\n *\n * ```markdown\n * | Alpha | B |\n * | ----- | ----- |\n * | C | Delta |\n * ```\n *\n * When `false`, there are no starting delimiters:\n *\n * ```markdown\n * Alpha | B |\n * ----- | ----- |\n * C | Delta |\n * ```\n * @property {boolean | null | undefined} [padding=true]\n * Whether to add a space of padding between delimiters and cells\n * (default: `true`).\n *\n * When `true`, there is padding:\n *\n * ```markdown\n * | Alpha | B |\n * | ----- | ----- |\n * | C | Delta |\n * ```\n *\n * When `false`, there is no padding:\n *\n * ```markdown\n * |Alpha|B |\n * |-----|-----|\n * |C |Delta|\n * ```\n * @property {((value: string) => number) | null | undefined} [stringLength]\n * Function to detect the length of table cell content (optional);\n * this is used when aligning the delimiters (`|`) between table cells;\n * full-width characters and emoji mess up delimiter alignment when viewing\n * the markdown source;\n * to fix this, you can pass this function,\n * which receives the cell content and returns its “visible” size;\n * note that what is and isn’t visible depends on where the text is displayed.\n *\n * Without such a function, the following:\n *\n * ```js\n * markdownTable([\n * ['Alpha', 'Bravo'],\n * ['中文', 'Charlie'],\n * ['👩❤️👩', 'Delta']\n * ])\n * ```\n *\n * Yields:\n *\n * ```markdown\n * | Alpha | Bravo |\n * | - | - |\n * | 中文 | Charlie |\n * | 👩❤️👩 | Delta |\n * ```\n *\n * With [`string-width`](https://github.com/sindresorhus/string-width):\n *\n * ```js\n * import stringWidth from 'string-width'\n *\n * markdownTable(\n * [\n * ['Alpha', 'Bravo'],\n * ['中文', 'Charlie'],\n * ['👩❤️👩', 'Delta']\n * ],\n * {stringLength: stringWidth}\n * )\n * ```\n *\n * Yields:\n *\n * ```markdown\n * | Alpha | Bravo |\n * | ----- | ------- |\n * | 中文 | Charlie |\n * | 👩❤️👩 | Delta |\n * ```\n */\n\n/**\n * @param {string} value\n * Cell value.\n * @returns {number}\n * Cell size.\n */\nfunction defaultStringLength(value) {\n return value.length\n}\n\n/**\n * Generate a markdown\n * ([GFM](https://docs.github.com/en/github/writing-on-github/working-with-advanced-formatting/organizing-information-with-tables))\n * table.\n *\n * @param {ReadonlyArray<ReadonlyArray<string | null | undefined>>} table\n * Table data (matrix of strings).\n * @param {Readonly<Options> | null | undefined} [options]\n * Configuration (optional).\n * @returns {string}\n * Result.\n */\nexport function markdownTable(table, options) {\n const settings = options || {}\n // To do: next major: change to spread.\n const align = (settings.align || []).concat()\n const stringLength = settings.stringLength || defaultStringLength\n /** @type {Array<number>} Character codes as symbols for alignment per column. */\n const alignments = []\n /** @type {Array<Array<string>>} Cells per row. */\n const cellMatrix = []\n /** @type {Array<Array<number>>} Sizes of each cell per row. */\n const sizeMatrix = []\n /** @type {Array<number>} */\n const longestCellByColumn = []\n let mostCellsPerRow = 0\n let rowIndex = -1\n\n // This is a superfluous loop if we don’t align delimiters, but otherwise we’d\n // do superfluous work when aligning, so optimize for aligning.\n while (++rowIndex < table.length) {\n /** @type {Array<string>} */\n const row = []\n /** @type {Array<number>} */\n const sizes = []\n let columnIndex = -1\n\n if (table[rowIndex].length > mostCellsPerRow) {\n mostCellsPerRow = table[rowIndex].length\n }\n\n while (++columnIndex < table[rowIndex].length) {\n const cell = serialize(table[rowIndex][columnIndex])\n\n if (settings.alignDelimiters !== false) {\n const size = stringLength(cell)\n sizes[columnIndex] = size\n\n if (\n longestCellByColumn[columnIndex] === undefined ||\n size > longestCellByColumn[columnIndex]\n ) {\n longestCellByColumn[columnIndex] = size\n }\n }\n\n row.push(cell)\n }\n\n cellMatrix[rowIndex] = row\n sizeMatrix[rowIndex] = sizes\n }\n\n // Figure out which alignments to use.\n let columnIndex = -1\n\n if (typeof align === 'object' && 'length' in align) {\n while (++columnIndex < mostCellsPerRow) {\n alignments[columnIndex] = toAlignment(align[columnIndex])\n }\n } else {\n const code = toAlignment(align)\n\n while (++columnIndex < mostCellsPerRow) {\n alignments[columnIndex] = code\n }\n }\n\n // Inject the alignment row.\n columnIndex = -1\n /** @type {Array<string>} */\n const row = []\n /** @type {Array<number>} */\n const sizes = []\n\n while (++columnIndex < mostCellsPerRow) {\n const code = alignments[columnIndex]\n let before = ''\n let after = ''\n\n if (code === 99 /* `c` */) {\n before = ':'\n after = ':'\n } else if (code === 108 /* `l` */) {\n before = ':'\n } else if (code === 114 /* `r` */) {\n after = ':'\n }\n\n // There *must* be at least one hyphen-minus in each alignment cell.\n let size =\n settings.alignDelimiters === false\n ? 1\n : Math.max(\n 1,\n longestCellByColumn[columnIndex] - before.length - after.length\n )\n\n const cell = before + '-'.repeat(size) + after\n\n if (settings.alignDelimiters !== false) {\n size = before.length + size + after.length\n\n if (size > longestCellByColumn[columnIndex]) {\n longestCellByColumn[columnIndex] = size\n }\n\n sizes[columnIndex] = size\n }\n\n row[columnIndex] = cell\n }\n\n // Inject the alignment row.\n cellMatrix.splice(1, 0, row)\n sizeMatrix.splice(1, 0, sizes)\n\n rowIndex = -1\n /** @type {Array<string>} */\n const lines = []\n\n while (++rowIndex < cellMatrix.length) {\n const row = cellMatrix[rowIndex]\n const sizes = sizeMatrix[rowIndex]\n columnIndex = -1\n /** @type {Array<string>} */\n const line = []\n\n while (++columnIndex < mostCellsPerRow) {\n const cell = row[columnIndex] || ''\n let before = ''\n let after = ''\n\n if (settings.alignDelimiters !== false) {\n const size =\n longestCellByColumn[columnIndex] - (sizes[columnIndex] || 0)\n const code = alignments[columnIndex]\n\n if (code === 114 /* `r` */) {\n before = ' '.repeat(size)\n } else if (code === 99 /* `c` */) {\n if (size % 2) {\n before = ' '.repeat(size / 2 + 0.5)\n after = ' '.repeat(size / 2 - 0.5)\n } else {\n before = ' '.repeat(size / 2)\n after = before\n }\n } else {\n after = ' '.repeat(size)\n }\n }\n\n if (settings.delimiterStart !== false && !columnIndex) {\n line.push('|')\n }\n\n if (\n settings.padding !== false &&\n // Don’t add the opening space if we’re not aligning and the cell is\n // empty: there will be a closing space.\n !(settings.alignDelimiters === false && cell === '') &&\n (settings.delimiterStart !== false || columnIndex)\n ) {\n line.push(' ')\n }\n\n if (settings.alignDelimiters !== false) {\n line.push(before)\n }\n\n line.push(cell)\n\n if (settings.alignDelimiters !== false) {\n line.push(after)\n }\n\n if (settings.padding !== false) {\n line.push(' ')\n }\n\n if (\n settings.delimiterEnd !== false ||\n columnIndex !== mostCellsPerRow - 1\n ) {\n line.push('|')\n }\n }\n\n lines.push(\n settings.delimiterEnd === false\n ? line.join('').replace(/ +$/, '')\n : line.join('')\n )\n }\n\n return lines.join('\\n')\n}\n\n/**\n * @param {string | null | undefined} [value]\n * Value to serialize.\n * @returns {string}\n * Result.\n */\nfunction serialize(value) {\n return value === null || value === undefined ? '' : String(value)\n}\n\n/**\n * @param {string | null | undefined} value\n * Value.\n * @returns {number}\n * Alignment.\n */\nfunction toAlignment(value) {\n const code = typeof value === 'string' ? value.codePointAt(0) : 0\n\n return code === 67 /* `C` */ || code === 99 /* `c` */\n ? 99 /* `c` */\n : code === 76 /* `L` */ || code === 108 /* `l` */\n ? 108 /* `l` */\n : code === 82 /* `R` */ || code === 114 /* `r` */\n ? 114 /* `r` */\n : 0\n}\n","/**\n * @import {Blockquote, Parents} from 'mdast'\n * @import {Info, Map, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {Blockquote} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function blockquote(node, _, state, info) {\n const exit = state.enter('blockquote')\n const tracker = state.createTracker(info)\n tracker.move('> ')\n tracker.shift(2)\n const value = state.indentLines(\n state.containerFlow(node, tracker.current()),\n map\n )\n exit()\n return value\n}\n\n/** @type {Map} */\nfunction map(line, _, blank) {\n return '>' + (blank ? '' : ' ') + line\n}\n","/**\n * @import {ConstructName, Unsafe} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {Array<ConstructName>} stack\n * @param {Unsafe} pattern\n * @returns {boolean}\n */\nexport function patternInScope(stack, pattern) {\n return (\n listInScope(stack, pattern.inConstruct, true) &&\n !listInScope(stack, pattern.notInConstruct, false)\n )\n}\n\n/**\n * @param {Array<ConstructName>} stack\n * @param {Unsafe['inConstruct']} list\n * @param {boolean} none\n * @returns {boolean}\n */\nfunction listInScope(stack, list, none) {\n if (typeof list === 'string') {\n list = [list]\n }\n\n if (!list || list.length === 0) {\n return none\n }\n\n let index = -1\n\n while (++index < list.length) {\n if (stack.includes(list[index])) {\n return true\n }\n }\n\n return false\n}\n","/**\n * @import {Break, Parents} from 'mdast'\n * @import {Info, State} from 'mdast-util-to-markdown'\n */\n\nimport {patternInScope} from '../util/pattern-in-scope.js'\n\n/**\n * @param {Break} _\n * @param {Parents | undefined} _1\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function hardBreak(_, _1, state, info) {\n let index = -1\n\n while (++index < state.unsafe.length) {\n // If we can’t put eols in this construct (setext headings, tables), use a\n // space instead.\n if (\n state.unsafe[index].character === '\\n' &&\n patternInScope(state.stack, state.unsafe[index])\n ) {\n return /[ \\t]/.test(info.before) ? '' : ' '\n }\n }\n\n return '\\\\\\n'\n}\n","/**\n * Get the count of the longest repeating streak of `substring` in `value`.\n *\n * @param {string} value\n * Content to search in.\n * @param {string} substring\n * Substring to look for, typically one character.\n * @returns {number}\n * Count of most frequent adjacent `substring`s in `value`.\n */\nexport function longestStreak(value, substring) {\n const source = String(value)\n let index = source.indexOf(substring)\n let expected = index\n let count = 0\n let max = 0\n\n if (typeof substring !== 'string') {\n throw new TypeError('Expected substring')\n }\n\n while (index !== -1) {\n if (index === expected) {\n if (++count > max) {\n max = count\n }\n } else {\n count = 1\n }\n\n expected = index + substring.length\n index = source.indexOf(substring, expected)\n }\n\n return max\n}\n","/**\n * @import {State} from 'mdast-util-to-markdown'\n * @import {Code} from 'mdast'\n */\n\n/**\n * @param {Code} node\n * @param {State} state\n * @returns {boolean}\n */\nexport function formatCodeAsIndented(node, state) {\n return Boolean(\n state.options.fences === false &&\n node.value &&\n // If there’s no info…\n !node.lang &&\n // And there’s a non-whitespace character…\n /[^ \\r\\n]/.test(node.value) &&\n // And the value doesn’t start or end in a blank…\n !/^[\\t ]*(?:[\\r\\n]|$)|(?:^|[\\r\\n])[\\t ]*$/.test(node.value)\n )\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['fence'], null | undefined>}\n */\nexport function checkFence(state) {\n const marker = state.options.fence || '`'\n\n if (marker !== '`' && marker !== '~') {\n throw new Error(\n 'Cannot serialize code with `' +\n marker +\n '` for `options.fence`, expected `` ` `` or `~`'\n )\n }\n\n return marker\n}\n","/**\n * @import {Info, Map, State} from 'mdast-util-to-markdown'\n * @import {Code, Parents} from 'mdast'\n */\n\nimport {longestStreak} from 'longest-streak'\nimport {formatCodeAsIndented} from '../util/format-code-as-indented.js'\nimport {checkFence} from '../util/check-fence.js'\n\n/**\n * @param {Code} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function code(node, _, state, info) {\n const marker = checkFence(state)\n const raw = node.value || ''\n const suffix = marker === '`' ? 'GraveAccent' : 'Tilde'\n\n if (formatCodeAsIndented(node, state)) {\n const exit = state.enter('codeIndented')\n const value = state.indentLines(raw, map)\n exit()\n return value\n }\n\n const tracker = state.createTracker(info)\n const sequence = marker.repeat(Math.max(longestStreak(raw, marker) + 1, 3))\n const exit = state.enter('codeFenced')\n let value = tracker.move(sequence)\n\n if (node.lang) {\n const subexit = state.enter(`codeFencedLang${suffix}`)\n value += tracker.move(\n state.safe(node.lang, {\n before: value,\n after: ' ',\n encode: ['`'],\n ...tracker.current()\n })\n )\n subexit()\n }\n\n if (node.lang && node.meta) {\n const subexit = state.enter(`codeFencedMeta${suffix}`)\n value += tracker.move(' ')\n value += tracker.move(\n state.safe(node.meta, {\n before: value,\n after: '\\n',\n encode: ['`'],\n ...tracker.current()\n })\n )\n subexit()\n }\n\n value += tracker.move('\\n')\n\n if (raw) {\n value += tracker.move(raw + '\\n')\n }\n\n value += tracker.move(sequence)\n exit()\n return value\n}\n\n/** @type {Map} */\nfunction map(line, _, blank) {\n return (blank ? '' : ' ') + line\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['quote'], null | undefined>}\n */\nexport function checkQuote(state) {\n const marker = state.options.quote || '\"'\n\n if (marker !== '\"' && marker !== \"'\") {\n throw new Error(\n 'Cannot serialize title with `' +\n marker +\n '` for `options.quote`, expected `\"`, or `\\'`'\n )\n }\n\n return marker\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Definition, Parents} from 'mdast'\n */\n\nimport {checkQuote} from '../util/check-quote.js'\n\n/**\n * @param {Definition} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function definition(node, _, state, info) {\n const quote = checkQuote(state)\n const suffix = quote === '\"' ? 'Quote' : 'Apostrophe'\n const exit = state.enter('definition')\n let subexit = state.enter('label')\n const tracker = state.createTracker(info)\n let value = tracker.move('[')\n value += tracker.move(\n state.safe(state.associationId(node), {\n before: value,\n after: ']',\n ...tracker.current()\n })\n )\n value += tracker.move(']: ')\n\n subexit()\n\n if (\n // If there’s no url, or…\n !node.url ||\n // If there are control characters or whitespace.\n /[\\0- \\u007F]/.test(node.url)\n ) {\n subexit = state.enter('destinationLiteral')\n value += tracker.move('<')\n value += tracker.move(\n state.safe(node.url, {before: value, after: '>', ...tracker.current()})\n )\n value += tracker.move('>')\n } else {\n // No whitespace, raw is prettier.\n subexit = state.enter('destinationRaw')\n value += tracker.move(\n state.safe(node.url, {\n before: value,\n after: node.title ? ' ' : '\\n',\n ...tracker.current()\n })\n )\n }\n\n subexit()\n\n if (node.title) {\n subexit = state.enter(`title${suffix}`)\n value += tracker.move(' ' + quote)\n value += tracker.move(\n state.safe(node.title, {\n before: value,\n after: quote,\n ...tracker.current()\n })\n )\n value += tracker.move(quote)\n subexit()\n }\n\n exit()\n\n return value\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['emphasis'], null | undefined>}\n */\nexport function checkEmphasis(state) {\n const marker = state.options.emphasis || '*'\n\n if (marker !== '*' && marker !== '_') {\n throw new Error(\n 'Cannot serialize emphasis with `' +\n marker +\n '` for `options.emphasis`, expected `*`, or `_`'\n )\n }\n\n return marker\n}\n","/**\n * Encode a code point as a character reference.\n *\n * @param {number} code\n * Code point to encode.\n * @returns {string}\n * Encoded character reference.\n */\nexport function encodeCharacterReference(code) {\n return '&#x' + code.toString(16).toUpperCase() + ';'\n}\n","/**\n * @import {Code} from 'micromark-util-types'\n */\n\nimport { markdownLineEndingOrSpace, unicodePunctuation, unicodeWhitespace } from 'micromark-util-character';\n/**\n * Classify whether a code represents whitespace, punctuation, or something\n * else.\n *\n * Used for attention (emphasis, strong), whose sequences can open or close\n * based on the class of surrounding characters.\n *\n * > 👉 **Note**: eof (`null`) is seen as whitespace.\n *\n * @param {Code} code\n * Code.\n * @returns {typeof constants.characterGroupWhitespace | typeof constants.characterGroupPunctuation | undefined}\n * Group.\n */\nexport function classifyCharacter(code) {\n if (code === null || markdownLineEndingOrSpace(code) || unicodeWhitespace(code)) {\n return 1;\n }\n if (unicodePunctuation(code)) {\n return 2;\n }\n}","/**\n * @import {EncodeSides} from '../types.js'\n */\n\nimport {classifyCharacter} from 'micromark-util-classify-character'\n\n/**\n * Check whether to encode (as a character reference) the characters\n * surrounding an attention run.\n *\n * Which characters are around an attention run influence whether it works or\n * not.\n *\n * See <https://github.com/orgs/syntax-tree/discussions/60> for more info.\n * See this markdown in a particular renderer to see what works:\n *\n * ```markdown\n * | | A (letter inside) | B (punctuation inside) | C (whitespace inside) | D (nothing inside) |\n * | ----------------------- | ----------------- | ---------------------- | --------------------- | ------------------ |\n * | 1 (letter outside) | x*y*z | x*.*z | x* *z | x**z |\n * | 2 (punctuation outside) | .*y*. | .*.*. | .* *. | .**. |\n * | 3 (whitespace outside) | x *y* z | x *.* z | x * * z | x ** z |\n * | 4 (nothing outside) | *x* | *.* | * * | ** |\n * ```\n *\n * @param {number} outside\n * Code point on the outer side of the run.\n * @param {number} inside\n * Code point on the inner side of the run.\n * @param {'*' | '_'} marker\n * Marker of the run.\n * Underscores are handled more strictly (they form less often) than\n * asterisks.\n * @returns {EncodeSides}\n * Whether to encode characters.\n */\n// Important: punctuation must never be encoded.\n// Punctuation is solely used by markdown constructs.\n// And by encoding itself.\n// Encoding them will break constructs or double encode things.\nexport function encodeInfo(outside, inside, marker) {\n const outsideKind = classifyCharacter(outside)\n const insideKind = classifyCharacter(inside)\n\n // Letter outside:\n if (outsideKind === undefined) {\n return insideKind === undefined\n ? // Letter inside:\n // we have to encode *both* letters for `_` as it is looser.\n // it already forms for `*` (and GFMs `~`).\n marker === '_'\n ? {inside: true, outside: true}\n : {inside: false, outside: false}\n : insideKind === 1\n ? // Whitespace inside: encode both (letter, whitespace).\n {inside: true, outside: true}\n : // Punctuation inside: encode outer (letter)\n {inside: false, outside: true}\n }\n\n // Whitespace outside:\n if (outsideKind === 1) {\n return insideKind === undefined\n ? // Letter inside: already forms.\n {inside: false, outside: false}\n : insideKind === 1\n ? // Whitespace inside: encode both (whitespace).\n {inside: true, outside: true}\n : // Punctuation inside: already forms.\n {inside: false, outside: false}\n }\n\n // Punctuation outside:\n return insideKind === undefined\n ? // Letter inside: already forms.\n {inside: false, outside: false}\n : insideKind === 1\n ? // Whitespace inside: encode inner (whitespace).\n {inside: true, outside: false}\n : // Punctuation inside: already forms.\n {inside: false, outside: false}\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Emphasis, Parents} from 'mdast'\n */\n\nimport {checkEmphasis} from '../util/check-emphasis.js'\nimport {encodeCharacterReference} from '../util/encode-character-reference.js'\nimport {encodeInfo} from '../util/encode-info.js'\n\nemphasis.peek = emphasisPeek\n\n/**\n * @param {Emphasis} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function emphasis(node, _, state, info) {\n const marker = checkEmphasis(state)\n const exit = state.enter('emphasis')\n const tracker = state.createTracker(info)\n const before = tracker.move(marker)\n\n let between = tracker.move(\n state.containerPhrasing(node, {\n after: marker,\n before,\n ...tracker.current()\n })\n )\n const betweenHead = between.charCodeAt(0)\n const open = encodeInfo(\n info.before.charCodeAt(info.before.length - 1),\n betweenHead,\n marker\n )\n\n if (open.inside) {\n between = encodeCharacterReference(betweenHead) + between.slice(1)\n }\n\n const betweenTail = between.charCodeAt(between.length - 1)\n const close = encodeInfo(info.after.charCodeAt(0), betweenTail, marker)\n\n if (close.inside) {\n between = between.slice(0, -1) + encodeCharacterReference(betweenTail)\n }\n\n const after = tracker.move(marker)\n\n exit()\n\n state.attentionEncodeSurroundingInfo = {\n after: close.outside,\n before: open.outside\n }\n return before + between + after\n}\n\n/**\n * @param {Emphasis} _\n * @param {Parents | undefined} _1\n * @param {State} state\n * @returns {string}\n */\nfunction emphasisPeek(_, _1, state) {\n return state.options.emphasis || '*'\n}\n","/**\n * @typedef {import('unist').Node} UnistNode\n * @typedef {import('unist').Parent} UnistParent\n * @typedef {import('unist-util-visit-parents').VisitorResult} VisitorResult\n */\n\n/**\n * @typedef {Exclude<import('unist-util-is').Test, undefined> | undefined} Test\n * Test from `unist-util-is`.\n *\n * Note: we have remove and add `undefined`, because otherwise when generating\n * automatic `.d.ts` files, TS tries to flatten paths from a local perspective,\n * which doesn’t work when publishing on npm.\n */\n\n// To do: use types from `unist-util-visit-parents` when it’s released.\n\n/**\n * @typedef {(\n * Fn extends (value: any) => value is infer Thing\n * ? Thing\n * : Fallback\n * )} Predicate\n * Get the value of a type guard `Fn`.\n * @template Fn\n * Value; typically function that is a type guard (such as `(x): x is Y`).\n * @template Fallback\n * Value to yield if `Fn` is not a type guard.\n */\n\n/**\n * @typedef {(\n * Check extends null | undefined // No test.\n * ? Value\n * : Value extends {type: Check} // String (type) test.\n * ? Value\n * : Value extends Check // Partial test.\n * ? Value\n * : Check extends Function // Function test.\n * ? Predicate<Check, Value> extends Value\n * ? Predicate<Check, Value>\n * : never\n * : never // Some other test?\n * )} MatchesOne\n * Check whether a node matches a primitive check in the type system.\n * @template Value\n * Value; typically unist `Node`.\n * @template Check\n * Value; typically `unist-util-is`-compatible test, but not arrays.\n */\n\n/**\n * @typedef {(\n * Check extends Array<any>\n * ? MatchesOne<Value, Check[keyof Check]>\n * : MatchesOne<Value, Check>\n * )} Matches\n * Check whether a node matches a check in the type system.\n * @template Value\n * Value; typically unist `Node`.\n * @template Check\n * Value; typically `unist-util-is`-compatible test.\n */\n\n/**\n * @typedef {0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10} Uint\n * Number; capped reasonably.\n */\n\n/**\n * @typedef {I extends 0 ? 1 : I extends 1 ? 2 : I extends 2 ? 3 : I extends 3 ? 4 : I extends 4 ? 5 : I extends 5 ? 6 : I extends 6 ? 7 : I extends 7 ? 8 : I extends 8 ? 9 : 10} Increment\n * Increment a number in the type system.\n * @template {Uint} [I=0]\n * Index.\n */\n\n/**\n * @typedef {(\n * Node extends UnistParent\n * ? Node extends {children: Array<infer Children>}\n * ? Child extends Children ? Node : never\n * : never\n * : never\n * )} InternalParent\n * Collect nodes that can be parents of `Child`.\n * @template {UnistNode} Node\n * All node types in a tree.\n * @template {UnistNode} Child\n * Node to search for.\n */\n\n/**\n * @typedef {InternalParent<InclusiveDescendant<Tree>, Child>} Parent\n * Collect nodes in `Tree` that can be parents of `Child`.\n * @template {UnistNode} Tree\n * All node types in a tree.\n * @template {UnistNode} Child\n * Node to search for.\n */\n\n/**\n * @typedef {(\n * Depth extends Max\n * ? never\n * :\n * | InternalParent<Node, Child>\n * | InternalAncestor<Node, InternalParent<Node, Child>, Max, Increment<Depth>>\n * )} InternalAncestor\n * Collect nodes in `Tree` that can be ancestors of `Child`.\n * @template {UnistNode} Node\n * All node types in a tree.\n * @template {UnistNode} Child\n * Node to search for.\n * @template {Uint} [Max=10]\n * Max; searches up to this depth.\n * @template {Uint} [Depth=0]\n * Current depth.\n */\n\n/**\n * @typedef {(\n * Tree extends UnistParent\n * ? Depth extends Max\n * ? Tree\n * : Tree | InclusiveDescendant<Tree['children'][number], Max, Increment<Depth>>\n * : Tree\n * )} InclusiveDescendant\n * Collect all (inclusive) descendants of `Tree`.\n *\n * > 👉 **Note**: for performance reasons, this seems to be the fastest way to\n * > recurse without actually running into an infinite loop, which the\n * > previous version did.\n * >\n * > Practically, a max of `2` is typically enough assuming a `Root` is\n * > passed, but it doesn’t improve performance.\n * > It gets higher with `List > ListItem > Table > TableRow > TableCell`.\n * > Using up to `10` doesn’t hurt or help either.\n * @template {UnistNode} Tree\n * Tree type.\n * @template {Uint} [Max=10]\n * Max; searches up to this depth.\n * @template {Uint} [Depth=0]\n * Current depth.\n */\n\n/**\n * @callback Visitor\n * Handle a node (matching `test`, if given).\n *\n * Visitors are free to transform `node`.\n * They can also transform `parent`.\n *\n * Replacing `node` itself, if `SKIP` is not returned, still causes its\n * descendants to be walked (which is a bug).\n *\n * When adding or removing previous siblings of `node` (or next siblings, in\n * case of reverse), the `Visitor` should return a new `Index` to specify the\n * sibling to traverse after `node` is traversed.\n * Adding or removing next siblings of `node` (or previous siblings, in case\n * of reverse) is handled as expected without needing to return a new `Index`.\n *\n * Removing the children property of `parent` still results in them being\n * traversed.\n * @param {Visited} node\n * Found node.\n * @param {Visited extends UnistNode ? number | undefined : never} index\n * Index of `node` in `parent`.\n * @param {Ancestor extends UnistParent ? Ancestor | undefined : never} parent\n * Parent of `node`.\n * @returns {VisitorResult}\n * What to do next.\n *\n * An `Index` is treated as a tuple of `[CONTINUE, Index]`.\n * An `Action` is treated as a tuple of `[Action]`.\n *\n * Passing a tuple back only makes sense if the `Action` is `SKIP`.\n * When the `Action` is `EXIT`, that action can be returned.\n * When the `Action` is `CONTINUE`, `Index` can be returned.\n * @template {UnistNode} [Visited=UnistNode]\n * Visited node type.\n * @template {UnistParent} [Ancestor=UnistParent]\n * Ancestor type.\n */\n\n/**\n * @typedef {Visitor<Visited, Parent<Ancestor, Visited>>} BuildVisitorFromMatch\n * Build a typed `Visitor` function from a node and all possible parents.\n *\n * It will infer which values are passed as `node` and which as `parent`.\n * @template {UnistNode} Visited\n * Node type.\n * @template {UnistParent} Ancestor\n * Parent type.\n */\n\n/**\n * @typedef {(\n * BuildVisitorFromMatch<\n * Matches<Descendant, Check>,\n * Extract<Descendant, UnistParent>\n * >\n * )} BuildVisitorFromDescendants\n * Build a typed `Visitor` function from a list of descendants and a test.\n *\n * It will infer which values are passed as `node` and which as `parent`.\n * @template {UnistNode} Descendant\n * Node type.\n * @template {Test} Check\n * Test type.\n */\n\n/**\n * @typedef {(\n * BuildVisitorFromDescendants<\n * InclusiveDescendant<Tree>,\n * Check\n * >\n * )} BuildVisitor\n * Build a typed `Visitor` function from a tree and a test.\n *\n * It will infer which values are passed as `node` and which as `parent`.\n * @template {UnistNode} [Tree=UnistNode]\n * Node type.\n * @template {Test} [Check=Test]\n * Test type.\n */\n\nimport {visitParents} from 'unist-util-visit-parents'\n\nexport {CONTINUE, EXIT, SKIP} from 'unist-util-visit-parents'\n\n/**\n * Visit nodes.\n *\n * This algorithm performs *depth-first* *tree traversal* in *preorder*\n * (**NLR**) or if `reverse` is given, in *reverse preorder* (**NRL**).\n *\n * You can choose for which nodes `visitor` is called by passing a `test`.\n * For complex tests, you should test yourself in `visitor`, as it will be\n * faster and will have improved type information.\n *\n * Walking the tree is an intensive task.\n * Make use of the return values of the visitor when possible.\n * Instead of walking a tree multiple times, walk it once, use `unist-util-is`\n * to check if a node matches, and then perform different operations.\n *\n * You can change the tree.\n * See `Visitor` for more info.\n *\n * @overload\n * @param {Tree} tree\n * @param {Check} check\n * @param {BuildVisitor<Tree, Check>} visitor\n * @param {boolean | null | undefined} [reverse]\n * @returns {undefined}\n *\n * @overload\n * @param {Tree} tree\n * @param {BuildVisitor<Tree>} visitor\n * @param {boolean | null | undefined} [reverse]\n * @returns {undefined}\n *\n * @param {UnistNode} tree\n * Tree to traverse.\n * @param {Visitor | Test} testOrVisitor\n * `unist-util-is`-compatible test (optional, omit to pass a visitor).\n * @param {Visitor | boolean | null | undefined} [visitorOrReverse]\n * Handle each node (when test is omitted, pass `reverse`).\n * @param {boolean | null | undefined} [maybeReverse=false]\n * Traverse in reverse preorder (NRL) instead of the default preorder (NLR).\n * @returns {undefined}\n * Nothing.\n *\n * @template {UnistNode} Tree\n * Node type.\n * @template {Test} Check\n * `unist-util-is`-compatible test.\n */\nexport function visit(tree, testOrVisitor, visitorOrReverse, maybeReverse) {\n /** @type {boolean | null | undefined} */\n let reverse\n /** @type {Test} */\n let test\n /** @type {Visitor} */\n let visitor\n\n if (\n typeof testOrVisitor === 'function' &&\n typeof visitorOrReverse !== 'function'\n ) {\n test = undefined\n visitor = testOrVisitor\n reverse = visitorOrReverse\n } else {\n // @ts-expect-error: assume the overload with test was given.\n test = testOrVisitor\n // @ts-expect-error: assume the overload with test was given.\n visitor = visitorOrReverse\n reverse = maybeReverse\n }\n\n visitParents(tree, test, overload, reverse)\n\n /**\n * @param {UnistNode} node\n * @param {Array<UnistParent>} parents\n */\n function overload(node, parents) {\n const parent = parents[parents.length - 1]\n const index = parent ? parent.children.indexOf(node) : undefined\n return visitor(node, index, parent)\n }\n}\n","/**\n * @typedef {import('mdast').Nodes} Nodes\n *\n * @typedef Options\n * Configuration (optional).\n * @property {boolean | null | undefined} [includeImageAlt=true]\n * Whether to use `alt` for `image`s (default: `true`).\n * @property {boolean | null | undefined} [includeHtml=true]\n * Whether to use `value` of HTML (default: `true`).\n */\n\n/** @type {Options} */\nconst emptyOptions = {}\n\n/**\n * Get the text content of a node or list of nodes.\n *\n * Prefers the node’s plain-text fields, otherwise serializes its children,\n * and if the given value is an array, serialize the nodes in it.\n *\n * @param {unknown} [value]\n * Thing to serialize, typically `Node`.\n * @param {Options | null | undefined} [options]\n * Configuration (optional).\n * @returns {string}\n * Serialized `value`.\n */\nexport function toString(value, options) {\n const settings = options || emptyOptions\n const includeImageAlt =\n typeof settings.includeImageAlt === 'boolean'\n ? settings.includeImageAlt\n : true\n const includeHtml =\n typeof settings.includeHtml === 'boolean' ? settings.includeHtml : true\n\n return one(value, includeImageAlt, includeHtml)\n}\n\n/**\n * One node or several nodes.\n *\n * @param {unknown} value\n * Thing to serialize.\n * @param {boolean} includeImageAlt\n * Include image `alt`s.\n * @param {boolean} includeHtml\n * Include HTML.\n * @returns {string}\n * Serialized node.\n */\nfunction one(value, includeImageAlt, includeHtml) {\n if (node(value)) {\n if ('value' in value) {\n return value.type === 'html' && !includeHtml ? '' : value.value\n }\n\n if (includeImageAlt && 'alt' in value && value.alt) {\n return value.alt\n }\n\n if ('children' in value) {\n return all(value.children, includeImageAlt, includeHtml)\n }\n }\n\n if (Array.isArray(value)) {\n return all(value, includeImageAlt, includeHtml)\n }\n\n return ''\n}\n\n/**\n * Serialize a list of nodes.\n *\n * @param {Array<unknown>} values\n * Thing to serialize.\n * @param {boolean} includeImageAlt\n * Include image `alt`s.\n * @param {boolean} includeHtml\n * Include HTML.\n * @returns {string}\n * Serialized nodes.\n */\nfunction all(values, includeImageAlt, includeHtml) {\n /** @type {Array<string>} */\n const result = []\n let index = -1\n\n while (++index < values.length) {\n result[index] = one(values[index], includeImageAlt, includeHtml)\n }\n\n return result.join('')\n}\n\n/**\n * Check if `value` looks like a node.\n *\n * @param {unknown} value\n * Thing.\n * @returns {value is Nodes}\n * Whether `value` is a node.\n */\nfunction node(value) {\n return Boolean(value && typeof value === 'object')\n}\n","/**\n * @import {State} from 'mdast-util-to-markdown'\n * @import {Heading} from 'mdast'\n */\n\nimport {EXIT, visit} from 'unist-util-visit'\nimport {toString} from 'mdast-util-to-string'\n\n/**\n * @param {Heading} node\n * @param {State} state\n * @returns {boolean}\n */\nexport function formatHeadingAsSetext(node, state) {\n let literalWithBreak = false\n\n // Look for literals with a line break.\n // Note that this also\n visit(node, function (node) {\n if (\n ('value' in node && /\\r?\\n|\\r/.test(node.value)) ||\n node.type === 'break'\n ) {\n literalWithBreak = true\n return EXIT\n }\n })\n\n return Boolean(\n (!node.depth || node.depth < 3) &&\n toString(node) &&\n (state.options.setext || literalWithBreak)\n )\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Heading, Parents} from 'mdast'\n */\n\nimport {encodeCharacterReference} from '../util/encode-character-reference.js'\nimport {formatHeadingAsSetext} from '../util/format-heading-as-setext.js'\n\n/**\n * @param {Heading} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function heading(node, _, state, info) {\n const rank = Math.max(Math.min(6, node.depth || 1), 1)\n const tracker = state.createTracker(info)\n\n if (formatHeadingAsSetext(node, state)) {\n const exit = state.enter('headingSetext')\n const subexit = state.enter('phrasing')\n const value = state.containerPhrasing(node, {\n ...tracker.current(),\n before: '\\n',\n after: '\\n'\n })\n subexit()\n exit()\n\n return (\n value +\n '\\n' +\n (rank === 1 ? '=' : '-').repeat(\n // The whole size…\n value.length -\n // Minus the position of the character after the last EOL (or\n // 0 if there is none)…\n (Math.max(value.lastIndexOf('\\r'), value.lastIndexOf('\\n')) + 1)\n )\n )\n }\n\n const sequence = '#'.repeat(rank)\n const exit = state.enter('headingAtx')\n const subexit = state.enter('phrasing')\n\n // Note: for proper tracking, we should reset the output positions when there\n // is no content returned, because then the space is not output.\n // Practically, in that case, there is no content, so it doesn’t matter that\n // we’ve tracked one too many characters.\n tracker.move(sequence + ' ')\n\n let value = state.containerPhrasing(node, {\n before: '# ',\n after: '\\n',\n ...tracker.current()\n })\n\n if (/^[\\t ]/.test(value)) {\n // To do: what effect has the character reference on tracking?\n value = encodeCharacterReference(value.charCodeAt(0)) + value.slice(1)\n }\n\n value = value ? sequence + ' ' + value : sequence\n\n if (state.options.closeAtx) {\n value += ' ' + sequence\n }\n\n subexit()\n exit()\n\n return value\n}\n","/**\n * @import {Html} from 'mdast'\n */\n\nhtml.peek = htmlPeek\n\n/**\n * @param {Html} node\n * @returns {string}\n */\nexport function html(node) {\n return node.value || ''\n}\n\n/**\n * @returns {string}\n */\nfunction htmlPeek() {\n return '<'\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Image, Parents} from 'mdast'\n */\n\nimport {checkQuote} from '../util/check-quote.js'\n\nimage.peek = imagePeek\n\n/**\n * @param {Image} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function image(node, _, state, info) {\n const quote = checkQuote(state)\n const suffix = quote === '\"' ? 'Quote' : 'Apostrophe'\n const exit = state.enter('image')\n let subexit = state.enter('label')\n const tracker = state.createTracker(info)\n let value = tracker.move('![')\n value += tracker.move(\n state.safe(node.alt, {before: value, after: ']', ...tracker.current()})\n )\n value += tracker.move('](')\n\n subexit()\n\n if (\n // If there’s no url but there is a title…\n (!node.url && node.title) ||\n // If there are control characters or whitespace.\n /[\\0- \\u007F]/.test(node.url)\n ) {\n subexit = state.enter('destinationLiteral')\n value += tracker.move('<')\n value += tracker.move(\n state.safe(node.url, {before: value, after: '>', ...tracker.current()})\n )\n value += tracker.move('>')\n } else {\n // No whitespace, raw is prettier.\n subexit = state.enter('destinationRaw')\n value += tracker.move(\n state.safe(node.url, {\n before: value,\n after: node.title ? ' ' : ')',\n ...tracker.current()\n })\n )\n }\n\n subexit()\n\n if (node.title) {\n subexit = state.enter(`title${suffix}`)\n value += tracker.move(' ' + quote)\n value += tracker.move(\n state.safe(node.title, {\n before: value,\n after: quote,\n ...tracker.current()\n })\n )\n value += tracker.move(quote)\n subexit()\n }\n\n value += tracker.move(')')\n exit()\n\n return value\n}\n\n/**\n * @returns {string}\n */\nfunction imagePeek() {\n return '!'\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {ImageReference, Parents} from 'mdast'\n */\n\nimageReference.peek = imageReferencePeek\n\n/**\n * @param {ImageReference} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function imageReference(node, _, state, info) {\n const type = node.referenceType\n const exit = state.enter('imageReference')\n let subexit = state.enter('label')\n const tracker = state.createTracker(info)\n let value = tracker.move('![')\n const alt = state.safe(node.alt, {\n before: value,\n after: ']',\n ...tracker.current()\n })\n value += tracker.move(alt + '][')\n\n subexit()\n // Hide the fact that we’re in phrasing, because escapes don’t work.\n const stack = state.stack\n state.stack = []\n subexit = state.enter('reference')\n // Note: for proper tracking, we should reset the output positions when we end\n // up making a `shortcut` reference, because then there is no brace output.\n // Practically, in that case, there is no content, so it doesn’t matter that\n // we’ve tracked one too many characters.\n const reference = state.safe(state.associationId(node), {\n before: value,\n after: ']',\n ...tracker.current()\n })\n subexit()\n state.stack = stack\n exit()\n\n if (type === 'full' || !alt || alt !== reference) {\n value += tracker.move(reference + ']')\n } else if (type === 'shortcut') {\n // Remove the unwanted `[`.\n value = value.slice(0, -1)\n } else {\n value += tracker.move(']')\n }\n\n return value\n}\n\n/**\n * @returns {string}\n */\nfunction imageReferencePeek() {\n return '!'\n}\n","/**\n * @import {State} from 'mdast-util-to-markdown'\n * @import {InlineCode, Parents} from 'mdast'\n */\n\ninlineCode.peek = inlineCodePeek\n\n/**\n * @param {InlineCode} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @returns {string}\n */\nexport function inlineCode(node, _, state) {\n let value = node.value || ''\n let sequence = '`'\n let index = -1\n\n // If there is a single grave accent on its own in the code, use a fence of\n // two.\n // If there are two in a row, use one.\n while (new RegExp('(^|[^`])' + sequence + '([^`]|$)').test(value)) {\n sequence += '`'\n }\n\n // If this is not just spaces or eols (tabs don’t count), and either the\n // first or last character are a space, eol, or tick, then pad with spaces.\n if (\n /[^ \\r\\n]/.test(value) &&\n ((/^[ \\r\\n]/.test(value) && /[ \\r\\n]$/.test(value)) || /^`|`$/.test(value))\n ) {\n value = ' ' + value + ' '\n }\n\n // We have a potential problem: certain characters after eols could result in\n // blocks being seen.\n // For example, if someone injected the string `'\\n# b'`, then that would\n // result in an ATX heading.\n // We can’t escape characters in `inlineCode`, but because eols are\n // transformed to spaces when going from markdown to HTML anyway, we can swap\n // them out.\n while (++index < state.unsafe.length) {\n const pattern = state.unsafe[index]\n const expression = state.compilePattern(pattern)\n /** @type {RegExpExecArray | null} */\n let match\n\n // Only look for `atBreak`s.\n // Btw: note that `atBreak` patterns will always start the regex at LF or\n // CR.\n if (!pattern.atBreak) continue\n\n while ((match = expression.exec(value))) {\n let position = match.index\n\n // Support CRLF (patterns only look for one of the characters).\n if (\n value.charCodeAt(position) === 10 /* `\\n` */ &&\n value.charCodeAt(position - 1) === 13 /* `\\r` */\n ) {\n position--\n }\n\n value = value.slice(0, position) + ' ' + value.slice(match.index + 1)\n }\n }\n\n return sequence + value + sequence\n}\n\n/**\n * @returns {string}\n */\nfunction inlineCodePeek() {\n return '`'\n}\n","/**\n * @import {State} from 'mdast-util-to-markdown'\n * @import {Link} from 'mdast'\n */\n\nimport {toString} from 'mdast-util-to-string'\n\n/**\n * @param {Link} node\n * @param {State} state\n * @returns {boolean}\n */\nexport function formatLinkAsAutolink(node, state) {\n const raw = toString(node)\n\n return Boolean(\n !state.options.resourceLink &&\n // If there’s a url…\n node.url &&\n // And there’s a no title…\n !node.title &&\n // And the content of `node` is a single text node…\n node.children &&\n node.children.length === 1 &&\n node.children[0].type === 'text' &&\n // And if the url is the same as the content…\n (raw === node.url || 'mailto:' + raw === node.url) &&\n // And that starts w/ a protocol…\n /^[a-z][a-z+.-]+:/i.test(node.url) &&\n // And that doesn’t contain ASCII control codes (character escapes and\n // references don’t work), space, or angle brackets…\n !/[\\0- <>\\u007F]/.test(node.url)\n )\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Link, Parents} from 'mdast'\n * @import {Exit} from '../types.js'\n */\n\nimport {checkQuote} from '../util/check-quote.js'\nimport {formatLinkAsAutolink} from '../util/format-link-as-autolink.js'\n\nlink.peek = linkPeek\n\n/**\n * @param {Link} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function link(node, _, state, info) {\n const quote = checkQuote(state)\n const suffix = quote === '\"' ? 'Quote' : 'Apostrophe'\n const tracker = state.createTracker(info)\n /** @type {Exit} */\n let exit\n /** @type {Exit} */\n let subexit\n\n if (formatLinkAsAutolink(node, state)) {\n // Hide the fact that we’re in phrasing, because escapes don’t work.\n const stack = state.stack\n state.stack = []\n exit = state.enter('autolink')\n let value = tracker.move('<')\n value += tracker.move(\n state.containerPhrasing(node, {\n before: value,\n after: '>',\n ...tracker.current()\n })\n )\n value += tracker.move('>')\n exit()\n state.stack = stack\n return value\n }\n\n exit = state.enter('link')\n subexit = state.enter('label')\n let value = tracker.move('[')\n value += tracker.move(\n state.containerPhrasing(node, {\n before: value,\n after: '](',\n ...tracker.current()\n })\n )\n value += tracker.move('](')\n subexit()\n\n if (\n // If there’s no url but there is a title…\n (!node.url && node.title) ||\n // If there are control characters or whitespace.\n /[\\0- \\u007F]/.test(node.url)\n ) {\n subexit = state.enter('destinationLiteral')\n value += tracker.move('<')\n value += tracker.move(\n state.safe(node.url, {before: value, after: '>', ...tracker.current()})\n )\n value += tracker.move('>')\n } else {\n // No whitespace, raw is prettier.\n subexit = state.enter('destinationRaw')\n value += tracker.move(\n state.safe(node.url, {\n before: value,\n after: node.title ? ' ' : ')',\n ...tracker.current()\n })\n )\n }\n\n subexit()\n\n if (node.title) {\n subexit = state.enter(`title${suffix}`)\n value += tracker.move(' ' + quote)\n value += tracker.move(\n state.safe(node.title, {\n before: value,\n after: quote,\n ...tracker.current()\n })\n )\n value += tracker.move(quote)\n subexit()\n }\n\n value += tracker.move(')')\n\n exit()\n return value\n}\n\n/**\n * @param {Link} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @returns {string}\n */\nfunction linkPeek(node, _, state) {\n return formatLinkAsAutolink(node, state) ? '<' : '['\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {LinkReference, Parents} from 'mdast'\n */\n\nlinkReference.peek = linkReferencePeek\n\n/**\n * @param {LinkReference} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function linkReference(node, _, state, info) {\n const type = node.referenceType\n const exit = state.enter('linkReference')\n let subexit = state.enter('label')\n const tracker = state.createTracker(info)\n let value = tracker.move('[')\n const text = state.containerPhrasing(node, {\n before: value,\n after: ']',\n ...tracker.current()\n })\n value += tracker.move(text + '][')\n\n subexit()\n // Hide the fact that we’re in phrasing, because escapes don’t work.\n const stack = state.stack\n state.stack = []\n subexit = state.enter('reference')\n // Note: for proper tracking, we should reset the output positions when we end\n // up making a `shortcut` reference, because then there is no brace output.\n // Practically, in that case, there is no content, so it doesn’t matter that\n // we’ve tracked one too many characters.\n const reference = state.safe(state.associationId(node), {\n before: value,\n after: ']',\n ...tracker.current()\n })\n subexit()\n state.stack = stack\n exit()\n\n if (type === 'full' || !text || text !== reference) {\n value += tracker.move(reference + ']')\n } else if (type === 'shortcut') {\n // Remove the unwanted `[`.\n value = value.slice(0, -1)\n } else {\n value += tracker.move(']')\n }\n\n return value\n}\n\n/**\n * @returns {string}\n */\nfunction linkReferencePeek() {\n return '['\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['bullet'], null | undefined>}\n */\nexport function checkBullet(state) {\n const marker = state.options.bullet || '*'\n\n if (marker !== '*' && marker !== '+' && marker !== '-') {\n throw new Error(\n 'Cannot serialize items with `' +\n marker +\n '` for `options.bullet`, expected `*`, `+`, or `-`'\n )\n }\n\n return marker\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\nimport {checkBullet} from './check-bullet.js'\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['bullet'], null | undefined>}\n */\nexport function checkBulletOther(state) {\n const bullet = checkBullet(state)\n const bulletOther = state.options.bulletOther\n\n if (!bulletOther) {\n return bullet === '*' ? '-' : '*'\n }\n\n if (bulletOther !== '*' && bulletOther !== '+' && bulletOther !== '-') {\n throw new Error(\n 'Cannot serialize items with `' +\n bulletOther +\n '` for `options.bulletOther`, expected `*`, `+`, or `-`'\n )\n }\n\n if (bulletOther === bullet) {\n throw new Error(\n 'Expected `bullet` (`' +\n bullet +\n '`) and `bulletOther` (`' +\n bulletOther +\n '`) to be different'\n )\n }\n\n return bulletOther\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['bulletOrdered'], null | undefined>}\n */\nexport function checkBulletOrdered(state) {\n const marker = state.options.bulletOrdered || '.'\n\n if (marker !== '.' && marker !== ')') {\n throw new Error(\n 'Cannot serialize items with `' +\n marker +\n '` for `options.bulletOrdered`, expected `.` or `)`'\n )\n }\n\n return marker\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['rule'], null | undefined>}\n */\nexport function checkRule(state) {\n const marker = state.options.rule || '*'\n\n if (marker !== '*' && marker !== '-' && marker !== '_') {\n throw new Error(\n 'Cannot serialize rules with `' +\n marker +\n '` for `options.rule`, expected `*`, `-`, or `_`'\n )\n }\n\n return marker\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {List, Parents} from 'mdast'\n */\n\nimport {checkBullet} from '../util/check-bullet.js'\nimport {checkBulletOther} from '../util/check-bullet-other.js'\nimport {checkBulletOrdered} from '../util/check-bullet-ordered.js'\nimport {checkRule} from '../util/check-rule.js'\n\n/**\n * @param {List} node\n * @param {Parents | undefined} parent\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function list(node, parent, state, info) {\n const exit = state.enter('list')\n const bulletCurrent = state.bulletCurrent\n /** @type {string} */\n let bullet = node.ordered ? checkBulletOrdered(state) : checkBullet(state)\n /** @type {string} */\n const bulletOther = node.ordered\n ? bullet === '.'\n ? ')'\n : '.'\n : checkBulletOther(state)\n let useDifferentMarker =\n parent && state.bulletLastUsed ? bullet === state.bulletLastUsed : false\n\n if (!node.ordered) {\n const firstListItem = node.children ? node.children[0] : undefined\n\n // If there’s an empty first list item directly in two list items,\n // we have to use a different bullet:\n //\n // ```markdown\n // * - *\n // ```\n //\n // …because otherwise it would become one big thematic break.\n if (\n // Bullet could be used as a thematic break marker:\n (bullet === '*' || bullet === '-') &&\n // Empty first list item:\n firstListItem &&\n (!firstListItem.children || !firstListItem.children[0]) &&\n // Directly in two other list items:\n state.stack[state.stack.length - 1] === 'list' &&\n state.stack[state.stack.length - 2] === 'listItem' &&\n state.stack[state.stack.length - 3] === 'list' &&\n state.stack[state.stack.length - 4] === 'listItem' &&\n // That are each the first child.\n state.indexStack[state.indexStack.length - 1] === 0 &&\n state.indexStack[state.indexStack.length - 2] === 0 &&\n state.indexStack[state.indexStack.length - 3] === 0\n ) {\n useDifferentMarker = true\n }\n\n // If there’s a thematic break at the start of the first list item,\n // we have to use a different bullet:\n //\n // ```markdown\n // * ---\n // ```\n //\n // …because otherwise it would become one big thematic break.\n if (checkRule(state) === bullet && firstListItem) {\n let index = -1\n\n while (++index < node.children.length) {\n const item = node.children[index]\n\n if (\n item &&\n item.type === 'listItem' &&\n item.children &&\n item.children[0] &&\n item.children[0].type === 'thematicBreak'\n ) {\n useDifferentMarker = true\n break\n }\n }\n }\n }\n\n if (useDifferentMarker) {\n bullet = bulletOther\n }\n\n state.bulletCurrent = bullet\n const value = state.containerFlow(node, info)\n state.bulletLastUsed = bullet\n state.bulletCurrent = bulletCurrent\n exit()\n return value\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['listItemIndent'], null | undefined>}\n */\nexport function checkListItemIndent(state) {\n const style = state.options.listItemIndent || 'one'\n\n if (style !== 'tab' && style !== 'one' && style !== 'mixed') {\n throw new Error(\n 'Cannot serialize items with `' +\n style +\n '` for `options.listItemIndent`, expected `tab`, `one`, or `mixed`'\n )\n }\n\n return style\n}\n","/**\n * @import {Info, Map, State} from 'mdast-util-to-markdown'\n * @import {ListItem, Parents} from 'mdast'\n */\n\nimport {checkBullet} from '../util/check-bullet.js'\nimport {checkListItemIndent} from '../util/check-list-item-indent.js'\n\n/**\n * @param {ListItem} node\n * @param {Parents | undefined} parent\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function listItem(node, parent, state, info) {\n const listItemIndent = checkListItemIndent(state)\n let bullet = state.bulletCurrent || checkBullet(state)\n\n // Add the marker value for ordered lists.\n if (parent && parent.type === 'list' && parent.ordered) {\n bullet =\n (typeof parent.start === 'number' && parent.start > -1\n ? parent.start\n : 1) +\n (state.options.incrementListMarker === false\n ? 0\n : parent.children.indexOf(node)) +\n bullet\n }\n\n let size = bullet.length + 1\n\n if (\n listItemIndent === 'tab' ||\n (listItemIndent === 'mixed' &&\n ((parent && parent.type === 'list' && parent.spread) || node.spread))\n ) {\n size = Math.ceil(size / 4) * 4\n }\n\n const tracker = state.createTracker(info)\n tracker.move(bullet + ' '.repeat(size - bullet.length))\n tracker.shift(size)\n const exit = state.enter('listItem')\n const value = state.indentLines(\n state.containerFlow(node, tracker.current()),\n map\n )\n exit()\n\n return value\n\n /** @type {Map} */\n function map(line, index, blank) {\n if (index) {\n return (blank ? '' : ' '.repeat(size)) + line\n }\n\n return (blank ? bullet : bullet + ' '.repeat(size - bullet.length)) + line\n }\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Paragraph, Parents} from 'mdast'\n */\n\n/**\n * @param {Paragraph} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function paragraph(node, _, state, info) {\n const exit = state.enter('paragraph')\n const subexit = state.enter('phrasing')\n const value = state.containerPhrasing(node, info)\n subexit()\n exit()\n return value\n}\n","/**\n * @typedef {import('mdast').Html} Html\n * @typedef {import('mdast').PhrasingContent} PhrasingContent\n */\n\nimport {convert} from 'unist-util-is'\n\n/**\n * Check if the given value is *phrasing content*.\n *\n * > 👉 **Note**: Excludes `html`, which can be both phrasing or flow.\n *\n * @param node\n * Thing to check, typically `Node`.\n * @returns\n * Whether `value` is phrasing content.\n */\n\nexport const phrasing =\n /** @type {(node?: unknown) => node is Exclude<PhrasingContent, Html>} */\n (\n convert([\n 'break',\n 'delete',\n 'emphasis',\n // To do: next major: removed since footnotes were added to GFM.\n 'footnote',\n 'footnoteReference',\n 'image',\n 'imageReference',\n 'inlineCode',\n // Enabled by `mdast-util-math`:\n 'inlineMath',\n 'link',\n 'linkReference',\n // Enabled by `mdast-util-mdx`:\n 'mdxJsxTextElement',\n // Enabled by `mdast-util-mdx`:\n 'mdxTextExpression',\n 'strong',\n 'text',\n // Enabled by `mdast-util-directive`:\n 'textDirective'\n ])\n )\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Parents, Root} from 'mdast'\n */\n\nimport {phrasing} from 'mdast-util-phrasing'\n\n/**\n * @param {Root} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function root(node, _, state, info) {\n // Note: `html` nodes are ambiguous.\n const hasPhrasing = node.children.some(function (d) {\n return phrasing(d)\n })\n\n const container = hasPhrasing ? state.containerPhrasing : state.containerFlow\n return container.call(state, node, info)\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['strong'], null | undefined>}\n */\nexport function checkStrong(state) {\n const marker = state.options.strong || '*'\n\n if (marker !== '*' && marker !== '_') {\n throw new Error(\n 'Cannot serialize strong with `' +\n marker +\n '` for `options.strong`, expected `*`, or `_`'\n )\n }\n\n return marker\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Parents, Strong} from 'mdast'\n */\n\nimport {checkStrong} from '../util/check-strong.js'\nimport {encodeCharacterReference} from '../util/encode-character-reference.js'\nimport {encodeInfo} from '../util/encode-info.js'\n\nstrong.peek = strongPeek\n\n/**\n * @param {Strong} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function strong(node, _, state, info) {\n const marker = checkStrong(state)\n const exit = state.enter('strong')\n const tracker = state.createTracker(info)\n const before = tracker.move(marker + marker)\n\n let between = tracker.move(\n state.containerPhrasing(node, {\n after: marker,\n before,\n ...tracker.current()\n })\n )\n const betweenHead = between.charCodeAt(0)\n const open = encodeInfo(\n info.before.charCodeAt(info.before.length - 1),\n betweenHead,\n marker\n )\n\n if (open.inside) {\n between = encodeCharacterReference(betweenHead) + between.slice(1)\n }\n\n const betweenTail = between.charCodeAt(between.length - 1)\n const close = encodeInfo(info.after.charCodeAt(0), betweenTail, marker)\n\n if (close.inside) {\n between = between.slice(0, -1) + encodeCharacterReference(betweenTail)\n }\n\n const after = tracker.move(marker + marker)\n\n exit()\n\n state.attentionEncodeSurroundingInfo = {\n after: close.outside,\n before: open.outside\n }\n return before + between + after\n}\n\n/**\n * @param {Strong} _\n * @param {Parents | undefined} _1\n * @param {State} state\n * @returns {string}\n */\nfunction strongPeek(_, _1, state) {\n return state.options.strong || '*'\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Parents, Text} from 'mdast'\n */\n\n/**\n * @param {Text} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function text(node, _, state, info) {\n return state.safe(node.value, info)\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['ruleRepetition'], null | undefined>}\n */\nexport function checkRuleRepetition(state) {\n const repetition = state.options.ruleRepetition || 3\n\n if (repetition < 3) {\n throw new Error(\n 'Cannot serialize rules with repetition `' +\n repetition +\n '` for `options.ruleRepetition`, expected `3` or more'\n )\n }\n\n return repetition\n}\n","/**\n * @import {State} from 'mdast-util-to-markdown'\n * @import {Parents, ThematicBreak} from 'mdast'\n */\n\nimport {checkRuleRepetition} from '../util/check-rule-repetition.js'\nimport {checkRule} from '../util/check-rule.js'\n\n/**\n * @param {ThematicBreak} _\n * @param {Parents | undefined} _1\n * @param {State} state\n * @returns {string}\n */\nexport function thematicBreak(_, _1, state) {\n const value = (\n checkRule(state) + (state.options.ruleSpaces ? ' ' : '')\n ).repeat(checkRuleRepetition(state))\n\n return state.options.ruleSpaces ? value.slice(0, -1) : value\n}\n","import {blockquote} from './blockquote.js'\nimport {hardBreak} from './break.js'\nimport {code} from './code.js'\nimport {definition} from './definition.js'\nimport {emphasis} from './emphasis.js'\nimport {heading} from './heading.js'\nimport {html} from './html.js'\nimport {image} from './image.js'\nimport {imageReference} from './image-reference.js'\nimport {inlineCode} from './inline-code.js'\nimport {link} from './link.js'\nimport {linkReference} from './link-reference.js'\nimport {list} from './list.js'\nimport {listItem} from './list-item.js'\nimport {paragraph} from './paragraph.js'\nimport {root} from './root.js'\nimport {strong} from './strong.js'\nimport {text} from './text.js'\nimport {thematicBreak} from './thematic-break.js'\n\n/**\n * Default (CommonMark) handlers.\n */\nexport const handle = {\n blockquote,\n break: hardBreak,\n code,\n definition,\n emphasis,\n hardBreak,\n heading,\n html,\n image,\n imageReference,\n inlineCode,\n link,\n linkReference,\n list,\n listItem,\n paragraph,\n root,\n strong,\n text,\n thematicBreak\n}\n","/**\n * @typedef {import('mdast').InlineCode} InlineCode\n * @typedef {import('mdast').Table} Table\n * @typedef {import('mdast').TableCell} TableCell\n * @typedef {import('mdast').TableRow} TableRow\n *\n * @typedef {import('markdown-table').Options} MarkdownTableOptions\n *\n * @typedef {import('mdast-util-from-markdown').CompileContext} CompileContext\n * @typedef {import('mdast-util-from-markdown').Extension} FromMarkdownExtension\n * @typedef {import('mdast-util-from-markdown').Handle} FromMarkdownHandle\n *\n * @typedef {import('mdast-util-to-markdown').Options} ToMarkdownExtension\n * @typedef {import('mdast-util-to-markdown').Handle} ToMarkdownHandle\n * @typedef {import('mdast-util-to-markdown').State} State\n * @typedef {import('mdast-util-to-markdown').Info} Info\n */\n\n/**\n * @typedef Options\n * Configuration.\n * @property {boolean | null | undefined} [tableCellPadding=true]\n * Whether to add a space of padding between delimiters and cells (default:\n * `true`).\n * @property {boolean | null | undefined} [tablePipeAlign=true]\n * Whether to align the delimiters (default: `true`).\n * @property {MarkdownTableOptions['stringLength'] | null | undefined} [stringLength]\n * Function to detect the length of table cell content, used when aligning\n * the delimiters between cells (optional).\n */\n\nimport {ok as assert} from 'devlop'\nimport {markdownTable} from 'markdown-table'\nimport {defaultHandlers} from 'mdast-util-to-markdown'\n\n/**\n * Create an extension for `mdast-util-from-markdown` to enable GFM tables in\n * markdown.\n *\n * @returns {FromMarkdownExtension}\n * Extension for `mdast-util-from-markdown` to enable GFM tables.\n */\nexport function gfmTableFromMarkdown() {\n return {\n enter: {\n table: enterTable,\n tableData: enterCell,\n tableHeader: enterCell,\n tableRow: enterRow\n },\n exit: {\n codeText: exitCodeText,\n table: exitTable,\n tableData: exit,\n tableHeader: exit,\n tableRow: exit\n }\n }\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterTable(token) {\n const align = token._align\n assert(align, 'expected `_align` on table')\n this.enter(\n {\n type: 'table',\n align: align.map(function (d) {\n return d === 'none' ? null : d\n }),\n children: []\n },\n token\n )\n this.data.inTable = true\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitTable(token) {\n this.exit(token)\n this.data.inTable = undefined\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterRow(token) {\n this.enter({type: 'tableRow', children: []}, token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exit(token) {\n this.exit(token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterCell(token) {\n this.enter({type: 'tableCell', children: []}, token)\n}\n\n// Overwrite the default code text data handler to unescape escaped pipes when\n// they are in tables.\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitCodeText(token) {\n let value = this.resume()\n\n if (this.data.inTable) {\n value = value.replace(/\\\\([\\\\|])/g, replace)\n }\n\n const node = this.stack[this.stack.length - 1]\n assert(node.type === 'inlineCode')\n node.value = value\n this.exit(token)\n}\n\n/**\n * @param {string} $0\n * @param {string} $1\n * @returns {string}\n */\nfunction replace($0, $1) {\n // Pipes work, backslashes don’t (but can’t escape pipes).\n return $1 === '|' ? $1 : $0\n}\n\n/**\n * Create an extension for `mdast-util-to-markdown` to enable GFM tables in\n * markdown.\n *\n * @param {Options | null | undefined} [options]\n * Configuration.\n * @returns {ToMarkdownExtension}\n * Extension for `mdast-util-to-markdown` to enable GFM tables.\n */\nexport function gfmTableToMarkdown(options) {\n const settings = options || {}\n const padding = settings.tableCellPadding\n const alignDelimiters = settings.tablePipeAlign\n const stringLength = settings.stringLength\n const around = padding ? ' ' : '|'\n\n return {\n unsafe: [\n {character: '\\r', inConstruct: 'tableCell'},\n {character: '\\n', inConstruct: 'tableCell'},\n // A pipe, when followed by a tab or space (padding), or a dash or colon\n // (unpadded delimiter row), could result in a table.\n {atBreak: true, character: '|', after: '[\\t :-]'},\n // A pipe in a cell must be encoded.\n {character: '|', inConstruct: 'tableCell'},\n // A colon must be followed by a dash, in which case it could start a\n // delimiter row.\n {atBreak: true, character: ':', after: '-'},\n // A delimiter row can also start with a dash, when followed by more\n // dashes, a colon, or a pipe.\n // This is a stricter version than the built in check for lists, thematic\n // breaks, and setex heading underlines though:\n // <https://github.com/syntax-tree/mdast-util-to-markdown/blob/51a2038/lib/unsafe.js#L57>\n {atBreak: true, character: '-', after: '[:|-]'}\n ],\n handlers: {\n inlineCode: inlineCodeWithTable,\n table: handleTable,\n tableCell: handleTableCell,\n tableRow: handleTableRow\n }\n }\n\n /**\n * @type {ToMarkdownHandle}\n * @param {Table} node\n */\n function handleTable(node, _, state, info) {\n return serializeData(handleTableAsData(node, state, info), node.align)\n }\n\n /**\n * This function isn’t really used normally, because we handle rows at the\n * table level.\n * But, if someone passes in a table row, this ensures we make somewhat sense.\n *\n * @type {ToMarkdownHandle}\n * @param {TableRow} node\n */\n function handleTableRow(node, _, state, info) {\n const row = handleTableRowAsData(node, state, info)\n const value = serializeData([row])\n // `markdown-table` will always add an align row\n return value.slice(0, value.indexOf('\\n'))\n }\n\n /**\n * @type {ToMarkdownHandle}\n * @param {TableCell} node\n */\n function handleTableCell(node, _, state, info) {\n const exit = state.enter('tableCell')\n const subexit = state.enter('phrasing')\n const value = state.containerPhrasing(node, {\n ...info,\n before: around,\n after: around\n })\n subexit()\n exit()\n return value\n }\n\n /**\n * @param {Array<Array<string>>} matrix\n * @param {Array<string | null | undefined> | null | undefined} [align]\n */\n function serializeData(matrix, align) {\n return markdownTable(matrix, {\n align,\n // @ts-expect-error: `markdown-table` types should support `null`.\n alignDelimiters,\n // @ts-expect-error: `markdown-table` types should support `null`.\n padding,\n // @ts-expect-error: `markdown-table` types should support `null`.\n stringLength\n })\n }\n\n /**\n * @param {Table} node\n * @param {State} state\n * @param {Info} info\n */\n function handleTableAsData(node, state, info) {\n const children = node.children\n let index = -1\n /** @type {Array<Array<string>>} */\n const result = []\n const subexit = state.enter('table')\n\n while (++index < children.length) {\n result[index] = handleTableRowAsData(children[index], state, info)\n }\n\n subexit()\n\n return result\n }\n\n /**\n * @param {TableRow} node\n * @param {State} state\n * @param {Info} info\n */\n function handleTableRowAsData(node, state, info) {\n const children = node.children\n let index = -1\n /** @type {Array<string>} */\n const result = []\n const subexit = state.enter('tableRow')\n\n while (++index < children.length) {\n // Note: the positional info as used here is incorrect.\n // Making it correct would be impossible due to aligning cells?\n // And it would need copy/pasting `markdown-table` into this project.\n result[index] = handleTableCell(children[index], node, state, info)\n }\n\n subexit()\n\n return result\n }\n\n /**\n * @type {ToMarkdownHandle}\n * @param {InlineCode} node\n */\n function inlineCodeWithTable(node, parent, state) {\n let value = defaultHandlers.inlineCode(node, parent, state)\n\n if (state.stack.includes('tableCell')) {\n value = value.replace(/\\|/g, '\\\\$&')\n }\n\n return value\n }\n}\n","/**\n * @typedef {import('mdast').ListItem} ListItem\n * @typedef {import('mdast').Paragraph} Paragraph\n * @typedef {import('mdast-util-from-markdown').CompileContext} CompileContext\n * @typedef {import('mdast-util-from-markdown').Extension} FromMarkdownExtension\n * @typedef {import('mdast-util-from-markdown').Handle} FromMarkdownHandle\n * @typedef {import('mdast-util-to-markdown').Options} ToMarkdownExtension\n * @typedef {import('mdast-util-to-markdown').Handle} ToMarkdownHandle\n */\n\nimport {ok as assert} from 'devlop'\nimport {defaultHandlers} from 'mdast-util-to-markdown'\n\n/**\n * Create an extension for `mdast-util-from-markdown` to enable GFM task\n * list items in markdown.\n *\n * @returns {FromMarkdownExtension}\n * Extension for `mdast-util-from-markdown` to enable GFM task list items.\n */\nexport function gfmTaskListItemFromMarkdown() {\n return {\n exit: {\n taskListCheckValueChecked: exitCheck,\n taskListCheckValueUnchecked: exitCheck,\n paragraph: exitParagraphWithTaskListItem\n }\n }\n}\n\n/**\n * Create an extension for `mdast-util-to-markdown` to enable GFM task list\n * items in markdown.\n *\n * @returns {ToMarkdownExtension}\n * Extension for `mdast-util-to-markdown` to enable GFM task list items.\n */\nexport function gfmTaskListItemToMarkdown() {\n return {\n unsafe: [{atBreak: true, character: '-', after: '[:|-]'}],\n handlers: {listItem: listItemWithTaskListItem}\n }\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitCheck(token) {\n // We’re always in a paragraph, in a list item.\n const node = this.stack[this.stack.length - 2]\n assert(node.type === 'listItem')\n node.checked = token.type === 'taskListCheckValueChecked'\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitParagraphWithTaskListItem(token) {\n const parent = this.stack[this.stack.length - 2]\n\n if (\n parent &&\n parent.type === 'listItem' &&\n typeof parent.checked === 'boolean'\n ) {\n const node = this.stack[this.stack.length - 1]\n assert(node.type === 'paragraph')\n const head = node.children[0]\n\n if (head && head.type === 'text') {\n const siblings = parent.children\n let index = -1\n /** @type {Paragraph | undefined} */\n let firstParaghraph\n\n while (++index < siblings.length) {\n const sibling = siblings[index]\n if (sibling.type === 'paragraph') {\n firstParaghraph = sibling\n break\n }\n }\n\n if (firstParaghraph === node) {\n // Must start with a space or a tab.\n head.value = head.value.slice(1)\n\n if (head.value.length === 0) {\n node.children.shift()\n } else if (\n node.position &&\n head.position &&\n typeof head.position.start.offset === 'number'\n ) {\n head.position.start.column++\n head.position.start.offset++\n node.position.start = Object.assign({}, head.position.start)\n }\n }\n }\n }\n\n this.exit(token)\n}\n\n/**\n * @type {ToMarkdownHandle}\n * @param {ListItem} node\n */\nfunction listItemWithTaskListItem(node, parent, state, info) {\n const head = node.children[0]\n const checkable =\n typeof node.checked === 'boolean' && head && head.type === 'paragraph'\n const checkbox = '[' + (node.checked ? 'x' : ' ') + '] '\n const tracker = state.createTracker(info)\n\n if (checkable) {\n tracker.move(checkbox)\n }\n\n let value = defaultHandlers.listItem(node, parent, state, {\n ...info,\n ...tracker.current()\n })\n\n if (checkable) {\n value = value.replace(/^(?:[*+-]|\\d+\\.)([\\r\\n]| {1,3})/, check)\n }\n\n return value\n\n /**\n * @param {string} $0\n * @returns {string}\n */\n function check($0) {\n return $0 + checkbox\n }\n}\n","/**\n * @import {Extension as FromMarkdownExtension} from 'mdast-util-from-markdown'\n * @import {Options} from 'mdast-util-gfm'\n * @import {Options as ToMarkdownExtension} from 'mdast-util-to-markdown'\n */\n\nimport {\n gfmAutolinkLiteralFromMarkdown,\n gfmAutolinkLiteralToMarkdown\n} from 'mdast-util-gfm-autolink-literal'\nimport {\n gfmFootnoteFromMarkdown,\n gfmFootnoteToMarkdown\n} from 'mdast-util-gfm-footnote'\nimport {\n gfmStrikethroughFromMarkdown,\n gfmStrikethroughToMarkdown\n} from 'mdast-util-gfm-strikethrough'\nimport {gfmTableFromMarkdown, gfmTableToMarkdown} from 'mdast-util-gfm-table'\nimport {\n gfmTaskListItemFromMarkdown,\n gfmTaskListItemToMarkdown\n} from 'mdast-util-gfm-task-list-item'\n\n/**\n * Create an extension for `mdast-util-from-markdown` to enable GFM (autolink\n * literals, footnotes, strikethrough, tables, tasklists).\n *\n * @returns {Array<FromMarkdownExtension>}\n * Extension for `mdast-util-from-markdown` to enable GFM (autolink literals,\n * footnotes, strikethrough, tables, tasklists).\n */\nexport function gfmFromMarkdown() {\n return [\n gfmAutolinkLiteralFromMarkdown(),\n gfmFootnoteFromMarkdown(),\n gfmStrikethroughFromMarkdown(),\n gfmTableFromMarkdown(),\n gfmTaskListItemFromMarkdown()\n ]\n}\n\n/**\n * Create an extension for `mdast-util-to-markdown` to enable GFM (autolink\n * literals, footnotes, strikethrough, tables, tasklists).\n *\n * @param {Options | null | undefined} [options]\n * Configuration (optional).\n * @returns {ToMarkdownExtension}\n * Extension for `mdast-util-to-markdown` to enable GFM (autolink literals,\n * footnotes, strikethrough, tables, tasklists).\n */\nexport function gfmToMarkdown(options) {\n return {\n extensions: [\n gfmAutolinkLiteralToMarkdown(),\n gfmFootnoteToMarkdown(options),\n gfmStrikethroughToMarkdown(),\n gfmTableToMarkdown(options),\n gfmTaskListItemToMarkdown()\n ]\n }\n}\n","/**\n * Like `Array#splice`, but smarter for giant arrays.\n *\n * `Array#splice` takes all items to be inserted as individual argument which\n * causes a stack overflow in V8 when trying to insert 100k items for instance.\n *\n * Otherwise, this does not return the removed items, and takes `items` as an\n * array instead of rest parameters.\n *\n * @template {unknown} T\n * Item type.\n * @param {Array<T>} list\n * List to operate on.\n * @param {number} start\n * Index to remove/insert at (can be negative).\n * @param {number} remove\n * Number of items to remove.\n * @param {Array<T>} items\n * Items to inject into `list`.\n * @returns {undefined}\n * Nothing.\n */\nexport function splice(list, start, remove, items) {\n const end = list.length;\n let chunkStart = 0;\n /** @type {Array<unknown>} */\n let parameters;\n\n // Make start between zero and `end` (included).\n if (start < 0) {\n start = -start > end ? 0 : end + start;\n } else {\n start = start > end ? end : start;\n }\n remove = remove > 0 ? remove : 0;\n\n // No need to chunk the items if there’s only a couple (10k) items.\n if (items.length < 10000) {\n parameters = Array.from(items);\n parameters.unshift(start, remove);\n // @ts-expect-error Hush, it’s fine.\n list.splice(...parameters);\n } else {\n // Delete `remove` items starting from `start`\n if (remove) list.splice(start, remove);\n\n // Insert the items in chunks to not cause stack overflows.\n while (chunkStart < items.length) {\n parameters = items.slice(chunkStart, chunkStart + 10000);\n parameters.unshift(start, 0);\n // @ts-expect-error Hush, it’s fine.\n list.splice(...parameters);\n chunkStart += 10000;\n start += 10000;\n }\n }\n}\n\n/**\n * Append `items` (an array) at the end of `list` (another array).\n * When `list` was empty, returns `items` instead.\n *\n * This prevents a potentially expensive operation when `list` is empty,\n * and adds items in batches to prevent V8 from hanging.\n *\n * @template {unknown} T\n * Item type.\n * @param {Array<T>} list\n * List to operate on.\n * @param {Array<T>} items\n * Items to add to `list`.\n * @returns {Array<T>}\n * Either `list` or `items`.\n */\nexport function push(list, items) {\n if (list.length > 0) {\n splice(list, list.length, 0, items);\n return list;\n }\n return items;\n}","/**\n * @import {\n * Extension,\n * Handles,\n * HtmlExtension,\n * NormalizedExtension\n * } from 'micromark-util-types'\n */\n\nimport {splice} from 'micromark-util-chunked'\n\nconst hasOwnProperty = {}.hasOwnProperty\n\n/**\n * Combine multiple syntax extensions into one.\n *\n * @param {ReadonlyArray<Extension>} extensions\n * List of syntax extensions.\n * @returns {NormalizedExtension}\n * A single combined extension.\n */\nexport function combineExtensions(extensions) {\n /** @type {NormalizedExtension} */\n const all = {}\n let index = -1\n\n while (++index < extensions.length) {\n syntaxExtension(all, extensions[index])\n }\n\n return all\n}\n\n/**\n * Merge `extension` into `all`.\n *\n * @param {NormalizedExtension} all\n * Extension to merge into.\n * @param {Extension} extension\n * Extension to merge.\n * @returns {undefined}\n * Nothing.\n */\nfunction syntaxExtension(all, extension) {\n /** @type {keyof Extension} */\n let hook\n\n for (hook in extension) {\n const maybe = hasOwnProperty.call(all, hook) ? all[hook] : undefined\n /** @type {Record<string, unknown>} */\n const left = maybe || (all[hook] = {})\n /** @type {Record<string, unknown> | undefined} */\n const right = extension[hook]\n /** @type {string} */\n let code\n\n if (right) {\n for (code in right) {\n if (!hasOwnProperty.call(left, code)) left[code] = []\n const value = right[code]\n constructs(\n // @ts-expect-error Looks like a list.\n left[code],\n Array.isArray(value) ? value : value ? [value] : []\n )\n }\n }\n }\n}\n\n/**\n * Merge `list` into `existing` (both lists of constructs).\n * Mutates `existing`.\n *\n * @param {Array<unknown>} existing\n * List of constructs to merge into.\n * @param {Array<unknown>} list\n * List of constructs to merge.\n * @returns {undefined}\n * Nothing.\n */\nfunction constructs(existing, list) {\n let index = -1\n /** @type {Array<unknown>} */\n const before = []\n\n while (++index < list.length) {\n // @ts-expect-error Looks like an object.\n ;(list[index].add === 'after' ? existing : before).push(list[index])\n }\n\n splice(existing, 0, 0, before)\n}\n\n/**\n * Combine multiple HTML extensions into one.\n *\n * @param {ReadonlyArray<HtmlExtension>} htmlExtensions\n * List of HTML extensions.\n * @returns {HtmlExtension}\n * Single combined HTML extension.\n */\nexport function combineHtmlExtensions(htmlExtensions) {\n /** @type {HtmlExtension} */\n const handlers = {}\n let index = -1\n\n while (++index < htmlExtensions.length) {\n htmlExtension(handlers, htmlExtensions[index])\n }\n\n return handlers\n}\n\n/**\n * Merge `extension` into `all`.\n *\n * @param {HtmlExtension} all\n * Extension to merge into.\n * @param {HtmlExtension} extension\n * Extension to merge.\n * @returns {undefined}\n * Nothing.\n */\nfunction htmlExtension(all, extension) {\n /** @type {keyof HtmlExtension} */\n let hook\n\n for (hook in extension) {\n const maybe = hasOwnProperty.call(all, hook) ? all[hook] : undefined\n const left = maybe || (all[hook] = {})\n const right = extension[hook]\n /** @type {keyof Handles} */\n let type\n\n if (right) {\n for (type in right) {\n // @ts-expect-error assume document vs regular handler are managed correctly.\n left[type] = right[type]\n }\n }\n }\n}\n","/**\n * @import {Code, ConstructRecord, Event, Extension, Previous, State, TokenizeContext, Tokenizer} from 'micromark-util-types'\n */\n\nimport { asciiAlpha, asciiAlphanumeric, asciiControl, markdownLineEndingOrSpace, unicodePunctuation, unicodeWhitespace } from 'micromark-util-character';\nconst wwwPrefix = {\n tokenize: tokenizeWwwPrefix,\n partial: true\n};\nconst domain = {\n tokenize: tokenizeDomain,\n partial: true\n};\nconst path = {\n tokenize: tokenizePath,\n partial: true\n};\nconst trail = {\n tokenize: tokenizeTrail,\n partial: true\n};\nconst emailDomainDotTrail = {\n tokenize: tokenizeEmailDomainDotTrail,\n partial: true\n};\nconst wwwAutolink = {\n name: 'wwwAutolink',\n tokenize: tokenizeWwwAutolink,\n previous: previousWww\n};\nconst protocolAutolink = {\n name: 'protocolAutolink',\n tokenize: tokenizeProtocolAutolink,\n previous: previousProtocol\n};\nconst emailAutolink = {\n name: 'emailAutolink',\n tokenize: tokenizeEmailAutolink,\n previous: previousEmail\n};\n\n/** @type {ConstructRecord} */\nconst text = {};\n\n/**\n * Create an extension for `micromark` to support GitHub autolink literal\n * syntax.\n *\n * @returns {Extension}\n * Extension for `micromark` that can be passed in `extensions` to enable GFM\n * autolink literal syntax.\n */\nexport function gfmAutolinkLiteral() {\n return {\n text\n };\n}\n\n/** @type {Code} */\nlet code = 48;\n\n// Add alphanumerics.\nwhile (code < 123) {\n text[code] = emailAutolink;\n code++;\n if (code === 58) code = 65;else if (code === 91) code = 97;\n}\ntext[43] = emailAutolink;\ntext[45] = emailAutolink;\ntext[46] = emailAutolink;\ntext[95] = emailAutolink;\ntext[72] = [emailAutolink, protocolAutolink];\ntext[104] = [emailAutolink, protocolAutolink];\ntext[87] = [emailAutolink, wwwAutolink];\ntext[119] = [emailAutolink, wwwAutolink];\n\n// To do: perform email autolink literals on events, afterwards.\n// That’s where `markdown-rs` and `cmark-gfm` perform it.\n// It should look for `@`, then for atext backwards, and then for a label\n// forwards.\n// To do: `mailto:`, `xmpp:` protocol as prefix.\n\n/**\n * Email autolink literal.\n *\n * ```markdown\n * > | a contact@example.org b\n * ^^^^^^^^^^^^^^^^^^^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeEmailAutolink(effects, ok, nok) {\n const self = this;\n /** @type {boolean | undefined} */\n let dot;\n /** @type {boolean} */\n let data;\n return start;\n\n /**\n * Start of email autolink literal.\n *\n * ```markdown\n * > | a contact@example.org b\n * ^\n * ```\n *\n * @type {State}\n */\n function start(code) {\n if (!gfmAtext(code) || !previousEmail.call(self, self.previous) || previousUnbalanced(self.events)) {\n return nok(code);\n }\n effects.enter('literalAutolink');\n effects.enter('literalAutolinkEmail');\n return atext(code);\n }\n\n /**\n * In email atext.\n *\n * ```markdown\n * > | a contact@example.org b\n * ^\n * ```\n *\n * @type {State}\n */\n function atext(code) {\n if (gfmAtext(code)) {\n effects.consume(code);\n return atext;\n }\n if (code === 64) {\n effects.consume(code);\n return emailDomain;\n }\n return nok(code);\n }\n\n /**\n * In email domain.\n *\n * The reference code is a bit overly complex as it handles the `@`, of which\n * there may be just one.\n * Source: <https://github.com/github/cmark-gfm/blob/ef1cfcb/extensions/autolink.c#L318>\n *\n * ```markdown\n * > | a contact@example.org b\n * ^\n * ```\n *\n * @type {State}\n */\n function emailDomain(code) {\n // Dot followed by alphanumerical (not `-` or `_`).\n if (code === 46) {\n return effects.check(emailDomainDotTrail, emailDomainAfter, emailDomainDot)(code);\n }\n\n // Alphanumerical, `-`, and `_`.\n if (code === 45 || code === 95 || asciiAlphanumeric(code)) {\n data = true;\n effects.consume(code);\n return emailDomain;\n }\n\n // To do: `/` if xmpp.\n\n // Note: normally we’d truncate trailing punctuation from the link.\n // However, email autolink literals cannot contain any of those markers,\n // except for `.`, but that can only occur if it isn’t trailing.\n // So we can ignore truncating!\n return emailDomainAfter(code);\n }\n\n /**\n * In email domain, on dot that is not a trail.\n *\n * ```markdown\n * > | a contact@example.org b\n * ^\n * ```\n *\n * @type {State}\n */\n function emailDomainDot(code) {\n effects.consume(code);\n dot = true;\n return emailDomain;\n }\n\n /**\n * After email domain.\n *\n * ```markdown\n * > | a contact@example.org b\n * ^\n * ```\n *\n * @type {State}\n */\n function emailDomainAfter(code) {\n // Domain must not be empty, must include a dot, and must end in alphabetical.\n // Source: <https://github.com/github/cmark-gfm/blob/ef1cfcb/extensions/autolink.c#L332>.\n if (data && dot && asciiAlpha(self.previous)) {\n effects.exit('literalAutolinkEmail');\n effects.exit('literalAutolink');\n return ok(code);\n }\n return nok(code);\n }\n}\n\n/**\n * `www` autolink literal.\n *\n * ```markdown\n * > | a www.example.org b\n * ^^^^^^^^^^^^^^^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeWwwAutolink(effects, ok, nok) {\n const self = this;\n return wwwStart;\n\n /**\n * Start of www autolink literal.\n *\n * ```markdown\n * > | www.example.com/a?b#c\n * ^\n * ```\n *\n * @type {State}\n */\n function wwwStart(code) {\n if (code !== 87 && code !== 119 || !previousWww.call(self, self.previous) || previousUnbalanced(self.events)) {\n return nok(code);\n }\n effects.enter('literalAutolink');\n effects.enter('literalAutolinkWww');\n // Note: we *check*, so we can discard the `www.` we parsed.\n // If it worked, we consider it as a part of the domain.\n return effects.check(wwwPrefix, effects.attempt(domain, effects.attempt(path, wwwAfter), nok), nok)(code);\n }\n\n /**\n * After a www autolink literal.\n *\n * ```markdown\n * > | www.example.com/a?b#c\n * ^\n * ```\n *\n * @type {State}\n */\n function wwwAfter(code) {\n effects.exit('literalAutolinkWww');\n effects.exit('literalAutolink');\n return ok(code);\n }\n}\n\n/**\n * Protocol autolink literal.\n *\n * ```markdown\n * > | a https://example.org b\n * ^^^^^^^^^^^^^^^^^^^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeProtocolAutolink(effects, ok, nok) {\n const self = this;\n let buffer = '';\n let seen = false;\n return protocolStart;\n\n /**\n * Start of protocol autolink literal.\n *\n * ```markdown\n * > | https://example.com/a?b#c\n * ^\n * ```\n *\n * @type {State}\n */\n function protocolStart(code) {\n if ((code === 72 || code === 104) && previousProtocol.call(self, self.previous) && !previousUnbalanced(self.events)) {\n effects.enter('literalAutolink');\n effects.enter('literalAutolinkHttp');\n buffer += String.fromCodePoint(code);\n effects.consume(code);\n return protocolPrefixInside;\n }\n return nok(code);\n }\n\n /**\n * In protocol.\n *\n * ```markdown\n * > | https://example.com/a?b#c\n * ^^^^^\n * ```\n *\n * @type {State}\n */\n function protocolPrefixInside(code) {\n // `5` is size of `https`\n if (asciiAlpha(code) && buffer.length < 5) {\n // @ts-expect-error: definitely number.\n buffer += String.fromCodePoint(code);\n effects.consume(code);\n return protocolPrefixInside;\n }\n if (code === 58) {\n const protocol = buffer.toLowerCase();\n if (protocol === 'http' || protocol === 'https') {\n effects.consume(code);\n return protocolSlashesInside;\n }\n }\n return nok(code);\n }\n\n /**\n * In slashes.\n *\n * ```markdown\n * > | https://example.com/a?b#c\n * ^^\n * ```\n *\n * @type {State}\n */\n function protocolSlashesInside(code) {\n if (code === 47) {\n effects.consume(code);\n if (seen) {\n return afterProtocol;\n }\n seen = true;\n return protocolSlashesInside;\n }\n return nok(code);\n }\n\n /**\n * After protocol, before domain.\n *\n * ```markdown\n * > | https://example.com/a?b#c\n * ^\n * ```\n *\n * @type {State}\n */\n function afterProtocol(code) {\n // To do: this is different from `markdown-rs`:\n // https://github.com/wooorm/markdown-rs/blob/b3a921c761309ae00a51fe348d8a43adbc54b518/src/construct/gfm_autolink_literal.rs#L172-L182\n return code === null || asciiControl(code) || markdownLineEndingOrSpace(code) || unicodeWhitespace(code) || unicodePunctuation(code) ? nok(code) : effects.attempt(domain, effects.attempt(path, protocolAfter), nok)(code);\n }\n\n /**\n * After a protocol autolink literal.\n *\n * ```markdown\n * > | https://example.com/a?b#c\n * ^\n * ```\n *\n * @type {State}\n */\n function protocolAfter(code) {\n effects.exit('literalAutolinkHttp');\n effects.exit('literalAutolink');\n return ok(code);\n }\n}\n\n/**\n * `www` prefix.\n *\n * ```markdown\n * > | a www.example.org b\n * ^^^^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeWwwPrefix(effects, ok, nok) {\n let size = 0;\n return wwwPrefixInside;\n\n /**\n * In www prefix.\n *\n * ```markdown\n * > | www.example.com\n * ^^^^\n * ```\n *\n * @type {State}\n */\n function wwwPrefixInside(code) {\n if ((code === 87 || code === 119) && size < 3) {\n size++;\n effects.consume(code);\n return wwwPrefixInside;\n }\n if (code === 46 && size === 3) {\n effects.consume(code);\n return wwwPrefixAfter;\n }\n return nok(code);\n }\n\n /**\n * After www prefix.\n *\n * ```markdown\n * > | www.example.com\n * ^\n * ```\n *\n * @type {State}\n */\n function wwwPrefixAfter(code) {\n // If there is *anything*, we can link.\n return code === null ? nok(code) : ok(code);\n }\n}\n\n/**\n * Domain.\n *\n * ```markdown\n * > | a https://example.org b\n * ^^^^^^^^^^^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeDomain(effects, ok, nok) {\n /** @type {boolean | undefined} */\n let underscoreInLastSegment;\n /** @type {boolean | undefined} */\n let underscoreInLastLastSegment;\n /** @type {boolean | undefined} */\n let seen;\n return domainInside;\n\n /**\n * In domain.\n *\n * ```markdown\n * > | https://example.com/a\n * ^^^^^^^^^^^\n * ```\n *\n * @type {State}\n */\n function domainInside(code) {\n // Check whether this marker, which is a trailing punctuation\n // marker, optionally followed by more trailing markers, and then\n // followed by an end.\n if (code === 46 || code === 95) {\n return effects.check(trail, domainAfter, domainAtPunctuation)(code);\n }\n\n // GH documents that only alphanumerics (other than `-`, `.`, and `_`) can\n // occur, which sounds like ASCII only, but they also support `www.點看.com`,\n // so that’s Unicode.\n // Instead of some new production for Unicode alphanumerics, markdown\n // already has that for Unicode punctuation and whitespace, so use those.\n // Source: <https://github.com/github/cmark-gfm/blob/ef1cfcb/extensions/autolink.c#L12>.\n if (code === null || markdownLineEndingOrSpace(code) || unicodeWhitespace(code) || code !== 45 && unicodePunctuation(code)) {\n return domainAfter(code);\n }\n seen = true;\n effects.consume(code);\n return domainInside;\n }\n\n /**\n * In domain, at potential trailing punctuation, that was not trailing.\n *\n * ```markdown\n * > | https://example.com\n * ^\n * ```\n *\n * @type {State}\n */\n function domainAtPunctuation(code) {\n // There is an underscore in the last segment of the domain\n if (code === 95) {\n underscoreInLastSegment = true;\n }\n // Otherwise, it’s a `.`: save the last segment underscore in the\n // penultimate segment slot.\n else {\n underscoreInLastLastSegment = underscoreInLastSegment;\n underscoreInLastSegment = undefined;\n }\n effects.consume(code);\n return domainInside;\n }\n\n /**\n * After domain.\n *\n * ```markdown\n * > | https://example.com/a\n * ^\n * ```\n *\n * @type {State} */\n function domainAfter(code) {\n // Note: that’s GH says a dot is needed, but it’s not true:\n // <https://github.com/github/cmark-gfm/issues/279>\n if (underscoreInLastLastSegment || underscoreInLastSegment || !seen) {\n return nok(code);\n }\n return ok(code);\n }\n}\n\n/**\n * Path.\n *\n * ```markdown\n * > | a https://example.org/stuff b\n * ^^^^^^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizePath(effects, ok) {\n let sizeOpen = 0;\n let sizeClose = 0;\n return pathInside;\n\n /**\n * In path.\n *\n * ```markdown\n * > | https://example.com/a\n * ^^\n * ```\n *\n * @type {State}\n */\n function pathInside(code) {\n if (code === 40) {\n sizeOpen++;\n effects.consume(code);\n return pathInside;\n }\n\n // To do: `markdown-rs` also needs this.\n // If this is a paren, and there are less closings than openings,\n // we don’t check for a trail.\n if (code === 41 && sizeClose < sizeOpen) {\n return pathAtPunctuation(code);\n }\n\n // Check whether this trailing punctuation marker is optionally\n // followed by more trailing markers, and then followed\n // by an end.\n if (code === 33 || code === 34 || code === 38 || code === 39 || code === 41 || code === 42 || code === 44 || code === 46 || code === 58 || code === 59 || code === 60 || code === 63 || code === 93 || code === 95 || code === 126) {\n return effects.check(trail, ok, pathAtPunctuation)(code);\n }\n if (code === null || markdownLineEndingOrSpace(code) || unicodeWhitespace(code)) {\n return ok(code);\n }\n effects.consume(code);\n return pathInside;\n }\n\n /**\n * In path, at potential trailing punctuation, that was not trailing.\n *\n * ```markdown\n * > | https://example.com/a\"b\n * ^\n * ```\n *\n * @type {State}\n */\n function pathAtPunctuation(code) {\n // Count closing parens.\n if (code === 41) {\n sizeClose++;\n }\n effects.consume(code);\n return pathInside;\n }\n}\n\n/**\n * Trail.\n *\n * This calls `ok` if this *is* the trail, followed by an end, which means\n * the entire trail is not part of the link.\n * It calls `nok` if this *is* part of the link.\n *\n * ```markdown\n * > | https://example.com\").\n * ^^^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeTrail(effects, ok, nok) {\n return trail;\n\n /**\n * In trail of domain or path.\n *\n * ```markdown\n * > | https://example.com\").\n * ^\n * ```\n *\n * @type {State}\n */\n function trail(code) {\n // Regular trailing punctuation.\n if (code === 33 || code === 34 || code === 39 || code === 41 || code === 42 || code === 44 || code === 46 || code === 58 || code === 59 || code === 63 || code === 95 || code === 126) {\n effects.consume(code);\n return trail;\n }\n\n // `&` followed by one or more alphabeticals and then a `;`, is\n // as a whole considered as trailing punctuation.\n // In all other cases, it is considered as continuation of the URL.\n if (code === 38) {\n effects.consume(code);\n return trailCharacterReferenceStart;\n }\n\n // Needed because we allow literals after `[`, as we fix:\n // <https://github.com/github/cmark-gfm/issues/278>.\n // Check that it is not followed by `(` or `[`.\n if (code === 93) {\n effects.consume(code);\n return trailBracketAfter;\n }\n if (\n // `<` is an end.\n code === 60 ||\n // So is whitespace.\n code === null || markdownLineEndingOrSpace(code) || unicodeWhitespace(code)) {\n return ok(code);\n }\n return nok(code);\n }\n\n /**\n * In trail, after `]`.\n *\n * > 👉 **Note**: this deviates from `cmark-gfm` to fix a bug.\n * > See end of <https://github.com/github/cmark-gfm/issues/278> for more.\n *\n * ```markdown\n * > | https://example.com](\n * ^\n * ```\n *\n * @type {State}\n */\n function trailBracketAfter(code) {\n // Whitespace or something that could start a resource or reference is the end.\n // Switch back to trail otherwise.\n if (code === null || code === 40 || code === 91 || markdownLineEndingOrSpace(code) || unicodeWhitespace(code)) {\n return ok(code);\n }\n return trail(code);\n }\n\n /**\n * In character-reference like trail, after `&`.\n *\n * ```markdown\n * > | https://example.com&).\n * ^\n * ```\n *\n * @type {State}\n */\n function trailCharacterReferenceStart(code) {\n // When non-alpha, it’s not a trail.\n return asciiAlpha(code) ? trailCharacterReferenceInside(code) : nok(code);\n }\n\n /**\n * In character-reference like trail.\n *\n * ```markdown\n * > | https://example.com&).\n * ^\n * ```\n *\n * @type {State}\n */\n function trailCharacterReferenceInside(code) {\n // Switch back to trail if this is well-formed.\n if (code === 59) {\n effects.consume(code);\n return trail;\n }\n if (asciiAlpha(code)) {\n effects.consume(code);\n return trailCharacterReferenceInside;\n }\n\n // It’s not a trail.\n return nok(code);\n }\n}\n\n/**\n * Dot in email domain trail.\n *\n * This calls `ok` if this *is* the trail, followed by an end, which means\n * the trail is not part of the link.\n * It calls `nok` if this *is* part of the link.\n *\n * ```markdown\n * > | contact@example.org.\n * ^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeEmailDomainDotTrail(effects, ok, nok) {\n return start;\n\n /**\n * Dot.\n *\n * ```markdown\n * > | contact@example.org.\n * ^ ^\n * ```\n *\n * @type {State}\n */\n function start(code) {\n // Must be dot.\n effects.consume(code);\n return after;\n }\n\n /**\n * After dot.\n *\n * ```markdown\n * > | contact@example.org.\n * ^ ^\n * ```\n *\n * @type {State}\n */\n function after(code) {\n // Not a trail if alphanumeric.\n return asciiAlphanumeric(code) ? nok(code) : ok(code);\n }\n}\n\n/**\n * See:\n * <https://github.com/github/cmark-gfm/blob/ef1cfcb/extensions/autolink.c#L156>.\n *\n * @type {Previous}\n */\nfunction previousWww(code) {\n return code === null || code === 40 || code === 42 || code === 95 || code === 91 || code === 93 || code === 126 || markdownLineEndingOrSpace(code);\n}\n\n/**\n * See:\n * <https://github.com/github/cmark-gfm/blob/ef1cfcb/extensions/autolink.c#L214>.\n *\n * @type {Previous}\n */\nfunction previousProtocol(code) {\n return !asciiAlpha(code);\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Previous}\n */\nfunction previousEmail(code) {\n // Do not allow a slash “inside” atext.\n // The reference code is a bit weird, but that’s what it results in.\n // Source: <https://github.com/github/cmark-gfm/blob/ef1cfcb/extensions/autolink.c#L307>.\n // Other than slash, every preceding character is allowed.\n return !(code === 47 || gfmAtext(code));\n}\n\n/**\n * @param {Code} code\n * @returns {boolean}\n */\nfunction gfmAtext(code) {\n return code === 43 || code === 45 || code === 46 || code === 95 || asciiAlphanumeric(code);\n}\n\n/**\n * @param {Array<Event>} events\n * @returns {boolean}\n */\nfunction previousUnbalanced(events) {\n let index = events.length;\n let result = false;\n while (index--) {\n const token = events[index][1];\n if ((token.type === 'labelLink' || token.type === 'labelImage') && !token._balanced) {\n result = true;\n break;\n }\n\n // If we’ve seen this token, and it was marked as not having any unbalanced\n // bracket before it, we can exit.\n if (token._gfmAutolinkLiteralWalkedInto) {\n result = false;\n break;\n }\n }\n if (events.length > 0 && !result) {\n // Mark the last token as “walked into” w/o finding\n // anything.\n events[events.length - 1][1]._gfmAutolinkLiteralWalkedInto = true;\n }\n return result;\n}","/**\n * @import {Event, Resolver, TokenizeContext} from 'micromark-util-types'\n */\n\n/**\n * Call all `resolveAll`s.\n *\n * @param {ReadonlyArray<{resolveAll?: Resolver | undefined}>} constructs\n * List of constructs, optionally with `resolveAll`s.\n * @param {Array<Event>} events\n * List of events.\n * @param {TokenizeContext} context\n * Context used by `tokenize`.\n * @returns {Array<Event>}\n * Changed events.\n */\nexport function resolveAll(constructs, events, context) {\n /** @type {Array<Resolver>} */\n const called = []\n let index = -1\n\n while (++index < constructs.length) {\n const resolve = constructs[index].resolveAll\n\n if (resolve && !called.includes(resolve)) {\n events = resolve(events, context)\n called.push(resolve)\n }\n }\n\n return events\n}\n","/**\n * @import {Effects, State, TokenType} from 'micromark-util-types'\n */\n\nimport { markdownSpace } from 'micromark-util-character';\n\n// To do: implement `spaceOrTab`, `spaceOrTabMinMax`, `spaceOrTabWithOptions`.\n\n/**\n * Parse spaces and tabs.\n *\n * There is no `nok` parameter:\n *\n * * spaces in markdown are often optional, in which case this factory can be\n * used and `ok` will be switched to whether spaces were found or not\n * * one line ending or space can be detected with `markdownSpace(code)` right\n * before using `factorySpace`\n *\n * ###### Examples\n *\n * Where `␉` represents a tab (plus how much it expands) and `␠` represents a\n * single space.\n *\n * ```markdown\n * ␉\n * ␠␠␠␠\n * ␉␠\n * ```\n *\n * @param {Effects} effects\n * Context.\n * @param {State} ok\n * State switched to when successful.\n * @param {TokenType} type\n * Type (`' \\t'`).\n * @param {number | undefined} [max=Infinity]\n * Max (exclusive).\n * @returns {State}\n * Start state.\n */\nexport function factorySpace(effects, ok, type, max) {\n const limit = max ? max - 1 : Number.POSITIVE_INFINITY;\n let size = 0;\n return start;\n\n /** @type {State} */\n function start(code) {\n if (markdownSpace(code)) {\n effects.enter(type);\n return prefix(code);\n }\n return ok(code);\n }\n\n /** @type {State} */\n function prefix(code) {\n if (markdownSpace(code) && size++ < limit) {\n effects.consume(code);\n return prefix;\n }\n effects.exit(type);\n return ok(code);\n }\n}","/**\n * @import {\n * Construct,\n * State,\n * TokenizeContext,\n * Tokenizer\n * } from 'micromark-util-types'\n */\n\nimport { factorySpace } from 'micromark-factory-space';\nimport { markdownLineEnding, markdownSpace } from 'micromark-util-character';\n/** @type {Construct} */\nexport const blankLine = {\n partial: true,\n tokenize: tokenizeBlankLine\n};\n\n/**\n * @this {TokenizeContext}\n * Context.\n * @type {Tokenizer}\n */\nfunction tokenizeBlankLine(effects, ok, nok) {\n return start;\n\n /**\n * Start of blank line.\n *\n * > 👉 **Note**: `␠` represents a space character.\n *\n * ```markdown\n * > | ␠␠␊\n * ^\n * > | ␊\n * ^\n * ```\n *\n * @type {State}\n */\n function start(code) {\n return markdownSpace(code) ? factorySpace(effects, after, \"linePrefix\")(code) : after(code);\n }\n\n /**\n * At eof/eol, after optional whitespace.\n *\n * > 👉 **Note**: `␠` represents a space character.\n *\n * ```markdown\n * > | ␠␠␊\n * ^\n * > | ␊\n * ^\n * ```\n *\n * @type {State}\n */\n function after(code) {\n return code === null || markdownLineEnding(code) ? ok(code) : nok(code);\n }\n}","/**\n * @import {Event, Exiter, Extension, Resolver, State, Token, TokenizeContext, Tokenizer} from 'micromark-util-types'\n */\n\nimport { blankLine } from 'micromark-core-commonmark';\nimport { factorySpace } from 'micromark-factory-space';\nimport { markdownLineEndingOrSpace } from 'micromark-util-character';\nimport { normalizeIdentifier } from 'micromark-util-normalize-identifier';\nconst indent = {\n tokenize: tokenizeIndent,\n partial: true\n};\n\n// To do: micromark should support a `_hiddenGfmFootnoteSupport`, which only\n// affects label start (image).\n// That will let us drop `tokenizePotentialGfmFootnote*`.\n// It currently has a `_hiddenFootnoteSupport`, which affects that and more.\n// That can be removed when `micromark-extension-footnote` is archived.\n\n/**\n * Create an extension for `micromark` to enable GFM footnote syntax.\n *\n * @returns {Extension}\n * Extension for `micromark` that can be passed in `extensions` to\n * enable GFM footnote syntax.\n */\nexport function gfmFootnote() {\n /** @type {Extension} */\n return {\n document: {\n [91]: {\n name: 'gfmFootnoteDefinition',\n tokenize: tokenizeDefinitionStart,\n continuation: {\n tokenize: tokenizeDefinitionContinuation\n },\n exit: gfmFootnoteDefinitionEnd\n }\n },\n text: {\n [91]: {\n name: 'gfmFootnoteCall',\n tokenize: tokenizeGfmFootnoteCall\n },\n [93]: {\n name: 'gfmPotentialFootnoteCall',\n add: 'after',\n tokenize: tokenizePotentialGfmFootnoteCall,\n resolveTo: resolveToPotentialGfmFootnoteCall\n }\n }\n };\n}\n\n// To do: remove after micromark update.\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizePotentialGfmFootnoteCall(effects, ok, nok) {\n const self = this;\n let index = self.events.length;\n const defined = self.parser.gfmFootnotes || (self.parser.gfmFootnotes = []);\n /** @type {Token} */\n let labelStart;\n\n // Find an opening.\n while (index--) {\n const token = self.events[index][1];\n if (token.type === \"labelImage\") {\n labelStart = token;\n break;\n }\n\n // Exit if we’ve walked far enough.\n if (token.type === 'gfmFootnoteCall' || token.type === \"labelLink\" || token.type === \"label\" || token.type === \"image\" || token.type === \"link\") {\n break;\n }\n }\n return start;\n\n /**\n * @type {State}\n */\n function start(code) {\n if (!labelStart || !labelStart._balanced) {\n return nok(code);\n }\n const id = normalizeIdentifier(self.sliceSerialize({\n start: labelStart.end,\n end: self.now()\n }));\n if (id.codePointAt(0) !== 94 || !defined.includes(id.slice(1))) {\n return nok(code);\n }\n effects.enter('gfmFootnoteCallLabelMarker');\n effects.consume(code);\n effects.exit('gfmFootnoteCallLabelMarker');\n return ok(code);\n }\n}\n\n// To do: remove after micromark update.\n/** @type {Resolver} */\nfunction resolveToPotentialGfmFootnoteCall(events, context) {\n let index = events.length;\n /** @type {Token | undefined} */\n let labelStart;\n\n // Find an opening.\n while (index--) {\n if (events[index][1].type === \"labelImage\" && events[index][0] === 'enter') {\n labelStart = events[index][1];\n break;\n }\n }\n // Change the `labelImageMarker` to a `data`.\n events[index + 1][1].type = \"data\";\n events[index + 3][1].type = 'gfmFootnoteCallLabelMarker';\n\n // The whole (without `!`):\n /** @type {Token} */\n const call = {\n type: 'gfmFootnoteCall',\n start: Object.assign({}, events[index + 3][1].start),\n end: Object.assign({}, events[events.length - 1][1].end)\n };\n // The `^` marker\n /** @type {Token} */\n const marker = {\n type: 'gfmFootnoteCallMarker',\n start: Object.assign({}, events[index + 3][1].end),\n end: Object.assign({}, events[index + 3][1].end)\n };\n // Increment the end 1 character.\n marker.end.column++;\n marker.end.offset++;\n marker.end._bufferIndex++;\n /** @type {Token} */\n const string = {\n type: 'gfmFootnoteCallString',\n start: Object.assign({}, marker.end),\n end: Object.assign({}, events[events.length - 1][1].start)\n };\n /** @type {Token} */\n const chunk = {\n type: \"chunkString\",\n contentType: 'string',\n start: Object.assign({}, string.start),\n end: Object.assign({}, string.end)\n };\n\n /** @type {Array<Event>} */\n const replacement = [\n // Take the `labelImageMarker` (now `data`, the `!`)\n events[index + 1], events[index + 2], ['enter', call, context],\n // The `[`\n events[index + 3], events[index + 4],\n // The `^`.\n ['enter', marker, context], ['exit', marker, context],\n // Everything in between.\n ['enter', string, context], ['enter', chunk, context], ['exit', chunk, context], ['exit', string, context],\n // The ending (`]`, properly parsed and labelled).\n events[events.length - 2], events[events.length - 1], ['exit', call, context]];\n events.splice(index, events.length - index + 1, ...replacement);\n return events;\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeGfmFootnoteCall(effects, ok, nok) {\n const self = this;\n const defined = self.parser.gfmFootnotes || (self.parser.gfmFootnotes = []);\n let size = 0;\n /** @type {boolean} */\n let data;\n\n // Note: the implementation of `markdown-rs` is different, because it houses\n // core *and* extensions in one project.\n // Therefore, it can include footnote logic inside `label-end`.\n // We can’t do that, but luckily, we can parse footnotes in a simpler way than\n // needed for labels.\n return start;\n\n /**\n * Start of footnote label.\n *\n * ```markdown\n * > | a [^b] c\n * ^\n * ```\n *\n * @type {State}\n */\n function start(code) {\n effects.enter('gfmFootnoteCall');\n effects.enter('gfmFootnoteCallLabelMarker');\n effects.consume(code);\n effects.exit('gfmFootnoteCallLabelMarker');\n return callStart;\n }\n\n /**\n * After `[`, at `^`.\n *\n * ```markdown\n * > | a [^b] c\n * ^\n * ```\n *\n * @type {State}\n */\n function callStart(code) {\n if (code !== 94) return nok(code);\n effects.enter('gfmFootnoteCallMarker');\n effects.consume(code);\n effects.exit('gfmFootnoteCallMarker');\n effects.enter('gfmFootnoteCallString');\n effects.enter('chunkString').contentType = 'string';\n return callData;\n }\n\n /**\n * In label.\n *\n * ```markdown\n * > | a [^b] c\n * ^\n * ```\n *\n * @type {State}\n */\n function callData(code) {\n if (\n // Too long.\n size > 999 ||\n // Closing brace with nothing.\n code === 93 && !data ||\n // Space or tab is not supported by GFM for some reason.\n // `\\n` and `[` not being supported makes sense.\n code === null || code === 91 || markdownLineEndingOrSpace(code)) {\n return nok(code);\n }\n if (code === 93) {\n effects.exit('chunkString');\n const token = effects.exit('gfmFootnoteCallString');\n if (!defined.includes(normalizeIdentifier(self.sliceSerialize(token)))) {\n return nok(code);\n }\n effects.enter('gfmFootnoteCallLabelMarker');\n effects.consume(code);\n effects.exit('gfmFootnoteCallLabelMarker');\n effects.exit('gfmFootnoteCall');\n return ok;\n }\n if (!markdownLineEndingOrSpace(code)) {\n data = true;\n }\n size++;\n effects.consume(code);\n return code === 92 ? callEscape : callData;\n }\n\n /**\n * On character after escape.\n *\n * ```markdown\n * > | a [^b\\c] d\n * ^\n * ```\n *\n * @type {State}\n */\n function callEscape(code) {\n if (code === 91 || code === 92 || code === 93) {\n effects.consume(code);\n size++;\n return callData;\n }\n return callData(code);\n }\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeDefinitionStart(effects, ok, nok) {\n const self = this;\n const defined = self.parser.gfmFootnotes || (self.parser.gfmFootnotes = []);\n /** @type {string} */\n let identifier;\n let size = 0;\n /** @type {boolean | undefined} */\n let data;\n return start;\n\n /**\n * Start of GFM footnote definition.\n *\n * ```markdown\n * > | [^a]: b\n * ^\n * ```\n *\n * @type {State}\n */\n function start(code) {\n effects.enter('gfmFootnoteDefinition')._container = true;\n effects.enter('gfmFootnoteDefinitionLabel');\n effects.enter('gfmFootnoteDefinitionLabelMarker');\n effects.consume(code);\n effects.exit('gfmFootnoteDefinitionLabelMarker');\n return labelAtMarker;\n }\n\n /**\n * In label, at caret.\n *\n * ```markdown\n * > | [^a]: b\n * ^\n * ```\n *\n * @type {State}\n */\n function labelAtMarker(code) {\n if (code === 94) {\n effects.enter('gfmFootnoteDefinitionMarker');\n effects.consume(code);\n effects.exit('gfmFootnoteDefinitionMarker');\n effects.enter('gfmFootnoteDefinitionLabelString');\n effects.enter('chunkString').contentType = 'string';\n return labelInside;\n }\n return nok(code);\n }\n\n /**\n * In label.\n *\n * > 👉 **Note**: `cmark-gfm` prevents whitespace from occurring in footnote\n * > definition labels.\n *\n * ```markdown\n * > | [^a]: b\n * ^\n * ```\n *\n * @type {State}\n */\n function labelInside(code) {\n if (\n // Too long.\n size > 999 ||\n // Closing brace with nothing.\n code === 93 && !data ||\n // Space or tab is not supported by GFM for some reason.\n // `\\n` and `[` not being supported makes sense.\n code === null || code === 91 || markdownLineEndingOrSpace(code)) {\n return nok(code);\n }\n if (code === 93) {\n effects.exit('chunkString');\n const token = effects.exit('gfmFootnoteDefinitionLabelString');\n identifier = normalizeIdentifier(self.sliceSerialize(token));\n effects.enter('gfmFootnoteDefinitionLabelMarker');\n effects.consume(code);\n effects.exit('gfmFootnoteDefinitionLabelMarker');\n effects.exit('gfmFootnoteDefinitionLabel');\n return labelAfter;\n }\n if (!markdownLineEndingOrSpace(code)) {\n data = true;\n }\n size++;\n effects.consume(code);\n return code === 92 ? labelEscape : labelInside;\n }\n\n /**\n * After `\\`, at a special character.\n *\n * > 👉 **Note**: `cmark-gfm` currently does not support escaped brackets:\n * > <https://github.com/github/cmark-gfm/issues/240>\n *\n * ```markdown\n * > | [^a\\*b]: c\n * ^\n * ```\n *\n * @type {State}\n */\n function labelEscape(code) {\n if (code === 91 || code === 92 || code === 93) {\n effects.consume(code);\n size++;\n return labelInside;\n }\n return labelInside(code);\n }\n\n /**\n * After definition label.\n *\n * ```markdown\n * > | [^a]: b\n * ^\n * ```\n *\n * @type {State}\n */\n function labelAfter(code) {\n if (code === 58) {\n effects.enter('definitionMarker');\n effects.consume(code);\n effects.exit('definitionMarker');\n if (!defined.includes(identifier)) {\n defined.push(identifier);\n }\n\n // Any whitespace after the marker is eaten, forming indented code\n // is not possible.\n // No space is also fine, just like a block quote marker.\n return factorySpace(effects, whitespaceAfter, 'gfmFootnoteDefinitionWhitespace');\n }\n return nok(code);\n }\n\n /**\n * After definition prefix.\n *\n * ```markdown\n * > | [^a]: b\n * ^\n * ```\n *\n * @type {State}\n */\n function whitespaceAfter(code) {\n // `markdown-rs` has a wrapping token for the prefix that is closed here.\n return ok(code);\n }\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeDefinitionContinuation(effects, ok, nok) {\n /// Start of footnote definition continuation.\n ///\n /// ```markdown\n /// | [^a]: b\n /// > | c\n /// ^\n /// ```\n //\n // Either a blank line, which is okay, or an indented thing.\n return effects.check(blankLine, ok, effects.attempt(indent, ok, nok));\n}\n\n/** @type {Exiter} */\nfunction gfmFootnoteDefinitionEnd(effects) {\n effects.exit('gfmFootnoteDefinition');\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeIndent(effects, ok, nok) {\n const self = this;\n return factorySpace(effects, afterPrefix, 'gfmFootnoteDefinitionIndent', 4 + 1);\n\n /**\n * @type {State}\n */\n function afterPrefix(code) {\n const tail = self.events[self.events.length - 1];\n return tail && tail[1].type === 'gfmFootnoteDefinitionIndent' && tail[2].sliceSerialize(tail[1], true).length === 4 ? ok(code) : nok(code);\n }\n}","/**\n * @import {Options} from 'micromark-extension-gfm-strikethrough'\n * @import {Event, Extension, Resolver, State, Token, TokenizeContext, Tokenizer} from 'micromark-util-types'\n */\n\nimport { splice } from 'micromark-util-chunked';\nimport { classifyCharacter } from 'micromark-util-classify-character';\nimport { resolveAll } from 'micromark-util-resolve-all';\n/**\n * Create an extension for `micromark` to enable GFM strikethrough syntax.\n *\n * @param {Options | null | undefined} [options={}]\n * Configuration.\n * @returns {Extension}\n * Extension for `micromark` that can be passed in `extensions`, to\n * enable GFM strikethrough syntax.\n */\nexport function gfmStrikethrough(options) {\n const options_ = options || {};\n let single = options_.singleTilde;\n const tokenizer = {\n name: 'strikethrough',\n tokenize: tokenizeStrikethrough,\n resolveAll: resolveAllStrikethrough\n };\n if (single === null || single === undefined) {\n single = true;\n }\n return {\n text: {\n [126]: tokenizer\n },\n insideSpan: {\n null: [tokenizer]\n },\n attentionMarkers: {\n null: [126]\n }\n };\n\n /**\n * Take events and resolve strikethrough.\n *\n * @type {Resolver}\n */\n function resolveAllStrikethrough(events, context) {\n let index = -1;\n\n // Walk through all events.\n while (++index < events.length) {\n // Find a token that can close.\n if (events[index][0] === 'enter' && events[index][1].type === 'strikethroughSequenceTemporary' && events[index][1]._close) {\n let open = index;\n\n // Now walk back to find an opener.\n while (open--) {\n // Find a token that can open the closer.\n if (events[open][0] === 'exit' && events[open][1].type === 'strikethroughSequenceTemporary' && events[open][1]._open &&\n // If the sizes are the same:\n events[index][1].end.offset - events[index][1].start.offset === events[open][1].end.offset - events[open][1].start.offset) {\n events[index][1].type = 'strikethroughSequence';\n events[open][1].type = 'strikethroughSequence';\n\n /** @type {Token} */\n const strikethrough = {\n type: 'strikethrough',\n start: Object.assign({}, events[open][1].start),\n end: Object.assign({}, events[index][1].end)\n };\n\n /** @type {Token} */\n const text = {\n type: 'strikethroughText',\n start: Object.assign({}, events[open][1].end),\n end: Object.assign({}, events[index][1].start)\n };\n\n // Opening.\n /** @type {Array<Event>} */\n const nextEvents = [['enter', strikethrough, context], ['enter', events[open][1], context], ['exit', events[open][1], context], ['enter', text, context]];\n const insideSpan = context.parser.constructs.insideSpan.null;\n if (insideSpan) {\n // Between.\n splice(nextEvents, nextEvents.length, 0, resolveAll(insideSpan, events.slice(open + 1, index), context));\n }\n\n // Closing.\n splice(nextEvents, nextEvents.length, 0, [['exit', text, context], ['enter', events[index][1], context], ['exit', events[index][1], context], ['exit', strikethrough, context]]);\n splice(events, open - 1, index - open + 3, nextEvents);\n index = open + nextEvents.length - 2;\n break;\n }\n }\n }\n }\n index = -1;\n while (++index < events.length) {\n if (events[index][1].type === 'strikethroughSequenceTemporary') {\n events[index][1].type = \"data\";\n }\n }\n return events;\n }\n\n /**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\n function tokenizeStrikethrough(effects, ok, nok) {\n const previous = this.previous;\n const events = this.events;\n let size = 0;\n return start;\n\n /** @type {State} */\n function start(code) {\n if (previous === 126 && events[events.length - 1][1].type !== \"characterEscape\") {\n return nok(code);\n }\n effects.enter('strikethroughSequenceTemporary');\n return more(code);\n }\n\n /** @type {State} */\n function more(code) {\n const before = classifyCharacter(previous);\n if (code === 126) {\n // If this is the third marker, exit.\n if (size > 1) return nok(code);\n effects.consume(code);\n size++;\n return more;\n }\n if (size < 2 && !single) return nok(code);\n const token = effects.exit('strikethroughSequenceTemporary');\n const after = classifyCharacter(code);\n token._open = !after || after === 2 && Boolean(before);\n token._close = !before || before === 2 && Boolean(after);\n return ok(code);\n }\n }\n}","/**\n * @import {Event} from 'micromark-util-types'\n */\n\n// Port of `edit_map.rs` from `markdown-rs`.\n// This should move to `markdown-js` later.\n\n// Deal with several changes in events, batching them together.\n//\n// Preferably, changes should be kept to a minimum.\n// Sometimes, it’s needed to change the list of events, because parsing can be\n// messy, and it helps to expose a cleaner interface of events to the compiler\n// and other users.\n// It can also help to merge many adjacent similar events.\n// And, in other cases, it’s needed to parse subcontent: pass some events\n// through another tokenizer and inject the result.\n\n/**\n * @typedef {[number, number, Array<Event>]} Change\n * @typedef {[number, number, number]} Jump\n */\n\n/**\n * Tracks a bunch of edits.\n */\nexport class EditMap {\n /**\n * Create a new edit map.\n */\n constructor() {\n /**\n * Record of changes.\n *\n * @type {Array<Change>}\n */\n this.map = [];\n }\n\n /**\n * Create an edit: a remove and/or add at a certain place.\n *\n * @param {number} index\n * @param {number} remove\n * @param {Array<Event>} add\n * @returns {undefined}\n */\n add(index, remove, add) {\n addImplementation(this, index, remove, add);\n }\n\n // To do: add this when moving to `micromark`.\n // /**\n // * Create an edit: but insert `add` before existing additions.\n // *\n // * @param {number} index\n // * @param {number} remove\n // * @param {Array<Event>} add\n // * @returns {undefined}\n // */\n // addBefore(index, remove, add) {\n // addImplementation(this, index, remove, add, true)\n // }\n\n /**\n * Done, change the events.\n *\n * @param {Array<Event>} events\n * @returns {undefined}\n */\n consume(events) {\n this.map.sort(function (a, b) {\n return a[0] - b[0];\n });\n\n /* c8 ignore next 3 -- `resolve` is never called without tables, so without edits. */\n if (this.map.length === 0) {\n return;\n }\n\n // To do: if links are added in events, like they are in `markdown-rs`,\n // this is needed.\n // // Calculate jumps: where items in the current list move to.\n // /** @type {Array<Jump>} */\n // const jumps = []\n // let index = 0\n // let addAcc = 0\n // let removeAcc = 0\n // while (index < this.map.length) {\n // const [at, remove, add] = this.map[index]\n // removeAcc += remove\n // addAcc += add.length\n // jumps.push([at, removeAcc, addAcc])\n // index += 1\n // }\n //\n // . shiftLinks(events, jumps)\n\n let index = this.map.length;\n /** @type {Array<Array<Event>>} */\n const vecs = [];\n while (index > 0) {\n index -= 1;\n vecs.push(events.slice(this.map[index][0] + this.map[index][1]), this.map[index][2]);\n\n // Truncate rest.\n events.length = this.map[index][0];\n }\n vecs.push(events.slice());\n events.length = 0;\n let slice = vecs.pop();\n while (slice) {\n for (const element of slice) {\n events.push(element);\n }\n slice = vecs.pop();\n }\n\n // Truncate everything.\n this.map.length = 0;\n }\n}\n\n/**\n * Create an edit.\n *\n * @param {EditMap} editMap\n * @param {number} at\n * @param {number} remove\n * @param {Array<Event>} add\n * @returns {undefined}\n */\nfunction addImplementation(editMap, at, remove, add) {\n let index = 0;\n\n /* c8 ignore next 3 -- `resolve` is never called without tables, so without edits. */\n if (remove === 0 && add.length === 0) {\n return;\n }\n while (index < editMap.map.length) {\n if (editMap.map[index][0] === at) {\n editMap.map[index][1] += remove;\n\n // To do: before not used by tables, use when moving to micromark.\n // if (before) {\n // add.push(...editMap.map[index][2])\n // editMap.map[index][2] = add\n // } else {\n editMap.map[index][2].push(...add);\n // }\n\n return;\n }\n index += 1;\n }\n editMap.map.push([at, remove, add]);\n}\n\n// /**\n// * Shift `previous` and `next` links according to `jumps`.\n// *\n// * This fixes links in case there are events removed or added between them.\n// *\n// * @param {Array<Event>} events\n// * @param {Array<Jump>} jumps\n// */\n// function shiftLinks(events, jumps) {\n// let jumpIndex = 0\n// let index = 0\n// let add = 0\n// let rm = 0\n\n// while (index < events.length) {\n// const rmCurr = rm\n\n// while (jumpIndex < jumps.length && jumps[jumpIndex][0] <= index) {\n// add = jumps[jumpIndex][2]\n// rm = jumps[jumpIndex][1]\n// jumpIndex += 1\n// }\n\n// // Ignore items that will be removed.\n// if (rm > rmCurr) {\n// index += rm - rmCurr\n// } else {\n// // ?\n// // if let Some(link) = &events[index].link {\n// // if let Some(next) = link.next {\n// // events[next].link.as_mut().unwrap().previous = Some(index + add - rm);\n// // while jumpIndex < jumps.len() && jumps[jumpIndex].0 <= next {\n// // add = jumps[jumpIndex].2;\n// // rm = jumps[jumpIndex].1;\n// // jumpIndex += 1;\n// // }\n// // events[index].link.as_mut().unwrap().next = Some(next + add - rm);\n// // index = next;\n// // continue;\n// // }\n// // }\n// index += 1\n// }\n// }\n// }","/**\n * @import {Event} from 'micromark-util-types'\n */\n\n/**\n * @typedef {'center' | 'left' | 'none' | 'right'} Align\n */\n\n/**\n * Figure out the alignment of a GFM table.\n *\n * @param {Readonly<Array<Event>>} events\n * List of events.\n * @param {number} index\n * Table enter event.\n * @returns {Array<Align>}\n * List of aligns.\n */\nexport function gfmTableAlign(events, index) {\n let inDelimiterRow = false;\n /** @type {Array<Align>} */\n const align = [];\n while (index < events.length) {\n const event = events[index];\n if (inDelimiterRow) {\n if (event[0] === 'enter') {\n // Start of alignment value: set a new column.\n // To do: `markdown-rs` uses `tableDelimiterCellValue`.\n if (event[1].type === 'tableContent') {\n align.push(events[index + 1][1].type === 'tableDelimiterMarker' ? 'left' : 'none');\n }\n }\n // Exits:\n // End of alignment value: change the column.\n // To do: `markdown-rs` uses `tableDelimiterCellValue`.\n else if (event[1].type === 'tableContent') {\n if (events[index - 1][1].type === 'tableDelimiterMarker') {\n const alignIndex = align.length - 1;\n align[alignIndex] = align[alignIndex] === 'left' ? 'center' : 'right';\n }\n }\n // Done!\n else if (event[1].type === 'tableDelimiterRow') {\n break;\n }\n } else if (event[0] === 'enter' && event[1].type === 'tableDelimiterRow') {\n inDelimiterRow = true;\n }\n index += 1;\n }\n return align;\n}","/**\n * @import {Event, Extension, Point, Resolver, State, Token, TokenizeContext, Tokenizer} from 'micromark-util-types'\n */\n\n/**\n * @typedef {[number, number, number, number]} Range\n * Cell info.\n *\n * @typedef {0 | 1 | 2 | 3} RowKind\n * Where we are: `1` for head row, `2` for delimiter row, `3` for body row.\n */\n\nimport { factorySpace } from 'micromark-factory-space';\nimport { markdownLineEnding, markdownLineEndingOrSpace, markdownSpace } from 'micromark-util-character';\nimport { EditMap } from './edit-map.js';\nimport { gfmTableAlign } from './infer.js';\n\n/**\n * Create an HTML extension for `micromark` to support GitHub tables syntax.\n *\n * @returns {Extension}\n * Extension for `micromark` that can be passed in `extensions` to enable GFM\n * table syntax.\n */\nexport function gfmTable() {\n return {\n flow: {\n null: {\n name: 'table',\n tokenize: tokenizeTable,\n resolveAll: resolveTable\n }\n }\n };\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeTable(effects, ok, nok) {\n const self = this;\n let size = 0;\n let sizeB = 0;\n /** @type {boolean | undefined} */\n let seen;\n return start;\n\n /**\n * Start of a GFM table.\n *\n * If there is a valid table row or table head before, then we try to parse\n * another row.\n * Otherwise, we try to parse a head.\n *\n * ```markdown\n * > | | a |\n * ^\n * | | - |\n * > | | b |\n * ^\n * ```\n * @type {State}\n */\n function start(code) {\n let index = self.events.length - 1;\n while (index > -1) {\n const type = self.events[index][1].type;\n if (type === \"lineEnding\" ||\n // Note: markdown-rs uses `whitespace` instead of `linePrefix`\n type === \"linePrefix\") index--;else break;\n }\n const tail = index > -1 ? self.events[index][1].type : null;\n const next = tail === 'tableHead' || tail === 'tableRow' ? bodyRowStart : headRowBefore;\n\n // Don’t allow lazy body rows.\n if (next === bodyRowStart && self.parser.lazy[self.now().line]) {\n return nok(code);\n }\n return next(code);\n }\n\n /**\n * Before table head row.\n *\n * ```markdown\n * > | | a |\n * ^\n * | | - |\n * | | b |\n * ```\n *\n * @type {State}\n */\n function headRowBefore(code) {\n effects.enter('tableHead');\n effects.enter('tableRow');\n return headRowStart(code);\n }\n\n /**\n * Before table head row, after whitespace.\n *\n * ```markdown\n * > | | a |\n * ^\n * | | - |\n * | | b |\n * ```\n *\n * @type {State}\n */\n function headRowStart(code) {\n if (code === 124) {\n return headRowBreak(code);\n }\n\n // To do: micromark-js should let us parse our own whitespace in extensions,\n // like `markdown-rs`:\n //\n // ```js\n // // 4+ spaces.\n // if (markdownSpace(code)) {\n // return nok(code)\n // }\n // ```\n\n seen = true;\n // Count the first character, that isn’t a pipe, double.\n sizeB += 1;\n return headRowBreak(code);\n }\n\n /**\n * At break in table head row.\n *\n * ```markdown\n * > | | a |\n * ^\n * ^\n * ^\n * | | - |\n * | | b |\n * ```\n *\n * @type {State}\n */\n function headRowBreak(code) {\n if (code === null) {\n // Note: in `markdown-rs`, we need to reset, in `micromark-js` we don‘t.\n return nok(code);\n }\n if (markdownLineEnding(code)) {\n // If anything other than one pipe (ignoring whitespace) was used, it’s fine.\n if (sizeB > 1) {\n sizeB = 0;\n // To do: check if this works.\n // Feel free to interrupt:\n self.interrupt = true;\n effects.exit('tableRow');\n effects.enter(\"lineEnding\");\n effects.consume(code);\n effects.exit(\"lineEnding\");\n return headDelimiterStart;\n }\n\n // Note: in `markdown-rs`, we need to reset, in `micromark-js` we don‘t.\n return nok(code);\n }\n if (markdownSpace(code)) {\n // To do: check if this is fine.\n // effects.attempt(State::Next(StateName::GfmTableHeadRowBreak), State::Nok)\n // State::Retry(space_or_tab(tokenizer))\n return factorySpace(effects, headRowBreak, \"whitespace\")(code);\n }\n sizeB += 1;\n if (seen) {\n seen = false;\n // Header cell count.\n size += 1;\n }\n if (code === 124) {\n effects.enter('tableCellDivider');\n effects.consume(code);\n effects.exit('tableCellDivider');\n // Whether a delimiter was seen.\n seen = true;\n return headRowBreak;\n }\n\n // Anything else is cell data.\n effects.enter(\"data\");\n return headRowData(code);\n }\n\n /**\n * In table head row data.\n *\n * ```markdown\n * > | | a |\n * ^\n * | | - |\n * | | b |\n * ```\n *\n * @type {State}\n */\n function headRowData(code) {\n if (code === null || code === 124 || markdownLineEndingOrSpace(code)) {\n effects.exit(\"data\");\n return headRowBreak(code);\n }\n effects.consume(code);\n return code === 92 ? headRowEscape : headRowData;\n }\n\n /**\n * In table head row escape.\n *\n * ```markdown\n * > | | a\\-b |\n * ^\n * | | ---- |\n * | | c |\n * ```\n *\n * @type {State}\n */\n function headRowEscape(code) {\n if (code === 92 || code === 124) {\n effects.consume(code);\n return headRowData;\n }\n return headRowData(code);\n }\n\n /**\n * Before delimiter row.\n *\n * ```markdown\n * | | a |\n * > | | - |\n * ^\n * | | b |\n * ```\n *\n * @type {State}\n */\n function headDelimiterStart(code) {\n // Reset `interrupt`.\n self.interrupt = false;\n\n // Note: in `markdown-rs`, we need to handle piercing here too.\n if (self.parser.lazy[self.now().line]) {\n return nok(code);\n }\n effects.enter('tableDelimiterRow');\n // Track if we’ve seen a `:` or `|`.\n seen = false;\n if (markdownSpace(code)) {\n return factorySpace(effects, headDelimiterBefore, \"linePrefix\", self.parser.constructs.disable.null.includes('codeIndented') ? undefined : 4)(code);\n }\n return headDelimiterBefore(code);\n }\n\n /**\n * Before delimiter row, after optional whitespace.\n *\n * Reused when a `|` is found later, to parse another cell.\n *\n * ```markdown\n * | | a |\n * > | | - |\n * ^\n * | | b |\n * ```\n *\n * @type {State}\n */\n function headDelimiterBefore(code) {\n if (code === 45 || code === 58) {\n return headDelimiterValueBefore(code);\n }\n if (code === 124) {\n seen = true;\n // If we start with a pipe, we open a cell marker.\n effects.enter('tableCellDivider');\n effects.consume(code);\n effects.exit('tableCellDivider');\n return headDelimiterCellBefore;\n }\n\n // More whitespace / empty row not allowed at start.\n return headDelimiterNok(code);\n }\n\n /**\n * After `|`, before delimiter cell.\n *\n * ```markdown\n * | | a |\n * > | | - |\n * ^\n * ```\n *\n * @type {State}\n */\n function headDelimiterCellBefore(code) {\n if (markdownSpace(code)) {\n return factorySpace(effects, headDelimiterValueBefore, \"whitespace\")(code);\n }\n return headDelimiterValueBefore(code);\n }\n\n /**\n * Before delimiter cell value.\n *\n * ```markdown\n * | | a |\n * > | | - |\n * ^\n * ```\n *\n * @type {State}\n */\n function headDelimiterValueBefore(code) {\n // Align: left.\n if (code === 58) {\n sizeB += 1;\n seen = true;\n effects.enter('tableDelimiterMarker');\n effects.consume(code);\n effects.exit('tableDelimiterMarker');\n return headDelimiterLeftAlignmentAfter;\n }\n\n // Align: none.\n if (code === 45) {\n sizeB += 1;\n // To do: seems weird that this *isn’t* left aligned, but that state is used?\n return headDelimiterLeftAlignmentAfter(code);\n }\n if (code === null || markdownLineEnding(code)) {\n return headDelimiterCellAfter(code);\n }\n return headDelimiterNok(code);\n }\n\n /**\n * After delimiter cell left alignment marker.\n *\n * ```markdown\n * | | a |\n * > | | :- |\n * ^\n * ```\n *\n * @type {State}\n */\n function headDelimiterLeftAlignmentAfter(code) {\n if (code === 45) {\n effects.enter('tableDelimiterFiller');\n return headDelimiterFiller(code);\n }\n\n // Anything else is not ok after the left-align colon.\n return headDelimiterNok(code);\n }\n\n /**\n * In delimiter cell filler.\n *\n * ```markdown\n * | | a |\n * > | | - |\n * ^\n * ```\n *\n * @type {State}\n */\n function headDelimiterFiller(code) {\n if (code === 45) {\n effects.consume(code);\n return headDelimiterFiller;\n }\n\n // Align is `center` if it was `left`, `right` otherwise.\n if (code === 58) {\n seen = true;\n effects.exit('tableDelimiterFiller');\n effects.enter('tableDelimiterMarker');\n effects.consume(code);\n effects.exit('tableDelimiterMarker');\n return headDelimiterRightAlignmentAfter;\n }\n effects.exit('tableDelimiterFiller');\n return headDelimiterRightAlignmentAfter(code);\n }\n\n /**\n * After delimiter cell right alignment marker.\n *\n * ```markdown\n * | | a |\n * > | | -: |\n * ^\n * ```\n *\n * @type {State}\n */\n function headDelimiterRightAlignmentAfter(code) {\n if (markdownSpace(code)) {\n return factorySpace(effects, headDelimiterCellAfter, \"whitespace\")(code);\n }\n return headDelimiterCellAfter(code);\n }\n\n /**\n * After delimiter cell.\n *\n * ```markdown\n * | | a |\n * > | | -: |\n * ^\n * ```\n *\n * @type {State}\n */\n function headDelimiterCellAfter(code) {\n if (code === 124) {\n return headDelimiterBefore(code);\n }\n if (code === null || markdownLineEnding(code)) {\n // Exit when:\n // * there was no `:` or `|` at all (it’s a thematic break or setext\n // underline instead)\n // * the header cell count is not the delimiter cell count\n if (!seen || size !== sizeB) {\n return headDelimiterNok(code);\n }\n\n // Note: in markdown-rs`, a reset is needed here.\n effects.exit('tableDelimiterRow');\n effects.exit('tableHead');\n // To do: in `markdown-rs`, resolvers need to be registered manually.\n // effects.register_resolver(ResolveName::GfmTable)\n return ok(code);\n }\n return headDelimiterNok(code);\n }\n\n /**\n * In delimiter row, at a disallowed byte.\n *\n * ```markdown\n * | | a |\n * > | | x |\n * ^\n * ```\n *\n * @type {State}\n */\n function headDelimiterNok(code) {\n // Note: in `markdown-rs`, we need to reset, in `micromark-js` we don‘t.\n return nok(code);\n }\n\n /**\n * Before table body row.\n *\n * ```markdown\n * | | a |\n * | | - |\n * > | | b |\n * ^\n * ```\n *\n * @type {State}\n */\n function bodyRowStart(code) {\n // Note: in `markdown-rs` we need to manually take care of a prefix,\n // but in `micromark-js` that is done for us, so if we’re here, we’re\n // never at whitespace.\n effects.enter('tableRow');\n return bodyRowBreak(code);\n }\n\n /**\n * At break in table body row.\n *\n * ```markdown\n * | | a |\n * | | - |\n * > | | b |\n * ^\n * ^\n * ^\n * ```\n *\n * @type {State}\n */\n function bodyRowBreak(code) {\n if (code === 124) {\n effects.enter('tableCellDivider');\n effects.consume(code);\n effects.exit('tableCellDivider');\n return bodyRowBreak;\n }\n if (code === null || markdownLineEnding(code)) {\n effects.exit('tableRow');\n return ok(code);\n }\n if (markdownSpace(code)) {\n return factorySpace(effects, bodyRowBreak, \"whitespace\")(code);\n }\n\n // Anything else is cell content.\n effects.enter(\"data\");\n return bodyRowData(code);\n }\n\n /**\n * In table body row data.\n *\n * ```markdown\n * | | a |\n * | | - |\n * > | | b |\n * ^\n * ```\n *\n * @type {State}\n */\n function bodyRowData(code) {\n if (code === null || code === 124 || markdownLineEndingOrSpace(code)) {\n effects.exit(\"data\");\n return bodyRowBreak(code);\n }\n effects.consume(code);\n return code === 92 ? bodyRowEscape : bodyRowData;\n }\n\n /**\n * In table body row escape.\n *\n * ```markdown\n * | | a |\n * | | ---- |\n * > | | b\\-c |\n * ^\n * ```\n *\n * @type {State}\n */\n function bodyRowEscape(code) {\n if (code === 92 || code === 124) {\n effects.consume(code);\n return bodyRowData;\n }\n return bodyRowData(code);\n }\n}\n\n/** @type {Resolver} */\n\nfunction resolveTable(events, context) {\n let index = -1;\n let inFirstCellAwaitingPipe = true;\n /** @type {RowKind} */\n let rowKind = 0;\n /** @type {Range} */\n let lastCell = [0, 0, 0, 0];\n /** @type {Range} */\n let cell = [0, 0, 0, 0];\n let afterHeadAwaitingFirstBodyRow = false;\n let lastTableEnd = 0;\n /** @type {Token | undefined} */\n let currentTable;\n /** @type {Token | undefined} */\n let currentBody;\n /** @type {Token | undefined} */\n let currentCell;\n const map = new EditMap();\n while (++index < events.length) {\n const event = events[index];\n const token = event[1];\n if (event[0] === 'enter') {\n // Start of head.\n if (token.type === 'tableHead') {\n afterHeadAwaitingFirstBodyRow = false;\n\n // Inject previous (body end and) table end.\n if (lastTableEnd !== 0) {\n flushTableEnd(map, context, lastTableEnd, currentTable, currentBody);\n currentBody = undefined;\n lastTableEnd = 0;\n }\n\n // Inject table start.\n currentTable = {\n type: 'table',\n start: Object.assign({}, token.start),\n // Note: correct end is set later.\n end: Object.assign({}, token.end)\n };\n map.add(index, 0, [['enter', currentTable, context]]);\n } else if (token.type === 'tableRow' || token.type === 'tableDelimiterRow') {\n inFirstCellAwaitingPipe = true;\n currentCell = undefined;\n lastCell = [0, 0, 0, 0];\n cell = [0, index + 1, 0, 0];\n\n // Inject table body start.\n if (afterHeadAwaitingFirstBodyRow) {\n afterHeadAwaitingFirstBodyRow = false;\n currentBody = {\n type: 'tableBody',\n start: Object.assign({}, token.start),\n // Note: correct end is set later.\n end: Object.assign({}, token.end)\n };\n map.add(index, 0, [['enter', currentBody, context]]);\n }\n rowKind = token.type === 'tableDelimiterRow' ? 2 : currentBody ? 3 : 1;\n }\n // Cell data.\n else if (rowKind && (token.type === \"data\" || token.type === 'tableDelimiterMarker' || token.type === 'tableDelimiterFiller')) {\n inFirstCellAwaitingPipe = false;\n\n // First value in cell.\n if (cell[2] === 0) {\n if (lastCell[1] !== 0) {\n cell[0] = cell[1];\n currentCell = flushCell(map, context, lastCell, rowKind, undefined, currentCell);\n lastCell = [0, 0, 0, 0];\n }\n cell[2] = index;\n }\n } else if (token.type === 'tableCellDivider') {\n if (inFirstCellAwaitingPipe) {\n inFirstCellAwaitingPipe = false;\n } else {\n if (lastCell[1] !== 0) {\n cell[0] = cell[1];\n currentCell = flushCell(map, context, lastCell, rowKind, undefined, currentCell);\n }\n lastCell = cell;\n cell = [lastCell[1], index, 0, 0];\n }\n }\n }\n // Exit events.\n else if (token.type === 'tableHead') {\n afterHeadAwaitingFirstBodyRow = true;\n lastTableEnd = index;\n } else if (token.type === 'tableRow' || token.type === 'tableDelimiterRow') {\n lastTableEnd = index;\n if (lastCell[1] !== 0) {\n cell[0] = cell[1];\n currentCell = flushCell(map, context, lastCell, rowKind, index, currentCell);\n } else if (cell[1] !== 0) {\n currentCell = flushCell(map, context, cell, rowKind, index, currentCell);\n }\n rowKind = 0;\n } else if (rowKind && (token.type === \"data\" || token.type === 'tableDelimiterMarker' || token.type === 'tableDelimiterFiller')) {\n cell[3] = index;\n }\n }\n if (lastTableEnd !== 0) {\n flushTableEnd(map, context, lastTableEnd, currentTable, currentBody);\n }\n map.consume(context.events);\n\n // To do: move this into `html`, when events are exposed there.\n // That’s what `markdown-rs` does.\n // That needs updates to `mdast-util-gfm-table`.\n index = -1;\n while (++index < context.events.length) {\n const event = context.events[index];\n if (event[0] === 'enter' && event[1].type === 'table') {\n event[1]._align = gfmTableAlign(context.events, index);\n }\n }\n return events;\n}\n\n/**\n * Generate a cell.\n *\n * @param {EditMap} map\n * @param {Readonly<TokenizeContext>} context\n * @param {Readonly<Range>} range\n * @param {RowKind} rowKind\n * @param {number | undefined} rowEnd\n * @param {Token | undefined} previousCell\n * @returns {Token | undefined}\n */\n// eslint-disable-next-line max-params\nfunction flushCell(map, context, range, rowKind, rowEnd, previousCell) {\n // `markdown-rs` uses:\n // rowKind === 2 ? 'tableDelimiterCell' : 'tableCell'\n const groupName = rowKind === 1 ? 'tableHeader' : rowKind === 2 ? 'tableDelimiter' : 'tableData';\n // `markdown-rs` uses:\n // rowKind === 2 ? 'tableDelimiterCellValue' : 'tableCellText'\n const valueName = 'tableContent';\n\n // Insert an exit for the previous cell, if there is one.\n //\n // ```markdown\n // > | | aa | bb | cc |\n // ^-- exit\n // ^^^^-- this cell\n // ```\n if (range[0] !== 0) {\n previousCell.end = Object.assign({}, getPoint(context.events, range[0]));\n map.add(range[0], 0, [['exit', previousCell, context]]);\n }\n\n // Insert enter of this cell.\n //\n // ```markdown\n // > | | aa | bb | cc |\n // ^-- enter\n // ^^^^-- this cell\n // ```\n const now = getPoint(context.events, range[1]);\n previousCell = {\n type: groupName,\n start: Object.assign({}, now),\n // Note: correct end is set later.\n end: Object.assign({}, now)\n };\n map.add(range[1], 0, [['enter', previousCell, context]]);\n\n // Insert text start at first data start and end at last data end, and\n // remove events between.\n //\n // ```markdown\n // > | | aa | bb | cc |\n // ^-- enter\n // ^-- exit\n // ^^^^-- this cell\n // ```\n if (range[2] !== 0) {\n const relatedStart = getPoint(context.events, range[2]);\n const relatedEnd = getPoint(context.events, range[3]);\n /** @type {Token} */\n const valueToken = {\n type: valueName,\n start: Object.assign({}, relatedStart),\n end: Object.assign({}, relatedEnd)\n };\n map.add(range[2], 0, [['enter', valueToken, context]]);\n if (rowKind !== 2) {\n // Fix positional info on remaining events\n const start = context.events[range[2]];\n const end = context.events[range[3]];\n start[1].end = Object.assign({}, end[1].end);\n start[1].type = \"chunkText\";\n start[1].contentType = \"text\";\n\n // Remove if needed.\n if (range[3] > range[2] + 1) {\n const a = range[2] + 1;\n const b = range[3] - range[2] - 1;\n map.add(a, b, []);\n }\n }\n map.add(range[3] + 1, 0, [['exit', valueToken, context]]);\n }\n\n // Insert an exit for the last cell, if at the row end.\n //\n // ```markdown\n // > | | aa | bb | cc |\n // ^-- exit\n // ^^^^^^-- this cell (the last one contains two “between” parts)\n // ```\n if (rowEnd !== undefined) {\n previousCell.end = Object.assign({}, getPoint(context.events, rowEnd));\n map.add(rowEnd, 0, [['exit', previousCell, context]]);\n previousCell = undefined;\n }\n return previousCell;\n}\n\n/**\n * Generate table end (and table body end).\n *\n * @param {Readonly<EditMap>} map\n * @param {Readonly<TokenizeContext>} context\n * @param {number} index\n * @param {Token} table\n * @param {Token | undefined} tableBody\n */\n// eslint-disable-next-line max-params\nfunction flushTableEnd(map, context, index, table, tableBody) {\n /** @type {Array<Event>} */\n const exits = [];\n const related = getPoint(context.events, index);\n if (tableBody) {\n tableBody.end = Object.assign({}, related);\n exits.push(['exit', tableBody, context]);\n }\n table.end = Object.assign({}, related);\n exits.push(['exit', table, context]);\n map.add(index + 1, 0, exits);\n}\n\n/**\n * @param {Readonly<Array<Event>>} events\n * @param {number} index\n * @returns {Readonly<Point>}\n */\nfunction getPoint(events, index) {\n const event = events[index];\n const side = event[0] === 'enter' ? 'start' : 'end';\n return event[1][side];\n}","/**\n * @import {Extension, State, TokenizeContext, Tokenizer} from 'micromark-util-types'\n */\n\nimport { factorySpace } from 'micromark-factory-space';\nimport { markdownLineEnding, markdownLineEndingOrSpace, markdownSpace } from 'micromark-util-character';\nconst tasklistCheck = {\n name: 'tasklistCheck',\n tokenize: tokenizeTasklistCheck\n};\n\n/**\n * Create an HTML extension for `micromark` to support GFM task list items\n * syntax.\n *\n * @returns {Extension}\n * Extension for `micromark` that can be passed in `htmlExtensions` to\n * support GFM task list items when serializing to HTML.\n */\nexport function gfmTaskListItem() {\n return {\n text: {\n [91]: tasklistCheck\n }\n };\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeTasklistCheck(effects, ok, nok) {\n const self = this;\n return open;\n\n /**\n * At start of task list item check.\n *\n * ```markdown\n * > | * [x] y.\n * ^\n * ```\n *\n * @type {State}\n */\n function open(code) {\n if (\n // Exit if there’s stuff before.\n self.previous !== null ||\n // Exit if not in the first content that is the first child of a list\n // item.\n !self._gfmTasklistFirstContentOfListItem) {\n return nok(code);\n }\n effects.enter('taskListCheck');\n effects.enter('taskListCheckMarker');\n effects.consume(code);\n effects.exit('taskListCheckMarker');\n return inside;\n }\n\n /**\n * In task list item check.\n *\n * ```markdown\n * > | * [x] y.\n * ^\n * ```\n *\n * @type {State}\n */\n function inside(code) {\n // Currently we match how GH works in files.\n // To match how GH works in comments, use `markdownSpace` (`[\\t ]`) instead\n // of `markdownLineEndingOrSpace` (`[\\t\\n\\r ]`).\n if (markdownLineEndingOrSpace(code)) {\n effects.enter('taskListCheckValueUnchecked');\n effects.consume(code);\n effects.exit('taskListCheckValueUnchecked');\n return close;\n }\n if (code === 88 || code === 120) {\n effects.enter('taskListCheckValueChecked');\n effects.consume(code);\n effects.exit('taskListCheckValueChecked');\n return close;\n }\n return nok(code);\n }\n\n /**\n * At close of task list item check.\n *\n * ```markdown\n * > | * [x] y.\n * ^\n * ```\n *\n * @type {State}\n */\n function close(code) {\n if (code === 93) {\n effects.enter('taskListCheckMarker');\n effects.consume(code);\n effects.exit('taskListCheckMarker');\n effects.exit('taskListCheck');\n return after;\n }\n return nok(code);\n }\n\n /**\n * @type {State}\n */\n function after(code) {\n // EOL in paragraph means there must be something else after it.\n if (markdownLineEnding(code)) {\n return ok(code);\n }\n\n // Space or tab?\n // Check what comes after.\n if (markdownSpace(code)) {\n return effects.check({\n tokenize: spaceThenNonSpace\n }, ok, nok)(code);\n }\n\n // EOF, or non-whitespace, both wrong.\n return nok(code);\n }\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction spaceThenNonSpace(effects, ok, nok) {\n return factorySpace(effects, after, \"whitespace\");\n\n /**\n * After whitespace, after task list item check.\n *\n * ```markdown\n * > | * [x] y.\n * ^\n * ```\n *\n * @type {State}\n */\n function after(code) {\n // EOF means there was nothing, so bad.\n // EOL means there’s content after it, so good.\n // Impossible to have more spaces.\n // Anything else is good.\n return code === null ? nok(code) : ok(code);\n }\n}","/**\n * @typedef {import('micromark-extension-gfm-footnote').HtmlOptions} HtmlOptions\n * @typedef {import('micromark-extension-gfm-strikethrough').Options} Options\n * @typedef {import('micromark-util-types').Extension} Extension\n * @typedef {import('micromark-util-types').HtmlExtension} HtmlExtension\n */\n\nimport {\n combineExtensions,\n combineHtmlExtensions\n} from 'micromark-util-combine-extensions'\nimport {\n gfmAutolinkLiteral,\n gfmAutolinkLiteralHtml\n} from 'micromark-extension-gfm-autolink-literal'\nimport {gfmFootnote, gfmFootnoteHtml} from 'micromark-extension-gfm-footnote'\nimport {\n gfmStrikethrough,\n gfmStrikethroughHtml\n} from 'micromark-extension-gfm-strikethrough'\nimport {gfmTable, gfmTableHtml} from 'micromark-extension-gfm-table'\nimport {gfmTagfilterHtml} from 'micromark-extension-gfm-tagfilter'\nimport {\n gfmTaskListItem,\n gfmTaskListItemHtml\n} from 'micromark-extension-gfm-task-list-item'\n\n/**\n * Create an extension for `micromark` to enable GFM syntax.\n *\n * @param {Options | null | undefined} [options]\n * Configuration (optional).\n *\n * Passed to `micromark-extens-gfm-strikethrough`.\n * @returns {Extension}\n * Extension for `micromark` that can be passed in `extensions` to enable GFM\n * syntax.\n */\nexport function gfm(options) {\n return combineExtensions([\n gfmAutolinkLiteral(),\n gfmFootnote(),\n gfmStrikethrough(options),\n gfmTable(),\n gfmTaskListItem()\n ])\n}\n\n/**\n * Create an extension for `micromark` to support GFM when serializing to HTML.\n *\n * @param {HtmlOptions | null | undefined} [options]\n * Configuration (optional).\n *\n * Passed to `micromark-extens-gfm-footnote`.\n * @returns {HtmlExtension}\n * Extension for `micromark` that can be passed in `htmlExtensions` to\n * support GFM when serializing to HTML.\n */\nexport function gfmHtml(options) {\n return combineHtmlExtensions([\n gfmAutolinkLiteralHtml(),\n gfmFootnoteHtml(options),\n gfmStrikethroughHtml(),\n gfmTableHtml(),\n gfmTagfilterHtml(),\n gfmTaskListItemHtml()\n ])\n}\n","/**\n * @import {Root} from 'mdast'\n * @import {Options} from 'remark-gfm'\n * @import {} from 'remark-parse'\n * @import {} from 'remark-stringify'\n * @import {Processor} from 'unified'\n */\n\nimport {gfmFromMarkdown, gfmToMarkdown} from 'mdast-util-gfm'\nimport {gfm} from 'micromark-extension-gfm'\n\n/** @type {Options} */\nconst emptyOptions = {}\n\n/**\n * Add support GFM (autolink literals, footnotes, strikethrough, tables,\n * tasklists).\n *\n * @param {Options | null | undefined} [options]\n * Configuration (optional).\n * @returns {undefined}\n * Nothing.\n */\nexport default function remarkGfm(options) {\n // @ts-expect-error: TS is wrong about `this`.\n // eslint-disable-next-line unicorn/no-this-assignment\n const self = /** @type {Processor<Root>} */ (this)\n const settings = options || emptyOptions\n const data = self.data()\n\n const micromarkExtensions =\n data.micromarkExtensions || (data.micromarkExtensions = [])\n const fromMarkdownExtensions =\n data.fromMarkdownExtensions || (data.fromMarkdownExtensions = [])\n const toMarkdownExtensions =\n data.toMarkdownExtensions || (data.toMarkdownExtensions = [])\n\n micromarkExtensions.push(gfm(settings))\n fromMarkdownExtensions.push(gfmFromMarkdown())\n toMarkdownExtensions.push(gfmToMarkdown(settings))\n}\n","export function cn(...classes: Array<string | false | null | undefined>): string {\n return classes.filter(Boolean).join(' ');\n}\n\n","'use client';\n\nimport '@assistant-ui/react-markdown/styles/dot.css';\n\nimport {\n type CodeHeaderProps,\n MarkdownTextPrimitive,\n unstable_memoizeMarkdownComponents as memoizeMarkdownComponents,\n useIsMarkdownCodeBlock,\n} from '@assistant-ui/react-markdown';\nimport remarkGfm from 'remark-gfm';\nimport { type FC, memo, useState } from 'react';\nimport { CheckIcon, CopyIcon } from 'lucide-react';\nimport { cn } from '../utils/cn';\n\ninterface MarkdownTextProps {\n content: string;\n}\n\nconst MarkdownTextImpl: FC<MarkdownTextProps> = ({ content }) => {\n return (\n <MarkdownTextPrimitive\n remarkPlugins={[remarkGfm]}\n className=\"cuadra-aui-md\"\n components={defaultComponents}\n {...({ children: content } as { children: string })}\n />\n );\n};\n\nexport const MarkdownText = memo(MarkdownTextImpl);\n\nconst CodeHeader: FC<CodeHeaderProps> = ({ language, code }) => {\n const { isCopied, copyToClipboard } = useCopyToClipboard();\n const onCopy = () => {\n if (!code || isCopied) return;\n copyToClipboard(code);\n };\n\n return (\n <div className=\"cuadra-flex cuadra-items-center cuadra-justify-between cuadra-gap-4 cuadra-mt-4 cuadra-rounded-t-lg cuadra-bg-muted/50 cuadra-border-b cuadra-border-border cuadra-px-4 cuadra-py-2 cuadra-text-sm cuadra-font-normal cuadra-text-foreground cuadra-font-brand\">\n <span className=\"cuadra-lowercase [&>span]:cuadra-text-xs cuadra-font-brand\">{language}</span>\n <button\n onClick={onCopy}\n className=\"cuadra-p-1 cuadra-rounded hover:cuadra-bg-muted cuadra-transition-colors\"\n aria-label=\"Copy code\"\n >\n {!isCopied && <CopyIcon className=\"cuadra-h-4 cuadra-w-4\" />}\n {isCopied && <CheckIcon className=\"cuadra-h-4 cuadra-w-4\" />}\n </button>\n </div>\n );\n};\n\nconst useCopyToClipboard = ({\n copiedDuration = 3000,\n}: {\n copiedDuration?: number;\n} = {}) => {\n const [isCopied, setIsCopied] = useState<boolean>(false);\n\n const copyToClipboard = (value: string) => {\n if (!value) return;\n\n navigator.clipboard.writeText(value).then(() => {\n setIsCopied(true);\n setTimeout(() => setIsCopied(false), copiedDuration);\n });\n };\n\n return { isCopied, copyToClipboard };\n};\n\nconst defaultComponents = memoizeMarkdownComponents({\n h1: ({ className, ...props }) => (\n <h1\n className={cn('cuadra-mb-6 cuadra-scroll-m-20 cuadra-text-2xl cuadra-font-normal cuadra-tracking-tight last:cuadra-mb-0 font-brand', className)}\n {...props}\n />\n ),\n h2: ({ className, ...props }) => (\n <h2\n className={cn(\n 'cuadra-mb-4 cuadra-mt-6 cuadra-scroll-m-20 cuadra-text-xl cuadra-font-normal cuadra-tracking-tight first:cuadra-mt-0 last:cuadra-mb-0 font-brand',\n className,\n )}\n {...props}\n />\n ),\n h3: ({ className, ...props }) => (\n <h3\n className={cn(\n 'cuadra-mb-3 cuadra-mt-5 cuadra-scroll-m-20 cuadra-text-lg cuadra-font-normal cuadra-tracking-tight first:cuadra-mt-0 last:cuadra-mb-0 font-brand',\n className,\n )}\n {...props}\n />\n ),\n h4: ({ className, ...props }) => (\n <h4\n className={cn(\n 'cuadra-mb-4 cuadra-mt-6 cuadra-scroll-m-20 cuadra-text-xl cuadra-font-normal cuadra-tracking-tight first:cuadra-mt-0 last:cuadra-mb-0 font-brand',\n className,\n )}\n {...props}\n />\n ),\n h5: ({ className, ...props }) => (\n <h5 className={cn('cuadra-my-4 cuadra-text-lg cuadra-font-normal first:cuadra-mt-0 last:cuadra-mb-0 font-brand', className)} {...props} />\n ),\n h6: ({ className, ...props }) => (\n <h6 className={cn('cuadra-my-4 cuadra-font-normal first:cuadra-mb-0 font-brand', className)} {...props} />\n ),\n p: ({ className, ...props }) => (\n <p className={cn('cuadra-mb-5 cuadra-mt-5 cuadra-leading-7 first:cuadra-mt-0 last:cuadra-mb-0 font-brand', className)} {...props} />\n ),\n a: ({ className, ...props }) => (\n <a\n className={cn('cuadra-text-primary cuadra-font-medium cuadra-underline cuadra-underline-offset-4 font-brand', className)}\n {...props}\n />\n ),\n blockquote: ({ className, ...props }) => (\n <blockquote className={cn('cuadra-border-l-2 cuadra-pl-6 cuadra-italic font-brand', className)} {...props} />\n ),\n ul: ({ className, ...props }) => (\n <ul className={cn('cuadra-my-5 cuadra-ml-6 cuadra-list-disc [&>li]:cuadra-mt-2 font-brand', className)} {...props} />\n ),\n ol: ({ className, ...props }) => (\n <ol className={cn('cuadra-my-5 cuadra-ml-6 cuadra-list-decimal [&>li]:cuadra-mt-2 font-brand', className)} {...props} />\n ),\n hr: ({ className, ...props }) => <hr className={cn('cuadra-my-5 cuadra-border-b', className)} {...props} />,\n table: ({ className, ...props }) => (\n <table\n className={cn('cuadra-my-5 cuadra-w-full cuadra-border-separate cuadra-border-spacing-0 cuadra-overflow-y-auto font-brand', className)}\n {...props}\n />\n ),\n th: ({ className, ...props }) => (\n <th\n className={cn(\n 'cuadra-bg-muted cuadra-px-4 cuadra-py-2 cuadra-text-left cuadra-font-normal first:cuadra-rounded-tl-lg last:cuadra-rounded-tr-lg [&[align=center]]:cuadra-text-center [&[align=right]]:cuadra-text-right font-brand',\n className,\n )}\n {...props}\n />\n ),\n td: ({ className, ...props }) => (\n <td\n className={cn(\n 'cuadra-border-b cuadra-border-l cuadra-px-4 cuadra-py-2 cuadra-text-left last:cuadra-border-r [&[align=center]]:cuadra-text-center [&[align=right]]:cuadra-text-right font-brand',\n className,\n )}\n {...props}\n />\n ),\n tr: ({ className, ...props }) => (\n <tr\n className={cn(\n 'cuadra-m-0 cuadra-border-b cuadra-p-0 first:cuadra-border-t [&:last-child>td:first-child]:cuadra-rounded-bl-lg [&:last-child>td:last-child]:cuadra-rounded-br-lg',\n className,\n )}\n {...props}\n />\n ),\n sup: ({ className, ...props }) => (\n <sup className={cn('[&>a]:cuadra-text-xs [&>a]:cuadra-no-underline font-brand', className)} {...props} />\n ),\n pre: ({ className, ...props }) => (\n <pre\n className={cn(\n 'cuadra-overflow-x-auto cuadra-whitespace-pre-wrap cuadra-break-words cuadra-rounded-b-lg !cuadra-rounded-t-none cuadra-bg-muted cuadra-p-4 cuadra-text-foreground font-brand',\n className,\n )}\n {...props}\n />\n ),\n code: function Code({ className, ...props }) {\n const isCodeBlock = useIsMarkdownCodeBlock();\n return (\n <code\n className={cn(!isCodeBlock && 'cuadra-bg-muted cuadra-rounded cuadra-border cuadra-font-normal font-brand', className)}\n {...props}\n />\n );\n },\n CodeHeader,\n});\n","import * as React from \"react\";\n\nexport interface BadgeProps extends React.HTMLAttributes<HTMLDivElement> {\n variant?: \"default\" | \"subtle\" | \"outline\" | \"ghost\";\n}\n\n/**\n * Badge component - simple, subtle styling\n */\nfunction Badge({ className = \"\", variant = \"default\", ...props }: BadgeProps) {\n const baseClasses = \n \"cuadra-inline-flex cuadra-items-center cuadra-rounded-md cuadra-px-3 cuadra-py-1.5 cuadra-text-xs cuadra-font-medium font-brand\";\n\n const variantClasses: Record<string, string> = {\n default:\n \"cuadra-border cuadra-border-border cuadra-bg-muted cuadra-text-muted-foreground\",\n subtle:\n \"cuadra-border cuadra-border-border/50 cuadra-bg-muted/50 cuadra-text-muted-foreground\",\n outline: \n \"cuadra-border cuadra-border-border cuadra-bg-background cuadra-text-foreground\",\n ghost:\n \"cuadra-text-muted-foreground\",\n };\n\n return (\n <div\n className={`${baseClasses} ${variantClasses[variant] || variantClasses.default} ${className}`}\n {...props}\n />\n );\n}\n\nexport { Badge };\n\n","import { FileSpreadsheet, FileText, FileType } from 'lucide-react';\nimport type { Source } from '../types/cuadra';\nimport { Badge } from './ui/badge';\n\nexport interface SourceCitationsProps {\n /** Array of source citations */\n sources: Source[];\n /** Maximum number of sources to show before collapsing */\n maxVisible?: number;\n}\n\n/**\n * Deduplicate sources by filename, keeping the highest score for each file\n */\nfunction deduplicateSources(sources: Source[]): Source[] {\n const byFilename = new Map<string, Source>();\n \n for (const source of sources) {\n const existing = byFilename.get(source.filename);\n if (!existing || source.score > existing.score) {\n byFilename.set(source.filename, source);\n }\n }\n \n // Sort by score descending\n return Array.from(byFilename.values()).sort((a, b) => b.score - a.score);\n}\n\n/**\n * Component to display RAG source citations\n * Shows document references that were used to generate the response\n * Deduplicates by filename, showing highest relevance score per file\n */\nexport function SourceCitations({ \n sources, \n maxVisible = 3,\n}: SourceCitationsProps) {\n if (!sources || sources.length === 0) return null;\n\n // Deduplicate by filename\n const uniqueSources = deduplicateSources(sources);\n const visibleSources = uniqueSources.slice(0, maxVisible);\n const hiddenCount = uniqueSources.length - maxVisible;\n\n return (\n <div className=\"cuadra-mt-4 cuadra-pt-3 cuadra-border-t cuadra-border-border\">\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-1.5 cuadra-mb-2.5\">\n <FileText className=\"cuadra-h-3.5 cuadra-w-3.5 cuadra-text-muted-foreground\" />\n <span className=\"cuadra-text-xs cuadra-font-medium cuadra-text-muted-foreground font-brand\">\n Sources\n </span>\n </div>\n \n <div className=\"cuadra-flex cuadra-flex-wrap cuadra-gap-2\">\n {visibleSources.map((source, index) => (\n <SourceBadge key={source.sourceId || index} source={source} />\n ))}\n {hiddenCount > 0 && (\n <Badge variant=\"subtle\">\n +{hiddenCount} more\n </Badge>\n )}\n </div>\n </div>\n );\n}\n\ninterface SourceBadgeProps {\n source: Source;\n}\n\n/**\n * Get icon based on file extension\n */\nfunction getFileIcon(filename: string): typeof FileText {\n const ext = filename.split('.').pop()?.toLowerCase();\n \n switch (ext) {\n case 'csv':\n case 'xlsx':\n case 'xls':\n return FileSpreadsheet;\n case 'md':\n case 'txt':\n return FileType;\n default:\n return FileText;\n }\n}\n\n/**\n * Individual source badge showing filename and optional relevance score\n */\nfunction SourceBadge({ source }: SourceBadgeProps) {\n const { filename, score } = source;\n const Icon = getFileIcon(filename);\n\n return (\n <Badge \n className=\"cuadra-gap-2 cuadra-cursor-default\"\n title={score < 1 ? `${Math.round(score * 100)}% relevant` : undefined}\n >\n <Icon className=\"cuadra-h-3.5 cuadra-w-3.5 cuadra-shrink-0 cuadra-opacity-60\" />\n <span className=\"cuadra-truncate cuadra-max-w-[180px]\">\n {filename}\n </span>\n {score < 1 && score > 0 && (\n <span className=\"cuadra-opacity-50 cuadra-text-[10px] cuadra-tabular-nums\">\n {Math.round(score * 100)}%\n </span>\n )}\n </Badge>\n );\n}\n\n","import { AttachmentPrimitive, ComposerPrimitive, MessagePrimitive, ThreadListItemPrimitive, ThreadListPrimitive, ThreadPrimitive, useAssistantRuntime, useAui, useAuiState, useThread, useThreadListItem, useThreadListItemRuntime, useThreadRuntime } from '@assistant-ui/react';\nimport { AlertCircle, ArrowUpIcon, Brain, Check, ChevronDown, EllipsisVertical, Eye, Loader2, MessageSquare, Paperclip, Pencil, Plus, Trash2, Upload, X } from 'lucide-react';\nimport { forwardRef, useCallback, useEffect, useImperativeHandle, useRef, useState, useSyncExternalStore } from 'react';\nimport { MarkdownText } from './MarkdownText';\nimport { SourceCitations } from './SourceCitations';\n// ThemeToggle import removed — theme is now controlled via the `theme` prop only\nimport { queuePreMadeResponse, streamingMetadataStore } from '../adapters/chatModelAdapter';\nimport { attachmentErrorStore } from '../adapters/attachmentErrorStore';\nimport '@assistant-ui/react-markdown/styles/dot.css';\n\n/**\n * Handle for controlling SimpleThread externally\n */\nexport interface SimpleThreadHandle {\n /** Send a user message programmatically (triggers normal API call) */\n sendMessage: (message: string) => void;\n /** Send a pre-made Q&A pair (instant, simulated streaming, no API call) */\n sendPreMadeQA: (question: string, answer: string, options?: { delay?: number; streamingSpeed?: number }) => void;\n /** Clear the current chat and start fresh */\n clearChat: () => void;\n}\n\n/** Suggestion with optional pre-made response */\nexport interface Suggestion {\n /** Suggestion prompt text */\n prompt: string;\n /** Pre-made response (optional) - if provided, response appears instantly with simulated streaming */\n response?: string;\n /** Delay before showing response in ms (overrides global preMadeResponseDelay) */\n responseDelay?: number;\n}\n\n/** Model option for the composer model selector */\nexport interface ModelOption {\n /** Unique model identifier */\n id: string;\n /** Display name shown in selector (e.g. \"GPT-4o\", \"Claude 3.5\") */\n name: string;\n /** Optional short description */\n description?: string;\n /** Whether the model supports vision/image inputs */\n supportsVision?: boolean;\n /** Whether the model supports reasoning/thinking */\n supportsReasoning?: boolean;\n}\n\nexport interface SimpleThreadProps {\n /** Welcome screen title */\n welcomeTitle?: string;\n /** Welcome screen subtitle */\n welcomeSubtitle?: string;\n /** Extra top padding for thread viewport (e.g., '1rem', '2rem') */\n extraTopPadding?: string;\n /** Suggestions to show in welcome screen */\n suggestions?: Suggestion[];\n /** Placeholder text for the input field */\n inputPlaceholder?: string;\n /** Enable file attachments in the chat input */\n enableAttachments?: boolean;\n /** Default delay before pre-made responses in ms (default: 1000) */\n preMadeResponseDelay?: number;\n /** Speed of simulated streaming in ms per word (default: 50) */\n streamingSpeed?: number;\n /** Apply safe area insets for mobile devices */\n safeArea?: boolean;\n\n // --- Claude-like features ---\n\n /** Show a visible border on the composer (default: false — uses shadow only) */\n composerBorder?: boolean;\n /** Available models for the composer model selector */\n models?: ModelOption[];\n /** Currently selected model ID */\n selectedModelId?: string;\n /** Callback when user selects a different model */\n onModelChange?: (modelId: string) => void;\n /** Show the chat title header at the top of the thread area */\n showChatHeader?: boolean;\n /** Show the theme toggle in the chat header */\n showThemeToggle?: boolean;\n /** Initial theme preference */\n theme?: 'light' | 'dark' | 'system';\n}\n\n/**\n * Simple Thread component wrapper\n * Provides a basic chat interface using assistant-ui primitives\n */\nexport const SimpleThread = forwardRef<SimpleThreadHandle, SimpleThreadProps>(\n function SimpleThread({\n welcomeTitle = 'What should we work on?',\n welcomeSubtitle,\n extraTopPadding,\n suggestions,\n inputPlaceholder = 'Reply...',\n enableAttachments = false,\n preMadeResponseDelay = 1000,\n streamingSpeed = 50,\n safeArea = false,\n composerBorder = false,\n models,\n selectedModelId,\n onModelChange,\n showChatHeader = false,\n // theme and showThemeToggle kept in interface for consumer API but not used internally\n }, ref) {\n const thread = useThreadRuntime();\n const aui = useAui();\n const [isLoadingMessages, setIsLoadingMessages] = useState(false);\n const [currentThreadId, setCurrentThreadId] = useState<string | null>(null);\n const [isProcessingPreMade, _setIsProcessingPreMade] = useState(false);\n\n // Subscribe to attachment validation errors (auto-dismiss after 4s)\n const attachmentError = useSyncExternalStore(\n attachmentErrorStore.subscribe,\n attachmentErrorStore.getSnapshot,\n );\n\n useEffect(() => {\n if (attachmentError) {\n const timer = setTimeout(() => attachmentErrorStore.clear(), 4000);\n return () => clearTimeout(timer);\n }\n }, [attachmentError]);\n\n // Track whether thread is empty for centered welcome layout\n const threadIsEmpty = useThread((state) => {\n const messages = (state as unknown as { messages?: readonly unknown[] }).messages;\n return !messages || messages.length === 0;\n });\n const showCenteredWelcome = threadIsEmpty && !isLoadingMessages;\n\n // Switch to a new thread when the model changes (multiChat mode).\n // This runs inside the AssistantRuntimeProvider so it has direct\n // hook access to the runtime — no fragile window globals needed.\n // We use a ref for the runtime + a short delay so the switch happens\n // after all cascading state updates from the model change have settled\n // (CuadraRuntimeProvider syncs selectedModelId via useEffect → extra render).\n const assistantRuntime = useAssistantRuntime();\n const assistantRuntimeRef = useRef(assistantRuntime);\n assistantRuntimeRef.current = assistantRuntime;\n const prevModelIdRef = useRef(selectedModelId);\n useEffect(() => {\n const prev = prevModelIdRef.current;\n prevModelIdRef.current = selectedModelId;\n\n if (prev && selectedModelId && prev !== selectedModelId) {\n const timer = setTimeout(() => {\n try {\n void assistantRuntimeRef.current.switchToNewThread();\n } catch (_error) {\n // Silently handle — user can manually create a new thread\n }\n }, 50);\n return () => clearTimeout(timer);\n }\n }, [selectedModelId]); // only depends on selectedModelId — not assistantRuntime\n\n /**\n * Send a pre-made Q&A with simulated streaming\n * Queues the response and triggers normal send flow - adapter intercepts and returns fake response\n */\n const sendPreMadeQAInternal = useCallback((\n question: string, \n answer: string, \n options?: { delay?: number; streamingSpeed?: number }\n ) => {\n if (isProcessingPreMade) return;\n \n try {\n // Queue the pre-made response - the adapter will intercept and return it\n // Also stores the Q&A for context in follow-up API calls\n const speed = options?.streamingSpeed ?? streamingSpeed;\n const delay = options?.delay ?? preMadeResponseDelay;\n queuePreMadeResponse(question, answer, speed, delay);\n \n // Trigger normal send flow with the question\n // The adapter will see the queued response and return it instead of calling API\n aui.composer().setText(question);\n setTimeout(() => {\n aui.composer().send();\n }, 10);\n } catch (_err) {\n // Pre-made QA error - silently ignore\n }\n }, [isProcessingPreMade, streamingSpeed, preMadeResponseDelay, aui]);\n\n /**\n * Send a message normally (triggers API call)\n */\n const sendMessageInternal = useCallback((message: string) => {\n try {\n aui.composer().setText(message);\n setTimeout(() => {\n aui.composer().send();\n }, 10);\n } catch {\n // Silently handle\n }\n }, [aui]);\n\n /**\n * Handle suggestion click - lifted to parent so it doesn't unmount mid-execution\n */\n const handleSuggestionClick = useCallback(async (suggestion: Suggestion) => {\n if (isProcessingPreMade || !thread) return;\n\n try {\n if (suggestion.response) {\n // Pre-made response mode\n const delay = suggestion.responseDelay ?? preMadeResponseDelay;\n await sendPreMadeQAInternal(suggestion.prompt, suggestion.response, { \n delay, \n streamingSpeed \n });\n } else {\n // Normal mode: use composer API\n sendMessageInternal(suggestion.prompt);\n }\n } catch {\n // Silently handle\n }\n }, [thread, isProcessingPreMade, preMadeResponseDelay, streamingSpeed, sendPreMadeQAInternal, sendMessageInternal]);\n\n /**\n * Clear the chat\n */\n const clearChatInternal = useCallback(async () => {\n try {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const runtime = (window as any).__cuadraThreadListRuntime;\n const threadListRuntime = runtime?.threadList;\n\n if (threadListRuntime?.switchToNewThread) {\n await threadListRuntime.switchToNewThread();\n }\n } catch {\n // Silently handle\n }\n }, []);\n\n // Expose methods via ref\n useImperativeHandle(ref, () => ({\n sendMessage: sendMessageInternal,\n sendPreMadeQA: sendPreMadeQAInternal,\n clearChat: clearChatInternal,\n }), [sendMessageInternal, sendPreMadeQAInternal, clearChatInternal]);\n\n // Track thread changes and loading state by subscribing to store changes\n useEffect(() => {\n if (!thread) {\n setIsLoadingMessages(false);\n return;\n }\n\n try {\n // Access runtime store to track loading state\n interface RuntimeStore {\n getState?: () => {\n threadId?: string;\n messages?: unknown[];\n isLoadingHistory?: boolean;\n isRunning?: boolean;\n };\n subscribe?: (callback: (state: { threadId?: string; messages?: unknown[]; isLoadingHistory?: boolean; isRunning?: boolean }) => void) => () => void;\n }\n const runtimeStore = (thread as { store?: RuntimeStore })?.store;\n if (!runtimeStore) {\n setIsLoadingMessages(false);\n return;\n }\n\n // Subscribe to store changes\n const unsubscribe = runtimeStore.subscribe?.((state) => {\n const threadId = state?.threadId;\n const messages = (state?.messages || []) as Array<{\n role?: string;\n content?: Array<{ type?: string; text?: string }> | string;\n }>;\n const isLoadingHistory = state?.isLoadingHistory || false;\n const messageCount = messages.length;\n\n // Check if thread changed\n if (threadId && threadId !== currentThreadId) {\n setCurrentThreadId(threadId);\n }\n\n // Only show loading if:\n // 1. We're actively loading history\n // 2. We have no messages yet\n const shouldShowLoading = isLoadingHistory && messageCount === 0;\n setIsLoadingMessages(shouldShowLoading);\n });\n\n // Initial check\n const state = runtimeStore.getState?.();\n if (state) {\n const threadId = state?.threadId;\n const messages = (state?.messages || []) as Array<{\n role?: string;\n content?: Array<{ type?: string; text?: string }> | string;\n }>;\n const isLoadingHistory = state?.isLoadingHistory || false;\n const messageCount = messages.length;\n\n if (threadId && threadId !== currentThreadId) {\n setCurrentThreadId(threadId);\n }\n\n const shouldShowLoading = isLoadingHistory && messageCount === 0;\n setIsLoadingMessages(shouldShowLoading);\n }\n\n return () => {\n if (unsubscribe) {\n unsubscribe();\n }\n };\n } catch {\n setIsLoadingMessages(false);\n }\n }, [thread, currentThreadId]);\n\n return (\n <ThreadPrimitive.Root className=\"cuadra-bg-inherit cuadra-flex cuadra-flex-col cuadra-w-full cuadra-h-full cuadra-relative\">\n {/* Top bar — centered model selector + optional thread selector */}\n {(showChatHeader || (models && models.length > 0)) && (\n <div className=\"cuadra-relative cuadra-flex cuadra-items-center cuadra-justify-center cuadra-px-3 cuadra-py-2 cuadra-min-h-[48px]\">\n {/* Thread selector: right on mobile, left on desktop (absolute so model stays centered) */}\n {showChatHeader && (\n <div className=\"cuadra-absolute cuadra-right-3 md:cuadra-right-auto md:cuadra-left-3 cuadra-top-0 cuadra-bottom-0 cuadra-flex cuadra-items-center\">\n <ChatHeader />\n </div>\n )}\n {/* Center: Model selector */}\n {models && models.length > 0 && (\n <ModelSelector\n models={models}\n selectedModelId={selectedModelId}\n onModelChange={onModelChange}\n />\n )}\n </div>\n )}\n {/* Drag-and-drop + content wrapper */}\n <ComposerPrimitive.AttachmentDropzone\n disabled={!enableAttachments}\n className={`cuadra-flex-1 cuadra-flex cuadra-flex-col cuadra-min-h-0 cuadra-relative ${showCenteredWelcome ? 'cuadra-justify-center' : ''}`}\n >\n {/* Drag overlay */}\n <div className=\"cuadra-dropzone-overlay cuadra-pointer-events-none cuadra-absolute cuadra-inset-0 cuadra-z-50 cuadra-flex cuadra-items-center cuadra-justify-center cuadra-rounded-lg cuadra-transition-opacity cuadra-duration-200\" style={{ opacity: 0 }}>\n <div className=\"cuadra-flex cuadra-flex-col cuadra-items-center cuadra-gap-3\">\n <div className=\"cuadra-flex cuadra-items-center cuadra-justify-center cuadra-size-14 cuadra-rounded-2xl\" style={{ backgroundColor: 'var(--cuadra-primary)', color: 'white' }}>\n <Upload className=\"cuadra-h-6 cuadra-w-6\" />\n </div>\n <p className=\"cuadra-text-sm cuadra-font-medium cuadra-text-foreground font-brand\">Drop files here</p>\n </div>\n </div>\n\n <ThreadPrimitive.Viewport \n className={`cuadra-w-full cuadra-bg-inherit cuadra-relative scrollbar-thin ${\n showCenteredWelcome \n ? 'cuadra-flex-none' \n : 'cuadra-flex-1 cuadra-min-h-0 cuadra-overflow-y-auto'\n }`}\n style={{ paddingTop: showCenteredWelcome ? '0' : (extraTopPadding || ((showChatHeader || (models && models.length > 0)) ? '0.5rem' : '2rem')) }}\n >\n {isLoadingMessages ? (\n <div className=\"cuadra-absolute cuadra-inset-0 cuadra-flex cuadra-items-center cuadra-justify-center\">\n <div className=\"cuadra-flex cuadra-flex-col cuadra-items-center cuadra-gap-3\">\n <Loader2 className=\"cuadra-h-5 cuadra-w-5 cuadra-animate-spin cuadra-text-muted-foreground\" />\n <p className=\"cuadra-text-sm cuadra-text-muted-foreground font-brand\">Loading messages...</p>\n </div>\n </div>\n ) : (\n <div className=\"cuadra-max-w-3xl cuadra-mx-auto cuadra-w-full cuadra-px-4\">\n <ThreadPrimitive.Empty>\n <WelcomeScreen title={welcomeTitle} subtitle={welcomeSubtitle} />\n </ThreadPrimitive.Empty>\n <ThreadPrimitive.Messages\n components={{\n UserMessage: UserMessage,\n AssistantMessage: AssistantMessage,\n }}\n />\n </div>\n )}\n </ThreadPrimitive.Viewport>\n\n {/* Composer area — Claude-like elevated surface with toolbar */}\n <div \n className={`cuadra-w-full cuadra-px-4 cuadra-pt-2 ${showCenteredWelcome ? 'cuadra-pb-2' : 'cuadra-pb-4'}`}\n style={safeArea ? { paddingBottom: 'calc(env(safe-area-inset-bottom) + 1rem)' } : undefined}\n >\n <div className=\"cuadra-max-w-3xl cuadra-mx-auto cuadra-w-full\">\n {/* Attachment validation error toast */}\n {attachmentError && (\n <div\n className=\"cuadra-flex cuadra-items-center cuadra-gap-2 cuadra-mb-2 cuadra-px-3 cuadra-py-2 cuadra-rounded-lg cuadra-text-xs font-brand cuadra-animate-in cuadra-fade-in cuadra-slide-in-from-bottom-2\"\n style={{\n backgroundColor: 'var(--cuadra-muted)',\n color: 'var(--cuadra-foreground)',\n opacity: 0.85,\n }}\n >\n <AlertCircle className=\"cuadra-h-3.5 cuadra-w-3.5 cuadra-flex-shrink-0 cuadra-text-muted-foreground\" />\n <span className=\"cuadra-flex-1 cuadra-text-muted-foreground\">{attachmentError.message}</span>\n <button\n type=\"button\"\n onClick={() => attachmentErrorStore.clear()}\n className=\"cuadra-flex-shrink-0 cuadra-p-0.5 cuadra-rounded hover:cuadra-bg-black/5 cuadra-transition-colors\"\n aria-label=\"Dismiss\"\n >\n <X className=\"cuadra-h-3 cuadra-w-3 cuadra-text-muted-foreground\" />\n </button>\n </div>\n )}\n {/* Composer attachments preview */}\n {enableAttachments && (\n <ComposerPrimitive.Attachments\n components={{\n Attachment: () => {\n const attachment = useAuiState(({ attachment }) => attachment);\n if (!attachment) return null;\n \n return (\n <div className=\"cuadra-relative cuadra-inline-flex cuadra-items-center cuadra-gap-2 cuadra-mb-2 cuadra-px-3 cuadra-py-1.5 cuadra-rounded-lg cuadra-transition-colors\"\n style={{ \n backgroundColor: 'var(--cuadra-muted)',\n border: '1px solid rgba(0,0,0,0.06)',\n }}\n >\n <Paperclip className=\"cuadra-h-3.5 cuadra-w-3.5 cuadra-text-muted-foreground\" strokeWidth={1.5} />\n <span className=\"cuadra-text-xs cuadra-text-foreground font-brand\">\n {attachment.name || 'Attachment'}\n </span>\n <AttachmentPrimitive.Remove asChild>\n <button\n type=\"button\"\n className=\"cuadra-flex cuadra-items-center cuadra-justify-center cuadra-size-4 cuadra-rounded-full hover:cuadra-bg-background/60 cuadra-transition-colors cuadra-ml-1\"\n aria-label=\"Remove attachment\"\n >\n <X className=\"cuadra-h-3 cuadra-w-3 cuadra-text-muted-foreground\" />\n </button>\n </AttachmentPrimitive.Remove>\n </div>\n );\n },\n }}\n />\n )}\n <ComposerPrimitive.Root \n className=\"cuadra-relative cuadra-w-full cuadra-rounded-2xl cuadra-transition-shadow cuadra-duration-300 cuadra-flex cuadra-flex-col\"\n style={{\n backgroundColor: 'var(--cuadra-composer)',\n boxShadow: 'var(--cuadra-composer-shadow)',\n border: composerBorder ? '1px solid hsl(var(--cuadra-border))' : 'none',\n }}\n >\n {/* Input area */}\n <ComposerPrimitive.Input\n asChild\n >\n <textarea\n rows={1}\n placeholder={inputPlaceholder}\n className=\"cuadra-flex cuadra-min-h-[44px] cuadra-w-full cuadra-rounded-t-2xl cuadra-border-0 cuadra-bg-transparent cuadra-px-4 cuadra-pt-3.5 cuadra-pb-1 cuadra-text-base cuadra-text-foreground placeholder:cuadra-text-muted-foreground focus-visible:cuadra-outline-none disabled:cuadra-cursor-not-allowed disabled:cuadra-opacity-50 cuadra-resize-none font-brand\"\n style={{ \n fontSize: '1rem',\n lineHeight: '1.5',\n }}\n />\n </ComposerPrimitive.Input>\n {/* Bottom toolbar — attach, model selector, send */}\n <div className=\"cuadra-flex cuadra-items-center cuadra-justify-between cuadra-px-3 cuadra-pb-2.5 cuadra-pt-1\">\n {/* Left side: attach button */}\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-1\">\n {enableAttachments && (\n <ComposerPrimitive.AddAttachment\n multiple\n asChild\n >\n <button\n type=\"button\"\n className=\"cuadra-group cuadra-relative cuadra-flex cuadra-items-center cuadra-justify-center cuadra-size-8 cuadra-rounded-lg hover:cuadra-bg-muted/60 cuadra-transition-colors\"\n aria-label=\"Attach files\"\n >\n <Plus className=\"cuadra-h-4 cuadra-w-4 cuadra-text-muted-foreground\" strokeWidth={1.5} />\n <span className=\"cuadra-pointer-events-none cuadra-absolute cuadra-bottom-full cuadra-left-1/2 cuadra--translate-x-1/2 cuadra-mb-1.5 cuadra-whitespace-nowrap cuadra-rounded-md cuadra-px-2.5 cuadra-py-1 cuadra-text-xs cuadra-text-foreground cuadra-opacity-0 group-hover:cuadra-opacity-100 cuadra-transition-opacity cuadra-duration-150 cuadra-delay-300 font-brand\" style={{ backgroundColor: 'var(--cuadra-popover)', boxShadow: '0 2px 8px rgba(0,0,0,0.15)' }}>\n Attach files\n </span>\n </button>\n </ComposerPrimitive.AddAttachment>\n )}\n </div>\n {/* Right side: send button */}\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-2\">\n <ComposerPrimitive.Send asChild>\n <button\n className=\"cuadra-flex cuadra-items-center cuadra-justify-center cuadra-size-8 cuadra-rounded-full cuadra-transition-all cuadra-duration-200 hover:cuadra-opacity-90 disabled:cuadra-opacity-30 disabled:cuadra-cursor-not-allowed\"\n style={{ \n backgroundColor: 'var(--cuadra-primary)',\n color: 'white',\n }}\n aria-label=\"Send message\"\n type=\"button\"\n >\n <ArrowUpIcon className=\"cuadra-h-4 cuadra-w-4\" strokeWidth={2} />\n </button>\n </ComposerPrimitive.Send>\n </div>\n </div>\n </ComposerPrimitive.Root>\n </div>\n </div>\n\n {/* Suggestion pills — below composer, only visible on empty thread */}\n {showCenteredWelcome && suggestions && suggestions.length > 0 && (\n <div className=\"cuadra-flex cuadra-flex-wrap cuadra-justify-center cuadra-gap-2 cuadra-px-4 cuadra-pb-6 cuadra-pt-1 cuadra-max-w-3xl cuadra-mx-auto cuadra-w-full\">\n {suggestions.map((suggestion, index) => (\n <button\n key={`suggestion-${index}`}\n type=\"button\"\n onClick={() => handleSuggestionClick(suggestion)}\n disabled={isProcessingPreMade}\n className=\"cuadra-px-3.5 cuadra-py-2 cuadra-rounded-full cuadra-text-sm cuadra-text-muted-foreground hover:cuadra-text-foreground cuadra-transition-all cuadra-duration-200 font-brand cuadra-cursor-pointer disabled:cuadra-opacity-50 disabled:cuadra-cursor-not-allowed\"\n style={{ \n backgroundColor: 'var(--cuadra-composer)',\n boxShadow: '0 0 0 1px rgba(0,0,0,0.06)',\n }}\n >\n {suggestion.prompt}\n </button>\n ))}\n </div>\n )}\n </ComposerPrimitive.AttachmentDropzone>\n </ThreadPrimitive.Root>\n );\n});\n\n/**\n * Welcome screen component — title + optional subtitle (suggestions are below the composer)\n */\nfunction WelcomeScreen({ title, subtitle }: { title: string; subtitle?: string }) {\n return (\n <div className=\"cuadra-flex cuadra-flex-col cuadra-items-center cuadra-text-center cuadra-py-6\">\n <h2\n className=\"cuadra-text-2xl md:cuadra-text-4xl cuadra-font-normal cuadra-text-foreground/80 cuadra-tracking-tight\"\n style={{ fontFamily: \"var(--cuadra-font-logo), Outfit, -apple-system, sans-serif\" }}\n >\n {title}\n </h2>\n {subtitle && (\n <p className=\"cuadra-text-sm md:cuadra-text-base cuadra-text-muted-foreground cuadra-mt-2 md:cuadra-mt-3 cuadra-max-w-lg cuadra-whitespace-pre-line\">\n {subtitle}\n </p>\n )}\n </div>\n );\n}\n\n/**\n * User message component — Claude-like right-aligned bubble\n */\nfunction UserMessage() {\n return (\n <MessagePrimitive.Root className=\"cuadra-flex cuadra-w-full cuadra-justify-end cuadra-py-2.5\">\n <div className=\"cuadra-flex cuadra-flex-col cuadra-items-end cuadra-gap-2 cuadra-max-w-[80%]\">\n {/* Display attachments */}\n <MessagePrimitive.Attachments\n components={{\n Attachment: () => {\n const attachment = useAuiState(({ attachment }) => attachment);\n if (!attachment) return null;\n \n const isImage = attachment.type === 'image';\n const file = attachment.file;\n const imageContent = attachment.content?.find((c): c is { type: 'image'; image: string } => c.type === 'image');\n const src = file ? URL.createObjectURL(file) : imageContent?.image;\n \n return (\n <div className=\"cuadra-mb-1\">\n {isImage && src ? (\n <img\n src={src}\n alt={attachment.name || 'Attachment'}\n className=\"cuadra-rounded-2xl cuadra-max-h-64 cuadra-max-w-full cuadra-object-contain\"\n style={{ border: '1px solid rgba(0,0,0,0.06)' }}\n />\n ) : (\n <div \n className=\"cuadra-flex cuadra-items-center cuadra-gap-2 cuadra-rounded-xl cuadra-px-3 cuadra-py-2\"\n style={{ backgroundColor: 'var(--cuadra-muted)', border: '1px solid rgba(0,0,0,0.06)' }}\n >\n <Paperclip className=\"cuadra-h-3.5 cuadra-w-3.5 cuadra-text-muted-foreground\" />\n <span className=\"cuadra-text-sm cuadra-text-foreground font-brand\">\n {attachment.name || 'Attachment'}\n </span>\n </div>\n )}\n </div>\n );\n },\n }}\n />\n <div \n className=\"cuadra-break-words cuadra-text-foreground font-brand\"\n style={{ \n borderRadius: '1.25rem',\n backgroundColor: 'var(--cuadra-muted)',\n paddingLeft: '1rem',\n paddingRight: '1rem',\n paddingTop: '0.625rem',\n paddingBottom: '0.625rem',\n wordBreak: 'break-word',\n fontSize: '0.9375rem',\n lineHeight: '1.625',\n }}\n >\n <MessagePrimitive.Parts\n components={{\n Text: ({ text }) => <MarkdownText content={text || ''} />,\n }}\n />\n </div>\n </div>\n </MessagePrimitive.Root>\n );\n}\n\n/**\n * Inline Reasoning component for MessagePrimitive.Parts\n * - While streaming: Shows actual reasoning content with shimmer effect\n * - After completion: Collapses into a minimal \"Reasoning\" toggle (no borders, no icons)\n */\nfunction Reasoning({ text }: { text: string }) {\n const [isOpen, setIsOpen] = useState(false);\n const isStreaming = useAuiState(({ message }) => message?.status?.type === \"running\");\n \n if (!text) return null;\n \n // While streaming: show actual reasoning content with shimmer\n if (isStreaming) {\n return (\n <div className=\"cuadra-mb-3 cuadra-py-1\">\n <div className=\"cuadra-text-sm cuadra-text-muted-foreground cuadra-leading-relaxed font-brand cuadra-animate-text-shimmer\">\n {text}\n </div>\n </div>\n );\n }\n \n // After completion: minimal collapsible\n return (\n <div className=\"cuadra-mb-3\">\n <button\n type=\"button\"\n onClick={() => setIsOpen(!isOpen)}\n className=\"cuadra-flex cuadra-items-center cuadra-gap-1.5 cuadra-py-1 cuadra-text-sm cuadra-text-muted-foreground hover:cuadra-text-foreground cuadra-transition-colors font-brand\"\n >\n <ChevronDown \n className={`cuadra-h-3.5 cuadra-w-3.5 cuadra-shrink-0 cuadra-transition-transform cuadra-duration-200 ${isOpen ? '' : 'cuadra--rotate-90'}`} \n />\n <span>Reasoning</span>\n </button>\n \n <div \n className={`cuadra-overflow-hidden cuadra-transition-all cuadra-duration-200 cuadra-ease-out ${\n isOpen ? 'cuadra-max-h-[2000px] cuadra-opacity-100' : 'cuadra-max-h-0 cuadra-opacity-0'\n }`}\n >\n <div className=\"cuadra-pl-5 cuadra-pt-2 cuadra-text-sm cuadra-text-muted-foreground cuadra-leading-relaxed font-brand\">\n <MarkdownText content={text} />\n </div>\n </div>\n </div>\n );\n}\n\n/**\n * Thinking indicator shown while assistant is generating response\n */\nfunction ThinkingIndicator() {\n return (\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-2.5 cuadra-text-muted-foreground cuadra-text-sm font-brand cuadra-py-3\">\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-1\">\n <span className=\"cuadra-inline-block cuadra-w-1.5 cuadra-h-1.5 cuadra-rounded-full cuadra-animate-bounce\" style={{ animationDelay: '0ms', backgroundColor: 'var(--cuadra-primary)', opacity: 0.7 }} />\n <span className=\"cuadra-inline-block cuadra-w-1.5 cuadra-h-1.5 cuadra-rounded-full cuadra-animate-bounce\" style={{ animationDelay: '150ms', backgroundColor: 'var(--cuadra-primary)', opacity: 0.5 }} />\n <span className=\"cuadra-inline-block cuadra-w-1.5 cuadra-h-1.5 cuadra-rounded-full cuadra-animate-bounce\" style={{ animationDelay: '300ms', backgroundColor: 'var(--cuadra-primary)', opacity: 0.3 }} />\n </div>\n <span>Thinking...</span>\n </div>\n );\n}\n\n/**\n * Assistant message component with native reasoning and custom sources\n */\nfunction AssistantMessage() {\n // Check if message is currently being generated (status.type === \"running\")\n // Also check if there's no content yet to show the thinking indicator\n const isInProgress = useAuiState(({ message }) => message?.status?.type === \"running\");\n const hasContent = useAuiState(({ message }) => {\n const content = message?.content;\n if (!content || !Array.isArray(content)) return false;\n // Check if any content part has actual text\n return content.some((part) => {\n if (part.type === 'text' && part.text && part.text.length > 0) return true;\n if (part.type === 'reasoning' && part.text && part.text.length > 0) return true;\n return false;\n });\n });\n\n // Capture the message ID when streaming actually starts (has content),\n // not at mount time (which is too early - shows thinking indicator before API call)\n // Use a ref to track if we've captured the ID and state to store it\n const hasCapturedRef = useRef(false);\n const [capturedMessageId, setCapturedMessageId] = useState<string | null>(null);\n \n // Capture the message ID when streaming starts (when we have content)\n useEffect(() => {\n if (isInProgress && hasContent && !hasCapturedRef.current) {\n const currentId = streamingMetadataStore.getCurrentMessageId();\n if (currentId) {\n hasCapturedRef.current = true;\n setCapturedMessageId(currentId);\n }\n }\n }, [isInProgress, hasContent]);\n \n // Subscribe to store changes and look up this message's sources\n // Only show sources if they were included in this specific message's response\n const sources = useSyncExternalStore(\n (callback) => streamingMetadataStore.subscribe(callback),\n () => {\n // If we have a captured message ID, use it\n if (capturedMessageId) {\n const msgSources = streamingMetadataStore.getForMessage(capturedMessageId);\n return msgSources.sources?.length ? msgSources.sources : null;\n }\n // Only check current streaming message if THIS message is currently in progress\n // This prevents completed messages from showing sources from new streaming messages\n if (isInProgress && hasContent) {\n const currentId = streamingMetadataStore.getCurrentMessageId();\n if (currentId) {\n const currentSources = streamingMetadataStore.getForMessage(currentId);\n return currentSources.sources?.length ? currentSources.sources : null;\n }\n }\n return null;\n },\n () => null\n );\n\n return (\n <MessagePrimitive.Root className=\"cuadra-flex cuadra-w-full cuadra-justify-start cuadra-py-2.5\">\n <div \n className=\"cuadra-text-foreground cuadra-max-w-full cuadra-break-words font-brand\"\n style={{ \n fontSize: '0.9375rem',\n lineHeight: '1.75',\n }}\n >\n {/* Content parts - includes native reasoning and text */}\n <MessagePrimitive.Parts\n components={{\n Text: ({ text }) => <MarkdownText content={text || ''} />,\n Reasoning: ({ text }) => <Reasoning text={text || ''} />,\n }}\n />\n \n {/* Show thinking indicator while message is being generated (no content yet) */}\n {isInProgress && !hasContent && <ThinkingIndicator />}\n \n {/* Source citations - custom component for RAG sources */}\n {sources && sources.length > 0 && (\n <SourceCitations sources={sources} />\n )}\n </div>\n </MessagePrimitive.Root>\n );\n}\n\n// ─── Icon Tooltip ────────────────────────────────────────────────\n\nfunction IconTooltip({ label, children }: { label: string; children: React.ReactNode }) {\n return (\n <span className=\"cuadra-relative cuadra-inline-flex cuadra-group/tip\">\n {children}\n <span\n className=\"cuadra-absolute cuadra-bottom-full cuadra-left-1/2 cuadra-mb-1.5 cuadra-hidden group-hover/tip:cuadra-flex cuadra-items-center cuadra-whitespace-nowrap cuadra-rounded-md cuadra-px-2 cuadra-py-1 cuadra-text-[10px] cuadra-pointer-events-none cuadra-z-50 font-brand\"\n style={{\n backgroundColor: 'var(--cuadra-popover)',\n color: 'var(--cuadra-foreground)',\n boxShadow: '0 2px 8px rgba(0,0,0,0.12), 0 0 0 1px rgba(0,0,0,0.04)',\n transform: 'translateX(-50%)',\n }}\n >\n {label}\n </span>\n </span>\n );\n}\n\n// ─── Model Selector ──────────────────────────────────────────────\n\ninterface ModelSelectorProps {\n models: ModelOption[];\n selectedModelId?: string;\n onModelChange?: (modelId: string) => void;\n}\n\nfunction ModelSelector({ models, selectedModelId, onModelChange }: ModelSelectorProps) {\n const [isOpen, setIsOpen] = useState(false);\n const selectedModel = models.find((m) => m.id === selectedModelId) || models[0];\n\n if (!selectedModel) return null;\n\n const handleModelSelect = (modelId: string) => {\n setIsOpen(false);\n if (modelId !== selectedModelId) {\n onModelChange?.(modelId);\n }\n };\n\n return (\n <div className=\"cuadra-relative\">\n <button\n type=\"button\"\n onClick={() => setIsOpen(!isOpen)}\n className=\"cuadra-flex cuadra-items-center cuadra-gap-1.5 cuadra-px-2.5 cuadra-py-1.5 cuadra-rounded-lg cuadra-text-sm md:cuadra-text-base cuadra-text-muted-foreground hover:cuadra-text-foreground cuadra-transition-colors font-brand\"\n >\n <span>{selectedModel.name}</span>\n <ChevronDown className=\"cuadra-h-3.5 cuadra-w-3.5\" />\n </button>\n\n {isOpen && (\n <>\n {/* Backdrop — desktop only */}\n <div \n className=\"cuadra-hidden md:cuadra-block cuadra-fixed cuadra-inset-0 cuadra-z-40\" \n onClick={() => setIsOpen(false)} \n />\n\n {/* Mobile: fullscreen overlay */}\n <div\n className=\"cuadra-fixed cuadra-inset-0 cuadra-z-50 cuadra-flex cuadra-flex-col md:cuadra-hidden\"\n style={{ backgroundColor: 'var(--cuadra-popover)' }}\n >\n {/* Mobile close button */}\n <div className=\"cuadra-flex cuadra-items-center cuadra-justify-end cuadra-px-4 cuadra-py-3 cuadra-min-h-[48px]\">\n <button\n type=\"button\"\n onClick={() => setIsOpen(false)}\n className=\"cuadra-flex cuadra-items-center cuadra-justify-center cuadra-p-1.5 cuadra-rounded-lg cuadra-text-muted-foreground hover:cuadra-bg-muted/60 cuadra-transition-colors\"\n >\n <X className=\"cuadra-h-5 cuadra-w-5\" />\n </button>\n </div>\n\n <div className=\"cuadra-flex-1 cuadra-min-h-0 cuadra-overflow-y-auto cuadra-py-1\">\n {models.map((model) => (\n <button\n key={model.id}\n type=\"button\"\n className=\"cuadra-w-full cuadra-flex cuadra-items-center cuadra-justify-between cuadra-gap-3 cuadra-px-4 cuadra-py-3 cuadra-text-sm cuadra-text-left hover:cuadra-bg-muted/60 cuadra-transition-colors cuadra-border-b cuadra-border-border/40 last:cuadra-border-b-0 font-brand\"\n onClick={() => handleModelSelect(model.id)}\n >\n <div className=\"cuadra-flex cuadra-flex-col cuadra-min-w-0\">\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-1.5\">\n <span className=\"cuadra-text-foreground cuadra-truncate\">{model.name}</span>\n {model.supportsReasoning && (\n <IconTooltip label=\"Reasoning\"><Brain className=\"cuadra-h-3 cuadra-w-3 cuadra-text-muted-foreground cuadra-flex-shrink-0\" /></IconTooltip>\n )}\n {model.supportsVision && (\n <IconTooltip label=\"Vision\"><Eye className=\"cuadra-h-3 cuadra-w-3 cuadra-text-muted-foreground cuadra-flex-shrink-0\" /></IconTooltip>\n )}\n </div>\n {model.description && (\n <span className=\"cuadra-text-xs cuadra-text-muted-foreground\">{model.description}</span>\n )}\n </div>\n {model.id === selectedModelId && (\n <Check className=\"cuadra-h-4 cuadra-w-4 cuadra-text-primary cuadra-flex-shrink-0\" />\n )}\n </button>\n ))}\n </div>\n </div>\n\n {/* Desktop: positioned dropdown (opens downward, centered) */}\n <div \n className=\"cuadra-hidden md:cuadra-block cuadra-absolute cuadra-top-full cuadra-left-1/2 cuadra--translate-x-1/2 cuadra-mt-2 cuadra-z-50 cuadra-rounded-xl cuadra-min-w-[200px] cuadra-py-1\"\n style={{ \n backgroundColor: 'var(--cuadra-popover)', \n boxShadow: '0 4px 24px rgba(0,0,0,0.12), 0 0 0 1px rgba(0,0,0,0.04)' \n }}\n >\n {models.map((model) => (\n <button\n key={model.id}\n type=\"button\"\n className=\"cuadra-w-full cuadra-flex cuadra-items-center cuadra-justify-between cuadra-gap-3 cuadra-px-3 cuadra-py-2 cuadra-text-sm cuadra-text-left hover:cuadra-bg-muted/60 cuadra-transition-colors font-brand\"\n onClick={() => handleModelSelect(model.id)}\n >\n <div className=\"cuadra-flex cuadra-flex-col cuadra-min-w-0\">\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-1.5\">\n <span className=\"cuadra-text-foreground cuadra-truncate\">{model.name}</span>\n {model.supportsReasoning && (\n <IconTooltip label=\"Reasoning\"><Brain className=\"cuadra-h-3 cuadra-w-3 cuadra-text-muted-foreground cuadra-flex-shrink-0\" /></IconTooltip>\n )}\n {model.supportsVision && (\n <IconTooltip label=\"Vision\"><Eye className=\"cuadra-h-3 cuadra-w-3 cuadra-text-muted-foreground cuadra-flex-shrink-0\" /></IconTooltip>\n )}\n </div>\n {model.description && (\n <span className=\"cuadra-text-xs cuadra-text-muted-foreground\">{model.description}</span>\n )}\n </div>\n {model.id === selectedModelId && (\n <Check className=\"cuadra-h-4 cuadra-w-4 cuadra-text-primary cuadra-flex-shrink-0\" />\n )}\n </button>\n ))}\n </div>\n </>\n )}\n </div>\n );\n}\n\n// ─── Chat Header with Thread List Dropdown ───────────────────────\n\nfunction ChatHeader() {\n const threadTitle = useAuiState(({ threadListItem }) => threadListItem?.title ?? null);\n const [isDropdownOpen, setIsDropdownOpen] = useState(false);\n const displayTitle = threadTitle || 'Untitled chat';\n\n return (\n <div className=\"cuadra-relative\">\n {/* Desktop: title + chevron */}\n <button\n type=\"button\"\n onClick={() => setIsDropdownOpen(!isDropdownOpen)}\n className=\"cuadra-hidden md:cuadra-flex cuadra-items-center cuadra-gap-1 cuadra-text-sm cuadra-text-foreground hover:cuadra-bg-muted/60 cuadra-rounded-lg cuadra-px-2 cuadra-py-1.5 cuadra-transition-colors font-brand\"\n >\n <span className=\"cuadra-font-medium cuadra-truncate cuadra-max-w-[200px]\">{displayTitle}</span>\n <ChevronDown className={`cuadra-h-3.5 cuadra-w-3.5 cuadra-text-muted-foreground cuadra-flex-shrink-0 cuadra-transition-transform cuadra-duration-200 ${isDropdownOpen ? 'cuadra-rotate-180' : ''}`} />\n </button>\n\n {/* Mobile: icon only */}\n <button\n type=\"button\"\n onClick={() => setIsDropdownOpen(!isDropdownOpen)}\n className=\"cuadra-flex md:cuadra-hidden cuadra-items-center cuadra-justify-center cuadra-text-foreground hover:cuadra-bg-muted/60 cuadra-rounded-lg cuadra-p-2 cuadra-transition-colors\"\n >\n <MessageSquare className=\"cuadra-h-5 cuadra-w-5\" />\n </button>\n\n {isDropdownOpen && (\n <ThreadListDropdown onClose={() => setIsDropdownOpen(false)} />\n )}\n </div>\n );\n}\n\n// ─── Thread List Dropdown ────────────────────────────────────────\n\nfunction ThreadListDropdown({ onClose }: { onClose: () => void }) {\n return (\n <>\n {/* Backdrop — visible on desktop only (mobile uses fullscreen panel) */}\n <div className=\"cuadra-hidden md:cuadra-block cuadra-fixed cuadra-inset-0 cuadra-z-40\" onClick={onClose} />\n\n {/* Mobile: fullscreen overlay */}\n <div\n className=\"cuadra-fixed cuadra-inset-0 cuadra-z-50 cuadra-flex cuadra-flex-col md:cuadra-hidden\"\n style={{ backgroundColor: 'var(--cuadra-popover)' }}\n >\n {/* Mobile close button */}\n <div className=\"cuadra-flex cuadra-items-center cuadra-justify-end cuadra-px-4 cuadra-py-3 cuadra-min-h-[48px]\">\n <button\n type=\"button\"\n onClick={onClose}\n className=\"cuadra-flex cuadra-items-center cuadra-justify-center cuadra-p-1.5 cuadra-rounded-lg cuadra-text-muted-foreground hover:cuadra-bg-muted/60 cuadra-transition-colors\"\n >\n <X className=\"cuadra-h-5 cuadra-w-5\" />\n </button>\n </div>\n\n <ThreadListPrimitive.Root className=\"cuadra-flex cuadra-flex-col cuadra-flex-1 cuadra-min-h-0\">\n {/* New Chat option */}\n <div className=\"cuadra-px-3 cuadra-py-2\">\n <ThreadListPrimitive.New\n className=\"cuadra-w-full cuadra-flex cuadra-items-center cuadra-gap-2.5 cuadra-px-3 cuadra-py-2.5 cuadra-rounded-lg cuadra-text-sm cuadra-text-foreground hover:cuadra-bg-muted cuadra-transition-colors cuadra-cursor-pointer font-brand\"\n onClick={onClose}\n >\n <Plus className=\"cuadra-h-4 cuadra-w-4 cuadra-text-muted-foreground\" />\n New Chat\n </ThreadListPrimitive.New>\n </div>\n\n {/* Thread list — scrollable, fills remaining space */}\n <DropdownThreadList onSelect={onClose} fullHeight />\n </ThreadListPrimitive.Root>\n </div>\n\n {/* Desktop: positioned dropdown (unchanged) */}\n <div\n className=\"cuadra-hidden md:cuadra-flex cuadra-absolute cuadra-left-0 cuadra-top-full cuadra-mt-1 cuadra-z-50 cuadra-w-[300px] cuadra-rounded-xl cuadra-overflow-hidden cuadra-flex-col\"\n style={{\n backgroundColor: 'var(--cuadra-popover)',\n boxShadow: '0 4px 24px rgba(0,0,0,0.12), 0 0 0 1px rgba(0,0,0,0.04)',\n }}\n >\n <ThreadListPrimitive.Root className=\"cuadra-flex cuadra-flex-col\">\n {/* New Chat option */}\n <div className=\"cuadra-px-2 cuadra-py-1.5\">\n <ThreadListPrimitive.New\n className=\"cuadra-w-full cuadra-flex cuadra-items-center cuadra-gap-2.5 cuadra-px-3 cuadra-py-2 cuadra-rounded-lg cuadra-text-sm cuadra-text-foreground hover:cuadra-bg-muted cuadra-transition-colors cuadra-cursor-pointer font-brand\"\n onClick={onClose}\n >\n <Plus className=\"cuadra-h-4 cuadra-w-4 cuadra-text-muted-foreground\" />\n New Chat\n </ThreadListPrimitive.New>\n </div>\n\n {/* Thread list — separator + scrollable area only when threads exist */}\n <DropdownThreadList onSelect={onClose} />\n </ThreadListPrimitive.Root>\n </div>\n </>\n );\n}\n\n// ─── Dropdown Thread List (separator + scrollable area, hidden when empty) ───\n\nfunction DropdownThreadList({ onSelect, fullHeight }: { onSelect?: () => void; fullHeight?: boolean }) {\n const threadIds = useAuiState(({ threads }) => threads?.threadIds ?? []);\n const newThreadId = useAuiState(({ threads }) => threads?.newThreadId);\n const hasChats = threadIds.some((id: string) => id !== newThreadId);\n\n if (!hasChats) return null;\n\n return (\n <>\n <div className=\"cuadra-mx-3 cuadra-my-0.5\" style={{ borderTop: '1px solid rgba(0,0,0,0.06)' }} />\n <div\n className={`cuadra-overflow-y-auto scrollbar-thin ${\n fullHeight\n ? 'cuadra-flex-1 cuadra-min-h-0 cuadra-px-3 cuadra-pb-3 cuadra-flex cuadra-flex-col cuadra-gap-2'\n : 'cuadra-px-2 cuadra-pb-1.5'\n }`}\n style={fullHeight ? undefined : { maxHeight: '340px' }}\n >\n <ThreadListPrimitive.Items\n components={{\n ThreadListItem: (props) => <DropdownThreadItem {...props} onSelect={onSelect} fullHeight={fullHeight} />,\n }}\n />\n </div>\n </>\n );\n}\n\n// ─── Dropdown Thread Item ────────────────────────────────────────\n\nfunction DropdownThreadItem({ onSelect, fullHeight }: { onSelect?: () => void; fullHeight?: boolean }) {\n const threadItem = useThreadListItem();\n const itemRuntime = useThreadListItemRuntime();\n const assistantRuntime = useAssistantRuntime();\n const currentThread = useThread((state) => state.threadId);\n const isActive = threadItem?.id === currentThread;\n const [isHovered, setIsHovered] = useState(false);\n const [isRenaming, setIsRenaming] = useState(false);\n const [nextTitle, setNextTitle] = useState(threadItem?.title || '');\n const [busy, setBusy] = useState(false);\n\n const handleRename = async () => {\n if (!threadItem?.remoteId || !nextTitle || nextTitle === threadItem.title) {\n setIsRenaming(false);\n return;\n }\n try {\n setBusy(true);\n // Use the runtime's rename so internal state updates and the UI\n // reflects the new title without needing a page refresh.\n await itemRuntime.rename(nextTitle);\n } catch {\n // Silently handle\n } finally {\n setBusy(false);\n setIsRenaming(false);\n }\n };\n\n const handleDelete = async (e: React.MouseEvent) => {\n e.stopPropagation();\n\n try {\n // Delete via the item runtime first — this calls the adapter and\n // removes the thread from the runtime's internal list.\n await itemRuntime.delete();\n\n // If this was the thread we were viewing, switch away now that\n // the runtime state is clean (avoids stale resource lookup).\n if (isActive) {\n await assistantRuntime.switchToNewThread();\n }\n } catch (err) {\n // eslint-disable-next-line no-console -- Intentional: surface delete failures for debugging\n console.error('[CuadraChat] Failed to delete thread:', err);\n }\n };\n\n useEffect(() => {\n if (threadItem?.title && !isRenaming) setNextTitle(threadItem.title);\n }, [threadItem?.title, isRenaming]);\n\n if (!threadItem) return null;\n\n // Mobile fullscreen: card style with three-dot menu\n if (fullHeight) {\n return (\n <ThreadListItemPrimitive.Root\n className={`cuadra-relative cuadra-flex cuadra-items-center cuadra-rounded-xl cuadra-transition-colors cuadra-cursor-pointer ${\n isActive\n ? 'cuadra-bg-primary/10 cuadra-ring-1 cuadra-ring-primary/20'\n : 'cuadra-bg-muted/40'\n }`}\n >\n {isRenaming ? (\n <div\n className=\"cuadra-flex-1 cuadra-px-4 cuadra-py-3.5 cuadra-min-w-0\"\n onClick={(e) => e.stopPropagation()}\n onMouseDown={(e) => e.stopPropagation()}\n onKeyDown={(e) => e.stopPropagation()}\n onKeyUp={(e) => e.stopPropagation()}\n onKeyPress={(e) => e.stopPropagation()}\n >\n <input\n className=\"cuadra-w-full cuadra-bg-transparent cuadra-border-0 cuadra-border-b cuadra-text-sm cuadra-text-foreground cuadra-py-0.5 cuadra-px-0 cuadra-outline-none font-brand\"\n style={{ borderBottomColor: 'var(--cuadra-primary)', borderBottomWidth: '1.5px' }}\n value={nextTitle}\n onChange={(e) => setNextTitle(e.target.value)}\n onKeyDown={(e) => {\n e.stopPropagation();\n if (e.key === 'Enter') void handleRename();\n if (e.key === 'Escape') { setIsRenaming(false); setNextTitle(threadItem.title || ''); }\n }}\n onBlur={() => void handleRename()}\n disabled={busy}\n autoFocus\n />\n </div>\n ) : (\n <ThreadListItemPrimitive.Trigger\n className=\"cuadra-flex-1 cuadra-px-4 cuadra-py-3.5 cuadra-min-w-0 cuadra-cursor-pointer\"\n onClick={() => onSelect?.()}\n >\n <span className=\"cuadra-text-sm cuadra-text-foreground cuadra-truncate cuadra-block cuadra-text-left font-brand\">\n {threadItem.title || 'Chat'}\n </span>\n </ThreadListItemPrimitive.Trigger>\n )}\n\n {/* Three-dot menu on mobile */}\n {!isRenaming && (\n <MobileThreadActions\n onRename={() => {\n setIsRenaming(true);\n setNextTitle(threadItem.title || 'Chat');\n }}\n onDelete={handleDelete}\n />\n )}\n </ThreadListItemPrimitive.Root>\n );\n }\n\n // Desktop: compact list with hover-to-reveal actions\n return (\n <ThreadListItemPrimitive.Root\n className={`cuadra-relative cuadra-flex cuadra-items-center cuadra-rounded-lg cuadra-transition-colors cuadra-cursor-pointer cuadra-border-b cuadra-border-border/40 last:cuadra-border-b-0 ${\n isActive ? 'cuadra-bg-muted' : 'hover:cuadra-bg-muted/60'\n }`}\n onMouseEnter={() => setIsHovered(true)}\n onMouseLeave={() => setIsHovered(false)}\n >\n {isRenaming ? (\n <div\n className=\"cuadra-flex-1 cuadra-px-3 cuadra-py-2 cuadra-min-w-0\"\n onClick={(e) => e.stopPropagation()}\n onMouseDown={(e) => e.stopPropagation()}\n onKeyDown={(e) => e.stopPropagation()}\n onKeyUp={(e) => e.stopPropagation()}\n onKeyPress={(e) => e.stopPropagation()}\n >\n <input\n className=\"cuadra-w-full cuadra-bg-transparent cuadra-border-0 cuadra-border-b cuadra-text-sm cuadra-text-foreground cuadra-py-0.5 cuadra-px-0 cuadra-outline-none font-brand\"\n style={{ borderBottomColor: 'var(--cuadra-primary)', borderBottomWidth: '1.5px' }}\n value={nextTitle}\n onChange={(e) => setNextTitle(e.target.value)}\n onKeyDown={(e) => {\n e.stopPropagation();\n if (e.key === 'Enter') void handleRename();\n if (e.key === 'Escape') { setIsRenaming(false); setNextTitle(threadItem.title || ''); }\n }}\n onBlur={() => void handleRename()}\n disabled={busy}\n autoFocus\n />\n </div>\n ) : (\n <ThreadListItemPrimitive.Trigger\n className=\"cuadra-flex-1 cuadra-px-3 cuadra-py-2 cuadra-min-w-0 cuadra-cursor-pointer\"\n onClick={() => onSelect?.()}\n >\n <span className=\"cuadra-text-sm cuadra-text-foreground cuadra-truncate cuadra-block cuadra-text-left font-brand\">\n {threadItem.title || 'Chat'}\n </span>\n </ThreadListItemPrimitive.Trigger>\n )}\n\n {/* Hover actions: rename + delete */}\n {(isHovered || isActive) && !isRenaming && (\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-0.5 cuadra-mr-1.5 cuadra-flex-shrink-0\">\n <button\n type=\"button\"\n className=\"cuadra-p-1 cuadra-rounded hover:cuadra-bg-background/60 cuadra-transition-colors\"\n onClick={(e) => {\n e.stopPropagation();\n setIsRenaming(true);\n setNextTitle(threadItem.title || 'Chat');\n }}\n title=\"Rename\"\n >\n <Pencil className=\"cuadra-h-3 cuadra-w-3 cuadra-text-muted-foreground\" />\n </button>\n <button\n type=\"button\"\n className=\"cuadra-p-1 cuadra-rounded hover:cuadra-bg-destructive/10 cuadra-transition-colors\"\n onClick={handleDelete}\n title=\"Delete\"\n >\n <Trash2 className=\"cuadra-h-3 cuadra-w-3 cuadra-text-destructive\" />\n </button>\n </div>\n )}\n </ThreadListItemPrimitive.Root>\n );\n}\n\n// ─── Mobile Thread Actions (three-dot menu) ─────────────────────\n\nfunction MobileThreadActions({ onRename, onDelete }: { onRename: () => void; onDelete: (e: React.MouseEvent) => void }) {\n const [isOpen, setIsOpen] = useState(false);\n\n return (\n <div className=\"cuadra-relative cuadra-flex-shrink-0 cuadra-mr-2\">\n <button\n type=\"button\"\n className=\"cuadra-p-2 cuadra-rounded-lg cuadra-text-muted-foreground hover:cuadra-bg-background/60 cuadra-transition-colors\"\n onClick={(e) => {\n e.stopPropagation();\n setIsOpen(!isOpen);\n }}\n >\n <EllipsisVertical className=\"cuadra-h-4 cuadra-w-4\" />\n </button>\n\n {isOpen && (\n <>\n <div className=\"cuadra-fixed cuadra-inset-0 cuadra-z-50\" onClick={(e) => { e.stopPropagation(); setIsOpen(false); }} />\n <div\n className=\"cuadra-absolute cuadra-right-0 cuadra-top-full cuadra-mt-1 cuadra-z-[60] cuadra-w-36 cuadra-rounded-xl cuadra-py-1 cuadra-overflow-hidden\"\n style={{\n backgroundColor: 'var(--cuadra-popover)',\n boxShadow: '0 4px 24px rgba(0,0,0,0.12), 0 0 0 1px rgba(0,0,0,0.04)',\n }}\n >\n <button\n type=\"button\"\n className=\"cuadra-w-full cuadra-flex cuadra-items-center cuadra-gap-2.5 cuadra-px-3 cuadra-py-2.5 cuadra-text-sm cuadra-text-foreground hover:cuadra-bg-muted cuadra-transition-colors font-brand\"\n onClick={(e) => {\n e.stopPropagation();\n setIsOpen(false);\n onRename();\n }}\n >\n <Pencil className=\"cuadra-h-4 cuadra-w-4 cuadra-text-muted-foreground\" />\n Rename\n </button>\n <button\n type=\"button\"\n className=\"cuadra-w-full cuadra-flex cuadra-items-center cuadra-gap-2.5 cuadra-px-3 cuadra-py-2.5 cuadra-text-sm cuadra-text-destructive hover:cuadra-bg-destructive/10 cuadra-transition-colors font-brand\"\n onClick={(e) => {\n e.stopPropagation();\n setIsOpen(false);\n onDelete(e);\n }}\n >\n <Trash2 className=\"cuadra-h-4 cuadra-w-4\" />\n Delete\n </button>\n </div>\n </>\n )}\n </div>\n );\n}\n\n","import React from 'react';\n\ninterface TexturedCardProps {\n children: React.ReactNode;\n className?: string;\n paddingX?: string;\n paddingY?: string;\n /** Border width in pixels. Set to 0 for borderless. */\n borderSizePx?: number;\n style?: React.CSSProperties;\n contentStyle?: React.CSSProperties;\n /** @deprecated No longer used — kept for backward compatibility */\n rotating?: boolean;\n}\n\nexport const TexturedCard: React.FC<TexturedCardProps> = ({\n children,\n className = '',\n paddingX = 'px-0',\n paddingY = 'py-0',\n borderSizePx = 2,\n style,\n contentStyle,\n}) => {\n return (\n <div\n className={`cuadra-relative cuadra-bg-background md:cuadra-rounded-lg cuadra-overflow-hidden cuadra-flex cuadra-flex-col ${paddingX} ${paddingY} ${className}`}\n style={{\n border: borderSizePx > 0 ? `${borderSizePx}px solid var(--cuadra-primary)` : 'none',\n ...style,\n ...contentStyle,\n }}\n >\n {children}\n </div>\n );\n};\n","import React, { createContext, useContext } from 'react';\n\ninterface CuadraWidgetContextType {\n currentModelId: string | null;\n models: Array<{ id: string; name: string }>;\n modelsLoading: boolean;\n modelsError: Error | null;\n onModelChange: (modelId: string) => void;\n refetchModels: () => void;\n}\n\nconst CuadraWidgetContext = createContext<CuadraWidgetContextType | undefined>(undefined);\n\nexport const useCuadraWidgetContext = () => {\n const context = useContext(CuadraWidgetContext);\n if (!context) {\n throw new Error('useCuadraWidgetContext must be used within CuadraWidgetProvider');\n }\n return context;\n};\n\ninterface CuadraWidgetProviderProps {\n children: React.ReactNode;\n value: CuadraWidgetContextType;\n}\n\nexport const CuadraWidgetProvider: React.FC<CuadraWidgetProviderProps> = ({ children, value }) => {\n return (\n <CuadraWidgetContext.Provider value={value}>{children}</CuadraWidgetContext.Provider>\n );\n};\n\n","import { useEffect } from 'react';\n\n/**\n * Hook to inject uikit CSS styles into the DOM\n * \n * IMPORTANT: For the CSS to work, consumers must import it:\n * import '@cuadra-ai/uikit/styles';\n * \n * This hook provides a fallback that tries to inject styles if they're not already loaded.\n */\nexport function useInjectStyles(): void {\n useEffect(() => {\n // Check if styles already injected or if CSS is already loaded\n const existingStyle = document.querySelector('style[data-cuadra-uikit]');\n if (existingStyle) {\n return;\n }\n\n // Check if CSS classes are already available (styles might be loaded via import)\n const testEl = document.createElement('div');\n testEl.className = 'cuadra-flex';\n document.body.appendChild(testEl);\n const stylesLoaded = window.getComputedStyle(testEl).display === 'flex';\n document.body.removeChild(testEl);\n\n if (stylesLoaded) {\n // Styles are already loaded, no need to inject\n return;\n }\n\n // Styles not loaded - try to inject them\n // Note: This is a fallback. The proper way is to import '@cuadra-ai/uikit/styles'\n // Silently attempt to load styles - user should import '@cuadra-ai/uikit/styles'\n\n // Create a link element to load the CSS\n const link = document.createElement('link');\n link.setAttribute('data-cuadra-uikit', 'true');\n link.rel = 'stylesheet';\n link.type = 'text/css';\n \n // Try to find the CSS file\n // In development with file: protocol, try relative path\n // In production, it should be in node_modules\n const possiblePaths = [\n '/node_modules/@cuadra-ai/uikit/dist/uikit.css',\n new URL('../dist/uikit.css', import.meta.url).href,\n ];\n\n let loaded = false;\n for (const path of possiblePaths) {\n link.href = path;\n link.onload = () => {\n loaded = true;\n };\n link.onerror = () => {\n // Try next path\n };\n document.head.appendChild(link);\n if (loaded) break;\n }\n }, []);\n}\n\n","import { forwardRef, useCallback, useEffect, useImperativeHandle, useMemo, useRef, useState } from 'react';\nimport { Loader2 } from 'lucide-react';\nimport { CuadraRuntimeProvider } from './CuadraRuntimeProvider';\nimport { SimpleThread, type SimpleThreadHandle } from './SimpleThread';\nimport { TexturedCard } from '../widget/components/TexturedCard';\nimport { CuadraWidgetProvider } from '../widget/context/CuadraWidgetContext';\nimport { CuadraChatClient } from '../lib/cuadraChatClient';\nimport type { ModelPage } from '../types/cuadra';\nimport { useInjectStyles } from '../utils/useInjectStyles';\n\n/**\n * Handle for controlling WidgetContent externally\n */\nexport interface WidgetContentHandle {\n /** Send a user message programmatically (triggers normal API call) */\n sendMessage: (message: string) => void;\n /** Send a pre-made Q&A pair (instant, simulated streaming, no API call) */\n sendPreMadeQA: (question: string, answer: string, options?: { delay?: number; streamingSpeed?: number }) => void;\n /** Clear the current chat and start fresh */\n clearChat: () => void;\n}\n\n// Export the config type for use in React components\nexport interface CuadraUIKitConfig {\n // API Configuration (choose one)\n /** Cuadra API base URL (e.g., 'https://api.cuadra.ai') */\n baseUrl?: string;\n /** Proxy URL for backend-handled authentication (e.g., '/api/chat') */\n proxyUrl?: string;\n \n // Authentication (optional if using proxyUrl)\n /** Bearer token for authentication (optional if backend handles auth) */\n sessionToken?: string | null;\n \n // Chat configuration\n /** Chat mode: single thread or multi-thread */\n mode?: 'singleChat' | 'multiChat';\n /** Model selection mode */\n modelMode?: 'selector' | 'fixed';\n /** Model ID (required if modelMode=\"fixed\") */\n modelId?: string;\n /** Callback when model changes (selector mode) */\n onModelChange?: (modelId: string) => void;\n /** Create temporary chats that auto-delete */\n ephemeral?: boolean;\n /** System prompt for the assistant */\n systemPrompt?: string;\n /** Enable reasoning/thinking output for supported models (e.g., o1, o3) */\n enableReasoning?: boolean;\n \n // Thread configuration\n /** Load existing thread (multiChat mode) */\n initialThreadId?: string;\n \n // UI Configuration\n /** Language/locale for i18n (e.g., 'en', 'es', 'fr') */\n language?: string;\n /** Welcome screen title */\n welcomeTitle?: string;\n /** Welcome screen subtitle */\n welcomeSubtitle?: string;\n /** Extra top padding for thread viewport (e.g., '1rem', '2rem') */\n extraTopPadding?: string;\n /** Suggestions to show in welcome screen (with optional pre-made responses) */\n suggestions?: Array<{\n /** Suggestion prompt text */\n prompt: string;\n /** Pre-made response (optional) - if provided, response appears instantly with simulated streaming */\n response?: string;\n /** Delay before showing response in ms (overrides preMadeResponseDelay) */\n responseDelay?: number;\n }>;\n /** Placeholder text for the input field */\n inputPlaceholder?: string;\n /** Default delay before pre-made responses in ms (default: 1000) */\n preMadeResponseDelay?: number;\n /** Speed of simulated streaming in ms per word (default: 50) */\n streamingSpeed?: number;\n \n // Theme configuration\n /** Show theme toggle button */\n showThemeToggle?: boolean;\n /** Initial theme ('light' | 'dark' | 'system') */\n theme?: 'light' | 'dark' | 'system';\n \n // Container\n /** Container element ID (defaults to 'cuadra-chat') */\n containerId?: string;\n /** CSS classes to add to container element */\n containerClass?: string;\n \n // Callbacks\n /** Called when an error occurs */\n onError?: (error: Error) => void;\n /** Called when chat is created */\n onChatCreated?: (chatId: string) => void;\n /** Called when a user message is sent */\n onUserMessage?: () => void;\n /** Called when thread ID updates */\n onThreadIdUpdate?: (oldId: string, newId: string) => void;\n /** Called when logout button is clicked */\n onLogout?: () => void;\n /** Enable file attachments in the chat input */\n enableAttachments?: boolean;\n /** Whether the current model supports vision/image inputs. When false, image attachments are rejected with an error. */\n supportsVision?: boolean;\n /** Initial message to send automatically when chat loads */\n initialMessage?: string;\n /** Initial pre-made Q&A to display when chat loads */\n initialPreMadeQA?: {\n question: string;\n answer: string;\n };\n /**\n * Interceptor called before each outgoing API request.\n * Return `true` to allow the request, `false` to block it.\n * Can be async (e.g., to show a login modal).\n */\n onBeforeRequest?: () => boolean | Promise<boolean>;\n /**\n * Mock response to show when onBeforeRequest returns false.\n * If set, instead of throwing an error, this message will be streamed as a response.\n * Can be a string or a function that returns a string (called each time a request is blocked).\n * Useful for showing helpful messages like \"Complete setup before chatting\".\n */\n mockResponseOnBlocked?: string | (() => string);\n /**\n * Hide the decorative border around the chat container\n * Useful for mobile apps or when embedding in a full-screen context\n * @default false\n */\n borderless?: boolean;\n /**\n * Apply safe area insets for mobile devices (notch, home indicator)\n * When true, adds padding to header, sidebar, and input areas\n * @default false\n */\n safeArea?: boolean;\n /**\n * Organization name to display in the sidebar header\n * Used for workspace branding in mobile apps\n */\n orgName?: string;\n /**\n * Primary color for branding (hex format)\n * Overrides the default primary color CSS variable\n * @example '#3B82F6'\n */\n primaryColor?: string;\n}\n\n/**\n * Main widget content component - shared between library and widget builds\n * This component does NOT use createRoot - it's a regular React component\n */\nexport const WidgetContent = forwardRef<WidgetContentHandle, { config: CuadraUIKitConfig }>(\n function WidgetContent({ config }, ref) {\n // Inject styles when component mounts\n useInjectStyles();\n \n // Ref to SimpleThread for external control\n const simpleThreadRef = useRef<SimpleThreadHandle>(null);\n \n // Track if initial message has been sent\n const initialMessageSentRef = useRef(false);\n\n const {\n baseUrl,\n proxyUrl,\n sessionToken,\n mode = 'multiChat',\n modelMode = 'fixed',\n modelId,\n onModelChange,\n ephemeral = false,\n systemPrompt,\n enableReasoning = false,\n initialThreadId,\n welcomeTitle,\n welcomeSubtitle,\n extraTopPadding,\n suggestions,\n inputPlaceholder,\n preMadeResponseDelay = 1000,\n streamingSpeed = 50,\n // showThemeToggle kept in interface for consumer API but not used internally\n initialMessage,\n initialPreMadeQA,\n theme: initialTheme = 'system',\n onError,\n onChatCreated,\n onUserMessage,\n onThreadIdUpdate,\n enableAttachments = false,\n supportsVision,\n onBeforeRequest,\n mockResponseOnBlocked,\n } = config;\n\n // Expose methods via ref\n useImperativeHandle(ref, () => ({\n sendMessage: (message: string) => {\n simpleThreadRef.current?.sendMessage(message);\n },\n sendPreMadeQA: (question: string, answer: string, options?: { delay?: number; streamingSpeed?: number }) => {\n simpleThreadRef.current?.sendPreMadeQA(question, answer, options);\n },\n clearChat: () => {\n simpleThreadRef.current?.clearChat();\n },\n }), []);\n\n // Handle initial message or pre-made Q&A on mount\n useEffect(() => {\n if (initialMessageSentRef.current) return;\n \n // Wait for chat to be ready before sending initial message\n const timer = setTimeout(() => {\n if (initialPreMadeQA) {\n initialMessageSentRef.current = true;\n simpleThreadRef.current?.sendPreMadeQA(\n initialPreMadeQA.question, \n initialPreMadeQA.answer,\n { delay: preMadeResponseDelay, streamingSpeed }\n );\n } else if (initialMessage) {\n initialMessageSentRef.current = true;\n simpleThreadRef.current?.sendMessage(initialMessage);\n }\n }, 500); // Wait 500ms for runtime to be ready\n \n return () => clearTimeout(timer);\n }, [initialMessage, initialPreMadeQA, preMadeResponseDelay, streamingSpeed]);\n\n // Determine the actual baseUrl and proxy mode\n const actualBaseUrl = proxyUrl || baseUrl || '';\n const isProxyMode = !!proxyUrl;\n\n // Create client for fetching models\n const client = useMemo(\n () => new CuadraChatClient(actualBaseUrl, sessionToken || undefined, isProxyMode),\n [actualBaseUrl, sessionToken, isProxyMode],\n );\n\n // Models state (replacing react-query)\n const [modelsData, setModelsData] = useState<ModelPage | null>(null);\n const [modelsLoading, setModelsLoading] = useState(false);\n const [modelsError, setModelsError] = useState<Error | null>(null);\n \n // Chats loading state (for multiChat mode)\n const [chatsLoading, setChatsLoading] = useState(false);\n \n // Set chatsLoading to true initially for multiChat mode\n useEffect(() => {\n if (mode === 'multiChat') {\n setChatsLoading(true);\n }\n }, [mode]);\n\n // Fetch models function\n const fetchModels = useCallback(async () => {\n if (modelMode !== 'selector') return;\n \n setModelsLoading(true);\n setModelsError(null);\n try {\n const data = await client.listModels();\n setModelsData(data);\n } catch (error) {\n setModelsError(error instanceof Error ? error : new Error('Failed to load models'));\n } finally {\n setModelsLoading(false);\n }\n }, [client, modelMode]);\n\n // Initial fetch and refetch function\n useEffect(() => {\n if (modelMode === 'selector') {\n fetchModels();\n }\n }, [modelMode, fetchModels]);\n\n // Expose refetch function on window for external access\n useEffect(() => {\n if (typeof window !== 'undefined') {\n (window as Window & { __cuadraUIKitRefetchModels?: typeof fetchModels }).__cuadraUIKitRefetchModels = fetchModels;\n }\n return () => {\n if (typeof window !== 'undefined') {\n delete (window as Window & { __cuadraUIKitRefetchModels?: typeof fetchModels }).__cuadraUIKitRefetchModels;\n }\n };\n }, [fetchModels]);\n\n // Track current model ID\n const [currentModelId, setCurrentModelId] = useState<string | null>(\n modelMode === 'fixed' ? modelId || null : null,\n );\n \n // Track previous model ID to detect changes\n const prevModelIdRef = useRef<string | null>(currentModelId);\n\n // Handle model change — only updates state. Thread switching is handled\n // inside SimpleThread (which lives inside the AssistantRuntimeProvider\n // and has direct hook access to the runtime).\n const handleModelChange = useCallback(\n (newModelId: string) => {\n setCurrentModelId(newModelId);\n prevModelIdRef.current = newModelId;\n onModelChange?.(newModelId);\n },\n [onModelChange, mode],\n );\n\n // Auto-select first model when models are loaded\n useEffect(() => {\n if (\n modelMode === 'selector' &&\n modelsData?.items &&\n modelsData.items.length > 0 &&\n !currentModelId\n ) {\n const firstModel = modelsData.items[0];\n if (firstModel?.id) {\n const modelId = firstModel.id as string;\n setCurrentModelId(modelId);\n onModelChange?.(modelId);\n }\n }\n }, [modelMode, modelsData?.items, currentModelId, onModelChange]);\n\n // Don't render chat UI until we have a model selected (for selector mode)\n const hasModel = modelMode === 'fixed' ? !!modelId : !!currentModelId;\n\n // Derive supportsVision from selected model data (selector mode) or use the explicit prop\n const resolvedSupportsVision = useMemo(() => {\n // If explicitly set by consumer, respect it\n if (supportsVision !== undefined) return supportsVision;\n // In selector mode, derive from the currently selected model's data\n if (modelMode === 'selector' && currentModelId && modelsData?.items) {\n const selectedModel = modelsData.items.find((m) => m.id === currentModelId);\n return selectedModel?.supportsVision ?? true;\n }\n // Default to true (allow images)\n return true;\n }, [supportsVision, modelMode, currentModelId, modelsData?.items]);\n\n // Prepare models for ChatHeader\n const models = useMemo(\n () =>\n (modelsData?.items || [])\n .filter((m) => m?.id && m?.displayName)\n .map((m) => ({\n id: m.id as string,\n name: m.displayName as string,\n supportsVision: m.supportsVision ?? false,\n supportsReasoning: m.supportsReasoning ?? false,\n })),\n [modelsData],\n );\n\n // Create widget context value\n const widgetContextValue = useMemo(\n () => ({\n currentModelId: currentModelId || '',\n models,\n modelsLoading,\n modelsError,\n onModelChange: handleModelChange,\n refetchModels: fetchModels,\n }),\n [currentModelId, models, modelsLoading, modelsError, handleModelChange, fetchModels],\n );\n\n // Stable callback for when chats finish loading (avoids recreating wrappedAdapter on every render)\n const handleChatsLoaded = useCallback(() => {\n if (mode === 'multiChat') {\n setChatsLoading(false);\n }\n }, [mode]);\n\n // Determine if we're still loading (models and/or chats)\n const isLoading = \n (modelMode === 'selector' && modelsLoading) || \n (mode === 'multiChat' && chatsLoading);\n\n // Show loading state if we're still loading models or chats\n if (isLoading) {\n return (\n <CuadraWidgetProvider value={widgetContextValue}>\n <TexturedCard \n className=\"cuadra-h-full\" \n paddingX=\"cuadra-px-0\" \n paddingY=\"cuadra-py-0\"\n borderSizePx={config.borderless ? 0 : 2}\n >\n <div className=\"cuadra-flex cuadra-items-center cuadra-justify-center cuadra-h-full\">\n <Loader2 className=\"cuadra-h-6 cuadra-w-6 cuadra-animate-spin cuadra-text-muted-foreground\" />\n </div>\n </TexturedCard>\n </CuadraWidgetProvider>\n );\n }\n\n // Show error state if we need a model but don't have one\n if (modelMode === 'selector' && !hasModel && modelsError) {\n return (\n <CuadraWidgetProvider value={widgetContextValue}>\n <TexturedCard className=\"cuadra-h-full\" paddingX=\"cuadra-px-0\" paddingY=\"cuadra-py-0\">\n <div className=\"cuadra-flex cuadra-items-center cuadra-justify-center cuadra-h-full\">\n <div className=\"cuadra-flex cuadra-flex-col cuadra-items-center cuadra-gap-2\">\n <p className=\"cuadra-text-sm cuadra-text-destructive font-brand\">Failed to load models</p>\n </div>\n </div>\n </TexturedCard>\n </CuadraWidgetProvider>\n );\n }\n\n return (\n <div className=\"cuadra-uikit cuadra-h-full cuadra-w-full\" style={{ height: '100%', width: '100%' }}>\n <CuadraWidgetProvider value={widgetContextValue}>\n <CuadraRuntimeProvider\n baseUrl={isProxyMode ? actualBaseUrl : (baseUrl || actualBaseUrl)}\n sessionToken={sessionToken}\n isProxyMode={isProxyMode}\n mode={mode}\n modelMode={modelMode}\n modelId={currentModelId || modelId || undefined}\n onModelChange={handleModelChange}\n ephemeral={ephemeral}\n systemPrompt={systemPrompt}\n enableReasoning={enableReasoning}\n initialThreadId={initialThreadId}\n enableAttachments={enableAttachments}\n supportsVision={resolvedSupportsVision}\n onError={onError}\n onChatCreated={onChatCreated}\n onUserMessage={onUserMessage}\n onThreadIdUpdate={onThreadIdUpdate}\n onBeforeRequest={onBeforeRequest}\n mockResponseOnBlocked={mockResponseOnBlocked}\n onChatsLoaded={handleChatsLoaded}\n >\n <TexturedCard\n paddingX=\"cuadra-px-0\"\n paddingY=\"cuadra-py-0\"\n className={`cuadra-h-full ${config.containerClass || ''}`}\n borderSizePx={config.borderless ? 0 : 2}\n style={{ height: '100%' }}\n >\n <div \n className=\"cuadra-h-full cuadra-flex cuadra-relative\" \n style={{ height: '100%' }}\n >\n <div className=\"cuadra-flex-1 cuadra-flex cuadra-flex-col\">\n <div className=\"cuadra-flex-1 cuadra-relative cuadra-min-h-0\">\n <SimpleThread \n ref={simpleThreadRef}\n welcomeTitle={welcomeTitle}\n welcomeSubtitle={welcomeSubtitle}\n extraTopPadding={extraTopPadding}\n suggestions={suggestions}\n inputPlaceholder={inputPlaceholder}\n enableAttachments={enableAttachments}\n preMadeResponseDelay={preMadeResponseDelay}\n streamingSpeed={streamingSpeed}\n safeArea={config.safeArea}\n showChatHeader={mode === 'multiChat'}\n models={modelMode === 'selector' ? models : undefined}\n selectedModelId={currentModelId || undefined}\n onModelChange={modelMode === 'selector' ? handleModelChange : undefined}\n theme={initialTheme}\n />\n </div>\n </div>\n </div>\n </TexturedCard>\n </CuadraRuntimeProvider>\n </CuadraWidgetProvider>\n </div>\n );\n});\n\n","import type { CuadraTheme } from '../types/theme';\n\n/**\n * Apply custom theme CSS variables.\n *\n * Variables are injected via a `<style>` element that targets `.cuadra-uikit`\n * (and `.dark .cuadra-uikit` for dark mode) so they override the scoped\n * defaults declared in the UIKit stylesheet.\n *\n * The variables are also set on `:root` as a fallback for any elements that\n * live outside the `.cuadra-uikit` wrapper (e.g. portalled menus).\n *\n * @param theme - Theme configuration object with light and/or dark mode colors\n * @param _scope - Unused, kept for backwards-compatible call sites\n */\nexport function applyTheme(theme: CuadraTheme, _scope: string = ':root'): void {\n if (typeof document === 'undefined') {\n return;\n }\n\n const root = document.documentElement;\n const parts: string[] = [];\n\n // Apply light theme\n if (theme.light) {\n // Inline styles on :root for portalled elements\n applyThemeColors(theme.light as Record<string, string | undefined>, root);\n\n // Style rule for .cuadra-uikit scope (overrides the scoped defaults)\n const lightVars = buildCSSVariables(theme.light as Record<string, string | undefined>);\n if (lightVars) {\n parts.push(`.cuadra-uikit { ${lightVars} }`);\n }\n }\n\n // Apply dark theme\n if (theme.dark) {\n const darkVars = buildCSSVariables(theme.dark as Record<string, string | undefined>);\n if (darkVars) {\n parts.push(`.dark { ${darkVars} }`);\n parts.push(`.dark .cuadra-uikit { ${darkVars} }`);\n }\n }\n\n // Create or update a single style element for all custom theme rules\n if (parts.length > 0) {\n let styleEl = document.getElementById('cuadra-theme-custom') as HTMLStyleElement;\n if (!styleEl) {\n styleEl = document.createElement('style');\n styleEl.id = 'cuadra-theme-custom';\n document.head.appendChild(styleEl);\n }\n styleEl.textContent = parts.join('\\n');\n }\n}\n\n/**\n * Map theme property names to CSS variable names\n */\nconst CSS_VAR_MAP: Record<string, string> = {\n background: 'background',\n foreground: 'foreground',\n card: 'card',\n cardForeground: 'card-foreground',\n popover: 'popover',\n popoverForeground: 'popover-foreground',\n primary: 'primary',\n primaryForeground: 'primary-foreground',\n primaryLight: 'primary-light',\n primaryLighter: 'primary-lighter',\n secondary: 'secondary',\n secondaryForeground: 'secondary-foreground',\n muted: 'muted',\n mutedForeground: 'muted-foreground',\n accent: 'accent',\n accentForeground: 'accent-foreground',\n border: 'border',\n input: 'input',\n ring: 'ring',\n destructive: 'destructive',\n destructiveForeground: 'destructive-foreground',\n success: 'success',\n successForeground: 'success-foreground',\n info: 'info',\n infoForeground: 'info-foreground',\n warning: 'warning',\n warningForeground: 'warning-foreground',\n};\n\n/**\n * Convert camelCase property name to CSS variable name\n */\nfunction getCSSVarName(key: string): string {\n const mapped = CSS_VAR_MAP[key];\n if (mapped) {\n return `--cuadra-${mapped}`;\n }\n // Fallback: convert camelCase to kebab-case\n return `--cuadra-${key.replace(/([A-Z])/g, '-$1').toLowerCase()}`;\n}\n\n/**\n * Apply theme colors to an element\n */\nfunction applyThemeColors(colors: Record<string, string | undefined>, element: HTMLElement): void {\n Object.entries(colors).forEach(([key, value]) => {\n if (value) {\n const cssVarName = getCSSVarName(key);\n element.style.setProperty(cssVarName, value);\n }\n });\n}\n\n/**\n * Build CSS variables string from theme colors\n */\nfunction buildCSSVariables(colors: Record<string, string | undefined>): string {\n return Object.entries(colors)\n .map(([key, value]) => {\n if (value) {\n const cssVarName = getCSSVarName(key);\n return `${cssVarName}: ${value};`;\n }\n return '';\n })\n .filter(Boolean)\n .join(' ');\n}\n\n/**\n * Remove custom theme (restore defaults)\n */\nexport function removeTheme(): void {\n if (typeof document === 'undefined') {\n return;\n }\n\n const root = document.documentElement;\n \n // Remove all custom CSS variables from root\n const customVars = Array.from(root.style).filter(prop => prop.startsWith('--cuadra-'));\n customVars.forEach(prop => root.style.removeProperty(prop));\n\n // Remove custom theme style element\n const customStyle = document.getElementById('cuadra-theme-custom');\n if (customStyle) {\n customStyle.remove();\n }\n\n // Remove legacy dark theme style element (from older versions)\n const darkStyle = document.getElementById('cuadra-theme-dark');\n if (darkStyle) {\n darkStyle.remove();\n }\n}\n\n","import { Component, type ErrorInfo, type ReactNode } from 'react';\n\n/**\n * Props for CuadraErrorBoundary\n */\nexport interface CuadraErrorBoundaryProps {\n /** Child components to wrap */\n children: ReactNode;\n /** \n * Custom fallback UI or render function\n * If not provided, DefaultErrorFallback is used\n */\n fallback?: ReactNode | ((error: Error, reset: () => void) => ReactNode);\n /** \n * Callback when an error is caught\n * Use for logging, analytics, etc.\n */\n onError?: (error: Error, errorInfo: ErrorInfo) => void;\n}\n\ninterface ErrorBoundaryState {\n hasError: boolean;\n error: Error | null;\n}\n\n/**\n * Default fallback UI shown when an error occurs\n */\nfunction DefaultErrorFallback({ \n onReset, \n error \n}: { \n onReset: () => void; \n error: Error | null;\n}) {\n return (\n <div className=\"cuadra-flex cuadra-flex-col cuadra-items-center cuadra-justify-center cuadra-h-full cuadra-p-6 cuadra-text-center\">\n <div className=\"cuadra-mb-4\">\n <svg \n className=\"cuadra-w-12 cuadra-h-12 cuadra-text-destructive cuadra-mx-auto\"\n fill=\"none\" \n viewBox=\"0 0 24 24\" \n stroke=\"currentColor\"\n >\n <path \n strokeLinecap=\"round\" \n strokeLinejoin=\"round\" \n strokeWidth={1.5} \n d=\"M12 9v2m0 4h.01m-6.938 4h13.856c1.54 0 2.502-1.667 1.732-3L13.732 4c-.77-1.333-2.694-1.333-3.464 0L3.34 16c-.77 1.333.192 3 1.732 3z\" \n />\n </svg>\n </div>\n <h3 className=\"cuadra-text-lg cuadra-font-medium cuadra-text-foreground cuadra-mb-2\">\n Something went wrong\n </h3>\n <p className=\"cuadra-text-sm cuadra-text-muted-foreground cuadra-mb-4 cuadra-max-w-sm\">\n {error?.message || 'An unexpected error occurred. Please try again.'}\n </p>\n <button\n onClick={onReset}\n className=\"cuadra-px-4 cuadra-py-2 cuadra-text-sm cuadra-font-medium cuadra-rounded-md cuadra-transition-colors\"\n style={{\n backgroundColor: 'var(--cuadra-primary)',\n color: 'hsl(var(--cuadra-primary-foreground))',\n }}\n >\n Try Again\n </button>\n </div>\n );\n}\n\n/**\n * Error boundary component for CuadraChat\n * \n * Catches JavaScript errors in child component tree and displays\n * a fallback UI instead of crashing the entire application.\n * \n * @example Basic usage (uses default fallback)\n * ```tsx\n * <CuadraErrorBoundary>\n * <CuadraChat {...props} />\n * </CuadraErrorBoundary>\n * ```\n * \n * @example With custom fallback\n * ```tsx\n * <CuadraErrorBoundary\n * fallback={<div>Oops! Something went wrong.</div>}\n * >\n * <CuadraChat {...props} />\n * </CuadraErrorBoundary>\n * ```\n * \n * @example With render function fallback\n * ```tsx\n * <CuadraErrorBoundary\n * fallback={(error, reset) => (\n * <div>\n * <p>Error: {error.message}</p>\n * <button onClick={reset}>Retry</button>\n * </div>\n * )}\n * onError={(error, info) => {\n * console.error('Chat error:', error);\n * analytics.track('chat_error', { message: error.message });\n * }}\n * >\n * <CuadraChat {...props} />\n * </CuadraErrorBoundary>\n * ```\n */\nexport class CuadraErrorBoundary extends Component<CuadraErrorBoundaryProps, ErrorBoundaryState> {\n constructor(props: CuadraErrorBoundaryProps) {\n super(props);\n this.state = { hasError: false, error: null };\n }\n\n static getDerivedStateFromError(error: Error): ErrorBoundaryState {\n return { hasError: true, error };\n }\n\n componentDidCatch(error: Error, errorInfo: ErrorInfo): void {\n this.props.onError?.(error, errorInfo);\n }\n\n /**\n * Reset the error state and attempt to re-render children\n */\n reset = (): void => {\n this.setState({ hasError: false, error: null });\n };\n\n render(): ReactNode {\n if (this.state.hasError) {\n const { fallback } = this.props;\n const { error } = this.state;\n\n // Render function fallback\n if (typeof fallback === 'function') {\n return fallback(error!, this.reset);\n }\n\n // Static ReactNode fallback\n if (fallback) {\n return fallback;\n }\n\n // Default fallback\n return <DefaultErrorFallback error={error} onReset={this.reset} />;\n }\n\n return this.props.children;\n }\n}\n\n/**\n * Hook-friendly wrapper for error boundary\n * Use when you need to reset error boundary from a child component\n */\nexport function withErrorBoundary<P extends object>(\n WrappedComponent: React.ComponentType<P>,\n errorBoundaryProps?: Omit<CuadraErrorBoundaryProps, 'children'>\n): React.FC<P> {\n const WithErrorBoundary: React.FC<P> = (props) => (\n <CuadraErrorBoundary {...errorBoundaryProps}>\n <WrappedComponent {...props} />\n </CuadraErrorBoundary>\n );\n\n WithErrorBoundary.displayName = `withErrorBoundary(${WrappedComponent.displayName || WrappedComponent.name || 'Component'})`;\n\n return WithErrorBoundary;\n}\n","import { createContext, type ReactNode, type RefObject, useCallback, useContext, useMemo, useState } from 'react';\nimport type { CuadraChatHandle } from './CuadraChat';\n\n/**\n * Context for accessing CuadraChat controls from any child component\n * \n * This provides a React-idiomatic way to control the chat component\n * from anywhere in your component tree without prop drilling.\n * \n * @example Using in a parent component\n * ```tsx\n * import { CuadraChat, CuadraChatProvider, useCuadraChat } from '@cuadra-ai/uikit';\n * \n * function App() {\n * return (\n * <CuadraChatProvider>\n * <CuadraChat connection={{ baseUrl: '...' }} />\n * <MyCustomControls />\n * </CuadraChatProvider>\n * );\n * }\n * \n * function MyCustomControls() {\n * const { controls } = useCuadraChat();\n * \n * return (\n * <button onClick={() => controls?.sendMessage('Hello!')}>\n * Send Hello\n * </button>\n * );\n * }\n * ```\n */\nexport interface CuadraChatContextValue {\n /**\n * Reference to the CuadraChat controls\n * May be null before the CuadraChat component mounts\n */\n controls: CuadraChatHandle | null;\n \n /**\n * Internal: Reference setter for CuadraChat to register itself\n * @internal\n */\n registerControls: (controls: CuadraChatHandle) => void;\n \n /**\n * Internal: Reference clearer for CuadraChat to unregister on unmount\n * @internal\n */\n unregisterControls: () => void;\n}\n\n/**\n * Context for CuadraChat controls\n * \n * Default value is null - use CuadraChatProvider to provide an actual value.\n * Consumers can check if context is null to detect missing provider.\n */\nexport const CuadraChatContext = createContext<CuadraChatContextValue | null>(null);\n\nCuadraChatContext.displayName = 'CuadraChatContext';\n\n/**\n * Hook to access CuadraChat controls from any child component\n * \n * @returns Object containing `controls` (CuadraChatHandle | null) and `isReady` boolean\n * @throws Error if used outside of CuadraChatProvider\n * \n * @example\n * ```tsx\n * function MyComponent() {\n * const { controls, isReady } = useCuadraChat();\n * \n * if (!isReady) {\n * return <span>Chat is loading...</span>;\n * }\n * \n * return (\n * <button onClick={() => controls.sendMessage('Hi!')}>\n * Send Message\n * </button>\n * );\n * }\n * ```\n */\nexport function useCuadraChat(): { controls: CuadraChatHandle | null; isReady: boolean } {\n const context = useContext(CuadraChatContext);\n \n if (context === null) {\n throw new Error(\n 'useCuadraChat must be used within a CuadraChatProvider. ' +\n 'Wrap your component tree with <CuadraChatProvider>...</CuadraChatProvider>'\n );\n }\n \n return {\n controls: context.controls,\n isReady: context.controls !== null,\n };\n}\n\n/**\n * Hook to optionally access CuadraChat controls\n * \n * Unlike useCuadraChat, this doesn't throw if the provider is missing.\n * Returns null if CuadraChatProvider is not present.\n * \n * @returns Object with controls and isReady, or null if no provider\n * \n * @example\n * ```tsx\n * function OptionalChatControl() {\n * const chat = useCuadraChatOptional();\n * \n * if (!chat) {\n * // No CuadraChatProvider in tree - gracefully handle\n * return null;\n * }\n * \n * return chat.isReady ? (\n * <button onClick={() => chat.controls?.sendMessage('Hi!')}>Send</button>\n * ) : null;\n * }\n * ```\n */\nexport function useCuadraChatOptional(): { controls: CuadraChatHandle | null; isReady: boolean } | null {\n const context = useContext(CuadraChatContext);\n \n if (context === null) {\n return null;\n }\n \n return {\n controls: context.controls,\n isReady: context.controls !== null,\n };\n}\n\n/**\n * Provider component for CuadraChat context\n * \n * Wrap your component tree with this to enable useCuadraChat hook\n * access from any descendant component.\n * \n * @example\n * ```tsx\n * function App() {\n * return (\n * <CuadraChatProvider>\n * <Header />\n * <CuadraChat connection={{ baseUrl: '...' }} />\n * <Sidebar />\n * </CuadraChatProvider>\n * );\n * }\n * ```\n */\nexport interface CuadraChatProviderProps {\n children: ReactNode;\n}\n\nexport function CuadraChatProvider({ children }: CuadraChatProviderProps) {\n const [controls, setControls] = useState<CuadraChatHandle | null>(null);\n \n const registerControls = useCallback((newControls: CuadraChatHandle) => {\n setControls(newControls);\n }, []);\n \n const unregisterControls = useCallback(() => {\n setControls(null);\n }, []);\n \n const value = useMemo<CuadraChatContextValue>(() => ({\n controls,\n registerControls,\n unregisterControls,\n }), [controls, registerControls, unregisterControls]);\n \n return (\n <CuadraChatContext.Provider value={value}>\n {children}\n </CuadraChatContext.Provider>\n );\n}\n\n/**\n * Helper to create a ref object that syncs with context\n * Used internally by CuadraChat to register itself with the context\n * @internal\n */\nexport function createContextualRef(\n registerControls: (controls: CuadraChatHandle) => void,\n unregisterControls: () => void\n): RefObject<CuadraChatHandle | null> {\n let currentValue: CuadraChatHandle | null = null;\n \n return {\n get current() {\n return currentValue;\n },\n set current(value: CuadraChatHandle | null) {\n currentValue = value;\n if (value) {\n registerControls(value);\n } else {\n unregisterControls();\n }\n },\n };\n}\n","/**\n * Props type definitions for CuadraChat component\n * \n * This module provides both the new grouped props interface (V2)\n * and backwards compatibility with the legacy flat props interface.\n */\n\nimport type { ReactNode } from 'react';\nimport type { CuadraTheme } from './theme';\n\n// ============================================\n// V2 PROPS - NEW GROUPED INTERFACE\n// ============================================\n\n/**\n * Connection configuration for the Cuadra API\n * \n * Either `baseUrl` (direct API access) or `proxyUrl` (backend-handled auth)\n * should be provided.\n * \n * @example Direct API access\n * ```ts\n * connection: {\n * baseUrl: 'https://api.cuadra.ai',\n * sessionToken: 'stytch_session_xxx',\n * }\n * ```\n * \n * @example Proxy mode (recommended for production)\n * ```ts\n * connection: {\n * proxyUrl: '/api/chat',\n * }\n * ```\n */\nexport interface CuadraConnectionConfig {\n /**\n * Cuadra API base URL for direct API access\n * @example 'https://api.cuadra.ai'\n */\n baseUrl?: string;\n\n /**\n * Proxy URL for backend-handled authentication\n * When set, requests go through your backend which adds auth headers\n * @example '/api/chat' or 'https://your-backend.com/api/chat'\n */\n proxyUrl?: string;\n\n /**\n * Bearer token for direct API authentication\n * Only needed when using `baseUrl` (not with `proxyUrl`)\n */\n sessionToken?: string | null;\n}\n\n/**\n * Chat behavior and model configuration\n */\nexport interface CuadraChatConfig {\n /**\n * Chat mode\n * - `singleChat`: Single conversation, no sidebar, simple UX\n * - `multiChat`: Multiple conversations with thread list sidebar\n * @default 'multiChat'\n */\n mode?: 'singleChat' | 'multiChat';\n\n /**\n * Model selection mode\n * - `fixed`: Use the specified `modelId`\n * - `selector`: Show dropdown to choose from available models\n * @default 'fixed'\n */\n modelMode?: 'fixed' | 'selector';\n\n /**\n * Model ID from your Cuadra Dashboard\n * Required when `modelMode` is `'fixed'`\n * @see https://dashboard.cuadra.ai\n */\n modelId?: string;\n\n /**\n * System prompt to prepend to all conversations\n * Use for setting assistant behavior, persona, or context\n */\n systemPrompt?: string;\n\n /**\n * Create ephemeral chats that auto-delete when browser closes\n * Useful for demo/preview scenarios\n * @default false\n */\n ephemeral?: boolean;\n\n /**\n * Enable reasoning/thinking output for supported models\n * Models like o1, o3 can show their \"thinking\" process\n * @default false\n */\n enableReasoning?: boolean;\n\n /**\n * Enable file attachments in the chat input\n * @default false\n */\n enableAttachments?: boolean;\n\n /**\n * Whether the current model supports vision/image inputs.\n * When false, image attachments will be rejected with a clear error message.\n * Only relevant when `enableAttachments` is true.\n * @default true\n */\n supportsVision?: boolean;\n\n /**\n * Load an existing thread on mount (multiChat mode only)\n * Useful for deep-linking to specific conversations\n */\n initialThreadId?: string;\n}\n\n/**\n * Suggestion shown in the welcome screen\n */\nexport interface CuadraSuggestion {\n /**\n * Suggestion prompt text shown as a button\n */\n prompt: string;\n\n /**\n * Pre-made response (optional)\n * If provided, the response appears instantly with simulated streaming\n * instead of making an API call. Great for demos and onboarding.\n */\n response?: string;\n\n /**\n * Delay before showing the pre-made response in milliseconds\n * Overrides the global `preMade.responseDelay` setting\n */\n responseDelay?: number;\n}\n\n/**\n * UI customization options\n */\nexport interface CuadraUIConfig {\n /**\n * Color theme mode\n * - `light`: Light theme\n * - `dark`: Dark theme\n * - `system`: Follow system preference\n * @default 'system'\n */\n theme?: 'light' | 'dark' | 'system';\n\n /**\n * Custom theme colors to override defaults\n * @see CuadraTheme for available color tokens\n */\n customTheme?: CuadraTheme;\n\n /**\n * Show the theme toggle button in the header\n * @default true\n */\n showThemeToggle?: boolean;\n\n /**\n * Language/locale for i18n\n * @example 'en', 'es', 'fr', 'de', 'pt'\n */\n language?: string;\n\n /**\n * Welcome screen title\n * @default \"Hi, how can I help you today?\"\n */\n welcomeTitle?: string;\n\n /**\n * Welcome screen subtitle\n * @default \"Start exploring our platform and discover what you can build.\"\n */\n welcomeSubtitle?: string;\n\n /**\n * Suggestion buttons shown in the welcome screen\n */\n suggestions?: CuadraSuggestion[];\n\n /**\n * Placeholder text for the chat input field\n * @default \"Type your message...\"\n */\n inputPlaceholder?: string;\n\n /**\n * Extra top padding for the thread viewport\n * Useful when embedding under a fixed header\n * @example '1rem', '2rem', '64px'\n */\n extraTopPadding?: string;\n\n /**\n * Hide the decorative border around the chat container\n * Useful for mobile apps or when embedding in a full-screen context\n * @default false\n */\n borderless?: boolean;\n\n /**\n * Apply safe area insets for mobile devices (notch, home indicator)\n * When true, adds padding to header, sidebar, and input areas\n * @default false\n */\n safeArea?: boolean;\n\n /**\n * Organization name to display in the sidebar header\n * Used for workspace branding in mobile apps\n */\n orgName?: string;\n\n /**\n * Primary color for branding (hex format)\n * Overrides the default primary color CSS variable\n * @example '#3B82F6'\n */\n primaryColor?: string;\n}\n\n/**\n * Pre-made response configuration\n * Controls timing for simulated streaming of pre-made responses\n */\nexport interface CuadraPreMadeConfig {\n /**\n * Default delay before pre-made responses start streaming\n * Simulates \"thinking\" time\n * @default 1000\n */\n responseDelay?: number;\n\n /**\n * Speed of simulated streaming in milliseconds per word\n * Lower = faster streaming\n * @default 50\n */\n streamingSpeed?: number;\n}\n\n/**\n * Initial state for the chat component\n */\nexport interface CuadraInitialState {\n /**\n * Message to send automatically when chat loads\n * Triggers a real API call\n */\n message?: string;\n\n /**\n * Pre-made Q&A to display when chat loads\n * Shows instantly with simulated streaming, no API call\n * Great for demos and onboarding\n */\n preMadeQA?: {\n question: string;\n answer: string;\n };\n}\n\n/**\n * Request interceptors for advanced control\n */\nexport interface CuadraInterceptors {\n /**\n * Called before each outgoing API request\n * Return `true` to allow the request, `false` to block it\n * Can be async (e.g., to show a login modal first)\n * \n * @example Block requests until user is logged in\n * ```ts\n * onBeforeRequest: async () => {\n * if (!isLoggedIn) {\n * showLoginModal();\n * return false; // Block the request\n * }\n * return true;\n * }\n * ```\n */\n onBeforeRequest?: () => boolean | Promise<boolean>;\n\n /**\n * Response to show when `onBeforeRequest` returns false\n * Instead of showing an error, this message will be streamed as a response\n * Can be a string or function returning a string\n * \n * @example\n * ```ts\n * mockResponseOnBlocked: \"Please complete setup before chatting!\"\n * ```\n */\n mockResponseOnBlocked?: string | (() => string);\n}\n\n/**\n * Event callbacks for monitoring and integration\n */\nexport interface CuadraCallbacks {\n /**\n * Called when a new chat is created\n * @param chatId - The server-assigned chat ID\n */\n onChatCreated?: (chatId: string) => void;\n\n /**\n * Called when an error occurs\n * Use for error tracking and user notification\n */\n onError?: (error: Error) => void;\n\n /**\n * Called when a user message is sent\n * Useful for analytics or rate limiting\n */\n onUserMessage?: () => void;\n\n /**\n * Called when the selected model changes (selector mode)\n * @param modelId - The newly selected model ID\n */\n onModelChange?: (modelId: string) => void;\n\n /**\n * Called when a local thread ID is mapped to a server chat ID\n * Useful for URL synchronization and deep linking\n */\n onThreadIdUpdate?: (oldId: string, newId: string) => void;\n\n /**\n * Called when the logout button is clicked\n * Use to trigger your app's logout flow\n */\n onLogout?: () => void;\n}\n\n/**\n * CuadraChat props with grouped configuration\n * \n * This is the recommended props format for new implementations.\n * It provides better discoverability and organization of options.\n * \n * @example Basic usage\n * ```tsx\n * <CuadraChat\n * connection={{ baseUrl: 'https://api.cuadra.ai', sessionToken: 'xxx' }}\n * chat={{ modelId: 'model-123' }}\n * />\n * ```\n * \n * @example Full configuration\n * ```tsx\n * <CuadraChat\n * connection={{ proxyUrl: '/api/chat' }}\n * chat={{\n * mode: 'multiChat',\n * modelMode: 'fixed',\n * modelId: 'model-123',\n * enableReasoning: true,\n * }}\n * ui={{\n * theme: 'dark',\n * welcomeTitle: 'Hello!',\n * suggestions: [{ prompt: 'What can you do?' }],\n * }}\n * callbacks={{\n * onChatCreated: (id) => console.log('Chat created:', id),\n * onError: (err) => console.error(err),\n * }}\n * className=\"my-chat\"\n * />\n * ```\n */\nexport interface CuadraChatPropsV2 {\n /**\n * API connection configuration\n * Either `baseUrl` or `proxyUrl` is required\n */\n connection: CuadraConnectionConfig;\n\n /**\n * Chat behavior configuration\n */\n chat?: CuadraChatConfig;\n\n /**\n * UI customization options\n */\n ui?: CuadraUIConfig;\n\n /**\n * Pre-made response timing configuration\n */\n preMade?: CuadraPreMadeConfig;\n\n /**\n * Initial state for auto-starting chat\n */\n initial?: CuadraInitialState;\n\n /**\n * Request interceptors\n */\n interceptors?: CuadraInterceptors;\n\n /**\n * Event callbacks\n */\n callbacks?: CuadraCallbacks;\n\n /**\n * CSS class name for the container element\n */\n className?: string;\n\n /**\n * Custom error fallback UI\n * Can be a ReactNode or render function with error and reset\n */\n errorFallback?: ReactNode | ((error: Error, reset: () => void) => ReactNode);\n}\n\n// ============================================\n// LEGACY PROPS - BACKWARDS COMPATIBILITY\n// ============================================\n\n/**\n * Legacy flat props interface\n * @deprecated Use CuadraChatPropsV2 with grouped props instead\n * \n * @example Migration\n * ```tsx\n * // Before (legacy)\n * <CuadraChat\n * baseUrl=\"https://api.cuadra.ai\"\n * sessionToken=\"xxx\"\n * modelId=\"model-123\"\n * welcomeTitle=\"Hi!\"\n * />\n * \n * // After (recommended)\n * <CuadraChat\n * connection={{ baseUrl: 'https://api.cuadra.ai', sessionToken: 'xxx' }}\n * chat={{ modelId: 'model-123' }}\n * ui={{ welcomeTitle: 'Hi!' }}\n * />\n * ```\n */\nexport interface CuadraChatPropsLegacy {\n /** @deprecated Use `connection.baseUrl` */\n baseUrl?: string;\n /** @deprecated Use `connection.proxyUrl` */\n proxyUrl?: string;\n /** @deprecated Use `connection.sessionToken` */\n sessionToken?: string | null;\n /** @deprecated Use `chat.mode` */\n mode?: 'singleChat' | 'multiChat';\n /** @deprecated Use `chat.modelMode` */\n modelMode?: 'fixed' | 'selector';\n /** @deprecated Use `chat.modelId` */\n modelId?: string;\n /** @deprecated Use `chat.systemPrompt` */\n systemPrompt?: string;\n /** @deprecated Use `chat.ephemeral` */\n ephemeral?: boolean;\n /** @deprecated Use `chat.enableReasoning` */\n enableReasoning?: boolean;\n /** @deprecated Use `chat.enableAttachments` */\n enableAttachments?: boolean;\n /** @deprecated Use `chat.supportsVision` */\n supportsVision?: boolean;\n /** @deprecated Use `chat.initialThreadId` */\n initialThreadId?: string;\n /** @deprecated Use `ui.theme` */\n theme?: 'light' | 'dark' | 'system';\n /** @deprecated Use `ui.customTheme` */\n customTheme?: CuadraTheme;\n /** @deprecated Use `ui.showThemeToggle` */\n showThemeToggle?: boolean;\n /** @deprecated Use `ui.language` */\n language?: string;\n /** @deprecated Use `ui.welcomeTitle` */\n welcomeTitle?: string;\n /** @deprecated Use `ui.welcomeSubtitle` */\n welcomeSubtitle?: string;\n /** @deprecated Use `ui.suggestions` */\n suggestions?: CuadraSuggestion[];\n /** @deprecated Use `ui.inputPlaceholder` */\n inputPlaceholder?: string;\n /** @deprecated Use `ui.extraTopPadding` */\n extraTopPadding?: string;\n /** @deprecated Use `preMade.responseDelay` */\n preMadeResponseDelay?: number;\n /** @deprecated Use `preMade.streamingSpeed` */\n streamingSpeed?: number;\n /** @deprecated Use `initial.message` */\n initialMessage?: string;\n /** @deprecated Use `initial.preMadeQA` */\n initialPreMadeQA?: { question: string; answer: string };\n /** @deprecated Use `interceptors.onBeforeRequest` */\n onBeforeRequest?: () => boolean | Promise<boolean>;\n /** @deprecated Use `interceptors.mockResponseOnBlocked` */\n mockResponseOnBlocked?: string | (() => string);\n /** @deprecated Use `callbacks.onChatCreated` */\n onChatCreated?: (chatId: string) => void;\n /** @deprecated Use `callbacks.onError` */\n onError?: (error: Error) => void;\n /** @deprecated Use `callbacks.onUserMessage` */\n onUserMessage?: () => void;\n /** @deprecated Use `callbacks.onModelChange` */\n onModelChange?: (modelId: string) => void;\n /** @deprecated Use `callbacks.onThreadIdUpdate` */\n onThreadIdUpdate?: (oldId: string, newId: string) => void;\n /** @deprecated Use `callbacks.onLogout` */\n onLogout?: () => void;\n /** Keep at top level */\n className?: string;\n /** Keep at top level */\n errorFallback?: ReactNode | ((error: Error, reset: () => void) => ReactNode);\n}\n\n// ============================================\n// TYPE UTILITIES\n// ============================================\n\n/**\n * Union type supporting both V2 (grouped) and legacy (flat) props\n * The component will automatically detect which format is used\n */\nexport type CuadraChatProps = CuadraChatPropsV2 | CuadraChatPropsLegacy;\n\n/**\n * Type guard to detect V2 props format\n * V2 props have a `connection` object at the top level\n * \n * @example\n * ```ts\n * if (isV2Props(props)) {\n * // TypeScript knows props is CuadraChatPropsV2\n * const { baseUrl } = props.connection;\n * }\n * ```\n */\nexport function isV2Props(props: CuadraChatProps): props is CuadraChatPropsV2 {\n return 'connection' in props && typeof props.connection === 'object';\n}\n\n/**\n * Normalized internal config used by CuadraChat components\n * Always uses the V2 structure internally\n */\nexport interface NormalizedCuadraChatConfig {\n connection: Required<Pick<CuadraConnectionConfig, 'baseUrl' | 'proxyUrl'>> & {\n sessionToken: string | null;\n };\n chat: Required<Omit<CuadraChatConfig, 'initialThreadId' | 'supportsVision'>> & {\n initialThreadId?: string;\n supportsVision?: boolean;\n };\n ui: Required<Omit<CuadraUIConfig, 'customTheme' | 'language' | 'suggestions' | 'extraTopPadding' | 'borderless' | 'safeArea' | 'orgName' | 'primaryColor'>> & {\n customTheme?: CuadraTheme;\n language?: string;\n suggestions?: CuadraSuggestion[];\n extraTopPadding?: string;\n borderless?: boolean;\n safeArea?: boolean;\n orgName?: string;\n primaryColor?: string;\n };\n preMade: Required<CuadraPreMadeConfig>;\n initial: CuadraInitialState;\n interceptors: CuadraInterceptors;\n callbacks: CuadraCallbacks;\n className?: string;\n errorFallback?: ReactNode | ((error: Error, reset: () => void) => ReactNode);\n}\n\n/**\n * Normalize CuadraChat props to internal config format\n * Handles both V2 (grouped) and legacy (flat) props\n * \n * @param props - Either V2 or legacy props\n * @returns Normalized configuration with defaults applied\n */\nexport function normalizeCuadraChatProps(props: CuadraChatProps): NormalizedCuadraChatConfig {\n if (isV2Props(props)) {\n return normalizeV2Props(props);\n }\n return normalizeLegacyProps(props);\n}\n\n/**\n * Normalize V2 props with defaults\n */\nfunction normalizeV2Props(props: CuadraChatPropsV2): NormalizedCuadraChatConfig {\n return {\n connection: {\n baseUrl: props.connection.baseUrl || '',\n proxyUrl: props.connection.proxyUrl || '',\n sessionToken: props.connection.sessionToken ?? null,\n },\n chat: {\n mode: props.chat?.mode ?? 'multiChat',\n modelMode: props.chat?.modelMode ?? 'fixed',\n modelId: props.chat?.modelId ?? '',\n systemPrompt: props.chat?.systemPrompt ?? '',\n ephemeral: props.chat?.ephemeral ?? false,\n enableReasoning: props.chat?.enableReasoning ?? false,\n enableAttachments: props.chat?.enableAttachments ?? false,\n supportsVision: props.chat?.supportsVision,\n initialThreadId: props.chat?.initialThreadId,\n },\n ui: {\n theme: props.ui?.theme ?? 'system',\n customTheme: props.ui?.customTheme,\n showThemeToggle: props.ui?.showThemeToggle ?? true,\n language: props.ui?.language,\n welcomeTitle: props.ui?.welcomeTitle ?? 'Hi, how can I help you today?',\n welcomeSubtitle: props.ui?.welcomeSubtitle ?? 'Start exploring our platform and discover what you can build.',\n suggestions: props.ui?.suggestions,\n inputPlaceholder: props.ui?.inputPlaceholder ?? 'Type your message...',\n extraTopPadding: props.ui?.extraTopPadding,\n borderless: props.ui?.borderless,\n safeArea: props.ui?.safeArea,\n orgName: props.ui?.orgName,\n primaryColor: props.ui?.primaryColor,\n },\n preMade: {\n responseDelay: props.preMade?.responseDelay ?? 1000,\n streamingSpeed: props.preMade?.streamingSpeed ?? 50,\n },\n initial: {\n message: props.initial?.message,\n preMadeQA: props.initial?.preMadeQA,\n },\n interceptors: {\n onBeforeRequest: props.interceptors?.onBeforeRequest,\n mockResponseOnBlocked: props.interceptors?.mockResponseOnBlocked,\n },\n callbacks: {\n onChatCreated: props.callbacks?.onChatCreated,\n onError: props.callbacks?.onError,\n onUserMessage: props.callbacks?.onUserMessage,\n onModelChange: props.callbacks?.onModelChange,\n onThreadIdUpdate: props.callbacks?.onThreadIdUpdate,\n onLogout: props.callbacks?.onLogout,\n },\n className: props.className,\n errorFallback: props.errorFallback,\n };\n}\n\n/**\n * Normalize legacy flat props to internal config format\n */\nfunction normalizeLegacyProps(props: CuadraChatPropsLegacy): NormalizedCuadraChatConfig {\n // Log deprecation warning in development\n if (typeof process !== 'undefined' && process.env?.NODE_ENV === 'development') {\n // eslint-disable-next-line no-console\n console.warn(\n '[CuadraUIKit] Using legacy flat props format. Consider migrating to grouped props for better organization.\\n' +\n 'See: https://docs.cuadra.ai/guides/uikit-migration'\n );\n }\n\n return {\n connection: {\n baseUrl: props.baseUrl || '',\n proxyUrl: props.proxyUrl || '',\n sessionToken: props.sessionToken ?? null,\n },\n chat: {\n mode: props.mode ?? 'multiChat',\n modelMode: props.modelMode ?? 'fixed',\n modelId: props.modelId ?? '',\n systemPrompt: props.systemPrompt ?? '',\n ephemeral: props.ephemeral ?? false,\n enableReasoning: props.enableReasoning ?? false,\n enableAttachments: props.enableAttachments ?? false,\n supportsVision: props.supportsVision,\n initialThreadId: props.initialThreadId,\n },\n ui: {\n theme: props.theme ?? 'system',\n customTheme: props.customTheme,\n showThemeToggle: props.showThemeToggle ?? true,\n language: props.language,\n welcomeTitle: props.welcomeTitle ?? 'Hi, how can I help you today?',\n welcomeSubtitle: props.welcomeSubtitle ?? 'Start exploring our platform and discover what you can build.',\n suggestions: props.suggestions,\n inputPlaceholder: props.inputPlaceholder ?? 'Type your message...',\n extraTopPadding: props.extraTopPadding,\n borderless: false, // Legacy props don't support borderless\n safeArea: false, // Legacy props don't support safeArea\n orgName: undefined, // Legacy props don't support orgName\n primaryColor: undefined, // Legacy props don't support primaryColor\n },\n preMade: {\n responseDelay: props.preMadeResponseDelay ?? 1000,\n streamingSpeed: props.streamingSpeed ?? 50,\n },\n initial: {\n message: props.initialMessage,\n preMadeQA: props.initialPreMadeQA,\n },\n interceptors: {\n onBeforeRequest: props.onBeforeRequest,\n mockResponseOnBlocked: props.mockResponseOnBlocked,\n },\n callbacks: {\n onChatCreated: props.onChatCreated,\n onError: props.onError,\n onUserMessage: props.onUserMessage,\n onModelChange: props.onModelChange,\n onThreadIdUpdate: props.onThreadIdUpdate,\n onLogout: props.onLogout,\n },\n className: props.className,\n errorFallback: props.errorFallback,\n };\n}\n","import { forwardRef, useContext, useEffect, useImperativeHandle, useMemo, useRef } from 'react';\nimport { type CuadraUIKitConfig, WidgetContent, type WidgetContentHandle } from './WidgetContent';\nimport { applyTheme } from '../utils/applyTheme';\nimport { CuadraErrorBoundary } from './ErrorBoundary';\nimport { CuadraChatContext } from './CuadraChatContext';\nimport { type CuadraChatProps, normalizeCuadraChatProps, type NormalizedCuadraChatConfig } from '../types/props';\n\n/**\n * Handle for controlling CuadraChat externally\n */\nexport interface CuadraChatHandle {\n /** Send a user message programmatically (triggers normal API call) */\n sendMessage: (message: string) => void;\n /** \n * Send a pre-made Q&A pair (instant, simulated streaming, no API call) \n * User can continue chatting normally after.\n */\n sendPreMadeQA: (question: string, answer: string, options?: { delay?: number; streamingSpeed?: number }) => void;\n /** Clear the current chat and start fresh */\n clearChat: () => void;\n}\n\n// Re-export the props types for convenience\nexport type { \n CuadraChatProps, \n CuadraChatPropsV2, \n CuadraChatPropsLegacy,\n CuadraConnectionConfig,\n CuadraChatConfig,\n CuadraUIConfig,\n CuadraCallbacks,\n CuadraInterceptors,\n CuadraPreMadeConfig,\n CuadraInitialState,\n CuadraSuggestion,\n} from '../types/props';\n\n// Module-level flag so the deprecation warning only logs once per session\nlet deprecationWarned = false;\n\n/**\n * Internal component that receives normalized config\n */\ninterface CuadraChatInnerProps {\n config: NormalizedCuadraChatConfig;\n}\n\nconst CuadraChatInner = forwardRef<CuadraChatHandle, CuadraChatInnerProps>(\n function CuadraChatInner({ config }, ref) {\n const widgetRef = useRef<WidgetContentHandle>(null);\n const chatContext = useContext(CuadraChatContext);\n\n // Apply custom theme if provided\n useEffect(() => {\n if (config.ui.customTheme) {\n applyTheme(config.ui.customTheme);\n }\n }, [config.ui.customTheme]);\n\n // Apply primary color override from branding\n useEffect(() => {\n if (config.ui.primaryColor && typeof document !== 'undefined') {\n const root = document.documentElement;\n // Convert hex to HSL for CSS variable compatibility\n const hexToHsl = (hex: string): string => {\n // Remove # if present\n const cleanHex = hex.replace('#', '');\n const r = parseInt(cleanHex.slice(0, 2), 16) / 255;\n const g = parseInt(cleanHex.slice(2, 4), 16) / 255;\n const b = parseInt(cleanHex.slice(4, 6), 16) / 255;\n\n const max = Math.max(r, g, b);\n const min = Math.min(r, g, b);\n let h = 0;\n let s = 0;\n const l = (max + min) / 2;\n\n if (max !== min) {\n const d = max - min;\n s = l > 0.5 ? d / (2 - max - min) : d / (max + min);\n switch (max) {\n case r: h = ((g - b) / d + (g < b ? 6 : 0)) / 6; break;\n case g: h = ((b - r) / d + 2) / 6; break;\n case b: h = ((r - g) / d + 4) / 6; break;\n }\n }\n\n return `${Math.round(h * 360)} ${Math.round(s * 100)}% ${Math.round(l * 100)}%`;\n };\n\n try {\n const hslValue = hexToHsl(config.ui.primaryColor);\n // --primary uses HSL components (for Tailwind: hsl(var(--primary)))\n root.style.setProperty('--primary', hslValue);\n // --cuadra-primary needs full color value (for inline styles)\n root.style.setProperty('--cuadra-primary', `hsl(${hslValue})`);\n } catch {\n // Invalid hex color, ignore\n }\n\n return () => {\n // Cleanup: remove the custom property on unmount\n root.style.removeProperty('--primary');\n root.style.removeProperty('--cuadra-primary');\n };\n }\n }, [config.ui.primaryColor]);\n\n // Create the controls object (memoized — methods delegate to widgetRef which is always current)\n const controls: CuadraChatHandle = useMemo(() => ({\n sendMessage: (message: string) => widgetRef.current?.sendMessage(message),\n sendPreMadeQA: (question: string, answer: string, options?: { delay?: number; streamingSpeed?: number }) => \n widgetRef.current?.sendPreMadeQA(question, answer, options),\n clearChat: () => widgetRef.current?.clearChat(),\n }), []);\n\n // Expose methods via ref\n useImperativeHandle(ref, () => controls, []);\n\n // Register with CuadraChatContext if available\n useEffect(() => {\n if (chatContext) {\n chatContext.registerControls(controls);\n return () => chatContext.unregisterControls();\n }\n }, [chatContext, controls]);\n\n // Also expose on window for non-React consumers (deprecated)\n useEffect(() => {\n if (typeof window !== 'undefined') {\n // Log deprecation warning once per session in development\n if (process.env.NODE_ENV === 'development' && !deprecationWarned) {\n deprecationWarned = true;\n // eslint-disable-next-line no-console\n console.warn(\n '[CuadraUIKit] window.__cuadraChatControls is deprecated. ' +\n 'Use CuadraChatProvider and useCuadraChat() hook instead. ' +\n 'See: https://docs.cuadra.ai/guides/uikit-migration'\n );\n }\n (window as Window & { __cuadraChatControls?: CuadraChatHandle }).__cuadraChatControls = controls;\n }\n return () => {\n if (typeof window !== 'undefined') {\n delete (window as Window & { __cuadraChatControls?: CuadraChatHandle }).__cuadraChatControls;\n }\n };\n }, [controls]);\n\n // Build WidgetContent config from normalized props\n const widgetConfig: CuadraUIKitConfig = {\n // Connection\n baseUrl: config.connection.baseUrl || undefined,\n proxyUrl: config.connection.proxyUrl || undefined,\n sessionToken: config.connection.sessionToken,\n \n // Chat behavior\n mode: config.chat.mode,\n modelMode: config.chat.modelMode,\n modelId: config.chat.modelId || undefined,\n systemPrompt: config.chat.systemPrompt || undefined,\n ephemeral: config.chat.ephemeral,\n enableReasoning: config.chat.enableReasoning,\n enableAttachments: config.chat.enableAttachments,\n supportsVision: config.chat.supportsVision,\n initialThreadId: config.chat.initialThreadId,\n \n // UI\n theme: config.ui.theme,\n showThemeToggle: config.ui.showThemeToggle,\n language: config.ui.language,\n welcomeTitle: config.ui.welcomeTitle,\n welcomeSubtitle: config.ui.welcomeSubtitle,\n suggestions: config.ui.suggestions,\n inputPlaceholder: config.ui.inputPlaceholder,\n extraTopPadding: config.ui.extraTopPadding,\n containerClass: config.className,\n borderless: config.ui.borderless,\n safeArea: config.ui.safeArea,\n orgName: config.ui.orgName,\n primaryColor: config.ui.primaryColor,\n \n // Pre-made\n preMadeResponseDelay: config.preMade.responseDelay,\n streamingSpeed: config.preMade.streamingSpeed,\n \n // Initial state\n initialMessage: config.initial.message,\n initialPreMadeQA: config.initial.preMadeQA,\n \n // Interceptors\n onBeforeRequest: config.interceptors.onBeforeRequest,\n mockResponseOnBlocked: config.interceptors.mockResponseOnBlocked,\n \n // Callbacks\n onChatCreated: config.callbacks.onChatCreated,\n onError: config.callbacks.onError,\n onUserMessage: config.callbacks.onUserMessage,\n onModelChange: config.callbacks.onModelChange,\n onThreadIdUpdate: config.callbacks.onThreadIdUpdate,\n onLogout: config.callbacks.onLogout,\n };\n\n return <WidgetContent ref={widgetRef} config={widgetConfig} />;\n }\n);\n\n/**\n * All-in-one Cuadra Chat component\n * \n * Supports both V2 grouped props (recommended) and legacy flat props.\n * \n * ## V2 Props (Recommended)\n * \n * ```tsx\n * <CuadraChat\n * connection={{\n * baseUrl: 'https://api.cuadra.ai',\n * sessionToken: 'your-token',\n * }}\n * chat={{\n * mode: 'multiChat',\n * modelId: 'your-model-id',\n * enableReasoning: true,\n * }}\n * ui={{\n * theme: 'dark',\n * welcomeTitle: 'Welcome!',\n * suggestions: [{ prompt: 'What can you do?' }],\n * }}\n * callbacks={{\n * onChatCreated: (id) => console.log('Chat:', id),\n * }}\n * />\n * ```\n * \n * ## Legacy Props (Still Supported)\n * \n * ```tsx\n * <CuadraChat\n * baseUrl=\"https://api.cuadra.ai\"\n * sessionToken=\"your-token\"\n * mode=\"multiChat\"\n * modelId=\"your-model-id\"\n * enableReasoning={true}\n * theme=\"dark\"\n * welcomeTitle=\"Welcome!\"\n * />\n * ```\n * \n * ## External Control\n * \n * Use `CuadraChatProvider` and `useCuadraChat()` hook for context-based control:\n * \n * ```tsx\n * <CuadraChatProvider>\n * <CuadraChat connection={{ baseUrl: '...' }} />\n * <MyControlsComponent /> {/* Can use useCuadraChat() *\\/}\n * </CuadraChatProvider>\n * ```\n * \n * Or use a ref for direct control:\n * \n * ```tsx\n * const chatRef = useRef<CuadraChatHandle>(null);\n * <CuadraChat ref={chatRef} connection={{ baseUrl: '...' }} />\n * chatRef.current?.sendMessage('Hello!');\n * ```\n */\nexport const CuadraChat = forwardRef<CuadraChatHandle, CuadraChatProps>(\n function CuadraChat(props, ref) {\n // Normalize props (handles both V2 and legacy formats)\n const config = normalizeCuadraChatProps(props);\n\n // Determine fallback - could be ReactNode or function\n const fallback = config.errorFallback;\n\n return (\n <CuadraErrorBoundary \n fallback={fallback}\n onError={(error, errorInfo) => {\n // Call user's onError callback if provided\n config.callbacks.onError?.(error);\n // Log in development\n if (process.env.NODE_ENV === 'development') {\n // eslint-disable-next-line no-console\n console.error('[CuadraChat] Error caught by boundary:', error, errorInfo);\n }\n }}\n >\n <CuadraChatInner ref={ref} config={config} />\n </CuadraErrorBoundary>\n );\n }\n);\n"],"names":["stylesInjected","injectionAttempted","autoInjectStyles","existingLink","existingStyle","testEl","stylesLoaded","injectStylesLink","link","currentUrl","cssUrl","alternatives","attemptIndex","tryNext","toKebabCase","string","toCamelCase","match","p1","p2","toPascalCase","camelCase","mergeClasses","classes","className","index","array","hasA11yProp","props","prop","defaultAttributes","Icon","forwardRef","color","size","strokeWidth","absoluteStrokeWidth","children","iconNode","rest","ref","createElement","tag","attrs","createLucideIcon","iconName","Component","__iconNode","ArrowUp","Brain","Check","ChevronDown","CircleAlert","Copy","EllipsisVertical","Eye","FileSpreadsheet","FileText","FileType","LoaderCircle","MessageSquare","Paperclip","Pencil","Plus","Trash2","Upload","X","INJECTION_PATTERNS","sanitizeSystemPrompt","prompt","sanitized","pattern","isValidUrl","url","parsed","validateBaseUrl","isLocalhostUrl","escapeHtml","text","htmlEntities","char","generateSecureUUID","bytes","hex","b","CuadraChatClient","baseUrl","sessionToken","isProxyMode","__publicField","token","endpoint","path","request","abortSignal","headers","body","response","errorData","chatId","params","expand","update","errorText","modelId","file","resourceType","resourceId","idempotencyKey","formData","parseSSEStream","stream","reader","decoder","buffer","done","value","lines","line","data","convertToCuadraMessages","messages","msg","content","textContent","c","convertFromCuadraMessage","promiseWithResolvers","resolve","reject","promise","res","rej","createMergeStream","list","sealed","controller","currentPull","handlePull","item","e","chunk","TextStreamControllerImpl","textDelta","createTextStream","readable","createTextStreamController","ToolCallStreamControllerImpl","_controller","hasArgsText","createToolCallStream","createToolCallStreamController","Counter","PathAppendEncoder","idx","PathAppendDecoder","idx2","PathMergeEncoder","counter","innerCounter","mapping","mappedIdx","customAlphabet","alphabet","defaultSize","id","i","generateId","AssistantStreamControllerImpl","state","callback","part","options","opt","toolName","toolCallId","parentId","createAssistantStream","localToRemoteIdMap","messageIdToChatIdMap","activeThreadRemoteId","resolveServerId","remoteId","mapped","createThreadListAdapter","client","onChatIdReceived","all","cursor","safety","items","chat","t","threadId","newTitle","actualId","_remoteId","_messages","title","error","rawText","m","p","localThreadId","serverChatId","threadListItem","useThreadListItem","isMain","React","history","converted","message","_message","adapters","jsx","RuntimeAdapterProvider","StreamingMetadataStore","messageId","existing","newSources","listener","streamingMetadataStore","preMadeResponseQueue","preMadeConversationHistory","queuePreMadeResponse","question","streamingSpeed","initialDelay","createChatModelAdapter","staticModelId","getModelId","systemPrompt","ephemeral","enableReasoning","onChatCreated","onUserMessage","onThreadIdUpdate","onBeforeRequest","mockResponseOnBlocked","resolveModelId","pendingPreMade","words","accumulatedText","speed","fakeChatId","messagesToSend","lastAssistantIndex","apiMessages","preMadeApiMessages","fileIds","lastUserMessage","attachments","attachment","uploaded","currentModelId","responseText","userMessages","userQuestion","pendingMock","accumulatedReasoning","sources","lastSourcesCount","hasAddedReasoningToTextDelay","streamMessageId","updateStoreIfNeeded","createYieldObject","chunkId","threadIdForStorage","chunkSources","source","s","chunkReasoning","delta","AttachmentValidationError","code","DEFAULT_ALLOWED_MIME_TYPES","DEFAULT_ALLOWED_EXTENSIONS","DEFAULT_MAX_FILE_SIZE","formatFileSize","k","sizes","getFileExtension","filename","mimeMatchesExtension","mimeType","extension","validExtensions","generateAttachmentId","createAttachmentAdapter","validation","supportsVision","onValidationError","onAttachmentAdded","onAttachmentRemoved","maxFileSizeBytes","allowedMimeTypes","allowedExtensions","validateContentType","acceptString","validateFile","ext","mime","prefix","type","dataURL","AttachmentErrorStore","l","attachmentErrorStore","CuadraRuntimeProvider","mode","_onModelChange","initialThreadId","onError","onChatsLoaded","enableAttachments","useMemo","selectedModelId","setSelectedModelId","useState","useEffect","handleChatCreated","useCallback","handleThreadIdUpdate","oldId","newId","modelAdapter","SingleChatProvider","MultiChatProvider","initialMessages","setInitialMessages","isLoadingThread","setIsLoadingThread","cancelled","loadThread","SingleChatRuntime","attachmentAdapter","localRuntime","useLocalRuntime","AssistantRuntimeProvider","modelAdapterOptions","_initialThreadId","_onError","threadListAdapter","handleThreadIdUpdateFromAdapter","onChatsLoadedRef","useRef","wrappedAdapter","result","modelIdRef","stableAdapterOptions","_modelId","runtimeHook","runtime","unstable_useRemoteThreadListRuntime","ccount","character","count","asciiAlpha","regexCheck","asciiAlphanumeric","asciiControl","markdownLineEnding","markdownLineEndingOrSpace","markdownSpace","unicodePunctuation","unicodeWhitespace","regex","check","escapeStringRegexp","convert","test","ok","castFactory","anyFactory","propertiesFactory","typeFactory","tests","checks","any","parameters","checkAsRecord","node","nodeAsRecord","key","testFunction","parent","looksLikeANode","empty","CONTINUE","EXIT","SKIP","visitParents","tree","visitor","reverse","is","step","factory","parents","name","visit","subresult","offset","grandparents","toResult","nodeAsParent","child","findAndReplace","ignored","pairs","toPairs","pairIndex","grandparent","siblings","handler","find","replace","start","change","nodes","position","matchObject","tupleOrList","tuple","toExpression","toFunction","escape","inConstruct","notInConstruct","gfmAutolinkLiteralFromMarkdown","transformGfmAutolinkLiterals","enterLiteralAutolink","enterLiteralAutolinkValue","exitLiteralAutolink","exitLiteralAutolinkEmail","exitLiteralAutolinkHttp","exitLiteralAutolinkWww","gfmAutolinkLiteralToMarkdown","findUrl","findEmail","_","protocol","domain","previous","isCorrectDomain","parts","splitUrl","atext","label","trailExec","trail","closingParenIndex","openingParens","closingParens","email","normalizeIdentifier","footnoteReference","footnoteReferencePeek","enterFootnoteCallString","enterFootnoteCall","enterFootnoteDefinitionLabelString","enterFootnoteDefinition","exitFootnoteCallString","exitFootnoteCall","exitFootnoteDefinitionLabelString","exitFootnoteDefinition","info","tracker","exit","subexit","gfmFootnoteFromMarkdown","gfmFootnoteToMarkdown","firstLineBlank","footnoteDefinition","mapAll","mapExceptFirst","blank","constructsWithoutStrikethrough","handleDelete","peekDelete","gfmStrikethroughFromMarkdown","enterStrikethrough","exitStrikethrough","gfmStrikethroughToMarkdown","defaultStringLength","markdownTable","table","settings","align","stringLength","alignments","cellMatrix","sizeMatrix","longestCellByColumn","mostCellsPerRow","rowIndex","row","columnIndex","cell","serialize","toAlignment","before","after","blockquote","map","patternInScope","stack","listInScope","none","hardBreak","_1","longestStreak","substring","expected","max","formatCodeAsIndented","checkFence","marker","raw","suffix","sequence","checkQuote","definition","quote","checkEmphasis","encodeCharacterReference","classifyCharacter","encodeInfo","outside","inside","outsideKind","insideKind","emphasis","emphasisPeek","between","betweenHead","open","betweenTail","close","testOrVisitor","visitorOrReverse","maybeReverse","overload","emptyOptions","toString","includeImageAlt","includeHtml","one","values","formatHeadingAsSetext","literalWithBreak","heading","rank","html","htmlPeek","image","imagePeek","imageReference","imageReferencePeek","alt","reference","inlineCode","inlineCodePeek","expression","formatLinkAsAutolink","linkPeek","linkReference","linkReferencePeek","checkBullet","checkBulletOther","bullet","bulletOther","checkBulletOrdered","checkRule","bulletCurrent","useDifferentMarker","firstListItem","checkListItemIndent","style","listItem","listItemIndent","paragraph","phrasing","root","d","checkStrong","strong","strongPeek","checkRuleRepetition","repetition","thematicBreak","handle","gfmTableFromMarkdown","enterTable","enterCell","enterRow","exitCodeText","exitTable","$0","$1","gfmTableToMarkdown","padding","alignDelimiters","around","inlineCodeWithTable","handleTable","handleTableCell","handleTableRow","serializeData","handleTableAsData","handleTableRowAsData","matrix","defaultHandlers","gfmTaskListItemFromMarkdown","exitCheck","exitParagraphWithTaskListItem","gfmTaskListItemToMarkdown","listItemWithTaskListItem","head","firstParaghraph","sibling","checkable","checkbox","gfmFromMarkdown","gfmToMarkdown","splice","remove","end","chunkStart","hasOwnProperty","combineExtensions","extensions","syntaxExtension","hook","left","right","constructs","wwwPrefix","tokenizeWwwPrefix","tokenizeDomain","tokenizePath","tokenizeTrail","emailDomainDotTrail","tokenizeEmailDomainDotTrail","wwwAutolink","tokenizeWwwAutolink","previousWww","protocolAutolink","tokenizeProtocolAutolink","previousProtocol","emailAutolink","tokenizeEmailAutolink","previousEmail","gfmAutolinkLiteral","effects","nok","self","dot","gfmAtext","previousUnbalanced","emailDomain","emailDomainAfter","emailDomainDot","wwwStart","wwwAfter","seen","protocolStart","protocolPrefixInside","protocolSlashesInside","afterProtocol","protocolAfter","wwwPrefixInside","wwwPrefixAfter","underscoreInLastSegment","underscoreInLastLastSegment","domainInside","domainAfter","domainAtPunctuation","sizeOpen","sizeClose","pathInside","pathAtPunctuation","trailCharacterReferenceStart","trailBracketAfter","trailCharacterReferenceInside","events","resolveAll","context","called","factorySpace","limit","blankLine","tokenizeBlankLine","indent","tokenizeIndent","gfmFootnote","tokenizeDefinitionStart","tokenizeDefinitionContinuation","gfmFootnoteDefinitionEnd","tokenizeGfmFootnoteCall","tokenizePotentialGfmFootnoteCall","resolveToPotentialGfmFootnoteCall","defined","labelStart","call","replacement","callStart","callData","callEscape","identifier","labelAtMarker","labelInside","labelAfter","labelEscape","whitespaceAfter","afterPrefix","tail","gfmStrikethrough","single","tokenizer","tokenizeStrikethrough","resolveAllStrikethrough","strikethrough","nextEvents","insideSpan","more","EditMap","add","addImplementation","a","vecs","slice","element","editMap","at","gfmTableAlign","inDelimiterRow","event","alignIndex","gfmTable","tokenizeTable","resolveTable","sizeB","next","bodyRowStart","headRowBefore","headRowStart","headRowBreak","headDelimiterStart","headRowData","headRowEscape","headDelimiterBefore","headDelimiterValueBefore","headDelimiterCellBefore","headDelimiterNok","headDelimiterLeftAlignmentAfter","headDelimiterCellAfter","headDelimiterFiller","headDelimiterRightAlignmentAfter","bodyRowBreak","bodyRowData","bodyRowEscape","inFirstCellAwaitingPipe","rowKind","lastCell","afterHeadAwaitingFirstBodyRow","lastTableEnd","currentTable","currentBody","currentCell","flushTableEnd","flushCell","range","rowEnd","previousCell","groupName","valueName","getPoint","now","relatedStart","relatedEnd","valueToken","tableBody","exits","related","side","tasklistCheck","tokenizeTasklistCheck","gfmTaskListItem","spaceThenNonSpace","gfm","remarkGfm","micromarkExtensions","fromMarkdownExtensions","toMarkdownExtensions","cn","MarkdownTextImpl","MarkdownTextPrimitive","defaultComponents","MarkdownText","memo","CodeHeader","language","isCopied","copyToClipboard","useCopyToClipboard","onCopy","jsxs","CopyIcon","CheckIcon","copiedDuration","setIsCopied","memoizeMarkdownComponents","isCodeBlock","useIsMarkdownCodeBlock","Badge","variant","baseClasses","variantClasses","deduplicateSources","byFilename","SourceCitations","maxVisible","uniqueSources","visibleSources","hiddenCount","SourceBadge","getFileIcon","score","SimpleThread","welcomeTitle","welcomeSubtitle","extraTopPadding","suggestions","inputPlaceholder","preMadeResponseDelay","safeArea","composerBorder","models","onModelChange","showChatHeader","thread","useThreadRuntime","aui","useAui","isLoadingMessages","setIsLoadingMessages","currentThreadId","setCurrentThreadId","isProcessingPreMade","_setIsProcessingPreMade","attachmentError","useSyncExternalStore","timer","showCenteredWelcome","useThread","assistantRuntime","useAssistantRuntime","assistantRuntimeRef","prevModelIdRef","prev","sendPreMadeQAInternal","answer","delay","sendMessageInternal","handleSuggestionClick","suggestion","clearChatInternal","threadListRuntime","useImperativeHandle","runtimeStore","unsubscribe","isLoadingHistory","messageCount","ThreadPrimitive","ChatHeader","ModelSelector","ComposerPrimitive","Loader2","WelcomeScreen","UserMessage","AssistantMessage","AlertCircle","useAuiState","AttachmentPrimitive","ArrowUpIcon","subtitle","MessagePrimitive","isImage","imageContent","src","Reasoning","isOpen","setIsOpen","isStreaming","ThinkingIndicator","isInProgress","hasContent","hasCapturedRef","capturedMessageId","setCapturedMessageId","currentId","msgSources","currentSources","IconTooltip","selectedModel","handleModelSelect","Fragment","model","threadTitle","isDropdownOpen","setIsDropdownOpen","displayTitle","ThreadListDropdown","onClose","ThreadListPrimitive","DropdownThreadList","onSelect","fullHeight","threadIds","threads","newThreadId","DropdownThreadItem","threadItem","itemRuntime","useThreadListItemRuntime","currentThread","isActive","isHovered","setIsHovered","isRenaming","setIsRenaming","nextTitle","setNextTitle","busy","setBusy","handleRename","err","ThreadListItemPrimitive","MobileThreadActions","onRename","onDelete","TexturedCard","paddingX","paddingY","borderSizePx","contentStyle","CuadraWidgetContext","createContext","CuadraWidgetProvider","useInjectStyles","possiblePaths","loaded","WidgetContent","config","simpleThreadRef","initialMessageSentRef","proxyUrl","modelMode","initialMessage","initialPreMadeQA","initialTheme","actualBaseUrl","modelsData","setModelsData","modelsLoading","setModelsLoading","modelsError","setModelsError","chatsLoading","setChatsLoading","fetchModels","setCurrentModelId","handleModelChange","newModelId","firstModel","hasModel","resolvedSupportsVision","widgetContextValue","handleChatsLoaded","applyTheme","theme","_scope","applyThemeColors","lightVars","buildCSSVariables","darkVars","styleEl","CSS_VAR_MAP","getCSSVarName","colors","cssVarName","removeTheme","customStyle","darkStyle","DefaultErrorFallback","onReset","CuadraErrorBoundary","errorInfo","fallback","withErrorBoundary","WrappedComponent","errorBoundaryProps","WithErrorBoundary","CuadraChatContext","useCuadraChat","useContext","useCuadraChatOptional","CuadraChatProvider","controls","setControls","registerControls","newControls","unregisterControls","isV2Props","normalizeCuadraChatProps","normalizeV2Props","normalizeLegacyProps","deprecationWarned","CuadraChatInner","widgetRef","chatContext","hexToHsl","cleanHex","r","g","min","h","hslValue","widgetConfig","CuadraChat"],"mappings":"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","x_google_ignoreList":[1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,27,28,29,30,31,32,33,34,35,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60,61,62,63,64,65,66,67,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,84,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,101,102,103,104,105,106,107,108,109,110]}
|
|
1
|
+
{"version":3,"file":"index.cjs","sources":["../src/utils/autoInjectStyles.ts","../node_modules/lucide-react/dist/esm/shared/src/utils.js","../node_modules/lucide-react/dist/esm/defaultAttributes.js","../node_modules/lucide-react/dist/esm/Icon.js","../node_modules/lucide-react/dist/esm/createLucideIcon.js","../node_modules/lucide-react/dist/esm/icons/arrow-up.js","../node_modules/lucide-react/dist/esm/icons/brain.js","../node_modules/lucide-react/dist/esm/icons/check.js","../node_modules/lucide-react/dist/esm/icons/chevron-down.js","../node_modules/lucide-react/dist/esm/icons/circle-alert.js","../node_modules/lucide-react/dist/esm/icons/copy.js","../node_modules/lucide-react/dist/esm/icons/ellipsis-vertical.js","../node_modules/lucide-react/dist/esm/icons/eye.js","../node_modules/lucide-react/dist/esm/icons/file-spreadsheet.js","../node_modules/lucide-react/dist/esm/icons/file-text.js","../node_modules/lucide-react/dist/esm/icons/file-type.js","../node_modules/lucide-react/dist/esm/icons/loader-circle.js","../node_modules/lucide-react/dist/esm/icons/message-square.js","../node_modules/lucide-react/dist/esm/icons/paperclip.js","../node_modules/lucide-react/dist/esm/icons/pencil.js","../node_modules/lucide-react/dist/esm/icons/plus.js","../node_modules/lucide-react/dist/esm/icons/trash-2.js","../node_modules/lucide-react/dist/esm/icons/upload.js","../node_modules/lucide-react/dist/esm/icons/x.js","../src/utils/sanitize.ts","../src/lib/cuadraChatClient.ts","../src/adapters/messageConverter.ts","../node_modules/assistant-stream/dist/utils/promiseWithResolvers.js","../node_modules/assistant-stream/dist/core/utils/stream/merge.js","../node_modules/assistant-stream/dist/core/modules/text.js","../node_modules/assistant-stream/dist/core/modules/tool-call.js","../node_modules/assistant-stream/dist/core/utils/Counter.js","../node_modules/assistant-stream/dist/core/utils/stream/path-utils.js","../node_modules/nanoid/non-secure/index.js","../node_modules/assistant-stream/dist/core/utils/generateId.js","../node_modules/assistant-stream/dist/core/modules/assistant-stream.js","../src/adapters/threadListAdapter.tsx","../src/adapters/streamingMetadataStore.ts","../src/adapters/chatModelAdapter.ts","../src/adapters/attachmentAdapter.ts","../src/adapters/attachmentErrorStore.ts","../src/components/CuadraRuntimeProvider.tsx","../node_modules/ccount/index.js","../node_modules/micromark-util-character/index.js","../node_modules/mdast-util-find-and-replace/node_modules/escape-string-regexp/index.js","../node_modules/unist-util-is/lib/index.js","../node_modules/unist-util-visit-parents/lib/index.js","../node_modules/mdast-util-find-and-replace/lib/index.js","../node_modules/mdast-util-gfm-autolink-literal/lib/index.js","../node_modules/micromark-util-normalize-identifier/index.js","../node_modules/mdast-util-gfm-footnote/lib/index.js","../node_modules/mdast-util-gfm-strikethrough/lib/index.js","../node_modules/markdown-table/index.js","../node_modules/mdast-util-to-markdown/lib/handle/blockquote.js","../node_modules/mdast-util-to-markdown/lib/util/pattern-in-scope.js","../node_modules/mdast-util-to-markdown/lib/handle/break.js","../node_modules/longest-streak/index.js","../node_modules/mdast-util-to-markdown/lib/util/format-code-as-indented.js","../node_modules/mdast-util-to-markdown/lib/util/check-fence.js","../node_modules/mdast-util-to-markdown/lib/handle/code.js","../node_modules/mdast-util-to-markdown/lib/util/check-quote.js","../node_modules/mdast-util-to-markdown/lib/handle/definition.js","../node_modules/mdast-util-to-markdown/lib/util/check-emphasis.js","../node_modules/mdast-util-to-markdown/lib/util/encode-character-reference.js","../node_modules/micromark-util-classify-character/index.js","../node_modules/mdast-util-to-markdown/lib/util/encode-info.js","../node_modules/mdast-util-to-markdown/lib/handle/emphasis.js","../node_modules/unist-util-visit/lib/index.js","../node_modules/mdast-util-to-string/lib/index.js","../node_modules/mdast-util-to-markdown/lib/util/format-heading-as-setext.js","../node_modules/mdast-util-to-markdown/lib/handle/heading.js","../node_modules/mdast-util-to-markdown/lib/handle/html.js","../node_modules/mdast-util-to-markdown/lib/handle/image.js","../node_modules/mdast-util-to-markdown/lib/handle/image-reference.js","../node_modules/mdast-util-to-markdown/lib/handle/inline-code.js","../node_modules/mdast-util-to-markdown/lib/util/format-link-as-autolink.js","../node_modules/mdast-util-to-markdown/lib/handle/link.js","../node_modules/mdast-util-to-markdown/lib/handle/link-reference.js","../node_modules/mdast-util-to-markdown/lib/util/check-bullet.js","../node_modules/mdast-util-to-markdown/lib/util/check-bullet-other.js","../node_modules/mdast-util-to-markdown/lib/util/check-bullet-ordered.js","../node_modules/mdast-util-to-markdown/lib/util/check-rule.js","../node_modules/mdast-util-to-markdown/lib/handle/list.js","../node_modules/mdast-util-to-markdown/lib/util/check-list-item-indent.js","../node_modules/mdast-util-to-markdown/lib/handle/list-item.js","../node_modules/mdast-util-to-markdown/lib/handle/paragraph.js","../node_modules/mdast-util-phrasing/lib/index.js","../node_modules/mdast-util-to-markdown/lib/handle/root.js","../node_modules/mdast-util-to-markdown/lib/util/check-strong.js","../node_modules/mdast-util-to-markdown/lib/handle/strong.js","../node_modules/mdast-util-to-markdown/lib/handle/text.js","../node_modules/mdast-util-to-markdown/lib/util/check-rule-repetition.js","../node_modules/mdast-util-to-markdown/lib/handle/thematic-break.js","../node_modules/mdast-util-to-markdown/lib/handle/index.js","../node_modules/mdast-util-gfm-table/lib/index.js","../node_modules/mdast-util-gfm-task-list-item/lib/index.js","../node_modules/mdast-util-gfm/lib/index.js","../node_modules/micromark-util-chunked/index.js","../node_modules/micromark-util-combine-extensions/index.js","../node_modules/micromark-extension-gfm-autolink-literal/lib/syntax.js","../node_modules/micromark-util-resolve-all/index.js","../node_modules/micromark-factory-space/index.js","../node_modules/micromark-core-commonmark/lib/blank-line.js","../node_modules/micromark-extension-gfm-footnote/lib/syntax.js","../node_modules/micromark-extension-gfm-strikethrough/lib/syntax.js","../node_modules/micromark-extension-gfm-table/lib/edit-map.js","../node_modules/micromark-extension-gfm-table/lib/infer.js","../node_modules/micromark-extension-gfm-table/lib/syntax.js","../node_modules/micromark-extension-gfm-task-list-item/lib/syntax.js","../node_modules/micromark-extension-gfm/index.js","../node_modules/remark-gfm/lib/index.js","../src/utils/cn.ts","../src/components/MarkdownText.tsx","../src/components/ui/badge.tsx","../src/components/SourceCitations.tsx","../src/components/SimpleThread.tsx","../src/widget/components/TexturedCard.tsx","../src/widget/context/CuadraWidgetContext.tsx","../src/utils/useInjectStyles.ts","../src/components/WidgetContent.tsx","../src/utils/applyTheme.ts","../src/components/ErrorBoundary.tsx","../src/components/CuadraChatContext.tsx","../src/types/props.ts","../src/components/CuadraChat.tsx"],"sourcesContent":["/**\n * Automatically inject UIKit styles when the package is imported\n * This ensures styles are loaded without requiring clients to manually import them\n * \n * The CSS import in index.ts should handle most cases, but this provides a runtime\n * fallback that injects a link tag to load the CSS file if needed.\n */\n\nlet stylesInjected = false;\nlet injectionAttempted = false;\n\n/**\n * Automatically inject styles into the DOM\n * This runs once when the module is first imported\n */\nfunction autoInjectStyles(): void {\n // Only attempt injection once\n if (injectionAttempted) {\n return;\n }\n injectionAttempted = true;\n\n // Only inject if we're in a browser environment\n if (typeof document === 'undefined' || typeof window === 'undefined') {\n return;\n }\n\n // Check if styles are already loaded (via manual import or previous injection)\n const existingLink = document.querySelector('link[data-cuadra-uikit-styles]');\n const existingStyle = document.querySelector('style[data-cuadra-uikit-styles]');\n \n if (existingLink || existingStyle) {\n stylesInjected = true;\n return;\n }\n\n // Check if CSS classes are already available (styles might be loaded via CSS import)\n // This is a quick check to see if Tailwind classes work\n try {\n const testEl = document.createElement('div');\n testEl.className = 'cuadra-flex';\n testEl.style.visibility = 'hidden';\n testEl.style.position = 'absolute';\n testEl.style.pointerEvents = 'none';\n document.body.appendChild(testEl);\n const stylesLoaded = window.getComputedStyle(testEl).display === 'flex';\n document.body.removeChild(testEl);\n\n if (stylesLoaded) {\n stylesInjected = true;\n return;\n }\n } catch (_error) {\n // If check fails, proceed to inject styles\n }\n\n // Inject styles as a fallback\n // The CSS import in index.ts should handle most bundlers, but this ensures\n // styles work even if the CSS import isn't processed correctly\n injectStylesLink();\n}\n\n/**\n * Inject a link tag to load the CSS file\n * This is a fallback for cases where CSS import doesn't work\n */\nfunction injectStylesLink(): void {\n if (stylesInjected) {\n return;\n }\n\n const link = document.createElement('link');\n link.setAttribute('data-cuadra-uikit-styles', 'true');\n link.rel = 'stylesheet';\n link.type = 'text/css';\n \n // Try to determine the correct path to the CSS file\n // In most bundlers, we can use import.meta.url to get the current module's URL\n try {\n // Try relative path from the built JS file\n // This works when the package is installed via npm\n const currentUrl = new URL(import.meta.url);\n const cssUrl = new URL('../dist/uikit.css', currentUrl);\n link.href = cssUrl.href;\n } catch (_error) {\n // Fallback to a common path\n link.href = '/node_modules/@cuadra-ai/uikit/dist/uikit.css';\n }\n\n // Handle load success\n link.onload = () => {\n stylesInjected = true;\n };\n\n // Handle load error - try alternative paths\n link.onerror = () => {\n // Try alternative paths\n const alternatives = [\n '/node_modules/@cuadra-ai/uikit/dist/uikit.css',\n 'https://unpkg.com/@cuadra-ai/uikit@latest/dist/uikit.css',\n ];\n\n let attemptIndex = 0;\n const tryNext = () => {\n if (attemptIndex < alternatives.length) {\n link.href = alternatives[attemptIndex++];\n link.onerror = tryNext;\n link.onload = () => {\n stylesInjected = true;\n };\n }\n };\n\n tryNext();\n };\n\n document.head.appendChild(link);\n}\n\n// Auto-inject styles when this module is imported\n// This runs immediately when the package is imported\n// We use a small delay to ensure DOM is ready\nif (typeof document !== 'undefined' && typeof window !== 'undefined') {\n if (document.readyState === 'loading') {\n document.addEventListener('DOMContentLoaded', autoInjectStyles);\n } else {\n // DOM is already ready, inject immediately\n autoInjectStyles();\n }\n}\n\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nconst toKebabCase = (string) => string.replace(/([a-z0-9])([A-Z])/g, \"$1-$2\").toLowerCase();\nconst toCamelCase = (string) => string.replace(\n /^([A-Z])|[\\s-_]+(\\w)/g,\n (match, p1, p2) => p2 ? p2.toUpperCase() : p1.toLowerCase()\n);\nconst toPascalCase = (string) => {\n const camelCase = toCamelCase(string);\n return camelCase.charAt(0).toUpperCase() + camelCase.slice(1);\n};\nconst mergeClasses = (...classes) => classes.filter((className, index, array) => {\n return Boolean(className) && className.trim() !== \"\" && array.indexOf(className) === index;\n}).join(\" \").trim();\nconst hasA11yProp = (props) => {\n for (const prop in props) {\n if (prop.startsWith(\"aria-\") || prop === \"role\" || prop === \"title\") {\n return true;\n }\n }\n};\n\nexport { hasA11yProp, mergeClasses, toCamelCase, toKebabCase, toPascalCase };\n//# sourceMappingURL=utils.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nvar defaultAttributes = {\n xmlns: \"http://www.w3.org/2000/svg\",\n width: 24,\n height: 24,\n viewBox: \"0 0 24 24\",\n fill: \"none\",\n stroke: \"currentColor\",\n strokeWidth: 2,\n strokeLinecap: \"round\",\n strokeLinejoin: \"round\"\n};\n\nexport { defaultAttributes as default };\n//# sourceMappingURL=defaultAttributes.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport { forwardRef, createElement } from 'react';\nimport defaultAttributes from './defaultAttributes.js';\nimport { mergeClasses, hasA11yProp } from './shared/src/utils.js';\n\nconst Icon = forwardRef(\n ({\n color = \"currentColor\",\n size = 24,\n strokeWidth = 2,\n absoluteStrokeWidth,\n className = \"\",\n children,\n iconNode,\n ...rest\n }, ref) => createElement(\n \"svg\",\n {\n ref,\n ...defaultAttributes,\n width: size,\n height: size,\n stroke: color,\n strokeWidth: absoluteStrokeWidth ? Number(strokeWidth) * 24 / Number(size) : strokeWidth,\n className: mergeClasses(\"lucide\", className),\n ...!children && !hasA11yProp(rest) && { \"aria-hidden\": \"true\" },\n ...rest\n },\n [\n ...iconNode.map(([tag, attrs]) => createElement(tag, attrs)),\n ...Array.isArray(children) ? children : [children]\n ]\n )\n);\n\nexport { Icon as default };\n//# sourceMappingURL=Icon.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport { forwardRef, createElement } from 'react';\nimport { mergeClasses, toKebabCase, toPascalCase } from './shared/src/utils.js';\nimport Icon from './Icon.js';\n\nconst createLucideIcon = (iconName, iconNode) => {\n const Component = forwardRef(\n ({ className, ...props }, ref) => createElement(Icon, {\n ref,\n iconNode,\n className: mergeClasses(\n `lucide-${toKebabCase(toPascalCase(iconName))}`,\n `lucide-${iconName}`,\n className\n ),\n ...props\n })\n );\n Component.displayName = toPascalCase(iconName);\n return Component;\n};\n\nexport { createLucideIcon as default };\n//# sourceMappingURL=createLucideIcon.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"path\", { d: \"m5 12 7-7 7 7\", key: \"hav0vg\" }],\n [\"path\", { d: \"M12 19V5\", key: \"x0mq9r\" }]\n];\nconst ArrowUp = createLucideIcon(\"arrow-up\", __iconNode);\n\nexport { __iconNode, ArrowUp as default };\n//# sourceMappingURL=arrow-up.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"path\", { d: \"M12 18V5\", key: \"adv99a\" }],\n [\"path\", { d: \"M15 13a4.17 4.17 0 0 1-3-4 4.17 4.17 0 0 1-3 4\", key: \"1e3is1\" }],\n [\"path\", { d: \"M17.598 6.5A3 3 0 1 0 12 5a3 3 0 1 0-5.598 1.5\", key: \"1gqd8o\" }],\n [\"path\", { d: \"M17.997 5.125a4 4 0 0 1 2.526 5.77\", key: \"iwvgf7\" }],\n [\"path\", { d: \"M18 18a4 4 0 0 0 2-7.464\", key: \"efp6ie\" }],\n [\"path\", { d: \"M19.967 17.483A4 4 0 1 1 12 18a4 4 0 1 1-7.967-.517\", key: \"1gq6am\" }],\n [\"path\", { d: \"M6 18a4 4 0 0 1-2-7.464\", key: \"k1g0md\" }],\n [\"path\", { d: \"M6.003 5.125a4 4 0 0 0-2.526 5.77\", key: \"q97ue3\" }]\n];\nconst Brain = createLucideIcon(\"brain\", __iconNode);\n\nexport { __iconNode, Brain as default };\n//# sourceMappingURL=brain.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [[\"path\", { d: \"M20 6 9 17l-5-5\", key: \"1gmf2c\" }]];\nconst Check = createLucideIcon(\"check\", __iconNode);\n\nexport { __iconNode, Check as default };\n//# sourceMappingURL=check.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [[\"path\", { d: \"m6 9 6 6 6-6\", key: \"qrunsl\" }]];\nconst ChevronDown = createLucideIcon(\"chevron-down\", __iconNode);\n\nexport { __iconNode, ChevronDown as default };\n//# sourceMappingURL=chevron-down.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"circle\", { cx: \"12\", cy: \"12\", r: \"10\", key: \"1mglay\" }],\n [\"line\", { x1: \"12\", x2: \"12\", y1: \"8\", y2: \"12\", key: \"1pkeuh\" }],\n [\"line\", { x1: \"12\", x2: \"12.01\", y1: \"16\", y2: \"16\", key: \"4dfq90\" }]\n];\nconst CircleAlert = createLucideIcon(\"circle-alert\", __iconNode);\n\nexport { __iconNode, CircleAlert as default };\n//# sourceMappingURL=circle-alert.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"rect\", { width: \"14\", height: \"14\", x: \"8\", y: \"8\", rx: \"2\", ry: \"2\", key: \"17jyea\" }],\n [\"path\", { d: \"M4 16c-1.1 0-2-.9-2-2V4c0-1.1.9-2 2-2h10c1.1 0 2 .9 2 2\", key: \"zix9uf\" }]\n];\nconst Copy = createLucideIcon(\"copy\", __iconNode);\n\nexport { __iconNode, Copy as default };\n//# sourceMappingURL=copy.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"circle\", { cx: \"12\", cy: \"12\", r: \"1\", key: \"41hilf\" }],\n [\"circle\", { cx: \"12\", cy: \"5\", r: \"1\", key: \"gxeob9\" }],\n [\"circle\", { cx: \"12\", cy: \"19\", r: \"1\", key: \"lyex9k\" }]\n];\nconst EllipsisVertical = createLucideIcon(\"ellipsis-vertical\", __iconNode);\n\nexport { __iconNode, EllipsisVertical as default };\n//# sourceMappingURL=ellipsis-vertical.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\n \"path\",\n {\n d: \"M2.062 12.348a1 1 0 0 1 0-.696 10.75 10.75 0 0 1 19.876 0 1 1 0 0 1 0 .696 10.75 10.75 0 0 1-19.876 0\",\n key: \"1nclc0\"\n }\n ],\n [\"circle\", { cx: \"12\", cy: \"12\", r: \"3\", key: \"1v7zrd\" }]\n];\nconst Eye = createLucideIcon(\"eye\", __iconNode);\n\nexport { __iconNode, Eye as default };\n//# sourceMappingURL=eye.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\n \"path\",\n {\n d: \"M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z\",\n key: \"1oefj6\"\n }\n ],\n [\"path\", { d: \"M14 2v5a1 1 0 0 0 1 1h5\", key: \"wfsgrz\" }],\n [\"path\", { d: \"M8 13h2\", key: \"yr2amv\" }],\n [\"path\", { d: \"M14 13h2\", key: \"un5t4a\" }],\n [\"path\", { d: \"M8 17h2\", key: \"2yhykz\" }],\n [\"path\", { d: \"M14 17h2\", key: \"10kma7\" }]\n];\nconst FileSpreadsheet = createLucideIcon(\"file-spreadsheet\", __iconNode);\n\nexport { __iconNode, FileSpreadsheet as default };\n//# sourceMappingURL=file-spreadsheet.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\n \"path\",\n {\n d: \"M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z\",\n key: \"1oefj6\"\n }\n ],\n [\"path\", { d: \"M14 2v5a1 1 0 0 0 1 1h5\", key: \"wfsgrz\" }],\n [\"path\", { d: \"M10 9H8\", key: \"b1mrlr\" }],\n [\"path\", { d: \"M16 13H8\", key: \"t4e002\" }],\n [\"path\", { d: \"M16 17H8\", key: \"z1uh3a\" }]\n];\nconst FileText = createLucideIcon(\"file-text\", __iconNode);\n\nexport { __iconNode, FileText as default };\n//# sourceMappingURL=file-text.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\n \"path\",\n {\n d: \"M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z\",\n key: \"1oefj6\"\n }\n ],\n [\"path\", { d: \"M14 2v5a1 1 0 0 0 1 1h5\", key: \"wfsgrz\" }],\n [\"path\", { d: \"M11 18h2\", key: \"12mj7e\" }],\n [\"path\", { d: \"M12 12v6\", key: \"3ahymv\" }],\n [\"path\", { d: \"M9 13v-.5a.5.5 0 0 1 .5-.5h5a.5.5 0 0 1 .5.5v.5\", key: \"qbrxap\" }]\n];\nconst FileType = createLucideIcon(\"file-type\", __iconNode);\n\nexport { __iconNode, FileType as default };\n//# sourceMappingURL=file-type.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [[\"path\", { d: \"M21 12a9 9 0 1 1-6.219-8.56\", key: \"13zald\" }]];\nconst LoaderCircle = createLucideIcon(\"loader-circle\", __iconNode);\n\nexport { __iconNode, LoaderCircle as default };\n//# sourceMappingURL=loader-circle.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\n \"path\",\n {\n d: \"M22 17a2 2 0 0 1-2 2H6.828a2 2 0 0 0-1.414.586l-2.202 2.202A.71.71 0 0 1 2 21.286V5a2 2 0 0 1 2-2h16a2 2 0 0 1 2 2z\",\n key: \"18887p\"\n }\n ]\n];\nconst MessageSquare = createLucideIcon(\"message-square\", __iconNode);\n\nexport { __iconNode, MessageSquare as default };\n//# sourceMappingURL=message-square.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\n \"path\",\n {\n d: \"m16 6-8.414 8.586a2 2 0 0 0 2.829 2.829l8.414-8.586a4 4 0 1 0-5.657-5.657l-8.379 8.551a6 6 0 1 0 8.485 8.485l8.379-8.551\",\n key: \"1miecu\"\n }\n ]\n];\nconst Paperclip = createLucideIcon(\"paperclip\", __iconNode);\n\nexport { __iconNode, Paperclip as default };\n//# sourceMappingURL=paperclip.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\n \"path\",\n {\n d: \"M21.174 6.812a1 1 0 0 0-3.986-3.987L3.842 16.174a2 2 0 0 0-.5.83l-1.321 4.352a.5.5 0 0 0 .623.622l4.353-1.32a2 2 0 0 0 .83-.497z\",\n key: \"1a8usu\"\n }\n ],\n [\"path\", { d: \"m15 5 4 4\", key: \"1mk7zo\" }]\n];\nconst Pencil = createLucideIcon(\"pencil\", __iconNode);\n\nexport { __iconNode, Pencil as default };\n//# sourceMappingURL=pencil.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"path\", { d: \"M5 12h14\", key: \"1ays0h\" }],\n [\"path\", { d: \"M12 5v14\", key: \"s699le\" }]\n];\nconst Plus = createLucideIcon(\"plus\", __iconNode);\n\nexport { __iconNode, Plus as default };\n//# sourceMappingURL=plus.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"path\", { d: \"M10 11v6\", key: \"nco0om\" }],\n [\"path\", { d: \"M14 11v6\", key: \"outv1u\" }],\n [\"path\", { d: \"M19 6v14a2 2 0 0 1-2 2H7a2 2 0 0 1-2-2V6\", key: \"miytrc\" }],\n [\"path\", { d: \"M3 6h18\", key: \"d0wm0j\" }],\n [\"path\", { d: \"M8 6V4a2 2 0 0 1 2-2h4a2 2 0 0 1 2 2v2\", key: \"e791ji\" }]\n];\nconst Trash2 = createLucideIcon(\"trash-2\", __iconNode);\n\nexport { __iconNode, Trash2 as default };\n//# sourceMappingURL=trash-2.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"path\", { d: \"M12 3v12\", key: \"1x0j5s\" }],\n [\"path\", { d: \"m17 8-5-5-5 5\", key: \"7q97r8\" }],\n [\"path\", { d: \"M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4\", key: \"ih7n3h\" }]\n];\nconst Upload = createLucideIcon(\"upload\", __iconNode);\n\nexport { __iconNode, Upload as default };\n//# sourceMappingURL=upload.js.map\n","/**\n * @license lucide-react v0.555.0 - ISC\n *\n * This source code is licensed under the ISC license.\n * See the LICENSE file in the root directory of this source tree.\n */\n\nimport createLucideIcon from '../createLucideIcon.js';\n\nconst __iconNode = [\n [\"path\", { d: \"M18 6 6 18\", key: \"1bl5f8\" }],\n [\"path\", { d: \"m6 6 12 12\", key: \"d8bk6v\" }]\n];\nconst X = createLucideIcon(\"x\", __iconNode);\n\nexport { __iconNode, X as default };\n//# sourceMappingURL=x.js.map\n","/**\n * Security utilities for input sanitization\n * \n * These utilities provide defense-in-depth protection against common\n * security issues. They should be used in addition to proper input\n * validation at the API level.\n */\n\n/**\n * Common prompt injection patterns to filter\n * These patterns are used by various LLM implementations\n */\nconst INJECTION_PATTERNS = [\n // Llama/Mistral instruction format\n /\\[INST\\]/gi,\n /\\[\\/INST\\]/gi,\n // Llama system prompt format\n /<<SYS>>/gi,\n /<<\\/SYS>>/gi,\n // Special token markers\n /<\\|.*?\\|>/g,\n // OpenAI-style system markers\n /<\\|im_start\\|>/gi,\n /<\\|im_end\\|>/gi,\n // Anthropic-style markers\n /\\[HUMAN\\]/gi,\n /\\[ASSISTANT\\]/gi,\n // Generic role injection attempts\n /^(system|user|assistant):\\s*/gim,\n];\n\n/**\n * Sanitize a system prompt to reduce prompt injection risks\n * \n * **Important:** This is defense-in-depth, not a complete solution.\n * Never use untrusted user input directly as a system prompt.\n * If you need dynamic prompts, use a server-side allow list.\n * \n * @param prompt - The system prompt to sanitize\n * @returns Sanitized prompt with common injection patterns removed\n * \n * @example\n * ```ts\n * const userInput = \"Hello [INST] ignore previous [/INST]\";\n * const safe = sanitizeSystemPrompt(userInput);\n * // Result: \"Hello ignore previous \"\n * ```\n */\nexport function sanitizeSystemPrompt(prompt: string): string {\n if (!prompt || typeof prompt !== 'string') {\n return '';\n }\n\n let sanitized = prompt;\n\n // Remove injection patterns\n for (const pattern of INJECTION_PATTERNS) {\n sanitized = sanitized.replace(pattern, '');\n }\n\n // Normalize whitespace (collapse multiple spaces/newlines)\n sanitized = sanitized\n .replace(/\\n{3,}/g, '\\n\\n')\n .replace(/ {2,}/g, ' ')\n .trim();\n\n return sanitized;\n}\n\n/**\n * Validate that a URL uses a safe protocol (http or https)\n * \n * Prevents javascript: protocol XSS attacks and other unsafe schemes.\n * \n * @param url - The URL to validate\n * @returns true if the URL uses http or https protocol\n * \n * @example\n * ```ts\n * isValidUrl('https://example.com'); // true\n * isValidUrl('javascript:alert(1)'); // false\n * isValidUrl('data:text/html,...'); // false\n * ```\n */\nexport function isValidUrl(url: string): boolean {\n if (!url || typeof url !== 'string') {\n return false;\n }\n\n try {\n const parsed = new URL(url);\n return ['http:', 'https:'].includes(parsed.protocol);\n } catch {\n return false;\n }\n}\n\n/**\n * Validation result for URL validation\n */\nexport interface UrlValidationResult {\n /** Whether the URL is valid */\n valid: boolean;\n /** Error message if invalid */\n error?: string;\n /** Parsed URL if valid */\n parsed?: URL;\n}\n\n/**\n * Validate a base URL for API configuration\n * \n * Ensures the URL is properly formatted and uses a secure protocol.\n * Returns detailed validation result for error messaging.\n * \n * @param url - The base URL to validate\n * @returns Validation result with error details\n * \n * @example\n * ```ts\n * const result = validateBaseUrl('https://api.example.com');\n * if (!result.valid) {\n * console.error(result.error);\n * }\n * ```\n */\nexport function validateBaseUrl(url: string): UrlValidationResult {\n if (!url) {\n return { valid: false, error: 'URL is required' };\n }\n\n if (typeof url !== 'string') {\n return { valid: false, error: 'URL must be a string' };\n }\n\n // Allow relative URLs for proxy mode\n if (url.startsWith('/') || url.startsWith('./')) {\n return { valid: true };\n }\n\n try {\n const parsed = new URL(url);\n\n if (!['http:', 'https:'].includes(parsed.protocol)) {\n return { \n valid: false, \n error: `Invalid protocol \"${parsed.protocol}\". Only http and https are allowed.` \n };\n }\n\n // Warn about non-HTTPS in production contexts\n if (parsed.protocol === 'http:' && !isLocalhostUrl(url)) {\n return {\n valid: true,\n parsed,\n error: 'Warning: Using HTTP for non-localhost URL. Consider using HTTPS.',\n };\n }\n\n return { valid: true, parsed };\n } catch {\n return { valid: false, error: 'Invalid URL format' };\n }\n}\n\n/**\n * Check if a URL is a localhost URL\n */\nfunction isLocalhostUrl(url: string): boolean {\n try {\n const parsed = new URL(url);\n return ['localhost', '127.0.0.1', '[::1]'].includes(parsed.hostname);\n } catch {\n return false;\n }\n}\n\n/**\n * Sanitize a string for safe display (escape HTML entities)\n * \n * Note: React already escapes content in JSX, so this is mainly\n * useful when constructing strings for non-React contexts.\n * \n * @param text - The text to sanitize\n * @returns Text with HTML entities escaped\n */\nexport function escapeHtml(text: string): string {\n if (!text || typeof text !== 'string') {\n return '';\n }\n\n const htmlEntities: Record<string, string> = {\n '&': '&',\n '<': '<',\n '>': '>',\n '\"': '"',\n \"'\": ''',\n '/': '/',\n };\n\n return text.replace(/[&<>\"'/]/g, (char) => htmlEntities[char] || char);\n}\n\n/**\n * Generate a cryptographically secure UUID v4\n * \n * Uses crypto.randomUUID when available, falls back to\n * crypto.getRandomValues for broader compatibility.\n * \n * @returns A UUID v4 string\n */\nexport function generateSecureUUID(): string {\n // Prefer native randomUUID (modern browsers, Node 19+)\n if (typeof crypto !== 'undefined' && crypto.randomUUID) {\n return crypto.randomUUID();\n }\n\n // Fallback using crypto.getRandomValues (still cryptographically secure)\n if (typeof crypto !== 'undefined' && crypto.getRandomValues) {\n const bytes = new Uint8Array(16);\n crypto.getRandomValues(bytes);\n\n // Set version (4) and variant (10xx) bits per RFC 4122\n bytes[6] = (bytes[6] & 0x0f) | 0x40;\n bytes[8] = (bytes[8] & 0x3f) | 0x80;\n\n const hex = Array.from(bytes, (b) => b.toString(16).padStart(2, '0')).join('');\n return `${hex.slice(0, 8)}-${hex.slice(8, 12)}-${hex.slice(12, 16)}-${hex.slice(16, 20)}-${hex.slice(20)}`;\n }\n\n // Last resort fallback (not cryptographically secure)\n // This should rarely be reached in modern environments\n if (typeof process !== 'undefined' && process.env?.NODE_ENV === 'development') {\n // eslint-disable-next-line no-console\n console.warn('[CuadraUIKit] Using non-cryptographic fallback for UUID generation');\n }\n\n return `${Date.now().toString(36)}-${Math.random().toString(36).slice(2)}-${Math.random().toString(36).slice(2)}`;\n}\n","import type {\n ChatOut,\n ChatRequest,\n ChatSummaryPage,\n ChatUpdate,\n GenerateTitleRequest,\n GenerateTitleResponse,\n ListChatsParams,\n ListModelsParams,\n ModelOut,\n ModelPage,\n Source,\n} from '../types/cuadra';\nimport { generateSecureUUID } from '../utils/sanitize';\n\n/**\n * SSE Chunk types for streaming responses\n * Supports current API format and AI SDK UI Message Stream Protocol\n */\nexport interface SSEChunk {\n // Common fields\n type?: string;\n id?: string; // Chat/message ID or block ID\n \n // Current API format fields\n delta?: string; // Delta content (incremental text)\n reasoning?: string; // Accumulated reasoning content\n sources?: Source[] | null; // RAG sources (sent in first chunk)\n toolCalls?: unknown[] | null; // Tool calls (future)\n finished?: boolean; // Whether the stream is finished\n usage?: {\n inputTokens: number;\n outputTokens: number;\n totalTokens: number;\n cost?: number | null;\n };\n \n // Legacy format fields\n content?: string;\n chatId?: string;\n message?: {\n role: string;\n content: string;\n };\n done?: boolean;\n \n // AI SDK Protocol: Source document event\n sourceId?: string;\n mediaType?: string;\n title?: string; // Filename for source-document events\n \n // AI SDK Protocol: Message metadata\n messageId?: string;\n \n // AI SDK Protocol: Finish event\n finishReason?: string;\n \n // AI SDK Protocol: Error event\n errorText?: string;\n}\n\n/**\n * CuadraChatClient - API client for Cuadra AI REST API\n * Handles authentication, HTTP requests, and SSE streaming\n */\nexport class CuadraChatClient {\n private baseUrl: string;\n private sessionToken: string | null = null;\n private isProxyMode: boolean = false;\n\n constructor(baseUrl: string, sessionToken?: string, isProxyMode: boolean = false) {\n this.baseUrl = baseUrl.replace(/\\/$/, ''); // Remove trailing slash\n this.isProxyMode = isProxyMode;\n if (sessionToken) {\n this.sessionToken = sessionToken;\n }\n }\n\n /**\n * Set or update the session token\n */\n setSessionToken(token: string | null): void {\n this.sessionToken = token;\n }\n\n /**\n * Create or continue a chat with streaming support\n * Returns a ReadableStream for SSE parsing\n */\n private getUrl(endpoint: string): string {\n if (this.isProxyMode) {\n // Proxy mode: remove /v1 prefix\n const path = `${this.baseUrl}${endpoint.replace('/v1', '')}`;\n // If baseUrl is relative, return as-is; otherwise construct full URL\n if (path.startsWith('/') || path.startsWith('./')) {\n return path;\n }\n return path;\n }\n return `${this.baseUrl}${endpoint}`;\n }\n\n async createOrContinueChat(\n request: ChatRequest,\n abortSignal?: AbortSignal,\n ): Promise<ReadableStream<Uint8Array>> {\n const url = this.getUrl('/v1/chats');\n \n // Generate idempotency key for create operation\n const idempotencyKey = this.generateIdempotencyKey();\n \n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n 'Accept': 'text/event-stream', // Tell server we expect SSE streaming\n 'Accept-Encoding': 'identity', // Disable compression to prevent buffering\n 'Cache-Control': 'no-cache', // Prevent caching of streaming response\n 'X-Accel-Buffering': 'no', // Disable nginx buffering if behind reverse proxy\n 'Idempotency-Key': idempotencyKey,\n };\n\n if (this.sessionToken) {\n headers['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n // Ensure stream is true for streaming\n const body: ChatRequest = {\n ...request,\n stream: true,\n };\n\n const response = await fetch(url, {\n method: 'POST',\n headers,\n body: JSON.stringify(body),\n signal: abortSignal,\n cache: 'no-store',\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n if (!response.body) {\n throw new Error('Response body is null');\n }\n\n return response.body;\n }\n\n /**\n * Get a specific chat by ID\n */\n async getChat(chatId: string): Promise<ChatOut> {\n const url = this.getUrl(`/v1/chats/${chatId}`);\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n };\n\n if (this.sessionToken) {\n headers['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url, {\n method: 'GET',\n headers,\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n return response.json();\n }\n\n /**\n * List chats with pagination\n */\n async listChats(params?: ListChatsParams): Promise<ChatSummaryPage> {\n const baseUrl = this.getUrl('/v1/chats');\n // Handle relative URLs (for proxy mode)\n const url = baseUrl.startsWith('/') || baseUrl.startsWith('./')\n ? new URL(baseUrl, window.location.origin)\n : new URL(baseUrl);\n \n if (params?.limit) {\n url.searchParams.set('limit', params.limit.toString());\n }\n if (params?.cursor) {\n url.searchParams.set('cursor', params.cursor);\n }\n if (params?.['expand[]']) {\n params['expand[]'].forEach((expand) => {\n url.searchParams.append('expand[]', expand);\n });\n }\n\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n };\n\n if (this.sessionToken) {\n headers['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url.toString(), {\n method: 'GET',\n headers,\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n return response.json();\n }\n\n /**\n * Generate a cryptographically secure UUID v4 idempotency key\n * Uses the shared secure UUID generator from sanitize utils\n */\n private generateIdempotencyKey(): string {\n return generateSecureUUID();\n }\n\n /**\n * Update a chat\n */\n async updateChat(chatId: string, update: ChatUpdate): Promise<ChatOut> {\n const endpoint = `/v1/chats/${chatId}`;\n const baseUrl = this.getUrl(endpoint);\n \n // Handle relative URLs (for proxy mode)\n const url = baseUrl.startsWith('/') || baseUrl.startsWith('./')\n ? new URL(baseUrl, window.location.origin)\n : new URL(baseUrl);\n \n // Generate idempotency key for update operation\n const idempotencyKey = this.generateIdempotencyKey();\n \n const headers: HeadersInit = {\n 'Content-Type': 'application/json',\n 'Idempotency-Key': idempotencyKey,\n };\n\n if (this.sessionToken) {\n (headers as Record<string, string>)['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url.toString(), {\n method: 'PATCH',\n headers,\n body: JSON.stringify(update),\n });\n\n if (!response.ok) {\n const errorText = await response.text();\n let errorData: { detail?: string; message?: string } = {};\n try {\n errorData = JSON.parse(errorText) as { detail?: string; message?: string };\n } catch {\n errorData = { detail: errorText };\n }\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n // Handle empty responses (e.g., 204 No Content)\n if (response.status === 204) {\n return {} as ChatOut;\n }\n\n const text = await response.text();\n if (!text) {\n return {} as ChatOut;\n }\n\n try {\n return JSON.parse(text);\n } catch {\n return {} as ChatOut;\n }\n }\n\n /**\n * Delete a chat\n */\n async deleteChat(chatId: string): Promise<void> {\n const endpoint = `/v1/chats/${chatId}`;\n const baseUrl = this.getUrl(endpoint);\n \n // Handle relative URLs (for proxy mode)\n const url = baseUrl.startsWith('/') || baseUrl.startsWith('./')\n ? new URL(baseUrl, window.location.origin)\n : new URL(baseUrl);\n \n // Generate idempotency key for delete operation\n const idempotencyKey = this.generateIdempotencyKey();\n \n const headers: HeadersInit = {\n 'Content-Type': 'application/json',\n 'Idempotency-Key': idempotencyKey,\n };\n\n if (this.sessionToken) {\n (headers as Record<string, string>)['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url.toString(), {\n method: 'DELETE',\n headers,\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n }\n\n /**\n * Generate an AI-powered title for a chat.\n * Uses a fast model on the backend to analyze the conversation.\n */\n async generateChatTitle(\n chatId: string,\n request?: GenerateTitleRequest,\n ): Promise<GenerateTitleResponse> {\n const endpoint = `/v1/chats/${chatId}/title`;\n const baseUrl = this.getUrl(endpoint);\n\n const url = baseUrl.startsWith('/') || baseUrl.startsWith('./')\n ? new URL(baseUrl, window.location.origin)\n : new URL(baseUrl);\n\n const idempotencyKey = this.generateIdempotencyKey();\n\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n 'Idempotency-Key': idempotencyKey,\n };\n\n if (this.sessionToken) {\n headers['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url.toString(), {\n method: 'POST',\n headers,\n body: request ? JSON.stringify(request) : undefined,\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n return response.json();\n }\n\n /**\n * List available models\n */\n async listModels(params?: ListModelsParams): Promise<ModelPage> {\n const baseUrl = this.getUrl('/v1/models');\n // Handle relative URLs (for proxy mode)\n const url = baseUrl.startsWith('/') || baseUrl.startsWith('./')\n ? new URL(baseUrl, window.location.origin)\n : new URL(baseUrl);\n \n if (params?.limit) {\n url.searchParams.set('limit', params.limit.toString());\n }\n if (params?.cursor) {\n url.searchParams.set('cursor', params.cursor);\n }\n if (params?.['expand[]']) {\n params['expand[]'].forEach((expand) => {\n url.searchParams.append('expand[]', expand);\n });\n }\n\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n };\n\n if (this.sessionToken) {\n headers['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url.toString(), {\n method: 'GET',\n headers,\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n return response.json();\n }\n\n /**\n * Get a specific model by ID\n */\n async getModel(modelId: string): Promise<ModelOut> {\n const url = this.getUrl(`/v1/models/${modelId}`);\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n };\n\n if (this.sessionToken) {\n headers['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url, {\n method: 'GET',\n headers,\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n return response.json();\n }\n\n /**\n * Upload a file to Cuadra API\n * @param file - The file to upload\n * @param resourceType - Optional resource type (e.g., 'chat')\n * @param resourceId - Optional resource ID to associate with\n * @returns Promise resolving to file metadata\n */\n async uploadFile(\n file: File,\n resourceType?: string,\n resourceId?: string,\n ): Promise<{ id: string; name: string; [key: string]: unknown }> {\n const url = this.getUrl('/v1/files');\n \n // Generate idempotency key\n const idempotencyKey = this.generateIdempotencyKey();\n \n const formData = new FormData();\n formData.append('file', file);\n \n if (resourceType) {\n formData.append('resource_type', resourceType);\n }\n if (resourceId) {\n formData.append('resource_id', resourceId);\n }\n\n const headers: Record<string, string> = {\n 'Idempotency-Key': idempotencyKey,\n };\n\n if (this.sessionToken) {\n headers['Authorization'] = `Bearer ${this.sessionToken}`;\n }\n\n const response = await fetch(url, {\n method: 'POST',\n headers,\n body: formData,\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.detail ||\n errorData.message ||\n `API error: ${response.status} ${response.statusText}`,\n );\n }\n\n return response.json();\n }\n}\n\n/**\n * Parse SSE stream into chunks\n */\nexport async function* parseSSEStream(\n stream: ReadableStream<Uint8Array>,\n abortSignal?: AbortSignal,\n): AsyncGenerator<SSEChunk> {\n const reader = stream.getReader();\n const decoder = new TextDecoder();\n let buffer = '';\n\n try {\n while (true) {\n if (abortSignal?.aborted) {\n reader.cancel();\n break;\n }\n\n const { done, value } = await reader.read();\n if (done) break;\n\n buffer += decoder.decode(value, { stream: true });\n const lines = buffer.split('\\n\\n');\n buffer = lines.pop() || '';\n\n for (const line of lines) {\n if (line.startsWith('data: ')) {\n const data = line.slice(6);\n if (data.trim() === '[DONE]') {\n yield { done: true };\n continue;\n }\n try {\n const parsed = JSON.parse(data);\n yield parsed;\n } catch {\n // Skip invalid JSON\n }\n }\n }\n }\n } finally {\n reader.releaseLock();\n }\n}\n\n","import type { ThreadMessage } from '@assistant-ui/react';\nimport type { MessageCreate } from '../types/cuadra';\n\n/**\n * Convert assistant-ui ThreadMessage to Cuadra API MessageCreate format\n * Note: Attachments are included in the message but may need to be uploaded separately\n * depending on API requirements\n */\nexport function convertToCuadraMessages(\n messages: readonly ThreadMessage[],\n): MessageCreate[] {\n return messages.map((msg) => {\n // Extract text content from assistant-ui message format\n let content = '';\n\n if (typeof msg.content === 'string') {\n content = msg.content;\n } else if (Array.isArray(msg.content)) {\n // Find text content in content array\n const textContent = msg.content.find((c) => c.type === 'text');\n if (\n textContent &&\n typeof textContent === 'object' &&\n 'text' in textContent\n ) {\n content = textContent.text as string;\n }\n }\n\n // Note: Attachments are available in msg.attachments\n // For now, we'll include them in the message structure\n // The actual file upload to Cuadra API should be handled separately\n // if the API requires file uploads before message creation\n \n return {\n role: msg.role as 'system' | 'user' | 'assistant',\n content,\n // Attachments will be handled separately - files are available in msg.attachments\n };\n });\n}\n\n/**\n * Convert Cuadra API MessageOut to assistant-ui ThreadMessage format\n */\nexport function convertFromCuadraMessage(\n msg: {\n role: string;\n content: string;\n id: string;\n createdAt: string;\n },\n): ThreadMessage {\n return {\n id: msg.id,\n role: msg.role as 'user' | 'assistant' | 'system',\n content: [\n {\n type: 'text' as const,\n text: msg.content,\n },\n ],\n createdAt: new Date(msg.createdAt),\n metadata: {\n custom: {},\n },\n ...(msg.role === 'assistant' && {\n status: {\n type: 'complete' as const,\n reason: 'stop' as const,\n },\n metadata: {\n unstable_state: null,\n unstable_annotations: [],\n unstable_data: [],\n steps: [],\n custom: {},\n },\n }),\n ...(msg.role === 'user' && {\n attachments: [],\n }),\n } as ThreadMessage;\n}\n\n","export const promiseWithResolvers = function () {\n let resolve;\n let reject;\n const promise = new Promise((res, rej) => {\n resolve = res;\n reject = rej;\n });\n if (!resolve || !reject)\n throw new Error(\"Failed to create promise\");\n return { promise, resolve, reject };\n};\n//# sourceMappingURL=promiseWithResolvers.js.map","import { promiseWithResolvers } from \"../../../utils/promiseWithResolvers.js\";\nexport const createMergeStream = () => {\n const list = [];\n let sealed = false;\n let controller;\n let currentPull;\n const handlePull = (item) => {\n if (!item.promise) {\n // TODO for most streams, we can directly pipeTo to avoid the microTask queue\n // add an option to eagerly pipe the stream to the merge stream\n // ideally, using assistant-stream w sync run method + piping to a sync WritableStream runs in the same microtask\n // this is useful because we often use AssistantStreams internally as a serialization utility, e. g. AssistantTransformStream\n // idea: avoid reader.read() by instead using a WritableStream & if (!hasPendingPull) await waitForPull()?\n item.promise = item.reader\n .read()\n .then(({ done, value }) => {\n item.promise = undefined;\n if (done) {\n list.splice(list.indexOf(item), 1);\n if (sealed && list.length === 0) {\n controller.close();\n }\n }\n else {\n controller.enqueue(value);\n }\n currentPull?.resolve();\n currentPull = undefined;\n })\n .catch((e) => {\n console.error(e);\n list.forEach((item) => {\n item.reader.cancel();\n });\n list.length = 0;\n controller.error(e);\n currentPull?.reject(e);\n currentPull = undefined;\n });\n }\n };\n const readable = new ReadableStream({\n start(c) {\n controller = c;\n },\n pull() {\n currentPull = promiseWithResolvers();\n list.forEach((item) => {\n handlePull(item);\n });\n return currentPull.promise;\n },\n cancel() {\n list.forEach((item) => {\n item.reader.cancel();\n });\n list.length = 0;\n },\n });\n return {\n readable,\n isSealed() {\n return sealed;\n },\n seal() {\n sealed = true;\n if (list.length === 0)\n controller.close();\n },\n addStream(stream) {\n if (sealed)\n throw new Error(\"Cannot add streams after the run callback has settled.\");\n const item = { reader: stream.getReader() };\n list.push(item);\n handlePull(item);\n },\n enqueue(chunk) {\n this.addStream(new ReadableStream({\n start(c) {\n c.enqueue(chunk);\n c.close();\n },\n }));\n },\n };\n};\n// TODO\n// export class SpanContainerMerger {\n// public get isSealed() {\n// return this.mergeStream.isSealed();\n// }\n// public get readable() {\n// return this.mergeStream.readable;\n// }\n// private subAllocator = new Counter();\n// private mergeStream = createMergeStream();\n// constructor() {\n// // id 0 is auto allocated\n// this.subAllocator.up();\n// }\n// add(stream: ReadableStream<AssistantStreamChunk>) {\n// this.mergeStream.addStream(\n// stream.pipeThrough(new SpanParentEncoder(this.subAllocator)),\n// );\n// }\n// enqueue(chunk: AssistantStreamChunk & { parentId: 0 }) {\n// this.mergeStream.addStream(\n// new ReadableStream({\n// start(c) {\n// c.enqueue(chunk);\n// c.close();\n// },\n// }),\n// );\n// }\n// seal() {\n// this.mergeStream.seal();\n// }\n// }\n// export class SpanContainerSplitter {\n// public writable;\n// private isSealed = false;\n// private writers = new Map<\n// number,\n// WritableStreamDefaultWriter<AssistantStreamChunk>\n// >();\n// private closeTasks: Promise<void>[] = [];\n// private allocator = new Counter();\n// private subAllocator = new Counter();\n// constructor() {\n// // id 0 is auto-allocated\n// this.allocator.up();\n// this.writable = new WritableStream({\n// write: (chunk) => {\n// const { type, parentId } = chunk;\n// const writer = this.writers.get(parentId);\n// if (writer === undefined) throw new Error(\"Parent id not found\");\n// writer.write(chunk);\n// if (type === \"span\") {\n// // allocate a new span id\n// this.writers.set(this.allocator.up(), writer);\n// }\n// if (type === \"finish\") {\n// this.writers.delete(parentId);\n// writer.close();\n// if (this.writers.size === 0) {\n// const closeTask = this.writable.close();\n// this.closeTasks.push(closeTask);\n// closeTask.then(() => {\n// this.closeTasks.splice(this.closeTasks.indexOf(closeTask), 1);\n// });\n// }\n// }\n// },\n// close: async () => {\n// if (this.writers.size > 0) throw new Error(\"Not all writers closed\");\n// // await and throw on any errors\n// await Promise.all(this.closeTasks);\n// },\n// });\n// }\n// add(stream: WritableStream<AssistantStreamChunk>) {\n// if (this.isSealed) throw new Error(\"Cannot add streams after sealing\");\n// const decoder = new SpanParentDecoder(this.subAllocator);\n// decoder.readable.pipeTo(stream);\n// this.writers.set(this.allocator.up(), decoder.writable.getWriter());\n// }\n// seal() {\n// this.isSealed = true;\n// if (this.writers.size === 0) this.writable.close();\n// }\n// }\n//# sourceMappingURL=merge.js.map","class TextStreamControllerImpl {\n _controller;\n _isClosed = false;\n constructor(controller) {\n this._controller = controller;\n }\n append(textDelta) {\n this._controller.enqueue({\n type: \"text-delta\",\n path: [],\n textDelta,\n });\n return this;\n }\n close() {\n if (this._isClosed)\n return;\n this._isClosed = true;\n this._controller.enqueue({\n type: \"part-finish\",\n path: [],\n });\n this._controller.close();\n }\n}\nexport const createTextStream = (readable) => {\n return new ReadableStream({\n start(c) {\n return readable.start?.(new TextStreamControllerImpl(c));\n },\n pull(c) {\n return readable.pull?.(new TextStreamControllerImpl(c));\n },\n cancel(c) {\n return readable.cancel?.(c);\n },\n });\n};\nexport const createTextStreamController = () => {\n let controller;\n const stream = createTextStream({\n start(c) {\n controller = c;\n },\n });\n return [stream, controller];\n};\n//# sourceMappingURL=text.js.map","import { createTextStream } from \"./text.js\";\nclass ToolCallStreamControllerImpl {\n _controller;\n _isClosed = false;\n _mergeTask;\n constructor(_controller) {\n this._controller = _controller;\n const stream = createTextStream({\n start: (c) => {\n this._argsTextController = c;\n },\n });\n let hasArgsText = false;\n this._mergeTask = stream.pipeTo(new WritableStream({\n write: (chunk) => {\n switch (chunk.type) {\n case \"text-delta\":\n hasArgsText = true;\n this._controller.enqueue(chunk);\n break;\n case \"part-finish\":\n if (!hasArgsText) {\n // if no argsText was provided, assume empty object\n this._controller.enqueue({\n type: \"text-delta\",\n textDelta: \"{}\",\n path: [],\n });\n }\n this._controller.enqueue({\n type: \"tool-call-args-text-finish\",\n path: [],\n });\n break;\n default:\n throw new Error(`Unexpected chunk type: ${chunk.type}`);\n }\n },\n }));\n }\n get argsText() {\n return this._argsTextController;\n }\n _argsTextController;\n async setResponse(response) {\n this._argsTextController.close();\n await Promise.resolve(); // flush microtask queue\n // TODO switch argsTextController to be something that doesn'#t require this\n this._controller.enqueue({\n type: \"result\",\n path: [],\n ...(response.artifact !== undefined\n ? { artifact: response.artifact }\n : {}),\n result: response.result,\n isError: response.isError ?? false,\n });\n }\n async close() {\n if (this._isClosed)\n return;\n this._isClosed = true;\n this._argsTextController.close();\n await this._mergeTask;\n this._controller.enqueue({\n type: \"part-finish\",\n path: [],\n });\n this._controller.close();\n }\n}\nexport const createToolCallStream = (readable) => {\n return new ReadableStream({\n start(c) {\n return readable.start?.(new ToolCallStreamControllerImpl(c));\n },\n pull(c) {\n return readable.pull?.(new ToolCallStreamControllerImpl(c));\n },\n cancel(c) {\n return readable.cancel?.(c);\n },\n });\n};\nexport const createToolCallStreamController = () => {\n let controller;\n const stream = createToolCallStream({\n start(c) {\n controller = c;\n },\n });\n return [stream, controller];\n};\n//# sourceMappingURL=tool-call.js.map","export class Counter {\n value = -1;\n up() {\n return ++this.value;\n }\n}\n//# sourceMappingURL=Counter.js.map","import { Counter } from \"../Counter.js\";\nexport class PathAppendEncoder extends TransformStream {\n constructor(idx) {\n super({\n transform(chunk, controller) {\n controller.enqueue({\n ...chunk,\n path: [idx, ...chunk.path],\n });\n },\n });\n }\n}\nexport class PathAppendDecoder extends TransformStream {\n constructor(idx) {\n super({\n transform(chunk, controller) {\n const { path: [idx2, ...path], } = chunk;\n if (idx !== idx2)\n throw new Error(`Path mismatch: expected ${idx}, got ${idx2}`);\n controller.enqueue({\n ...chunk,\n path,\n });\n },\n });\n }\n}\nexport class PathMergeEncoder extends TransformStream {\n constructor(counter) {\n const innerCounter = new Counter();\n const mapping = new Map();\n super({\n transform(chunk, controller) {\n if (chunk.type === \"part-start\" && chunk.path.length === 0) {\n mapping.set(innerCounter.up(), counter.up());\n }\n const [idx, ...path] = chunk.path;\n if (idx === undefined) {\n controller.enqueue(chunk);\n return;\n }\n const mappedIdx = mapping.get(idx);\n if (mappedIdx === undefined)\n throw new Error(\"Path not found\");\n controller.enqueue({\n ...chunk,\n path: [mappedIdx, ...path],\n });\n },\n });\n }\n}\n//# sourceMappingURL=path-utils.js.map","/* @ts-self-types=\"./index.d.ts\" */\nlet urlAlphabet =\n 'useandom-26T198340PX75pxJACKVERYMINDBUSHWOLF_GQZbfghjklqvwyzrict'\nexport let customAlphabet = (alphabet, defaultSize = 21) => {\n return (size = defaultSize) => {\n let id = ''\n let i = size | 0\n while (i--) {\n id += alphabet[(Math.random() * alphabet.length) | 0]\n }\n return id\n }\n}\nexport let nanoid = (size = 21) => {\n let id = ''\n let i = size | 0\n while (i--) {\n id += urlAlphabet[(Math.random() * 64) | 0]\n }\n return id\n}\n","import { customAlphabet } from \"nanoid/non-secure\";\nexport const generateId = customAlphabet(\"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\", 7);\n//# sourceMappingURL=generateId.js.map","import { AssistantStream } from \"../AssistantStream.js\";\nimport { createMergeStream } from \"../utils/stream/merge.js\";\nimport { createTextStreamController } from \"./text.js\";\nimport { createToolCallStreamController, } from \"./tool-call.js\";\nimport { Counter } from \"../utils/Counter.js\";\nimport { PathAppendEncoder, PathMergeEncoder, } from \"../utils/stream/path-utils.js\";\nimport { DataStreamEncoder } from \"../serialization/data-stream/DataStream.js\";\nimport { generateId } from \"../utils/generateId.js\";\nimport { promiseWithResolvers } from \"../../utils/promiseWithResolvers.js\";\nclass AssistantStreamControllerImpl {\n _state;\n _parentId;\n constructor(state) {\n this._state = state || {\n merger: createMergeStream(),\n contentCounter: new Counter(),\n };\n }\n get __internal_isClosed() {\n return this._state.merger.isSealed();\n }\n __internal_getReadable() {\n return this._state.merger.readable;\n }\n __internal_subscribeToClose(callback) {\n this._state.closeSubscriber = callback;\n }\n _addPart(part, stream) {\n if (this._state.append) {\n this._state.append.controller.close();\n this._state.append = undefined;\n }\n this.enqueue({\n type: \"part-start\",\n part,\n path: [],\n });\n this._state.merger.addStream(stream.pipeThrough(new PathAppendEncoder(this._state.contentCounter.value)));\n }\n merge(stream) {\n this._state.merger.addStream(stream.pipeThrough(new PathMergeEncoder(this._state.contentCounter)));\n }\n appendText(textDelta) {\n if (this._state.append?.kind !== \"text\") {\n this._state.append = {\n kind: \"text\",\n controller: this.addTextPart(),\n };\n }\n this._state.append.controller.append(textDelta);\n }\n appendReasoning(textDelta) {\n if (this._state.append?.kind !== \"reasoning\") {\n this._state.append = {\n kind: \"reasoning\",\n controller: this.addReasoningPart(),\n };\n }\n this._state.append.controller.append(textDelta);\n }\n addTextPart() {\n const [stream, controller] = createTextStreamController();\n this._addPart({ type: \"text\" }, stream);\n return controller;\n }\n addReasoningPart() {\n const [stream, controller] = createTextStreamController();\n this._addPart({ type: \"reasoning\" }, stream);\n return controller;\n }\n addToolCallPart(options) {\n const opt = typeof options === \"string\" ? { toolName: options } : options;\n const toolName = opt.toolName;\n const toolCallId = opt.toolCallId ?? generateId();\n const [stream, controller] = createToolCallStreamController();\n this._addPart({\n type: \"tool-call\",\n toolName,\n toolCallId,\n ...(this._parentId && { parentId: this._parentId }),\n }, stream);\n if (opt.argsText !== undefined) {\n controller.argsText.append(opt.argsText);\n controller.argsText.close();\n }\n if (opt.args !== undefined) {\n controller.argsText.append(JSON.stringify(opt.args));\n controller.argsText.close();\n }\n if (opt.response !== undefined) {\n controller.setResponse(opt.response);\n }\n return controller;\n }\n appendSource(options) {\n this._addPart({ ...options, ...(this._parentId && { parentId: this._parentId }) }, new ReadableStream({\n start(controller) {\n controller.enqueue({\n type: \"part-finish\",\n path: [],\n });\n controller.close();\n },\n }));\n }\n appendFile(options) {\n this._addPart(options, new ReadableStream({\n start(controller) {\n controller.enqueue({\n type: \"part-finish\",\n path: [],\n });\n controller.close();\n },\n }));\n }\n enqueue(chunk) {\n this._state.merger.enqueue(chunk);\n if (chunk.type === \"part-start\" && chunk.path.length === 0) {\n this._state.contentCounter.up();\n }\n }\n withParentId(parentId) {\n const controller = new AssistantStreamControllerImpl(this._state);\n controller._parentId = parentId;\n return controller;\n }\n close() {\n this._state.append?.controller?.close();\n this._state.merger.seal();\n this._state.closeSubscriber?.();\n }\n}\nexport function createAssistantStream(callback) {\n const controller = new AssistantStreamControllerImpl();\n const runTask = async () => {\n try {\n await callback(controller);\n }\n catch (e) {\n if (!controller.__internal_isClosed) {\n controller.enqueue({\n type: \"error\",\n path: [],\n error: String(e),\n });\n }\n throw e;\n }\n finally {\n if (!controller.__internal_isClosed) {\n controller.close();\n }\n }\n };\n runTask();\n return controller.__internal_getReadable();\n}\nexport function createAssistantStreamController() {\n const { resolve, promise } = promiseWithResolvers();\n let controller;\n const stream = createAssistantStream((c) => {\n controller = c;\n controller.__internal_subscribeToClose(resolve);\n return promise;\n });\n return [stream, controller];\n}\nexport function createAssistantStreamResponse(callback) {\n return AssistantStream.toResponse(createAssistantStream(callback), new DataStreamEncoder());\n}\n//# sourceMappingURL=assistant-stream.js.map","import {\n type unstable_RemoteThreadListAdapter as RemoteThreadListAdapter,\n RuntimeAdapterProvider,\n type ThreadHistoryAdapter,\n type ThreadMessage,\n useThreadListItem,\n} from '@assistant-ui/react';\nimport { createAssistantStream } from 'assistant-stream';\nimport React from 'react';\nimport { CuadraChatClient } from '../lib/cuadraChatClient';\nimport { convertFromCuadraMessage } from './messageConverter';\n\n/**\n * Store mapping of local thread IDs to server chat IDs\n * This is used to update remoteId when server chat ID is received\n */\nexport const localToRemoteIdMap = new Map<string, string>();\n\n/**\n * Store mapping of message IDs to their chat (remote) IDs.\n * Populated during history loading so the chat model adapter\n * can resolve which chat a conversation belongs to.\n */\nexport const messageIdToChatIdMap = new Map<string, string>();\n\n/**\n * The remoteId of the thread that is currently active (rendering).\n * Set by the unstable_Provider so the chat model adapter can map\n * server chat IDs back to local thread IDs without a reverse lookup.\n */\nexport let activeThreadRemoteId: string | undefined;\n\n/**\n * Resolve a remoteId (which may be a __LOCALID_*) to the actual server chat ID.\n * Uses multiple resolution strategies in order of reliability.\n */\nfunction resolveServerId(\n remoteId: string,\n messages?: readonly ThreadMessage[],\n): string {\n // 1. Direct lookup in localToRemoteIdMap\n const mapped = localToRemoteIdMap.get(remoteId);\n if (mapped) return mapped;\n\n // 2. Scan messages for a registered chatId (populated by the chat model\n // adapter when the SSE stream returns a chat ID)\n if (messages) {\n for (const msg of messages) {\n const chatId = messageIdToChatIdMap.get(msg.id);\n if (chatId) {\n // Back-fill so rename/delete/fetch also benefit\n localToRemoteIdMap.set(remoteId, chatId);\n return chatId;\n }\n }\n }\n\n // 3. Fall back to the remoteId as-is (valid for pre-existing chats\n // whose remoteId is already the server chat ID)\n return remoteId;\n}\n\nexport function createThreadListAdapter(\n client: CuadraChatClient,\n onChatIdReceived?: (localThreadId: string, serverChatId: string) => void,\n): RemoteThreadListAdapter & {\n updateRemoteId: (localThreadId: string, serverChatId: string) => void;\n} {\n return {\n async list() {\n const all = [];\n const limit = 50;\n let cursor: string | undefined = undefined;\n let safety = 0;\n\n // Fetch all chats with pagination\n do {\n const response = await client.listChats({ limit, cursor });\n const items = response.items || [];\n\n if (Array.isArray(items) && items.length > 0) {\n all.push(...items);\n }\n\n cursor = response.nextCursor || undefined;\n safety++;\n } while (cursor && safety < 50);\n\n // Map to assistant-ui thread format\n // Filter out deleted chats and chats without IDs\n const threads = all\n .map((chat) => {\n const id = chat.id;\n // Skip if no ID or if deleted\n if (!id || chat.deletedAt) return null;\n\n return {\n status: 'regular' as const,\n remoteId: id,\n title: chat.title || 'New Chat',\n createdAt: chat.createdAt ? new Date(chat.createdAt) : new Date(),\n updatedAt: chat.updatedAt ? new Date(chat.updatedAt) : new Date(),\n };\n })\n .filter((t) => t !== null) as Array<{\n status: 'regular' | 'archived';\n remoteId: string;\n title?: string;\n createdAt: Date;\n updatedAt: Date;\n }>;\n\n return { threads };\n },\n\n async initialize(threadId: string) {\n // For LocalRuntime, threadId is already generated locally (e.g., __LOCALID_...)\n // Check if we have a server chat ID for this local thread ID\n const serverChatId = localToRemoteIdMap.get(threadId);\n const remoteId = serverChatId || threadId;\n\n return { remoteId, externalId: undefined };\n },\n\n async rename(remoteId: string, newTitle: string) {\n const actualId = resolveServerId(remoteId);\n await client.updateChat(actualId, { title: newTitle });\n },\n\n async archive(_remoteId: string) {\n // Cuadra API uses soft delete, which we can treat as archive\n // If API adds explicit archive support, implement here\n },\n\n async unarchive(_remoteId: string) {\n // If Cuadra supports unarchive, implement here\n // Currently not supported\n },\n\n async delete(remoteId: string) {\n const actualId = resolveServerId(remoteId);\n await client.deleteChat(actualId);\n },\n\n async generateTitle(remoteId: string, _messages: readonly ThreadMessage[]) {\n const actualId = resolveServerId(remoteId, _messages);\n\n // Call the backend title generation endpoint.\n // The API uses a fast model to analyze the conversation and returns a title.\n let title: string | undefined;\n try {\n const result = await client.generateChatTitle(actualId);\n title = result.title;\n } catch (error) {\n // eslint-disable-next-line no-console -- Intentional: surface title-generation failures for debugging\n console.error('[generateTitle] API call failed, falling back to first message', error);\n // Fallback: extract from first user message if the endpoint fails\n const firstUserMsg = _messages.find((m) => m.role === 'user');\n const rawText = firstUserMsg?.content\n ?.filter((p): p is { type: 'text'; text: string } => p.type === 'text')\n .map((p) => p.text)\n .join(' ')\n .trim();\n title = rawText\n ? rawText.length > 60\n ? rawText.slice(0, 57) + '...'\n : rawText\n : undefined;\n }\n\n return createAssistantStream((controller) => {\n const text = controller.addTextPart();\n if (title) {\n text.append(title);\n }\n text.close();\n controller.close();\n });\n },\n\n async fetch(threadId: string) {\n try {\n const chat = await client.getChat(threadId);\n const id = chat.id || threadId;\n\n return {\n status: 'regular' as const,\n remoteId: id,\n title: chat.title || 'Chat',\n createdAt: chat.createdAt ? new Date(chat.createdAt) : new Date(),\n updatedAt: chat.updatedAt ? new Date(chat.updatedAt) : new Date(),\n };\n } catch {\n // Return default if fetch fails\n return {\n status: 'regular' as const,\n remoteId: threadId,\n createdAt: new Date(),\n updatedAt: new Date(),\n };\n }\n },\n\n // Add a method to update the remoteId mapping when chat ID is received\n updateRemoteId(localThreadId: string, serverChatId: string) {\n localToRemoteIdMap.set(localThreadId, serverChatId);\n onChatIdReceived?.(localThreadId, serverChatId);\n },\n\n // Provider component that adds thread-specific history adapter\n unstable_Provider: ({ children }: { children?: React.ReactNode }) => {\n const threadListItem = useThreadListItem();\n const remoteId = threadListItem?.remoteId;\n const isMain = threadListItem?.isMain;\n\n // Keep activeThreadRemoteId in sync so the chat model adapter\n // can look it up when a server chat ID arrives during streaming.\n // Only the active (main) thread sets the global — multiple Provider\n // instances are mounted simultaneously (one per thread).\n // We intentionally allow clearing to `undefined` when a new thread\n // is created (remoteId is undefined for local-only threads).\n React.useEffect(() => {\n if (isMain) {\n activeThreadRemoteId = remoteId;\n }\n }, [remoteId, isMain]);\n\n // Create thread-specific history adapter\n const history = React.useMemo<ThreadHistoryAdapter>(\n () => ({\n async load() {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n if (!remoteId) return { messages: [] } as any;\n\n try {\n // Fetch chat with messages expanded\n const chat = await client.getChat(remoteId);\n \n if (!chat.messages || chat.messages.length === 0) {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n return { messages: [] } as any;\n }\n\n // Convert API messages to ExportedMessageRepository format\n // Each item needs { message, parentId } where parentId links to the previous message\n const converted = chat.messages.map((msg) => convertFromCuadraMessage(msg));\n const messages = converted.map((message, idx) => ({\n message,\n parentId: idx > 0 ? converted[idx - 1]!.id : null,\n }));\n\n // Register every message ID → chatId so the model adapter\n // can resolve which chat this thread belongs to\n for (const msg of converted) {\n messageIdToChatIdMap.set(msg.id, remoteId);\n }\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n return { messages } as any;\n } catch (_error) {\n // Failed to load messages for thread\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n return { messages: [] } as any;\n }\n },\n\n async append(_message) {\n // Messages are automatically persisted by the API when sent\n // This method is called by the runtime but we don't need to do anything\n // since messages are already saved via the chat API\n },\n }),\n [remoteId, client],\n );\n\n const adapters = React.useMemo(() => ({ history }), [history]);\n\n return (\n <RuntimeAdapterProvider adapters={adapters}>\n {children}\n </RuntimeAdapterProvider>\n );\n },\n };\n}\n\n","import type { Source } from '../types/cuadra';\n\n/**\n * Metadata stored per message\n * Keyed by message/chat ID to persist across the conversation\n */\nexport interface MessageMetadata {\n /** RAG sources/citations used to generate the response */\n sources?: Source[];\n}\n\n/**\n * Store for per-message metadata (sources)\n * Keyed by message ID so each message retains its own sources\n * Reasoning is handled via native assistant-ui content parts\n */\nclass StreamingMetadataStore {\n /** Per-message metadata, keyed by message ID */\n private messageMetadata: Map<string, MessageMetadata> = new Map();\n /** Current streaming message ID */\n private currentMessageId: string | null = null;\n private listeners: Set<() => void> = new Set();\n\n /**\n * Set the current streaming message ID\n */\n setCurrentMessage(messageId: string): void {\n this.currentMessageId = messageId;\n }\n\n /**\n * Update sources for the current message\n */\n update(data: Partial<MessageMetadata>): void {\n if (!this.currentMessageId) return;\n \n const existing = this.messageMetadata.get(this.currentMessageId) || {};\n const newSources = data.sources;\n \n // Only update if sources actually changed\n if (newSources && newSources.length > 0) {\n this.messageMetadata.set(this.currentMessageId, {\n ...existing,\n sources: newSources,\n });\n this.notify();\n }\n }\n\n /**\n * Get metadata for the current streaming message\n */\n get(): MessageMetadata {\n if (!this.currentMessageId) return {};\n return this.messageMetadata.get(this.currentMessageId) || {};\n }\n\n /**\n * Get metadata for a specific message ID\n */\n getForMessage(messageId: string): MessageMetadata {\n return this.messageMetadata.get(messageId) || {};\n }\n\n /**\n * Get the current message ID\n */\n getCurrentMessageId(): string | null {\n return this.currentMessageId;\n }\n\n /**\n * Clear current message (but keep historical metadata)\n * Called when starting a new message stream\n */\n clear(): void {\n this.currentMessageId = null;\n // Don't clear messageMetadata - keep historical sources!\n }\n\n /**\n * Subscribe to changes\n */\n subscribe(listener: () => void): () => void {\n this.listeners.add(listener);\n return () => this.listeners.delete(listener);\n }\n\n private notify(): void {\n this.listeners.forEach(listener => listener());\n }\n}\n\n// Singleton instance\nexport const streamingMetadataStore = new StreamingMetadataStore();\n\n// Re-export the old type name for compatibility\nexport type StreamingMetadata = MessageMetadata;\n\n","import type { ChatModelAdapter, ChatModelRunOptions } from '@assistant-ui/react';\nimport { CuadraChatClient, parseSSEStream, type SSEChunk } from '../lib/cuadraChatClient';\nimport { convertToCuadraMessages } from './messageConverter';\nimport { activeThreadRemoteId, localToRemoteIdMap, messageIdToChatIdMap } from './threadListAdapter';\nimport { streamingMetadataStore } from './streamingMetadataStore';\nimport type { Source } from '../types/cuadra';\n\n// Re-export for convenience\nexport type { StreamingMetadata } from './streamingMetadataStore';\nexport { streamingMetadataStore } from './streamingMetadataStore';\n\n// Chat ID resolution uses a single strategy: messageIdToChatIdMap.\n// Thread identification uses unstable_threadId from run() options (v0.12+),\n// falling back to activeThreadRemoteId global for compatibility.\n\n/**\n * Queue for pre-made responses to intercept API calls\n * When a pre-made suggestion is clicked, the response is queued here\n * The adapter will return this response instead of calling the real API\n */\ninterface PreMadeResponse {\n question: string;\n response: string;\n streamingSpeed?: number;\n initialDelay?: number;\n}\n\nconst preMadeResponseQueue: PreMadeResponse[] = [];\n\n/**\n * Store for pre-made conversation history\n * These messages will be included in subsequent API calls for context\n */\ninterface PreMadeMessage {\n role: 'user' | 'assistant';\n content: string;\n}\n\nlet preMadeConversationHistory: PreMadeMessage[] = [];\n\n/**\n * Queue a pre-made response to be returned instead of calling the API\n * Call this before triggering the normal send flow\n * @param question - The user's question\n * @param response - The text to return as the assistant's response\n * @param streamingSpeed - Speed in ms between words (default: 50)\n * @param initialDelay - Delay in ms before streaming starts (default: 800)\n */\nexport function queuePreMadeResponse(question: string, response: string, streamingSpeed: number = 50, initialDelay: number = 800): void {\n preMadeResponseQueue.push({ question, response, streamingSpeed, initialDelay });\n}\n\n/**\n * Check if there's a pending pre-made response\n */\nexport function hasPendingPreMadeResponse(): boolean {\n return preMadeResponseQueue.length > 0;\n}\n\n/**\n * Clear all pending pre-made responses and conversation history\n */\nexport function clearPreMadeResponses(): void {\n preMadeResponseQueue.length = 0;\n preMadeConversationHistory = [];\n}\n\n/**\n * Get the pre-made conversation history for context\n */\nexport function getPreMadeConversationHistory(): PreMadeMessage[] {\n return preMadeConversationHistory;\n}\n\nexport interface ChatModelAdapterOptions {\n modelId?: string | null; // Optional - backend can resolve model ID in proxy mode\n /**\n * Lazy getter for modelId. When provided, called at request time instead of\n * using the static `modelId`. Useful to avoid recreating the adapter (and\n * rebuilding the runtime) every time the selected model changes.\n */\n getModelId?: () => string | null | undefined;\n systemPrompt?: string;\n ephemeral?: boolean;\n /** Enable reasoning/thinking output for supported models (e.g., o1, o3) */\n enableReasoning?: boolean;\n onChatCreated?: (chatId: string) => void;\n onUserMessage?: () => void; // Called when a user message is sent\n onThreadIdUpdate?: (localThreadId: string, serverChatId: string) => void; // Called when chat ID is received to update thread mapping\n /** Interceptor called before each API request. Return true to allow, false to block. */\n onBeforeRequest?: () => boolean | Promise<boolean>;\n /** \n * Mock response to show when onBeforeRequest returns false.\n * If set, instead of throwing an error, this message will be streamed as a response.\n * Can be a string or a function that returns a string (called each time a request is blocked).\n * Useful for showing helpful messages like \"Complete setup before chatting\".\n */\n mockResponseOnBlocked?: string | (() => string);\n}\n\n/**\n * Creates a ChatModelAdapter for LocalRuntime that connects to Cuadra API\n * Handles streaming responses and converts between assistant-ui and API formats\n */\nexport function createChatModelAdapter(\n client: CuadraChatClient,\n options: ChatModelAdapterOptions,\n): ChatModelAdapter {\n const { modelId: staticModelId, getModelId, systemPrompt, ephemeral, enableReasoning, onChatCreated, onUserMessage, onThreadIdUpdate, onBeforeRequest, mockResponseOnBlocked } = options;\n const resolveModelId = () => (getModelId ? getModelId() : staticModelId) ?? null;\n \n return {\n async *run(options: ChatModelRunOptions) {\n const { messages, abortSignal } = options;\n // unstable_threadId is the remoteId of the current thread, provided\n // directly by the runtime (available since @assistant-ui/react 0.12).\n // For brand-new threads it may be undefined at run start.\n const threadId = (options as { unstable_threadId?: string }).unstable_threadId;\n \n // Check if there's a pending pre-made response\n const pendingPreMade = preMadeResponseQueue.shift();\n if (pendingPreMade) {\n // Simulate the API response with streaming\n onUserMessage?.();\n \n // Initial \"thinking\" delay before streaming starts\n const initialDelay = pendingPreMade.initialDelay ?? 800;\n if (initialDelay > 0) {\n await new Promise(resolve => setTimeout(resolve, initialDelay));\n }\n \n // Check if aborted during delay\n if (abortSignal?.aborted) {\n throw new Error('Request aborted');\n }\n \n const words = pendingPreMade.response.split(' ');\n let accumulatedText = '';\n const speed = pendingPreMade.streamingSpeed || 50;\n \n // Generate a fake chat ID for ephemeral pre-made chats\n const fakeChatId = `premade-${Date.now()}`;\n \n // If ephemeral, we don't need a real chat ID\n // If not ephemeral, you might want to still create a real chat\n if (ephemeral) {\n onChatCreated?.(fakeChatId);\n }\n \n // Simulate streaming word by word\n for (let i = 0; i < words.length; i++) {\n if (abortSignal?.aborted) {\n throw new Error('Request aborted');\n }\n \n accumulatedText += (i === 0 ? '' : ' ') + words[i];\n \n yield {\n content: [{ type: 'text', text: accumulatedText }],\n };\n \n // Delay between words for streaming effect\n await new Promise(resolve => setTimeout(resolve, speed + Math.random() * (speed * 0.5)));\n }\n \n // Final yield\n yield {\n content: [{ type: 'text', text: accumulatedText }],\n };\n \n // Store the pre-made Q&A in conversation history for context in follow-up calls\n preMadeConversationHistory.push(\n { role: 'user', content: pendingPreMade.question },\n { role: 'assistant', content: pendingPreMade.response }\n );\n \n return;\n }\n \n // Resolve the server chat ID for this conversation.\n // The history loader registers every message ID → chatId in\n // messageIdToChatIdMap. We scan the current messages for a match.\n // For new threads (first message), no match is found and the server\n // creates a new chat.\n let serverChatId: string | undefined;\n\n for (const msg of messages) {\n const mapped = messageIdToChatIdMap.get(msg.id);\n if (mapped) {\n serverChatId = mapped;\n break;\n }\n }\n\n onUserMessage?.();\n\n // Convert assistant-ui ThreadMessage format to API MessageCreate format\n let messagesToSend = messages;\n \n // Find the last assistant message index\n let lastAssistantIndex = -1;\n for (let i = messages.length - 1; i >= 0; i--) {\n if (messages[i].role === 'assistant') {\n lastAssistantIndex = i;\n break;\n }\n }\n \n // If we have a chatId OR we have assistant messages (indicating we're continuing a conversation),\n // only send new messages (messages after the last assistant response)\n // This is because the backend already has the conversation history\n if (serverChatId || lastAssistantIndex >= 0) {\n // If we found an assistant message, only send messages after it (new user messages)\n if (lastAssistantIndex >= 0) {\n messagesToSend = messages.slice(lastAssistantIndex + 1);\n }\n // If we have chatId but no assistant message yet, keep all messages as fallback\n // This ensures we don't lose messages in edge cases\n }\n\n // If we have pre-made conversation history, prepend it to the API messages\n // This provides context for follow-up messages after a pre-made Q&A\n const apiMessages = convertToCuadraMessages(messagesToSend);\n \n if (preMadeConversationHistory.length > 0 && !serverChatId) {\n // Only include pre-made history for the first real API call (no chatId yet)\n // Convert pre-made history directly to API format and prepend\n const preMadeApiMessages = preMadeConversationHistory.map(msg => ({\n role: msg.role as 'user' | 'assistant',\n content: msg.content,\n }));\n \n // Prepend pre-made history to the API messages\n apiMessages.unshift(...preMadeApiMessages);\n \n // Clear the pre-made history after using it (it's now part of the real conversation)\n preMadeConversationHistory = [];\n }\n\n // Upload any file attachments from the last user message\n const fileIds: string[] = [];\n let lastUserMessage: (typeof messagesToSend)[number] | undefined;\n for (let i = messagesToSend.length - 1; i >= 0; i--) {\n if (messagesToSend[i]!.role === 'user') { lastUserMessage = messagesToSend[i]; break; }\n }\n if (lastUserMessage && 'attachments' in lastUserMessage) {\n const attachments = (lastUserMessage as { attachments?: Array<{ file?: File }> }).attachments;\n if (attachments && attachments.length > 0) {\n for (const attachment of attachments) {\n if (attachment.file) {\n try {\n const uploaded = await client.uploadFile(attachment.file);\n fileIds.push(uploaded.id);\n } catch (_uploadError) {\n // File upload is best-effort — don't block the message\n }\n }\n }\n }\n }\n\n // Create request\n // modelId is optional - backend can resolve it in proxy mode\n const request: {\n messages: typeof apiMessages;\n chatId: string | null;\n modelId?: string | null;\n systemPrompt?: string;\n ephemeral?: boolean;\n enableReasoning?: boolean;\n stream: boolean;\n fileIds?: string[];\n } = {\n messages: apiMessages,\n chatId: serverChatId || null, // Use server chatId if available\n systemPrompt,\n ephemeral,\n enableReasoning,\n stream: true,\n };\n \n // Only include modelId if provided (backend may resolve it)\n const currentModelId = resolveModelId();\n if (currentModelId && currentModelId.trim() !== '') {\n request.modelId = currentModelId;\n }\n\n // Include file IDs if any files were uploaded\n if (fileIds.length > 0) {\n request.fileIds = fileIds;\n }\n\n // Call interceptor before making the API request\n if (onBeforeRequest) {\n const shouldProceed = await onBeforeRequest();\n if (!shouldProceed) {\n // Request was blocked by interceptor\n // If mockResponseOnBlocked is set, show it instead of throwing\n if (mockResponseOnBlocked) {\n // Get the response text (call function if it's a function)\n const responseText = typeof mockResponseOnBlocked === 'function' \n ? mockResponseOnBlocked() \n : mockResponseOnBlocked;\n\n // Get the last user message to use as the \"question\"\n const userMessages = messages.filter(m => m.role === 'user');\n const lastUserMessage = userMessages.length > 0 ? userMessages[userMessages.length - 1] : undefined;\n let userQuestion = 'Message';\n if (lastUserMessage) {\n if (typeof lastUserMessage.content === 'string') {\n userQuestion = lastUserMessage.content;\n } else if (Array.isArray(lastUserMessage.content)) {\n const textPart = lastUserMessage.content.find(\n (c: { type: string; text?: string }) => c.type === 'text'\n ) as { type: string; text?: string } | undefined;\n userQuestion = textPart?.text || 'Message';\n }\n }\n \n // Queue the mock response and let the pre-made handler process it\n queuePreMadeResponse(userQuestion, responseText, 40, 400);\n \n // Process the queued response (same as the pre-made response flow above)\n const pendingMock = preMadeResponseQueue.shift();\n if (pendingMock) {\n onUserMessage?.();\n \n // Initial \"thinking\" delay before streaming starts\n const initialDelay = pendingMock.initialDelay ?? 400;\n if (initialDelay > 0) {\n await new Promise(resolve => setTimeout(resolve, initialDelay));\n }\n \n // Check if aborted during delay\n if (abortSignal?.aborted) {\n throw new Error('Request aborted');\n }\n \n const words = pendingMock.response.split(' ');\n let accumulatedText = '';\n const speed = pendingMock.streamingSpeed || 40;\n \n // Generate a fake chat ID for the mock response\n const fakeChatId = `mock-${Date.now()}`;\n if (ephemeral) {\n onChatCreated?.(fakeChatId);\n }\n \n // Simulate streaming word by word\n for (let i = 0; i < words.length; i++) {\n if (abortSignal?.aborted) {\n throw new Error('Request aborted');\n }\n \n accumulatedText += (i === 0 ? '' : ' ') + words[i];\n \n yield {\n content: [{ type: 'text', text: accumulatedText }],\n };\n \n // Delay between words for streaming effect\n await new Promise(resolve => setTimeout(resolve, speed + Math.random() * (speed * 0.3)));\n }\n \n // Final yield\n yield {\n content: [{ type: 'text', text: accumulatedText }],\n };\n \n return;\n }\n }\n \n throw new Error('Request blocked');\n }\n }\n\n // Stream from API\n const stream = await client.createOrContinueChat(request, abortSignal);\n\n let accumulatedText = '';\n let accumulatedReasoning = '';\n let chatId: string | undefined;\n const sources: Source[] = [];\n let lastSourcesCount = 0;\n let hasAddedReasoningToTextDelay = false; // Track if we've added delay between reasoning and text\n\n // Generate a unique ID for this message stream\n const streamMessageId = `msg_${Date.now()}_${Math.random().toString(36).substring(2, 9)}`;\n \n // Set up store for this new message (clears current but keeps history)\n streamingMetadataStore.clear();\n streamingMetadataStore.setCurrentMessage(streamMessageId);\n\n // Helper to update store only when sources change (reasoning now goes through content parts)\n const updateStoreIfNeeded = () => {\n const sourcesChanged = sources.length !== lastSourcesCount;\n \n if (sourcesChanged) {\n lastSourcesCount = sources.length;\n streamingMetadataStore.update({\n sources: sources.length > 0 ? [...sources] : undefined,\n });\n }\n };\n\n // Helper to create yield object with proper content parts\n // Uses native assistant-ui reasoning content parts\n const createYieldObject = () => {\n updateStoreIfNeeded();\n \n // Build content array with reasoning first (if any), then text\n const content: Array<{ type: 'reasoning' | 'text'; text: string }> = [];\n \n if (accumulatedReasoning) {\n content.push({ type: 'reasoning' as const, text: accumulatedReasoning });\n }\n \n if (accumulatedText) {\n content.push({ type: 'text' as const, text: accumulatedText });\n }\n \n // Fallback to empty text if nothing yet\n if (content.length === 0) {\n content.push({ type: 'text' as const, text: '' });\n }\n \n return { content };\n };\n\n // Parse SSE stream\n for await (const chunk of parseSSEStream(stream, abortSignal)) {\n // Check abort signal\n if (abortSignal?.aborted) {\n throw new Error('Request aborted');\n }\n\n // Handle chat ID from chunk\n const chunkId = chunk.id || chunk.chatId || chunk.messageId;\n if (chunkId && !chatId) {\n chatId = chunkId;\n // Prefer the threadId from run() options (stable, provided by runtime).\n // Fall back to activeThreadRemoteId which may have been updated by the\n // time the SSE chunk arrives (after HTTP roundtrip gives React time\n // to flush the useEffect that sets it).\n const threadIdForStorage = threadId || activeThreadRemoteId;\n \n onChatCreated?.(chunkId);\n \n // Register all current messages against this chat ID so\n // future calls to run() can resolve the chat\n for (const msg of messages) {\n messageIdToChatIdMap.set(msg.id, chunkId);\n }\n \n // Map the active local thread ID → server chat ID so that\n // rename / delete / generateTitle can resolve the real ID.\n if (threadIdForStorage && !localToRemoteIdMap.has(threadIdForStorage)) {\n localToRemoteIdMap.set(threadIdForStorage, chunkId);\n }\n\n // Notify the thread list adapter about the ID update\n if (onThreadIdUpdate && threadIdForStorage) {\n onThreadIdUpdate(threadIdForStorage, chunkId);\n }\n }\n\n // Handle sources array (sent in first chunk typically)\n const chunkSources = (chunk as SSEChunk & { sources?: Source[] | null }).sources;\n if (chunkSources && Array.isArray(chunkSources) && chunkSources.length > 0) {\n // Add new sources (avoid duplicates by sourceId)\n for (const source of chunkSources) {\n if (!sources.some(s => s.sourceId === source.sourceId)) {\n sources.push({\n sourceId: source.sourceId,\n filename: source.filename || 'Unknown',\n score: source.score ?? 1.0,\n chunkId: source.chunkId,\n datasetId: source.datasetId,\n });\n }\n }\n }\n\n // Handle reasoning field (string delta on each chunk)\n const chunkReasoning = (chunk as SSEChunk & { reasoning?: string }).reasoning;\n if (chunkReasoning && chunkReasoning.length > 0) {\n // Reasoning comes as delta - accumulate it\n accumulatedReasoning += chunkReasoning;\n // Yield immediately to show reasoning as it streams\n yield createYieldObject();\n }\n\n // Handle AI SDK Protocol events (for future compatibility)\n if (chunk.type) {\n switch (chunk.type) {\n case 'source-document':\n if (chunk.sourceId) {\n sources.push({\n sourceId: chunk.sourceId,\n filename: chunk.title || 'Unknown',\n score: 1.0,\n });\n }\n continue;\n \n case 'reasoning-start':\n // Reasoning now flows through content parts\n continue;\n \n case 'reasoning-delta':\n if (chunk.delta) {\n accumulatedReasoning += chunk.delta;\n }\n if (accumulatedText || accumulatedReasoning) {\n yield createYieldObject();\n }\n continue;\n \n case 'reasoning-end':\n // Reasoning now flows through content parts\n if (accumulatedText || accumulatedReasoning) {\n yield createYieldObject();\n }\n continue;\n \n case 'text-delta':\n if (chunk.delta) {\n // Add small delay between reasoning and first text chunk for better UX\n if (accumulatedReasoning && !hasAddedReasoningToTextDelay && !accumulatedText) {\n hasAddedReasoningToTextDelay = true;\n await new Promise(resolve => setTimeout(resolve, 400));\n }\n accumulatedText += chunk.delta;\n }\n if (accumulatedText) {\n yield createYieldObject();\n }\n continue;\n \n case 'error':\n throw new Error(chunk.errorText || 'Stream error');\n \n case 'text-start':\n case 'text-end':\n case 'start':\n case 'start-step':\n case 'finish-step':\n case 'finish':\n continue;\n }\n }\n\n // Handle current API format: {\"id\": \"...\", \"delta\": \"...\", \"reasoning\": \"\", \"sources\": [...], \"finished\": false}\n const delta = chunk.delta;\n if (delta !== undefined && delta !== null) {\n // Add small delay between reasoning and first text chunk for better UX\n if (accumulatedReasoning && !hasAddedReasoningToTextDelay && !accumulatedText) {\n hasAddedReasoningToTextDelay = true;\n await new Promise(resolve => setTimeout(resolve, 400));\n }\n accumulatedText += delta;\n }\n\n // Handle legacy formats for compatibility\n if (chunk.content) {\n accumulatedText += chunk.content;\n }\n\n // Handle message object format\n if (chunk.message) {\n accumulatedText += chunk.message.content || '';\n }\n\n // Check if finished\n if (chunk.finished) {\n // Stream is complete, yield final accumulated result\n if (accumulatedText || accumulatedReasoning || sources.length > 0) {\n yield createYieldObject();\n }\n break;\n }\n\n // Yield incremental updates\n if (accumulatedText || accumulatedReasoning || sources.length > 0) {\n yield createYieldObject();\n }\n }\n\n // Final yield with accumulated content\n if (accumulatedText || accumulatedReasoning || sources.length > 0) {\n yield createYieldObject();\n }\n },\n };\n}\n\n","import type {\n AttachmentAdapter,\n CompleteAttachment,\n PendingAttachment,\n} from '@assistant-ui/react';\n\n/**\n * Attachment validation configuration\n */\nexport interface AttachmentValidationConfig {\n /**\n * Maximum file size in bytes\n * @default 10 * 1024 * 1024 (10MB)\n */\n maxFileSizeBytes?: number;\n\n /**\n * Allowed MIME types (e.g., ['image/png', 'image/jpeg'])\n * If not provided, defaults to common document and image types\n */\n allowedMimeTypes?: string[];\n\n /**\n * Allowed file extensions (e.g., ['.pdf', '.docx'])\n * If not provided, defaults to common document and image extensions\n */\n allowedExtensions?: string[];\n\n /**\n * Maximum number of attachments per message\n * @default 5\n */\n maxAttachments?: number;\n\n /**\n * Whether to validate file content type matches extension\n * @default true\n */\n validateContentType?: boolean;\n}\n\n/**\n * Attachment validation error\n */\nexport class AttachmentValidationError extends Error {\n constructor(\n message: string,\n public readonly code: 'FILE_TOO_LARGE' | 'INVALID_TYPE' | 'TOO_MANY_FILES' | 'VALIDATION_FAILED' | 'VISION_NOT_SUPPORTED',\n public readonly file?: File\n ) {\n super(message);\n this.name = 'AttachmentValidationError';\n }\n}\n\n/**\n * Attachment adapter options\n */\nexport interface AttachmentAdapterOptions {\n /**\n * Validation configuration\n */\n validation?: AttachmentValidationConfig;\n\n /**\n * Whether the current model supports vision/image inputs.\n * When false, image attachments will be rejected with a clear error.\n * @default true\n */\n supportsVision?: boolean;\n\n /**\n * Callback when validation fails\n * Allows custom error handling (e.g., show toast notification)\n */\n onValidationError?: (error: AttachmentValidationError) => void;\n\n /**\n * Callback when attachment is added\n */\n onAttachmentAdded?: (attachment: PendingAttachment) => void;\n\n /**\n * Callback when attachment is removed\n */\n onAttachmentRemoved?: (attachmentId: string) => void;\n}\n\n/**\n * Default allowed MIME types\n */\nconst DEFAULT_ALLOWED_MIME_TYPES = [\n // Images\n 'image/jpeg',\n 'image/png',\n 'image/gif',\n 'image/webp',\n 'image/svg+xml',\n // Documents\n 'application/pdf',\n 'application/msword',\n 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',\n 'application/vnd.ms-excel',\n 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',\n // Text\n 'text/plain',\n 'text/markdown',\n 'text/csv',\n 'application/json',\n];\n\n/**\n * Default allowed file extensions\n */\nconst DEFAULT_ALLOWED_EXTENSIONS = [\n '.pdf',\n '.doc',\n '.docx',\n '.xls',\n '.xlsx',\n '.txt',\n '.md',\n '.csv',\n '.json',\n '.jpg',\n '.jpeg',\n '.png',\n '.gif',\n '.webp',\n '.svg',\n];\n\n/**\n * Default max file size (10MB)\n */\nconst DEFAULT_MAX_FILE_SIZE = 10 * 1024 * 1024;\n\n/**\n * Format bytes to human-readable size\n */\nfunction formatFileSize(bytes: number): string {\n if (bytes === 0) return '0 Bytes';\n \n const k = 1024;\n const sizes = ['Bytes', 'KB', 'MB', 'GB'];\n const i = Math.floor(Math.log(bytes) / Math.log(k));\n \n return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];\n}\n\n/**\n * Get file extension from filename\n */\nfunction getFileExtension(filename: string): string {\n const ext = filename.lastIndexOf('.') >= 0 \n ? filename.slice(filename.lastIndexOf('.')).toLowerCase() \n : '';\n return ext;\n}\n\n/**\n * Check if MIME type matches extension (basic validation)\n */\nfunction mimeMatchesExtension(mimeType: string, extension: string): boolean {\n const mimeToExtMap: Record<string, string[]> = {\n 'image/jpeg': ['.jpg', '.jpeg'],\n 'image/png': ['.png'],\n 'image/gif': ['.gif'],\n 'image/webp': ['.webp'],\n 'image/svg+xml': ['.svg'],\n 'application/pdf': ['.pdf'],\n 'text/plain': ['.txt'],\n 'text/markdown': ['.md'],\n 'text/csv': ['.csv'],\n 'application/json': ['.json'],\n };\n \n const validExtensions = mimeToExtMap[mimeType];\n if (!validExtensions) {\n // Unknown MIME type, allow it\n return true;\n }\n \n return validExtensions.includes(extension.toLowerCase());\n}\n\n/**\n * Generate a secure attachment ID\n */\nfunction generateAttachmentId(): string {\n // Use crypto.randomUUID if available\n if (typeof crypto !== 'undefined' && crypto.randomUUID) {\n return 'attachment-' + crypto.randomUUID();\n }\n \n // Fallback\n return 'attachment-' + Date.now() + '-' + Math.random().toString(36).slice(2, 11);\n}\n\n/**\n * Create an attachment adapter with enhanced validation\n */\nexport function createAttachmentAdapter(\n options: AttachmentAdapterOptions = {}\n): AttachmentAdapter {\n const {\n validation = {},\n supportsVision = true,\n onValidationError,\n onAttachmentAdded,\n onAttachmentRemoved,\n } = options;\n\n const {\n maxFileSizeBytes = DEFAULT_MAX_FILE_SIZE,\n allowedMimeTypes = DEFAULT_ALLOWED_MIME_TYPES,\n allowedExtensions = DEFAULT_ALLOWED_EXTENSIONS,\n validateContentType = true,\n } = validation;\n\n // Build accept string from allowed extensions\n const acceptString = allowedExtensions.join(',');\n\n /**\n * Validate a file before adding it as an attachment\n */\n function validateFile(file: File): void {\n // Check file size\n if (file.size > maxFileSizeBytes) {\n const error = new AttachmentValidationError(\n 'File \"' + file.name + '\" is too large (' + formatFileSize(file.size) + '). Maximum size is ' + formatFileSize(maxFileSizeBytes) + '.',\n 'FILE_TOO_LARGE',\n file\n );\n onValidationError?.(error);\n throw error;\n }\n\n // Check vision support for image files\n if (!supportsVision && file.type.startsWith('image/')) {\n const error = new AttachmentValidationError(\n 'The current model does not support image inputs. Please select a model with vision capabilities to attach images.',\n 'VISION_NOT_SUPPORTED',\n file\n );\n onValidationError?.(error);\n throw error;\n }\n\n // Check extension\n const extension = getFileExtension(file.name);\n if (extension && !allowedExtensions.some(ext => ext.toLowerCase() === extension)) {\n const error = new AttachmentValidationError(\n 'File type \"' + extension + '\" is not allowed. Allowed types: ' + allowedExtensions.join(', '),\n 'INVALID_TYPE',\n file\n );\n onValidationError?.(error);\n throw error;\n }\n\n // Check MIME type\n if (file.type && !allowedMimeTypes.some(mime => {\n // Support wildcards like 'image/*'\n if (mime.endsWith('/*')) {\n const prefix = mime.slice(0, -1);\n return file.type.startsWith(prefix);\n }\n return file.type === mime;\n })) {\n const error = new AttachmentValidationError(\n 'File type \"' + file.type + '\" is not allowed.',\n 'INVALID_TYPE',\n file\n );\n onValidationError?.(error);\n throw error;\n }\n\n // Validate MIME type matches extension\n if (validateContentType && file.type && extension) {\n if (!mimeMatchesExtension(file.type, extension)) {\n const error = new AttachmentValidationError(\n 'File extension \"' + extension + '\" does not match content type \"' + file.type + '\".',\n 'VALIDATION_FAILED',\n file\n );\n onValidationError?.(error);\n throw error;\n }\n }\n }\n\n return {\n accept: acceptString,\n\n async add({ file }): Promise<PendingAttachment> {\n // Validate the file\n validateFile(file);\n\n // Determine attachment type based on file\n let type: 'image' | 'document' | 'file' = 'file';\n\n if (file.type.startsWith('image/')) {\n type = 'image';\n } else if (\n file.type.includes('pdf') ||\n file.type.includes('document') ||\n file.type.includes('text') ||\n file.name.endsWith('.pdf') ||\n file.name.endsWith('.docx') ||\n file.name.endsWith('.txt') ||\n file.name.endsWith('.md') ||\n file.name.endsWith('.csv') ||\n file.name.endsWith('.json')\n ) {\n type = 'document';\n }\n\n const attachment: PendingAttachment = {\n id: generateAttachmentId(),\n type,\n name: file.name,\n contentType: file.type,\n file,\n status: { type: 'requires-action', reason: 'composer-send' },\n };\n\n onAttachmentAdded?.(attachment);\n\n return attachment;\n },\n\n async remove(attachment) {\n onAttachmentRemoved?.(attachment.id);\n // No other cleanup needed - files are handled by the browser\n },\n\n async send(attachment: PendingAttachment): Promise<CompleteAttachment> {\n // For images, create a data URL\n if (attachment.type === 'image' && attachment.file) {\n const dataURL = await new Promise<string>((resolve, reject) => {\n const reader = new FileReader();\n reader.onload = () => resolve(reader.result as string);\n reader.onerror = () => reject(new Error('Failed to read file'));\n reader.readAsDataURL(attachment.file!);\n });\n\n return {\n ...attachment,\n status: { type: 'complete' },\n content: [\n {\n type: 'image',\n image: dataURL,\n },\n ],\n };\n }\n\n // For other files, return as-is\n return {\n ...attachment,\n status: { type: 'complete' },\n content: [],\n };\n },\n };\n}\n","/**\n * Lightweight reactive store for surfacing attachment validation errors in the UI.\n *\n * The attachment adapter runs outside React's tree (inside @assistant-ui/react internals),\n * so we use a tiny pub/sub store — identical to the pattern used by streamingMetadataStore.\n */\n\ntype Listener = () => void;\n\nexport interface AttachmentError {\n message: string;\n code: string;\n timestamp: number;\n}\n\nclass AttachmentErrorStore {\n private current: AttachmentError | null = null;\n private listeners = new Set<Listener>();\n\n /** Push a new error (triggers subscribers). */\n push(message: string, code: string) {\n this.current = { message, code, timestamp: Date.now() };\n this.notify();\n }\n\n /** Clear the current error (triggers subscribers). */\n clear() {\n if (this.current) {\n this.current = null;\n this.notify();\n }\n }\n\n /** Get the current error (snapshot for useSyncExternalStore). */\n getSnapshot = (): AttachmentError | null => this.current;\n\n /** Subscribe (for useSyncExternalStore). Returns unsubscribe function. */\n subscribe = (listener: Listener): (() => void) => {\n this.listeners.add(listener);\n return () => this.listeners.delete(listener);\n };\n\n private notify() {\n this.listeners.forEach((l) => l());\n }\n}\n\nexport const attachmentErrorStore = new AttachmentErrorStore();\n","import React, { useCallback, useEffect, useMemo, useRef, useState } from 'react';\nimport {\n AssistantRuntimeProvider,\n type ThreadMessageLike,\n unstable_useRemoteThreadListRuntime,\n useLocalRuntime,\n} from '@assistant-ui/react';\nimport { CuadraChatClient } from '../lib/cuadraChatClient';\nimport { createChatModelAdapter } from '../adapters/chatModelAdapter';\nimport { createThreadListAdapter, messageIdToChatIdMap } from '../adapters/threadListAdapter';\nimport { convertFromCuadraMessage } from '../adapters/messageConverter';\nimport { type AttachmentValidationError, createAttachmentAdapter } from '../adapters/attachmentAdapter';\nimport { attachmentErrorStore } from '../adapters/attachmentErrorStore';\n\nexport interface CuadraRuntimeProviderProps {\n children: React.ReactNode;\n baseUrl: string;\n sessionToken?: string | null;\n isProxyMode?: boolean; // If true, uses proxy URL patterns (removes /v1 prefix)\n\n // Mode selection\n mode?: 'singleChat' | 'multiChat'; // Default: 'multiChat'\n\n // Model configuration\n modelMode?: 'selector' | 'fixed'; // Default: 'fixed'\n modelId?: string; // Required if modelMode === 'fixed'\n onModelChange?: (modelId: string) => void; // Called when model changes (selector mode)\n\n // Chat configuration\n ephemeral?: boolean; // Default: false - creates temporary chats that auto-delete\n systemPrompt?: string;\n /** Enable reasoning/thinking output for supported models (e.g., o1, o3) */\n enableReasoning?: boolean;\n\n // Thread configuration\n initialThreadId?: string; // Load existing thread (singleChat fetches messages; multiChat uses thread list)\n\n // Callbacks\n onError?: (error: Error) => void;\n onChatCreated?: (chatId: string) => void;\n onUserMessage?: () => void; // Called when a user message is sent\n onThreadIdUpdate?: (\n oldThreadId: string,\n newThreadId: string,\n ) => void; // Callback when thread ID is updated from local to server\n onChatsLoaded?: () => void; // Callback when chats/thread list is loaded\n /** Enable file attachments in the chat input */\n enableAttachments?: boolean;\n /** Whether the current model supports vision/image inputs. When false, image attachments are rejected. */\n supportsVision?: boolean;\n /** Interceptor called before each API request. Return true to allow, false to block. */\n onBeforeRequest?: () => boolean | Promise<boolean>;\n /** \n * Mock response to show when onBeforeRequest returns false.\n * If set, instead of throwing an error, this message will be streamed as a response.\n * Can be a string or a function that returns a string (called each time a request is blocked).\n * Useful for showing helpful messages like \"Complete setup before chatting\".\n */\n mockResponseOnBlocked?: string | (() => string);\n}\n\n/**\n * Main provider component that sets up assistant-ui runtime with Cuadra API\n * Supports single/multi chat modes, fixed/selector model modes, and ephemeral chats\n */\nexport function CuadraRuntimeProvider({\n children,\n baseUrl,\n sessionToken,\n isProxyMode = false,\n mode = 'multiChat',\n modelId,\n onModelChange: _onModelChange,\n ephemeral = false,\n systemPrompt,\n enableReasoning = false,\n initialThreadId,\n onError,\n onChatCreated,\n onUserMessage,\n onThreadIdUpdate,\n onChatsLoaded,\n enableAttachments = false,\n supportsVision = true,\n onBeforeRequest,\n mockResponseOnBlocked,\n}: CuadraRuntimeProviderProps) {\n // Create API client\n const client = useMemo(\n () => new CuadraChatClient(baseUrl, sessionToken || undefined, isProxyMode),\n [baseUrl, sessionToken, isProxyMode],\n );\n\n // Track selected model ID (syncs with prop)\n const [selectedModelId, setSelectedModelId] = useState<string | null>(modelId || null);\n\n // Sync selectedModelId with modelId prop\n useEffect(() => {\n if (modelId) {\n setSelectedModelId(modelId);\n }\n }, [modelId]);\n\n // Handle chat created callback\n const handleChatCreated = useCallback(\n (chatId: string) => {\n onChatCreated?.(chatId);\n },\n [onChatCreated],\n );\n\n // Handle thread ID update\n const handleThreadIdUpdate = useCallback(\n (oldId: string, newId: string) => {\n onThreadIdUpdate?.(oldId, newId);\n },\n [onThreadIdUpdate],\n );\n\n // For singleChat mode, create model adapter and use useLocalRuntime\n const modelAdapter = useMemo(() => {\n return createChatModelAdapter(client, {\n modelId: selectedModelId || undefined, // Pass undefined if not set (backend resolves it)\n systemPrompt,\n ephemeral,\n enableReasoning,\n onChatCreated: handleChatCreated,\n onUserMessage,\n onBeforeRequest,\n mockResponseOnBlocked,\n });\n }, [client, selectedModelId, systemPrompt, ephemeral, enableReasoning, handleChatCreated, onUserMessage, onBeforeRequest, mockResponseOnBlocked]);\n\n if (mode === 'singleChat') {\n return (\n <SingleChatProvider\n client={client}\n initialThreadId={initialThreadId}\n modelAdapter={modelAdapter}\n enableAttachments={enableAttachments}\n supportsVision={supportsVision}\n onChatCreated={handleChatCreated}\n >\n {children}\n </SingleChatProvider>\n );\n }\n\n // For multiChat mode, use RemoteThreadListRuntime\n // Pass adapter options instead of the adapter itself so we can add thread ID update callback\n return (\n <MultiChatProvider\n client={client}\n modelAdapterOptions={{\n modelId: selectedModelId || undefined,\n systemPrompt,\n ephemeral,\n enableReasoning,\n onChatCreated: handleChatCreated,\n onUserMessage,\n onBeforeRequest,\n mockResponseOnBlocked,\n }}\n initialThreadId={initialThreadId}\n onChatsLoaded={onChatsLoaded}\n onThreadIdUpdate={handleThreadIdUpdate}\n onError={onError}\n enableAttachments={enableAttachments}\n supportsVision={supportsVision}\n >\n {children}\n </MultiChatProvider>\n );\n}\n\n/**\n * Internal provider for singleChat mode using useLocalRuntime.\n *\n * Reacts to `initialThreadId` changes without requiring the parent to\n * change the `key` prop on CuadraChat. Only the inner\n * `SingleChatRuntime` is remounted (via `runtimeKey`) when messages\n * finish loading, keeping the outer chrome (border, theme, etc.) stable.\n */\nfunction SingleChatProvider({\n client,\n initialThreadId,\n modelAdapter,\n enableAttachments,\n supportsVision = true,\n onChatCreated,\n children,\n}: {\n client: CuadraChatClient;\n initialThreadId?: string;\n modelAdapter: ReturnType<typeof createChatModelAdapter>;\n enableAttachments?: boolean;\n supportsVision?: boolean;\n onChatCreated?: (chatId: string) => void;\n children: React.ReactNode;\n}) {\n const [initialMessages, setInitialMessages] = useState<ThreadMessageLike[] | undefined>(undefined);\n const [isLoadingThread, setIsLoadingThread] = useState(!!initialThreadId);\n\n // Key that remounts only SingleChatRuntime when a different thread is loaded.\n const [runtimeKey, setRuntimeKey] = useState<string>(initialThreadId ?? 'new');\n\n // Use a ref for onChatCreated so the fetch effect doesn't re-run\n // when the callback reference changes (which happens on every parent render).\n const onChatCreatedRef = useRef(onChatCreated);\n onChatCreatedRef.current = onChatCreated;\n\n // Fetch existing thread messages when initialThreadId is provided or changes.\n useEffect(() => {\n if (!initialThreadId) {\n setInitialMessages(undefined);\n setRuntimeKey('new');\n setIsLoadingThread(false);\n return;\n }\n\n setIsLoadingThread(true);\n let cancelled = false;\n\n async function loadThread() {\n try {\n const chat = await client.getChat(initialThreadId!);\n\n if (cancelled) return;\n\n if (chat.messages && chat.messages.length > 0) {\n const converted: ThreadMessageLike[] = chat.messages.map((msg) =>\n convertFromCuadraMessage(msg),\n );\n\n // Pre-register every message ID → chatId so the adapter's run()\n // resolves the server chat and appends to the existing thread.\n for (const msg of chat.messages) {\n messageIdToChatIdMap.set(msg.id, chat.id);\n }\n\n setInitialMessages(converted);\n } else {\n setInitialMessages(undefined);\n }\n\n // Notify parent about the loaded chat\n onChatCreatedRef.current?.(chat.id);\n } catch (_err) {\n // If fetching fails, fall back to a fresh chat silently\n setInitialMessages(undefined);\n } finally {\n if (!cancelled) {\n setRuntimeKey(initialThreadId!);\n setIsLoadingThread(false);\n }\n }\n }\n\n loadThread();\n\n return () => {\n cancelled = true;\n };\n }, [client, initialThreadId]);\n\n // Always render the runtime — never return null.\n // During loading we show a centered spinner; this keeps the outer\n // chrome (border, card, theme wrapper) stable across thread switches.\n return (\n <SingleChatRuntime\n key={runtimeKey}\n modelAdapter={modelAdapter}\n initialMessages={isLoadingThread ? undefined : initialMessages}\n enableAttachments={enableAttachments}\n supportsVision={supportsVision}\n >\n {isLoadingThread ? <ThreadLoadingIndicator /> : children}\n </SingleChatRuntime>\n );\n}\n\n/** Minimal loading indicator shown while fetching an existing thread. */\nfunction ThreadLoadingIndicator() {\n return (\n <div className=\"flex items-center justify-center h-full w-full\">\n <div className=\"h-6 w-6 animate-spin rounded-full border-2 border-muted-foreground/30 border-t-muted-foreground\" />\n </div>\n );\n}\n\n/**\n * Thin wrapper that calls useLocalRuntime with the resolved initialMessages.\n * Separated so that SingleChatProvider can delay rendering until the fetch\n * completes, keeping hooks unconditional.\n */\nfunction SingleChatRuntime({\n modelAdapter,\n initialMessages,\n enableAttachments,\n supportsVision = true,\n children,\n}: {\n modelAdapter: ReturnType<typeof createChatModelAdapter>;\n initialMessages?: ThreadMessageLike[];\n enableAttachments?: boolean;\n supportsVision?: boolean;\n children: React.ReactNode;\n}) {\n // Create attachment adapter only if enabled\n const attachmentAdapter = useMemo(() => {\n return enableAttachments ? createAttachmentAdapter({\n supportsVision,\n onValidationError: (error: AttachmentValidationError) => {\n attachmentErrorStore.push(error.message, error.code);\n },\n }) : undefined;\n }, [enableAttachments, supportsVision]);\n \n const localRuntime = useLocalRuntime(modelAdapter, {\n initialMessages,\n adapters: {\n ...(attachmentAdapter && { attachments: attachmentAdapter }),\n },\n });\n\n return (\n <AssistantRuntimeProvider runtime={localRuntime}>\n {children}\n </AssistantRuntimeProvider>\n );\n}\n\n/**\n * Internal provider for multiChat mode using RemoteThreadListRuntime\n */\nfunction MultiChatProvider({\n client,\n modelAdapterOptions,\n initialThreadId: _initialThreadId,\n onChatsLoaded,\n onThreadIdUpdate,\n onError: _onError,\n enableAttachments,\n supportsVision,\n children,\n}: {\n client: CuadraChatClient;\n modelAdapterOptions: {\n modelId?: string | null;\n systemPrompt?: string;\n ephemeral?: boolean;\n enableReasoning?: boolean;\n onChatCreated?: (chatId: string) => void;\n onUserMessage?: () => void;\n onBeforeRequest?: () => boolean | Promise<boolean>;\n mockResponseOnBlocked?: string | (() => string);\n };\n initialThreadId?: string;\n onChatsLoaded?: () => void;\n onThreadIdUpdate?: (oldId: string, newId: string) => void;\n onError?: (error: Error) => void;\n enableAttachments?: boolean;\n supportsVision?: boolean;\n children: React.ReactNode;\n}) {\n // Create thread list adapter\n const threadListAdapter = useMemo(() => {\n return createThreadListAdapter(client, (localThreadId, serverChatId) => {\n onThreadIdUpdate?.(localThreadId, serverChatId);\n });\n }, [client, onThreadIdUpdate]);\n\n // Create a callback to update thread ID mapping when chat is created\n const handleThreadIdUpdateFromAdapter = useCallback(\n (localThreadId: string, serverChatId: string) => {\n // Update the adapter's remote ID mapping\n if (threadListAdapter.updateRemoteId) {\n threadListAdapter.updateRemoteId(localThreadId, serverChatId);\n }\n // Also call the prop callback\n onThreadIdUpdate?.(localThreadId, serverChatId);\n },\n [threadListAdapter, onThreadIdUpdate],\n );\n\n // Use a ref for onChatsLoaded so wrappedAdapter stays stable\n const onChatsLoadedRef = useRef(onChatsLoaded);\n onChatsLoadedRef.current = onChatsLoaded;\n\n // Wrap list method to call onChatsLoaded (stable — only depends on threadListAdapter)\n const wrappedAdapter = useMemo(() => {\n return {\n ...threadListAdapter,\n async list() {\n const result = await threadListAdapter.list();\n onChatsLoadedRef.current?.();\n return result;\n },\n };\n }, [threadListAdapter]);\n\n // Expose adapter globally for external access (e.g., rename/delete)\n useEffect(() => {\n (window as Window & { __cuadraThreadListAdapter?: typeof wrappedAdapter }).__cuadraThreadListAdapter =\n wrappedAdapter;\n (window as Window & { __cuadraThreadListRuntime?: unknown }).__cuadraThreadListRuntime = null;\n }, [wrappedAdapter]);\n\n // Keep modelId in a ref so changing it doesn't recreate the adapter\n // (which would rebuild the runtime and lose the current thread).\n // The adapter reads the ref lazily at request time via getModelId.\n const modelIdRef = useRef(modelAdapterOptions.modelId);\n modelIdRef.current = modelAdapterOptions.modelId;\n\n // Stable options that exclude modelId (modelId is read via ref)\n const stableAdapterOptions = useMemo(() => {\n const { modelId: _modelId, ...rest } = modelAdapterOptions;\n return rest;\n }, [\n modelAdapterOptions.systemPrompt,\n modelAdapterOptions.ephemeral,\n modelAdapterOptions.enableReasoning,\n modelAdapterOptions.onChatCreated,\n modelAdapterOptions.onUserMessage,\n modelAdapterOptions.onBeforeRequest,\n modelAdapterOptions.mockResponseOnBlocked,\n ]);\n\n // Create model adapter with thread ID update callback for multiChat mode\n const modelAdapter = useMemo(() => {\n return createChatModelAdapter(client, {\n ...stableAdapterOptions,\n getModelId: () => modelIdRef.current,\n onThreadIdUpdate: handleThreadIdUpdateFromAdapter,\n });\n }, [client, stableAdapterOptions, handleThreadIdUpdateFromAdapter]);\n\n // Create attachment adapter only if enabled\n const attachmentAdapter = useMemo(() => {\n return enableAttachments ? createAttachmentAdapter({\n supportsVision,\n onValidationError: (error: AttachmentValidationError) => {\n attachmentErrorStore.push(error.message, error.code);\n },\n }) : undefined;\n }, [enableAttachments, supportsVision]);\n \n // Create runtime hook that uses the model adapter\n const runtimeHook = useCallback(() => {\n return useLocalRuntime(modelAdapter, {\n adapters: {\n ...(attachmentAdapter && { attachments: attachmentAdapter }),\n },\n });\n }, [modelAdapter, attachmentAdapter]);\n\n // Use RemoteThreadListRuntime hook\n const runtime = unstable_useRemoteThreadListRuntime({\n adapter: wrappedAdapter,\n runtimeHook,\n });\n\n // Expose runtime globally for external refresh\n useEffect(() => {\n (window as Window & { __cuadraThreadListRuntime?: typeof runtime }).__cuadraThreadListRuntime =\n runtime;\n }, [runtime]);\n\n return (\n <AssistantRuntimeProvider runtime={runtime}>{children}</AssistantRuntimeProvider>\n );\n}\n","/**\n * Count how often a character (or substring) is used in a string.\n *\n * @param {string} value\n * Value to search in.\n * @param {string} character\n * Character (or substring) to look for.\n * @return {number}\n * Number of times `character` occurred in `value`.\n */\nexport function ccount(value, character) {\n const source = String(value)\n\n if (typeof character !== 'string') {\n throw new TypeError('Expected character')\n }\n\n let count = 0\n let index = source.indexOf(character)\n\n while (index !== -1) {\n count++\n index = source.indexOf(character, index + character.length)\n }\n\n return count\n}\n","/**\n * @import {Code} from 'micromark-util-types'\n */\n\n/**\n * Check whether the character code represents an ASCII alpha (`a` through `z`,\n * case insensitive).\n *\n * An **ASCII alpha** is an ASCII upper alpha or ASCII lower alpha.\n *\n * An **ASCII upper alpha** is a character in the inclusive range U+0041 (`A`)\n * to U+005A (`Z`).\n *\n * An **ASCII lower alpha** is a character in the inclusive range U+0061 (`a`)\n * to U+007A (`z`).\n *\n * @param code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport const asciiAlpha = regexCheck(/[A-Za-z]/);\n\n/**\n * Check whether the character code represents an ASCII alphanumeric (`a`\n * through `z`, case insensitive, or `0` through `9`).\n *\n * An **ASCII alphanumeric** is an ASCII digit (see `asciiDigit`) or ASCII alpha\n * (see `asciiAlpha`).\n *\n * @param code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport const asciiAlphanumeric = regexCheck(/[\\dA-Za-z]/);\n\n/**\n * Check whether the character code represents an ASCII atext.\n *\n * atext is an ASCII alphanumeric (see `asciiAlphanumeric`), or a character in\n * the inclusive ranges U+0023 NUMBER SIGN (`#`) to U+0027 APOSTROPHE (`'`),\n * U+002A ASTERISK (`*`), U+002B PLUS SIGN (`+`), U+002D DASH (`-`), U+002F\n * SLASH (`/`), U+003D EQUALS TO (`=`), U+003F QUESTION MARK (`?`), U+005E\n * CARET (`^`) to U+0060 GRAVE ACCENT (`` ` ``), or U+007B LEFT CURLY BRACE\n * (`{`) to U+007E TILDE (`~`).\n *\n * See:\n * **\\[RFC5322]**:\n * [Internet Message Format](https://tools.ietf.org/html/rfc5322).\n * P. Resnick.\n * IETF.\n *\n * @param code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport const asciiAtext = regexCheck(/[#-'*+\\--9=?A-Z^-~]/);\n\n/**\n * Check whether a character code is an ASCII control character.\n *\n * An **ASCII control** is a character in the inclusive range U+0000 NULL (NUL)\n * to U+001F (US), or U+007F (DEL).\n *\n * @param {Code} code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport function asciiControl(code) {\n return (\n // Special whitespace codes (which have negative values), C0 and Control\n // character DEL\n code !== null && (code < 32 || code === 127)\n );\n}\n\n/**\n * Check whether the character code represents an ASCII digit (`0` through `9`).\n *\n * An **ASCII digit** is a character in the inclusive range U+0030 (`0`) to\n * U+0039 (`9`).\n *\n * @param code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport const asciiDigit = regexCheck(/\\d/);\n\n/**\n * Check whether the character code represents an ASCII hex digit (`a` through\n * `f`, case insensitive, or `0` through `9`).\n *\n * An **ASCII hex digit** is an ASCII digit (see `asciiDigit`), ASCII upper hex\n * digit, or an ASCII lower hex digit.\n *\n * An **ASCII upper hex digit** is a character in the inclusive range U+0041\n * (`A`) to U+0046 (`F`).\n *\n * An **ASCII lower hex digit** is a character in the inclusive range U+0061\n * (`a`) to U+0066 (`f`).\n *\n * @param code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport const asciiHexDigit = regexCheck(/[\\dA-Fa-f]/);\n\n/**\n * Check whether the character code represents ASCII punctuation.\n *\n * An **ASCII punctuation** is a character in the inclusive ranges U+0021\n * EXCLAMATION MARK (`!`) to U+002F SLASH (`/`), U+003A COLON (`:`) to U+0040 AT\n * SIGN (`@`), U+005B LEFT SQUARE BRACKET (`[`) to U+0060 GRAVE ACCENT\n * (`` ` ``), or U+007B LEFT CURLY BRACE (`{`) to U+007E TILDE (`~`).\n *\n * @param code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport const asciiPunctuation = regexCheck(/[!-/:-@[-`{-~]/);\n\n/**\n * Check whether a character code is a markdown line ending.\n *\n * A **markdown line ending** is the virtual characters M-0003 CARRIAGE RETURN\n * LINE FEED (CRLF), M-0004 LINE FEED (LF) and M-0005 CARRIAGE RETURN (CR).\n *\n * In micromark, the actual character U+000A LINE FEED (LF) and U+000D CARRIAGE\n * RETURN (CR) are replaced by these virtual characters depending on whether\n * they occurred together.\n *\n * @param {Code} code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport function markdownLineEnding(code) {\n return code !== null && code < -2;\n}\n\n/**\n * Check whether a character code is a markdown line ending (see\n * `markdownLineEnding`) or markdown space (see `markdownSpace`).\n *\n * @param {Code} code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport function markdownLineEndingOrSpace(code) {\n return code !== null && (code < 0 || code === 32);\n}\n\n/**\n * Check whether a character code is a markdown space.\n *\n * A **markdown space** is the concrete character U+0020 SPACE (SP) and the\n * virtual characters M-0001 VIRTUAL SPACE (VS) and M-0002 HORIZONTAL TAB (HT).\n *\n * In micromark, the actual character U+0009 CHARACTER TABULATION (HT) is\n * replaced by one M-0002 HORIZONTAL TAB (HT) and between 0 and 3 M-0001 VIRTUAL\n * SPACE (VS) characters, depending on the column at which the tab occurred.\n *\n * @param {Code} code\n * Code.\n * @returns {boolean}\n * Whether it matches.\n */\nexport function markdownSpace(code) {\n return code === -2 || code === -1 || code === 32;\n}\n\n// Size note: removing ASCII from the regex and using `asciiPunctuation` here\n// In fact adds to the bundle size.\n/**\n * Check whether the character code represents Unicode punctuation.\n *\n * A **Unicode punctuation** is a character in the Unicode `Pc` (Punctuation,\n * Connector), `Pd` (Punctuation, Dash), `Pe` (Punctuation, Close), `Pf`\n * (Punctuation, Final quote), `Pi` (Punctuation, Initial quote), `Po`\n * (Punctuation, Other), or `Ps` (Punctuation, Open) categories, or an ASCII\n * punctuation (see `asciiPunctuation`).\n *\n * See:\n * **\\[UNICODE]**:\n * [The Unicode Standard](https://www.unicode.org/versions/).\n * Unicode Consortium.\n *\n * @param code\n * Code.\n * @returns\n * Whether it matches.\n */\nexport const unicodePunctuation = regexCheck(/\\p{P}|\\p{S}/u);\n\n/**\n * Check whether the character code represents Unicode whitespace.\n *\n * Note that this does handle micromark specific markdown whitespace characters.\n * See `markdownLineEndingOrSpace` to check that.\n *\n * A **Unicode whitespace** is a character in the Unicode `Zs` (Separator,\n * Space) category, or U+0009 CHARACTER TABULATION (HT), U+000A LINE FEED (LF),\n * U+000C (FF), or U+000D CARRIAGE RETURN (CR) (**\\[UNICODE]**).\n *\n * See:\n * **\\[UNICODE]**:\n * [The Unicode Standard](https://www.unicode.org/versions/).\n * Unicode Consortium.\n *\n * @param code\n * Code.\n * @returns\n * Whether it matches.\n */\nexport const unicodeWhitespace = regexCheck(/\\s/);\n\n/**\n * Create a code check from a regex.\n *\n * @param {RegExp} regex\n * Expression.\n * @returns {(code: Code) => boolean}\n * Check.\n */\nfunction regexCheck(regex) {\n return check;\n\n /**\n * Check whether a code matches the bound regex.\n *\n * @param {Code} code\n * Character code.\n * @returns {boolean}\n * Whether the character code matches the bound regex.\n */\n function check(code) {\n return code !== null && code > -1 && regex.test(String.fromCharCode(code));\n }\n}","export default function escapeStringRegexp(string) {\n\tif (typeof string !== 'string') {\n\t\tthrow new TypeError('Expected a string');\n\t}\n\n\t// Escape characters with special meaning either inside or outside character sets.\n\t// Use a simple backslash escape when it’s always valid, and a `\\xnn` escape when the simpler form would be disallowed by Unicode patterns’ stricter grammar.\n\treturn string\n\t\t.replace(/[|\\\\{}()[\\]^$+*?.]/g, '\\\\$&')\n\t\t.replace(/-/g, '\\\\x2d');\n}\n","/**\n * @import {Node, Parent} from 'unist'\n */\n\n/**\n * @template Fn\n * @template Fallback\n * @typedef {Fn extends (value: any) => value is infer Thing ? Thing : Fallback} Predicate\n */\n\n/**\n * @callback Check\n * Check that an arbitrary value is a node.\n * @param {unknown} this\n * The given context.\n * @param {unknown} [node]\n * Anything (typically a node).\n * @param {number | null | undefined} [index]\n * The node’s position in its parent.\n * @param {Parent | null | undefined} [parent]\n * The node’s parent.\n * @returns {boolean}\n * Whether this is a node and passes a test.\n *\n * @typedef {Record<string, unknown> | Node} Props\n * Object to check for equivalence.\n *\n * Note: `Node` is included as it is common but is not indexable.\n *\n * @typedef {Array<Props | TestFunction | string> | ReadonlyArray<Props | TestFunction | string> | Props | TestFunction | string | null | undefined} Test\n * Check for an arbitrary node.\n *\n * @callback TestFunction\n * Check if a node passes a test.\n * @param {unknown} this\n * The given context.\n * @param {Node} node\n * A node.\n * @param {number | undefined} [index]\n * The node’s position in its parent.\n * @param {Parent | undefined} [parent]\n * The node’s parent.\n * @returns {boolean | undefined | void}\n * Whether this node passes the test.\n *\n * Note: `void` is included until TS sees no return as `undefined`.\n */\n\n/**\n * Check if `node` is a `Node` and whether it passes the given test.\n *\n * @param {unknown} node\n * Thing to check, typically `Node`.\n * @param {Test} test\n * A check for a specific node.\n * @param {number | null | undefined} index\n * The node’s position in its parent.\n * @param {Parent | null | undefined} parent\n * The node’s parent.\n * @param {unknown} context\n * Context object (`this`) to pass to `test` functions.\n * @returns {boolean}\n * Whether `node` is a node and passes a test.\n */\nexport const is =\n // Note: overloads in JSDoc can’t yet use different `@template`s.\n /**\n * @type {(\n * (<Condition extends ReadonlyArray<string>>(node: unknown, test: Condition, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & {type: Condition[number]}) &\n * (<Condition extends Array<string>>(node: unknown, test: Condition, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & {type: Condition[number]}) &\n * (<Condition extends string>(node: unknown, test: Condition, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & {type: Condition}) &\n * (<Condition extends Props>(node: unknown, test: Condition, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & Condition) &\n * (<Condition extends TestFunction>(node: unknown, test: Condition, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & Predicate<Condition, Node>) &\n * ((node?: null | undefined) => false) &\n * ((node: unknown, test?: null | undefined, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node) &\n * ((node: unknown, test?: Test, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => boolean)\n * )}\n */\n (\n /**\n * @param {unknown} [node]\n * @param {Test} [test]\n * @param {number | null | undefined} [index]\n * @param {Parent | null | undefined} [parent]\n * @param {unknown} [context]\n * @returns {boolean}\n */\n // eslint-disable-next-line max-params\n function (node, test, index, parent, context) {\n const check = convert(test)\n\n if (\n index !== undefined &&\n index !== null &&\n (typeof index !== 'number' ||\n index < 0 ||\n index === Number.POSITIVE_INFINITY)\n ) {\n throw new Error('Expected positive finite index')\n }\n\n if (\n parent !== undefined &&\n parent !== null &&\n (!is(parent) || !parent.children)\n ) {\n throw new Error('Expected parent node')\n }\n\n if (\n (parent === undefined || parent === null) !==\n (index === undefined || index === null)\n ) {\n throw new Error('Expected both parent and index')\n }\n\n return looksLikeANode(node)\n ? check.call(context, node, index, parent)\n : false\n }\n )\n\n/**\n * Generate an assertion from a test.\n *\n * Useful if you’re going to test many nodes, for example when creating a\n * utility where something else passes a compatible test.\n *\n * The created function is a bit faster because it expects valid input only:\n * a `node`, `index`, and `parent`.\n *\n * @param {Test} test\n * * when nullish, checks if `node` is a `Node`.\n * * when `string`, works like passing `(node) => node.type === test`.\n * * when `function` checks if function passed the node is true.\n * * when `object`, checks that all keys in test are in node, and that they have (strictly) equal values.\n * * when `array`, checks if any one of the subtests pass.\n * @returns {Check}\n * An assertion.\n */\nexport const convert =\n // Note: overloads in JSDoc can’t yet use different `@template`s.\n /**\n * @type {(\n * (<Condition extends string>(test: Condition) => (node: unknown, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & {type: Condition}) &\n * (<Condition extends Props>(test: Condition) => (node: unknown, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & Condition) &\n * (<Condition extends TestFunction>(test: Condition) => (node: unknown, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node & Predicate<Condition, Node>) &\n * ((test?: null | undefined) => (node?: unknown, index?: number | null | undefined, parent?: Parent | null | undefined, context?: unknown) => node is Node) &\n * ((test?: Test) => Check)\n * )}\n */\n (\n /**\n * @param {Test} [test]\n * @returns {Check}\n */\n function (test) {\n if (test === null || test === undefined) {\n return ok\n }\n\n if (typeof test === 'function') {\n return castFactory(test)\n }\n\n if (typeof test === 'object') {\n return Array.isArray(test)\n ? anyFactory(test)\n : // Cast because `ReadonlyArray` goes into the above but `isArray`\n // narrows to `Array`.\n propertiesFactory(/** @type {Props} */ (test))\n }\n\n if (typeof test === 'string') {\n return typeFactory(test)\n }\n\n throw new Error('Expected function, string, or object as test')\n }\n )\n\n/**\n * @param {Array<Props | TestFunction | string>} tests\n * @returns {Check}\n */\nfunction anyFactory(tests) {\n /** @type {Array<Check>} */\n const checks = []\n let index = -1\n\n while (++index < tests.length) {\n checks[index] = convert(tests[index])\n }\n\n return castFactory(any)\n\n /**\n * @this {unknown}\n * @type {TestFunction}\n */\n function any(...parameters) {\n let index = -1\n\n while (++index < checks.length) {\n if (checks[index].apply(this, parameters)) return true\n }\n\n return false\n }\n}\n\n/**\n * Turn an object into a test for a node with a certain fields.\n *\n * @param {Props} check\n * @returns {Check}\n */\nfunction propertiesFactory(check) {\n const checkAsRecord = /** @type {Record<string, unknown>} */ (check)\n\n return castFactory(all)\n\n /**\n * @param {Node} node\n * @returns {boolean}\n */\n function all(node) {\n const nodeAsRecord = /** @type {Record<string, unknown>} */ (\n /** @type {unknown} */ (node)\n )\n\n /** @type {string} */\n let key\n\n for (key in check) {\n if (nodeAsRecord[key] !== checkAsRecord[key]) return false\n }\n\n return true\n }\n}\n\n/**\n * Turn a string into a test for a node with a certain type.\n *\n * @param {string} check\n * @returns {Check}\n */\nfunction typeFactory(check) {\n return castFactory(type)\n\n /**\n * @param {Node} node\n */\n function type(node) {\n return node && node.type === check\n }\n}\n\n/**\n * Turn a custom test into a test for a node that passes that test.\n *\n * @param {TestFunction} testFunction\n * @returns {Check}\n */\nfunction castFactory(testFunction) {\n return check\n\n /**\n * @this {unknown}\n * @type {Check}\n */\n function check(value, index, parent) {\n return Boolean(\n looksLikeANode(value) &&\n testFunction.call(\n this,\n value,\n typeof index === 'number' ? index : undefined,\n parent || undefined\n )\n )\n }\n}\n\nfunction ok() {\n return true\n}\n\n/**\n * @param {unknown} value\n * @returns {value is Node}\n */\nfunction looksLikeANode(value) {\n return value !== null && typeof value === 'object' && 'type' in value\n}\n","/**\n * @import {Node as UnistNode, Parent as UnistParent} from 'unist'\n */\n\n/**\n * @typedef {Exclude<import('unist-util-is').Test, undefined> | undefined} Test\n * Test from `unist-util-is`.\n *\n * Note: we have remove and add `undefined`, because otherwise when generating\n * automatic `.d.ts` files, TS tries to flatten paths from a local perspective,\n * which doesn’t work when publishing on npm.\n */\n\n/**\n * @typedef {(\n * Fn extends (value: any) => value is infer Thing\n * ? Thing\n * : Fallback\n * )} Predicate\n * Get the value of a type guard `Fn`.\n * @template Fn\n * Value; typically function that is a type guard (such as `(x): x is Y`).\n * @template Fallback\n * Value to yield if `Fn` is not a type guard.\n */\n\n/**\n * @typedef {(\n * Check extends null | undefined // No test.\n * ? Value\n * : Value extends {type: Check} // String (type) test.\n * ? Value\n * : Value extends Check // Partial test.\n * ? Value\n * : Check extends Function // Function test.\n * ? Predicate<Check, Value> extends Value\n * ? Predicate<Check, Value>\n * : never\n * : never // Some other test?\n * )} MatchesOne\n * Check whether a node matches a primitive check in the type system.\n * @template Value\n * Value; typically unist `Node`.\n * @template Check\n * Value; typically `unist-util-is`-compatible test, but not arrays.\n */\n\n/**\n * @typedef {(\n * Check extends ReadonlyArray<infer T>\n * ? MatchesOne<Value, T>\n * : Check extends Array<infer T>\n * ? MatchesOne<Value, T>\n * : MatchesOne<Value, Check>\n * )} Matches\n * Check whether a node matches a check in the type system.\n * @template Value\n * Value; typically unist `Node`.\n * @template Check\n * Value; typically `unist-util-is`-compatible test.\n */\n\n/**\n * @typedef {0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10} Uint\n * Number; capped reasonably.\n */\n\n/**\n * @typedef {I extends 0 ? 1 : I extends 1 ? 2 : I extends 2 ? 3 : I extends 3 ? 4 : I extends 4 ? 5 : I extends 5 ? 6 : I extends 6 ? 7 : I extends 7 ? 8 : I extends 8 ? 9 : 10} Increment\n * Increment a number in the type system.\n * @template {Uint} [I=0]\n * Index.\n */\n\n/**\n * @typedef {(\n * Node extends UnistParent\n * ? Node extends {children: Array<infer Children>}\n * ? Child extends Children ? Node : never\n * : never\n * : never\n * )} InternalParent\n * Collect nodes that can be parents of `Child`.\n * @template {UnistNode} Node\n * All node types in a tree.\n * @template {UnistNode} Child\n * Node to search for.\n */\n\n/**\n * @typedef {InternalParent<InclusiveDescendant<Tree>, Child>} Parent\n * Collect nodes in `Tree` that can be parents of `Child`.\n * @template {UnistNode} Tree\n * All node types in a tree.\n * @template {UnistNode} Child\n * Node to search for.\n */\n\n/**\n * @typedef {(\n * Depth extends Max\n * ? never\n * :\n * | InternalParent<Node, Child>\n * | InternalAncestor<Node, InternalParent<Node, Child>, Max, Increment<Depth>>\n * )} InternalAncestor\n * Collect nodes in `Tree` that can be ancestors of `Child`.\n * @template {UnistNode} Node\n * All node types in a tree.\n * @template {UnistNode} Child\n * Node to search for.\n * @template {Uint} [Max=10]\n * Max; searches up to this depth.\n * @template {Uint} [Depth=0]\n * Current depth.\n */\n\n/**\n * @typedef {InternalAncestor<InclusiveDescendant<Tree>, Child>} Ancestor\n * Collect nodes in `Tree` that can be ancestors of `Child`.\n * @template {UnistNode} Tree\n * All node types in a tree.\n * @template {UnistNode} Child\n * Node to search for.\n */\n\n/**\n * @typedef {(\n * Tree extends UnistParent\n * ? Depth extends Max\n * ? Tree\n * : Tree | InclusiveDescendant<Tree['children'][number], Max, Increment<Depth>>\n * : Tree\n * )} InclusiveDescendant\n * Collect all (inclusive) descendants of `Tree`.\n *\n * > 👉 **Note**: for performance reasons, this seems to be the fastest way to\n * > recurse without actually running into an infinite loop, which the\n * > previous version did.\n * >\n * > Practically, a max of `2` is typically enough assuming a `Root` is\n * > passed, but it doesn’t improve performance.\n * > It gets higher with `List > ListItem > Table > TableRow > TableCell`.\n * > Using up to `10` doesn’t hurt or help either.\n * @template {UnistNode} Tree\n * Tree type.\n * @template {Uint} [Max=10]\n * Max; searches up to this depth.\n * @template {Uint} [Depth=0]\n * Current depth.\n */\n\n/**\n * @typedef {'skip' | boolean} Action\n * Union of the action types.\n *\n * @typedef {number} Index\n * Move to the sibling at `index` next (after node itself is completely\n * traversed).\n *\n * Useful if mutating the tree, such as removing the node the visitor is\n * currently on, or any of its previous siblings.\n * Results less than 0 or greater than or equal to `children.length` stop\n * traversing the parent.\n *\n * @typedef {[(Action | null | undefined | void)?, (Index | null | undefined)?]} ActionTuple\n * List with one or two values, the first an action, the second an index.\n *\n * @typedef {Action | ActionTuple | Index | null | undefined | void} VisitorResult\n * Any value that can be returned from a visitor.\n */\n\n/**\n * @callback Visitor\n * Handle a node (matching `test`, if given).\n *\n * Visitors are free to transform `node`.\n * They can also transform the parent of node (the last of `ancestors`).\n *\n * Replacing `node` itself, if `SKIP` is not returned, still causes its\n * descendants to be walked (which is a bug).\n *\n * When adding or removing previous siblings of `node` (or next siblings, in\n * case of reverse), the `Visitor` should return a new `Index` to specify the\n * sibling to traverse after `node` is traversed.\n * Adding or removing next siblings of `node` (or previous siblings, in case\n * of reverse) is handled as expected without needing to return a new `Index`.\n *\n * Removing the children property of an ancestor still results in them being\n * traversed.\n * @param {Visited} node\n * Found node.\n * @param {Array<VisitedParents>} ancestors\n * Ancestors of `node`.\n * @returns {VisitorResult}\n * What to do next.\n *\n * An `Index` is treated as a tuple of `[CONTINUE, Index]`.\n * An `Action` is treated as a tuple of `[Action]`.\n *\n * Passing a tuple back only makes sense if the `Action` is `SKIP`.\n * When the `Action` is `EXIT`, that action can be returned.\n * When the `Action` is `CONTINUE`, `Index` can be returned.\n * @template {UnistNode} [Visited=UnistNode]\n * Visited node type.\n * @template {UnistParent} [VisitedParents=UnistParent]\n * Ancestor type.\n */\n\n/**\n * @typedef {Visitor<Matches<InclusiveDescendant<Tree>, Check>, Ancestor<Tree, Matches<InclusiveDescendant<Tree>, Check>>>} BuildVisitor\n * Build a typed `Visitor` function from a tree and a test.\n *\n * It will infer which values are passed as `node` and which as `parents`.\n * @template {UnistNode} [Tree=UnistNode]\n * Tree type.\n * @template {Test} [Check=Test]\n * Test type.\n */\n\nimport {convert} from 'unist-util-is'\nimport {color} from 'unist-util-visit-parents/do-not-use-color'\n\n/** @type {Readonly<ActionTuple>} */\nconst empty = []\n\n/**\n * Continue traversing as normal.\n */\nexport const CONTINUE = true\n\n/**\n * Stop traversing immediately.\n */\nexport const EXIT = false\n\n/**\n * Do not traverse this node’s children.\n */\nexport const SKIP = 'skip'\n\n/**\n * Visit nodes, with ancestral information.\n *\n * This algorithm performs *depth-first* *tree traversal* in *preorder*\n * (**NLR**) or if `reverse` is given, in *reverse preorder* (**NRL**).\n *\n * You can choose for which nodes `visitor` is called by passing a `test`.\n * For complex tests, you should test yourself in `visitor`, as it will be\n * faster and will have improved type information.\n *\n * Walking the tree is an intensive task.\n * Make use of the return values of the visitor when possible.\n * Instead of walking a tree multiple times, walk it once, use `unist-util-is`\n * to check if a node matches, and then perform different operations.\n *\n * You can change the tree.\n * See `Visitor` for more info.\n *\n * @overload\n * @param {Tree} tree\n * @param {Check} check\n * @param {BuildVisitor<Tree, Check>} visitor\n * @param {boolean | null | undefined} [reverse]\n * @returns {undefined}\n *\n * @overload\n * @param {Tree} tree\n * @param {BuildVisitor<Tree>} visitor\n * @param {boolean | null | undefined} [reverse]\n * @returns {undefined}\n *\n * @param {UnistNode} tree\n * Tree to traverse.\n * @param {Visitor | Test} test\n * `unist-util-is`-compatible test\n * @param {Visitor | boolean | null | undefined} [visitor]\n * Handle each node.\n * @param {boolean | null | undefined} [reverse]\n * Traverse in reverse preorder (NRL) instead of the default preorder (NLR).\n * @returns {undefined}\n * Nothing.\n *\n * @template {UnistNode} Tree\n * Node type.\n * @template {Test} Check\n * `unist-util-is`-compatible test.\n */\nexport function visitParents(tree, test, visitor, reverse) {\n /** @type {Test} */\n let check\n\n if (typeof test === 'function' && typeof visitor !== 'function') {\n reverse = visitor\n // @ts-expect-error no visitor given, so `visitor` is test.\n visitor = test\n } else {\n // @ts-expect-error visitor given, so `test` isn’t a visitor.\n check = test\n }\n\n const is = convert(check)\n const step = reverse ? -1 : 1\n\n factory(tree, undefined, [])()\n\n /**\n * @param {UnistNode} node\n * @param {number | undefined} index\n * @param {Array<UnistParent>} parents\n */\n function factory(node, index, parents) {\n const value = /** @type {Record<string, unknown>} */ (\n node && typeof node === 'object' ? node : {}\n )\n\n if (typeof value.type === 'string') {\n const name =\n // `hast`\n typeof value.tagName === 'string'\n ? value.tagName\n : // `xast`\n typeof value.name === 'string'\n ? value.name\n : undefined\n\n Object.defineProperty(visit, 'name', {\n value:\n 'node (' + color(node.type + (name ? '<' + name + '>' : '')) + ')'\n })\n }\n\n return visit\n\n function visit() {\n /** @type {Readonly<ActionTuple>} */\n let result = empty\n /** @type {Readonly<ActionTuple>} */\n let subresult\n /** @type {number} */\n let offset\n /** @type {Array<UnistParent>} */\n let grandparents\n\n if (!test || is(node, index, parents[parents.length - 1] || undefined)) {\n // @ts-expect-error: `visitor` is now a visitor.\n result = toResult(visitor(node, parents))\n\n if (result[0] === EXIT) {\n return result\n }\n }\n\n if ('children' in node && node.children) {\n const nodeAsParent = /** @type {UnistParent} */ (node)\n\n if (nodeAsParent.children && result[0] !== SKIP) {\n offset = (reverse ? nodeAsParent.children.length : -1) + step\n grandparents = parents.concat(nodeAsParent)\n\n while (offset > -1 && offset < nodeAsParent.children.length) {\n const child = nodeAsParent.children[offset]\n\n subresult = factory(child, offset, grandparents)()\n\n if (subresult[0] === EXIT) {\n return subresult\n }\n\n offset =\n typeof subresult[1] === 'number' ? subresult[1] : offset + step\n }\n }\n }\n\n return result\n }\n }\n}\n\n/**\n * Turn a return value into a clean result.\n *\n * @param {VisitorResult} value\n * Valid return values from visitors.\n * @returns {Readonly<ActionTuple>}\n * Clean result.\n */\nfunction toResult(value) {\n if (Array.isArray(value)) {\n return value\n }\n\n if (typeof value === 'number') {\n return [CONTINUE, value]\n }\n\n return value === null || value === undefined ? empty : [value]\n}\n","/**\n * @import {Nodes, Parents, PhrasingContent, Root, Text} from 'mdast'\n * @import {BuildVisitor, Test, VisitorResult} from 'unist-util-visit-parents'\n */\n\n/**\n * @typedef RegExpMatchObject\n * Info on the match.\n * @property {number} index\n * The index of the search at which the result was found.\n * @property {string} input\n * A copy of the search string in the text node.\n * @property {[...Array<Parents>, Text]} stack\n * All ancestors of the text node, where the last node is the text itself.\n *\n * @typedef {RegExp | string} Find\n * Pattern to find.\n *\n * Strings are escaped and then turned into global expressions.\n *\n * @typedef {Array<FindAndReplaceTuple>} FindAndReplaceList\n * Several find and replaces, in array form.\n *\n * @typedef {[Find, Replace?]} FindAndReplaceTuple\n * Find and replace in tuple form.\n *\n * @typedef {ReplaceFunction | string | null | undefined} Replace\n * Thing to replace with.\n *\n * @callback ReplaceFunction\n * Callback called when a search matches.\n * @param {...any} parameters\n * The parameters are the result of corresponding search expression:\n *\n * * `value` (`string`) — whole match\n * * `...capture` (`Array<string>`) — matches from regex capture groups\n * * `match` (`RegExpMatchObject`) — info on the match\n * @returns {Array<PhrasingContent> | PhrasingContent | string | false | null | undefined}\n * Thing to replace with.\n *\n * * when `null`, `undefined`, `''`, remove the match\n * * …or when `false`, do not replace at all\n * * …or when `string`, replace with a text node of that value\n * * …or when `Node` or `Array<Node>`, replace with those nodes\n *\n * @typedef {[RegExp, ReplaceFunction]} Pair\n * Normalized find and replace.\n *\n * @typedef {Array<Pair>} Pairs\n * All find and replaced.\n *\n * @typedef Options\n * Configuration.\n * @property {Test | null | undefined} [ignore]\n * Test for which nodes to ignore (optional).\n */\n\nimport escape from 'escape-string-regexp'\nimport {visitParents} from 'unist-util-visit-parents'\nimport {convert} from 'unist-util-is'\n\n/**\n * Find patterns in a tree and replace them.\n *\n * The algorithm searches the tree in *preorder* for complete values in `Text`\n * nodes.\n * Partial matches are not supported.\n *\n * @param {Nodes} tree\n * Tree to change.\n * @param {FindAndReplaceList | FindAndReplaceTuple} list\n * Patterns to find.\n * @param {Options | null | undefined} [options]\n * Configuration (when `find` is not `Find`).\n * @returns {undefined}\n * Nothing.\n */\nexport function findAndReplace(tree, list, options) {\n const settings = options || {}\n const ignored = convert(settings.ignore || [])\n const pairs = toPairs(list)\n let pairIndex = -1\n\n while (++pairIndex < pairs.length) {\n visitParents(tree, 'text', visitor)\n }\n\n /** @type {BuildVisitor<Root, 'text'>} */\n function visitor(node, parents) {\n let index = -1\n /** @type {Parents | undefined} */\n let grandparent\n\n while (++index < parents.length) {\n const parent = parents[index]\n /** @type {Array<Nodes> | undefined} */\n const siblings = grandparent ? grandparent.children : undefined\n\n if (\n ignored(\n parent,\n siblings ? siblings.indexOf(parent) : undefined,\n grandparent\n )\n ) {\n return\n }\n\n grandparent = parent\n }\n\n if (grandparent) {\n return handler(node, parents)\n }\n }\n\n /**\n * Handle a text node which is not in an ignored parent.\n *\n * @param {Text} node\n * Text node.\n * @param {Array<Parents>} parents\n * Parents.\n * @returns {VisitorResult}\n * Result.\n */\n function handler(node, parents) {\n const parent = parents[parents.length - 1]\n const find = pairs[pairIndex][0]\n const replace = pairs[pairIndex][1]\n let start = 0\n /** @type {Array<Nodes>} */\n const siblings = parent.children\n const index = siblings.indexOf(node)\n let change = false\n /** @type {Array<PhrasingContent>} */\n let nodes = []\n\n find.lastIndex = 0\n\n let match = find.exec(node.value)\n\n while (match) {\n const position = match.index\n /** @type {RegExpMatchObject} */\n const matchObject = {\n index: match.index,\n input: match.input,\n stack: [...parents, node]\n }\n let value = replace(...match, matchObject)\n\n if (typeof value === 'string') {\n value = value.length > 0 ? {type: 'text', value} : undefined\n }\n\n // It wasn’t a match after all.\n if (value === false) {\n // False acts as if there was no match.\n // So we need to reset `lastIndex`, which currently being at the end of\n // the current match, to the beginning.\n find.lastIndex = position + 1\n } else {\n if (start !== position) {\n nodes.push({\n type: 'text',\n value: node.value.slice(start, position)\n })\n }\n\n if (Array.isArray(value)) {\n nodes.push(...value)\n } else if (value) {\n nodes.push(value)\n }\n\n start = position + match[0].length\n change = true\n }\n\n if (!find.global) {\n break\n }\n\n match = find.exec(node.value)\n }\n\n if (change) {\n if (start < node.value.length) {\n nodes.push({type: 'text', value: node.value.slice(start)})\n }\n\n parent.children.splice(index, 1, ...nodes)\n } else {\n nodes = [node]\n }\n\n return index + nodes.length\n }\n}\n\n/**\n * Turn a tuple or a list of tuples into pairs.\n *\n * @param {FindAndReplaceList | FindAndReplaceTuple} tupleOrList\n * Schema.\n * @returns {Pairs}\n * Clean pairs.\n */\nfunction toPairs(tupleOrList) {\n /** @type {Pairs} */\n const result = []\n\n if (!Array.isArray(tupleOrList)) {\n throw new TypeError('Expected find and replace tuple or list of tuples')\n }\n\n /** @type {FindAndReplaceList} */\n // @ts-expect-error: correct.\n const list =\n !tupleOrList[0] || Array.isArray(tupleOrList[0])\n ? tupleOrList\n : [tupleOrList]\n\n let index = -1\n\n while (++index < list.length) {\n const tuple = list[index]\n result.push([toExpression(tuple[0]), toFunction(tuple[1])])\n }\n\n return result\n}\n\n/**\n * Turn a find into an expression.\n *\n * @param {Find} find\n * Find.\n * @returns {RegExp}\n * Expression.\n */\nfunction toExpression(find) {\n return typeof find === 'string' ? new RegExp(escape(find), 'g') : find\n}\n\n/**\n * Turn a replace into a function.\n *\n * @param {Replace} replace\n * Replace.\n * @returns {ReplaceFunction}\n * Function.\n */\nfunction toFunction(replace) {\n return typeof replace === 'function'\n ? replace\n : function () {\n return replace\n }\n}\n","/**\n * @import {RegExpMatchObject, ReplaceFunction} from 'mdast-util-find-and-replace'\n * @import {CompileContext, Extension as FromMarkdownExtension, Handle as FromMarkdownHandle, Transform as FromMarkdownTransform} from 'mdast-util-from-markdown'\n * @import {ConstructName, Options as ToMarkdownExtension} from 'mdast-util-to-markdown'\n * @import {Link, PhrasingContent} from 'mdast'\n */\n\nimport {ccount} from 'ccount'\nimport {ok as assert} from 'devlop'\nimport {unicodePunctuation, unicodeWhitespace} from 'micromark-util-character'\nimport {findAndReplace} from 'mdast-util-find-and-replace'\n\n/** @type {ConstructName} */\nconst inConstruct = 'phrasing'\n/** @type {Array<ConstructName>} */\nconst notInConstruct = ['autolink', 'link', 'image', 'label']\n\n/**\n * Create an extension for `mdast-util-from-markdown` to enable GFM autolink\n * literals in markdown.\n *\n * @returns {FromMarkdownExtension}\n * Extension for `mdast-util-to-markdown` to enable GFM autolink literals.\n */\nexport function gfmAutolinkLiteralFromMarkdown() {\n return {\n transforms: [transformGfmAutolinkLiterals],\n enter: {\n literalAutolink: enterLiteralAutolink,\n literalAutolinkEmail: enterLiteralAutolinkValue,\n literalAutolinkHttp: enterLiteralAutolinkValue,\n literalAutolinkWww: enterLiteralAutolinkValue\n },\n exit: {\n literalAutolink: exitLiteralAutolink,\n literalAutolinkEmail: exitLiteralAutolinkEmail,\n literalAutolinkHttp: exitLiteralAutolinkHttp,\n literalAutolinkWww: exitLiteralAutolinkWww\n }\n }\n}\n\n/**\n * Create an extension for `mdast-util-to-markdown` to enable GFM autolink\n * literals in markdown.\n *\n * @returns {ToMarkdownExtension}\n * Extension for `mdast-util-to-markdown` to enable GFM autolink literals.\n */\nexport function gfmAutolinkLiteralToMarkdown() {\n return {\n unsafe: [\n {\n character: '@',\n before: '[+\\\\-.\\\\w]',\n after: '[\\\\-.\\\\w]',\n inConstruct,\n notInConstruct\n },\n {\n character: '.',\n before: '[Ww]',\n after: '[\\\\-.\\\\w]',\n inConstruct,\n notInConstruct\n },\n {\n character: ':',\n before: '[ps]',\n after: '\\\\/',\n inConstruct,\n notInConstruct\n }\n ]\n }\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterLiteralAutolink(token) {\n this.enter({type: 'link', title: null, url: '', children: []}, token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterLiteralAutolinkValue(token) {\n this.config.enter.autolinkProtocol.call(this, token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitLiteralAutolinkHttp(token) {\n this.config.exit.autolinkProtocol.call(this, token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitLiteralAutolinkWww(token) {\n this.config.exit.data.call(this, token)\n const node = this.stack[this.stack.length - 1]\n assert(node.type === 'link')\n node.url = 'http://' + this.sliceSerialize(token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitLiteralAutolinkEmail(token) {\n this.config.exit.autolinkEmail.call(this, token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitLiteralAutolink(token) {\n this.exit(token)\n}\n\n/** @type {FromMarkdownTransform} */\nfunction transformGfmAutolinkLiterals(tree) {\n findAndReplace(\n tree,\n [\n [/(https?:\\/\\/|www(?=\\.))([-.\\w]+)([^ \\t\\r\\n]*)/gi, findUrl],\n [/(?<=^|\\s|\\p{P}|\\p{S})([-.\\w+]+)@([-\\w]+(?:\\.[-\\w]+)+)/gu, findEmail]\n ],\n {ignore: ['link', 'linkReference']}\n )\n}\n\n/**\n * @type {ReplaceFunction}\n * @param {string} _\n * @param {string} protocol\n * @param {string} domain\n * @param {string} path\n * @param {RegExpMatchObject} match\n * @returns {Array<PhrasingContent> | Link | false}\n */\n// eslint-disable-next-line max-params\nfunction findUrl(_, protocol, domain, path, match) {\n let prefix = ''\n\n // Not an expected previous character.\n if (!previous(match)) {\n return false\n }\n\n // Treat `www` as part of the domain.\n if (/^w/i.test(protocol)) {\n domain = protocol + domain\n protocol = ''\n prefix = 'http://'\n }\n\n if (!isCorrectDomain(domain)) {\n return false\n }\n\n const parts = splitUrl(domain + path)\n\n if (!parts[0]) return false\n\n /** @type {Link} */\n const result = {\n type: 'link',\n title: null,\n url: prefix + protocol + parts[0],\n children: [{type: 'text', value: protocol + parts[0]}]\n }\n\n if (parts[1]) {\n return [result, {type: 'text', value: parts[1]}]\n }\n\n return result\n}\n\n/**\n * @type {ReplaceFunction}\n * @param {string} _\n * @param {string} atext\n * @param {string} label\n * @param {RegExpMatchObject} match\n * @returns {Link | false}\n */\nfunction findEmail(_, atext, label, match) {\n if (\n // Not an expected previous character.\n !previous(match, true) ||\n // Label ends in not allowed character.\n /[-\\d_]$/.test(label)\n ) {\n return false\n }\n\n return {\n type: 'link',\n title: null,\n url: 'mailto:' + atext + '@' + label,\n children: [{type: 'text', value: atext + '@' + label}]\n }\n}\n\n/**\n * @param {string} domain\n * @returns {boolean}\n */\nfunction isCorrectDomain(domain) {\n const parts = domain.split('.')\n\n if (\n parts.length < 2 ||\n (parts[parts.length - 1] &&\n (/_/.test(parts[parts.length - 1]) ||\n !/[a-zA-Z\\d]/.test(parts[parts.length - 1]))) ||\n (parts[parts.length - 2] &&\n (/_/.test(parts[parts.length - 2]) ||\n !/[a-zA-Z\\d]/.test(parts[parts.length - 2])))\n ) {\n return false\n }\n\n return true\n}\n\n/**\n * @param {string} url\n * @returns {[string, string | undefined]}\n */\nfunction splitUrl(url) {\n const trailExec = /[!\"&'),.:;<>?\\]}]+$/.exec(url)\n\n if (!trailExec) {\n return [url, undefined]\n }\n\n url = url.slice(0, trailExec.index)\n\n let trail = trailExec[0]\n let closingParenIndex = trail.indexOf(')')\n const openingParens = ccount(url, '(')\n let closingParens = ccount(url, ')')\n\n while (closingParenIndex !== -1 && openingParens > closingParens) {\n url += trail.slice(0, closingParenIndex + 1)\n trail = trail.slice(closingParenIndex + 1)\n closingParenIndex = trail.indexOf(')')\n closingParens++\n }\n\n return [url, trail]\n}\n\n/**\n * @param {RegExpMatchObject} match\n * @param {boolean | null | undefined} [email=false]\n * @returns {boolean}\n */\nfunction previous(match, email) {\n const code = match.input.charCodeAt(match.index - 1)\n\n return (\n (match.index === 0 ||\n unicodeWhitespace(code) ||\n unicodePunctuation(code)) &&\n // If it’s an email, the previous character should not be a slash.\n (!email || code !== 47)\n )\n}\n","/**\n * Normalize an identifier (as found in references, definitions).\n *\n * Collapses markdown whitespace, trim, and then lower- and uppercase.\n *\n * Some characters are considered “uppercase”, such as U+03F4 (`ϴ`), but if their\n * lowercase counterpart (U+03B8 (`θ`)) is uppercased will result in a different\n * uppercase character (U+0398 (`Θ`)).\n * So, to get a canonical form, we perform both lower- and uppercase.\n *\n * Using uppercase last makes sure keys will never interact with default\n * prototypal values (such as `constructor`): nothing in the prototype of\n * `Object` is uppercase.\n *\n * @param {string} value\n * Identifier to normalize.\n * @returns {string}\n * Normalized identifier.\n */\nexport function normalizeIdentifier(value) {\n return value\n // Collapse markdown whitespace.\n .replace(/[\\t\\n\\r ]+/g, \" \")\n // Trim.\n .replace(/^ | $/g, '')\n // Some characters are considered “uppercase”, but if their lowercase\n // counterpart is uppercased will result in a different uppercase\n // character.\n // Hence, to get that form, we perform both lower- and uppercase.\n // Upper case makes sure keys will not interact with default prototypal\n // methods: no method is uppercase.\n .toLowerCase().toUpperCase();\n}","/**\n * @import {\n * CompileContext,\n * Extension as FromMarkdownExtension,\n * Handle as FromMarkdownHandle\n * } from 'mdast-util-from-markdown'\n * @import {ToMarkdownOptions} from 'mdast-util-gfm-footnote'\n * @import {\n * Handle as ToMarkdownHandle,\n * Map,\n * Options as ToMarkdownExtension\n * } from 'mdast-util-to-markdown'\n * @import {FootnoteDefinition, FootnoteReference} from 'mdast'\n */\n\nimport {ok as assert} from 'devlop'\nimport {normalizeIdentifier} from 'micromark-util-normalize-identifier'\n\nfootnoteReference.peek = footnoteReferencePeek\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterFootnoteCallString() {\n this.buffer()\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterFootnoteCall(token) {\n this.enter({type: 'footnoteReference', identifier: '', label: ''}, token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterFootnoteDefinitionLabelString() {\n this.buffer()\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterFootnoteDefinition(token) {\n this.enter(\n {type: 'footnoteDefinition', identifier: '', label: '', children: []},\n token\n )\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitFootnoteCallString(token) {\n const label = this.resume()\n const node = this.stack[this.stack.length - 1]\n assert(node.type === 'footnoteReference')\n node.identifier = normalizeIdentifier(\n this.sliceSerialize(token)\n ).toLowerCase()\n node.label = label\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitFootnoteCall(token) {\n this.exit(token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitFootnoteDefinitionLabelString(token) {\n const label = this.resume()\n const node = this.stack[this.stack.length - 1]\n assert(node.type === 'footnoteDefinition')\n node.identifier = normalizeIdentifier(\n this.sliceSerialize(token)\n ).toLowerCase()\n node.label = label\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitFootnoteDefinition(token) {\n this.exit(token)\n}\n\n/** @type {ToMarkdownHandle} */\nfunction footnoteReferencePeek() {\n return '['\n}\n\n/**\n * @type {ToMarkdownHandle}\n * @param {FootnoteReference} node\n */\nfunction footnoteReference(node, _, state, info) {\n const tracker = state.createTracker(info)\n let value = tracker.move('[^')\n const exit = state.enter('footnoteReference')\n const subexit = state.enter('reference')\n value += tracker.move(\n state.safe(state.associationId(node), {after: ']', before: value})\n )\n subexit()\n exit()\n value += tracker.move(']')\n return value\n}\n\n/**\n * Create an extension for `mdast-util-from-markdown` to enable GFM footnotes\n * in markdown.\n *\n * @returns {FromMarkdownExtension}\n * Extension for `mdast-util-from-markdown`.\n */\nexport function gfmFootnoteFromMarkdown() {\n return {\n enter: {\n gfmFootnoteCallString: enterFootnoteCallString,\n gfmFootnoteCall: enterFootnoteCall,\n gfmFootnoteDefinitionLabelString: enterFootnoteDefinitionLabelString,\n gfmFootnoteDefinition: enterFootnoteDefinition\n },\n exit: {\n gfmFootnoteCallString: exitFootnoteCallString,\n gfmFootnoteCall: exitFootnoteCall,\n gfmFootnoteDefinitionLabelString: exitFootnoteDefinitionLabelString,\n gfmFootnoteDefinition: exitFootnoteDefinition\n }\n }\n}\n\n/**\n * Create an extension for `mdast-util-to-markdown` to enable GFM footnotes\n * in markdown.\n *\n * @param {ToMarkdownOptions | null | undefined} [options]\n * Configuration (optional).\n * @returns {ToMarkdownExtension}\n * Extension for `mdast-util-to-markdown`.\n */\nexport function gfmFootnoteToMarkdown(options) {\n // To do: next major: change default.\n let firstLineBlank = false\n\n if (options && options.firstLineBlank) {\n firstLineBlank = true\n }\n\n return {\n handlers: {footnoteDefinition, footnoteReference},\n // This is on by default already.\n unsafe: [{character: '[', inConstruct: ['label', 'phrasing', 'reference']}]\n }\n\n /**\n * @type {ToMarkdownHandle}\n * @param {FootnoteDefinition} node\n */\n function footnoteDefinition(node, _, state, info) {\n const tracker = state.createTracker(info)\n let value = tracker.move('[^')\n const exit = state.enter('footnoteDefinition')\n const subexit = state.enter('label')\n value += tracker.move(\n state.safe(state.associationId(node), {before: value, after: ']'})\n )\n subexit()\n\n value += tracker.move(']:')\n\n if (node.children && node.children.length > 0) {\n tracker.shift(4)\n\n value += tracker.move(\n (firstLineBlank ? '\\n' : ' ') +\n state.indentLines(\n state.containerFlow(node, tracker.current()),\n firstLineBlank ? mapAll : mapExceptFirst\n )\n )\n }\n\n exit()\n\n return value\n }\n}\n\n/** @type {Map} */\nfunction mapExceptFirst(line, index, blank) {\n return index === 0 ? line : mapAll(line, index, blank)\n}\n\n/** @type {Map} */\nfunction mapAll(line, index, blank) {\n return (blank ? '' : ' ') + line\n}\n","/**\n * @typedef {import('mdast').Delete} Delete\n *\n * @typedef {import('mdast-util-from-markdown').CompileContext} CompileContext\n * @typedef {import('mdast-util-from-markdown').Extension} FromMarkdownExtension\n * @typedef {import('mdast-util-from-markdown').Handle} FromMarkdownHandle\n *\n * @typedef {import('mdast-util-to-markdown').ConstructName} ConstructName\n * @typedef {import('mdast-util-to-markdown').Handle} ToMarkdownHandle\n * @typedef {import('mdast-util-to-markdown').Options} ToMarkdownExtension\n */\n\n/**\n * List of constructs that occur in phrasing (paragraphs, headings), but cannot\n * contain strikethrough.\n * So they sort of cancel each other out.\n * Note: could use a better name.\n *\n * Note: keep in sync with: <https://github.com/syntax-tree/mdast-util-to-markdown/blob/8ce8dbf/lib/unsafe.js#L14>\n *\n * @type {Array<ConstructName>}\n */\nconst constructsWithoutStrikethrough = [\n 'autolink',\n 'destinationLiteral',\n 'destinationRaw',\n 'reference',\n 'titleQuote',\n 'titleApostrophe'\n]\n\nhandleDelete.peek = peekDelete\n\n/**\n * Create an extension for `mdast-util-from-markdown` to enable GFM\n * strikethrough in markdown.\n *\n * @returns {FromMarkdownExtension}\n * Extension for `mdast-util-from-markdown` to enable GFM strikethrough.\n */\nexport function gfmStrikethroughFromMarkdown() {\n return {\n canContainEols: ['delete'],\n enter: {strikethrough: enterStrikethrough},\n exit: {strikethrough: exitStrikethrough}\n }\n}\n\n/**\n * Create an extension for `mdast-util-to-markdown` to enable GFM\n * strikethrough in markdown.\n *\n * @returns {ToMarkdownExtension}\n * Extension for `mdast-util-to-markdown` to enable GFM strikethrough.\n */\nexport function gfmStrikethroughToMarkdown() {\n return {\n unsafe: [\n {\n character: '~',\n inConstruct: 'phrasing',\n notInConstruct: constructsWithoutStrikethrough\n }\n ],\n handlers: {delete: handleDelete}\n }\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterStrikethrough(token) {\n this.enter({type: 'delete', children: []}, token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitStrikethrough(token) {\n this.exit(token)\n}\n\n/**\n * @type {ToMarkdownHandle}\n * @param {Delete} node\n */\nfunction handleDelete(node, _, state, info) {\n const tracker = state.createTracker(info)\n const exit = state.enter('strikethrough')\n let value = tracker.move('~~')\n value += state.containerPhrasing(node, {\n ...tracker.current(),\n before: value,\n after: '~'\n })\n value += tracker.move('~~')\n exit()\n return value\n}\n\n/** @type {ToMarkdownHandle} */\nfunction peekDelete() {\n return '~'\n}\n","// To do: next major: remove.\n/**\n * @typedef {Options} MarkdownTableOptions\n * Configuration.\n */\n\n/**\n * @typedef Options\n * Configuration.\n * @property {boolean | null | undefined} [alignDelimiters=true]\n * Whether to align the delimiters (default: `true`);\n * they are aligned by default:\n *\n * ```markdown\n * | Alpha | B |\n * | ----- | ----- |\n * | C | Delta |\n * ```\n *\n * Pass `false` to make them staggered:\n *\n * ```markdown\n * | Alpha | B |\n * | - | - |\n * | C | Delta |\n * ```\n * @property {ReadonlyArray<string | null | undefined> | string | null | undefined} [align]\n * How to align columns (default: `''`);\n * one style for all columns or styles for their respective columns;\n * each style is either `'l'` (left), `'r'` (right), or `'c'` (center);\n * other values are treated as `''`, which doesn’t place the colon in the\n * alignment row but does align left;\n * *only the lowercased first character is used, so `Right` is fine.*\n * @property {boolean | null | undefined} [delimiterEnd=true]\n * Whether to end each row with the delimiter (default: `true`).\n *\n * > 👉 **Note**: please don’t use this: it could create fragile structures\n * > that aren’t understandable to some markdown parsers.\n *\n * When `true`, there are ending delimiters:\n *\n * ```markdown\n * | Alpha | B |\n * | ----- | ----- |\n * | C | Delta |\n * ```\n *\n * When `false`, there are no ending delimiters:\n *\n * ```markdown\n * | Alpha | B\n * | ----- | -----\n * | C | Delta\n * ```\n * @property {boolean | null | undefined} [delimiterStart=true]\n * Whether to begin each row with the delimiter (default: `true`).\n *\n * > 👉 **Note**: please don’t use this: it could create fragile structures\n * > that aren’t understandable to some markdown parsers.\n *\n * When `true`, there are starting delimiters:\n *\n * ```markdown\n * | Alpha | B |\n * | ----- | ----- |\n * | C | Delta |\n * ```\n *\n * When `false`, there are no starting delimiters:\n *\n * ```markdown\n * Alpha | B |\n * ----- | ----- |\n * C | Delta |\n * ```\n * @property {boolean | null | undefined} [padding=true]\n * Whether to add a space of padding between delimiters and cells\n * (default: `true`).\n *\n * When `true`, there is padding:\n *\n * ```markdown\n * | Alpha | B |\n * | ----- | ----- |\n * | C | Delta |\n * ```\n *\n * When `false`, there is no padding:\n *\n * ```markdown\n * |Alpha|B |\n * |-----|-----|\n * |C |Delta|\n * ```\n * @property {((value: string) => number) | null | undefined} [stringLength]\n * Function to detect the length of table cell content (optional);\n * this is used when aligning the delimiters (`|`) between table cells;\n * full-width characters and emoji mess up delimiter alignment when viewing\n * the markdown source;\n * to fix this, you can pass this function,\n * which receives the cell content and returns its “visible” size;\n * note that what is and isn’t visible depends on where the text is displayed.\n *\n * Without such a function, the following:\n *\n * ```js\n * markdownTable([\n * ['Alpha', 'Bravo'],\n * ['中文', 'Charlie'],\n * ['👩❤️👩', 'Delta']\n * ])\n * ```\n *\n * Yields:\n *\n * ```markdown\n * | Alpha | Bravo |\n * | - | - |\n * | 中文 | Charlie |\n * | 👩❤️👩 | Delta |\n * ```\n *\n * With [`string-width`](https://github.com/sindresorhus/string-width):\n *\n * ```js\n * import stringWidth from 'string-width'\n *\n * markdownTable(\n * [\n * ['Alpha', 'Bravo'],\n * ['中文', 'Charlie'],\n * ['👩❤️👩', 'Delta']\n * ],\n * {stringLength: stringWidth}\n * )\n * ```\n *\n * Yields:\n *\n * ```markdown\n * | Alpha | Bravo |\n * | ----- | ------- |\n * | 中文 | Charlie |\n * | 👩❤️👩 | Delta |\n * ```\n */\n\n/**\n * @param {string} value\n * Cell value.\n * @returns {number}\n * Cell size.\n */\nfunction defaultStringLength(value) {\n return value.length\n}\n\n/**\n * Generate a markdown\n * ([GFM](https://docs.github.com/en/github/writing-on-github/working-with-advanced-formatting/organizing-information-with-tables))\n * table.\n *\n * @param {ReadonlyArray<ReadonlyArray<string | null | undefined>>} table\n * Table data (matrix of strings).\n * @param {Readonly<Options> | null | undefined} [options]\n * Configuration (optional).\n * @returns {string}\n * Result.\n */\nexport function markdownTable(table, options) {\n const settings = options || {}\n // To do: next major: change to spread.\n const align = (settings.align || []).concat()\n const stringLength = settings.stringLength || defaultStringLength\n /** @type {Array<number>} Character codes as symbols for alignment per column. */\n const alignments = []\n /** @type {Array<Array<string>>} Cells per row. */\n const cellMatrix = []\n /** @type {Array<Array<number>>} Sizes of each cell per row. */\n const sizeMatrix = []\n /** @type {Array<number>} */\n const longestCellByColumn = []\n let mostCellsPerRow = 0\n let rowIndex = -1\n\n // This is a superfluous loop if we don’t align delimiters, but otherwise we’d\n // do superfluous work when aligning, so optimize for aligning.\n while (++rowIndex < table.length) {\n /** @type {Array<string>} */\n const row = []\n /** @type {Array<number>} */\n const sizes = []\n let columnIndex = -1\n\n if (table[rowIndex].length > mostCellsPerRow) {\n mostCellsPerRow = table[rowIndex].length\n }\n\n while (++columnIndex < table[rowIndex].length) {\n const cell = serialize(table[rowIndex][columnIndex])\n\n if (settings.alignDelimiters !== false) {\n const size = stringLength(cell)\n sizes[columnIndex] = size\n\n if (\n longestCellByColumn[columnIndex] === undefined ||\n size > longestCellByColumn[columnIndex]\n ) {\n longestCellByColumn[columnIndex] = size\n }\n }\n\n row.push(cell)\n }\n\n cellMatrix[rowIndex] = row\n sizeMatrix[rowIndex] = sizes\n }\n\n // Figure out which alignments to use.\n let columnIndex = -1\n\n if (typeof align === 'object' && 'length' in align) {\n while (++columnIndex < mostCellsPerRow) {\n alignments[columnIndex] = toAlignment(align[columnIndex])\n }\n } else {\n const code = toAlignment(align)\n\n while (++columnIndex < mostCellsPerRow) {\n alignments[columnIndex] = code\n }\n }\n\n // Inject the alignment row.\n columnIndex = -1\n /** @type {Array<string>} */\n const row = []\n /** @type {Array<number>} */\n const sizes = []\n\n while (++columnIndex < mostCellsPerRow) {\n const code = alignments[columnIndex]\n let before = ''\n let after = ''\n\n if (code === 99 /* `c` */) {\n before = ':'\n after = ':'\n } else if (code === 108 /* `l` */) {\n before = ':'\n } else if (code === 114 /* `r` */) {\n after = ':'\n }\n\n // There *must* be at least one hyphen-minus in each alignment cell.\n let size =\n settings.alignDelimiters === false\n ? 1\n : Math.max(\n 1,\n longestCellByColumn[columnIndex] - before.length - after.length\n )\n\n const cell = before + '-'.repeat(size) + after\n\n if (settings.alignDelimiters !== false) {\n size = before.length + size + after.length\n\n if (size > longestCellByColumn[columnIndex]) {\n longestCellByColumn[columnIndex] = size\n }\n\n sizes[columnIndex] = size\n }\n\n row[columnIndex] = cell\n }\n\n // Inject the alignment row.\n cellMatrix.splice(1, 0, row)\n sizeMatrix.splice(1, 0, sizes)\n\n rowIndex = -1\n /** @type {Array<string>} */\n const lines = []\n\n while (++rowIndex < cellMatrix.length) {\n const row = cellMatrix[rowIndex]\n const sizes = sizeMatrix[rowIndex]\n columnIndex = -1\n /** @type {Array<string>} */\n const line = []\n\n while (++columnIndex < mostCellsPerRow) {\n const cell = row[columnIndex] || ''\n let before = ''\n let after = ''\n\n if (settings.alignDelimiters !== false) {\n const size =\n longestCellByColumn[columnIndex] - (sizes[columnIndex] || 0)\n const code = alignments[columnIndex]\n\n if (code === 114 /* `r` */) {\n before = ' '.repeat(size)\n } else if (code === 99 /* `c` */) {\n if (size % 2) {\n before = ' '.repeat(size / 2 + 0.5)\n after = ' '.repeat(size / 2 - 0.5)\n } else {\n before = ' '.repeat(size / 2)\n after = before\n }\n } else {\n after = ' '.repeat(size)\n }\n }\n\n if (settings.delimiterStart !== false && !columnIndex) {\n line.push('|')\n }\n\n if (\n settings.padding !== false &&\n // Don’t add the opening space if we’re not aligning and the cell is\n // empty: there will be a closing space.\n !(settings.alignDelimiters === false && cell === '') &&\n (settings.delimiterStart !== false || columnIndex)\n ) {\n line.push(' ')\n }\n\n if (settings.alignDelimiters !== false) {\n line.push(before)\n }\n\n line.push(cell)\n\n if (settings.alignDelimiters !== false) {\n line.push(after)\n }\n\n if (settings.padding !== false) {\n line.push(' ')\n }\n\n if (\n settings.delimiterEnd !== false ||\n columnIndex !== mostCellsPerRow - 1\n ) {\n line.push('|')\n }\n }\n\n lines.push(\n settings.delimiterEnd === false\n ? line.join('').replace(/ +$/, '')\n : line.join('')\n )\n }\n\n return lines.join('\\n')\n}\n\n/**\n * @param {string | null | undefined} [value]\n * Value to serialize.\n * @returns {string}\n * Result.\n */\nfunction serialize(value) {\n return value === null || value === undefined ? '' : String(value)\n}\n\n/**\n * @param {string | null | undefined} value\n * Value.\n * @returns {number}\n * Alignment.\n */\nfunction toAlignment(value) {\n const code = typeof value === 'string' ? value.codePointAt(0) : 0\n\n return code === 67 /* `C` */ || code === 99 /* `c` */\n ? 99 /* `c` */\n : code === 76 /* `L` */ || code === 108 /* `l` */\n ? 108 /* `l` */\n : code === 82 /* `R` */ || code === 114 /* `r` */\n ? 114 /* `r` */\n : 0\n}\n","/**\n * @import {Blockquote, Parents} from 'mdast'\n * @import {Info, Map, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {Blockquote} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function blockquote(node, _, state, info) {\n const exit = state.enter('blockquote')\n const tracker = state.createTracker(info)\n tracker.move('> ')\n tracker.shift(2)\n const value = state.indentLines(\n state.containerFlow(node, tracker.current()),\n map\n )\n exit()\n return value\n}\n\n/** @type {Map} */\nfunction map(line, _, blank) {\n return '>' + (blank ? '' : ' ') + line\n}\n","/**\n * @import {ConstructName, Unsafe} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {Array<ConstructName>} stack\n * @param {Unsafe} pattern\n * @returns {boolean}\n */\nexport function patternInScope(stack, pattern) {\n return (\n listInScope(stack, pattern.inConstruct, true) &&\n !listInScope(stack, pattern.notInConstruct, false)\n )\n}\n\n/**\n * @param {Array<ConstructName>} stack\n * @param {Unsafe['inConstruct']} list\n * @param {boolean} none\n * @returns {boolean}\n */\nfunction listInScope(stack, list, none) {\n if (typeof list === 'string') {\n list = [list]\n }\n\n if (!list || list.length === 0) {\n return none\n }\n\n let index = -1\n\n while (++index < list.length) {\n if (stack.includes(list[index])) {\n return true\n }\n }\n\n return false\n}\n","/**\n * @import {Break, Parents} from 'mdast'\n * @import {Info, State} from 'mdast-util-to-markdown'\n */\n\nimport {patternInScope} from '../util/pattern-in-scope.js'\n\n/**\n * @param {Break} _\n * @param {Parents | undefined} _1\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function hardBreak(_, _1, state, info) {\n let index = -1\n\n while (++index < state.unsafe.length) {\n // If we can’t put eols in this construct (setext headings, tables), use a\n // space instead.\n if (\n state.unsafe[index].character === '\\n' &&\n patternInScope(state.stack, state.unsafe[index])\n ) {\n return /[ \\t]/.test(info.before) ? '' : ' '\n }\n }\n\n return '\\\\\\n'\n}\n","/**\n * Get the count of the longest repeating streak of `substring` in `value`.\n *\n * @param {string} value\n * Content to search in.\n * @param {string} substring\n * Substring to look for, typically one character.\n * @returns {number}\n * Count of most frequent adjacent `substring`s in `value`.\n */\nexport function longestStreak(value, substring) {\n const source = String(value)\n let index = source.indexOf(substring)\n let expected = index\n let count = 0\n let max = 0\n\n if (typeof substring !== 'string') {\n throw new TypeError('Expected substring')\n }\n\n while (index !== -1) {\n if (index === expected) {\n if (++count > max) {\n max = count\n }\n } else {\n count = 1\n }\n\n expected = index + substring.length\n index = source.indexOf(substring, expected)\n }\n\n return max\n}\n","/**\n * @import {State} from 'mdast-util-to-markdown'\n * @import {Code} from 'mdast'\n */\n\n/**\n * @param {Code} node\n * @param {State} state\n * @returns {boolean}\n */\nexport function formatCodeAsIndented(node, state) {\n return Boolean(\n state.options.fences === false &&\n node.value &&\n // If there’s no info…\n !node.lang &&\n // And there’s a non-whitespace character…\n /[^ \\r\\n]/.test(node.value) &&\n // And the value doesn’t start or end in a blank…\n !/^[\\t ]*(?:[\\r\\n]|$)|(?:^|[\\r\\n])[\\t ]*$/.test(node.value)\n )\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['fence'], null | undefined>}\n */\nexport function checkFence(state) {\n const marker = state.options.fence || '`'\n\n if (marker !== '`' && marker !== '~') {\n throw new Error(\n 'Cannot serialize code with `' +\n marker +\n '` for `options.fence`, expected `` ` `` or `~`'\n )\n }\n\n return marker\n}\n","/**\n * @import {Info, Map, State} from 'mdast-util-to-markdown'\n * @import {Code, Parents} from 'mdast'\n */\n\nimport {longestStreak} from 'longest-streak'\nimport {formatCodeAsIndented} from '../util/format-code-as-indented.js'\nimport {checkFence} from '../util/check-fence.js'\n\n/**\n * @param {Code} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function code(node, _, state, info) {\n const marker = checkFence(state)\n const raw = node.value || ''\n const suffix = marker === '`' ? 'GraveAccent' : 'Tilde'\n\n if (formatCodeAsIndented(node, state)) {\n const exit = state.enter('codeIndented')\n const value = state.indentLines(raw, map)\n exit()\n return value\n }\n\n const tracker = state.createTracker(info)\n const sequence = marker.repeat(Math.max(longestStreak(raw, marker) + 1, 3))\n const exit = state.enter('codeFenced')\n let value = tracker.move(sequence)\n\n if (node.lang) {\n const subexit = state.enter(`codeFencedLang${suffix}`)\n value += tracker.move(\n state.safe(node.lang, {\n before: value,\n after: ' ',\n encode: ['`'],\n ...tracker.current()\n })\n )\n subexit()\n }\n\n if (node.lang && node.meta) {\n const subexit = state.enter(`codeFencedMeta${suffix}`)\n value += tracker.move(' ')\n value += tracker.move(\n state.safe(node.meta, {\n before: value,\n after: '\\n',\n encode: ['`'],\n ...tracker.current()\n })\n )\n subexit()\n }\n\n value += tracker.move('\\n')\n\n if (raw) {\n value += tracker.move(raw + '\\n')\n }\n\n value += tracker.move(sequence)\n exit()\n return value\n}\n\n/** @type {Map} */\nfunction map(line, _, blank) {\n return (blank ? '' : ' ') + line\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['quote'], null | undefined>}\n */\nexport function checkQuote(state) {\n const marker = state.options.quote || '\"'\n\n if (marker !== '\"' && marker !== \"'\") {\n throw new Error(\n 'Cannot serialize title with `' +\n marker +\n '` for `options.quote`, expected `\"`, or `\\'`'\n )\n }\n\n return marker\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Definition, Parents} from 'mdast'\n */\n\nimport {checkQuote} from '../util/check-quote.js'\n\n/**\n * @param {Definition} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function definition(node, _, state, info) {\n const quote = checkQuote(state)\n const suffix = quote === '\"' ? 'Quote' : 'Apostrophe'\n const exit = state.enter('definition')\n let subexit = state.enter('label')\n const tracker = state.createTracker(info)\n let value = tracker.move('[')\n value += tracker.move(\n state.safe(state.associationId(node), {\n before: value,\n after: ']',\n ...tracker.current()\n })\n )\n value += tracker.move(']: ')\n\n subexit()\n\n if (\n // If there’s no url, or…\n !node.url ||\n // If there are control characters or whitespace.\n /[\\0- \\u007F]/.test(node.url)\n ) {\n subexit = state.enter('destinationLiteral')\n value += tracker.move('<')\n value += tracker.move(\n state.safe(node.url, {before: value, after: '>', ...tracker.current()})\n )\n value += tracker.move('>')\n } else {\n // No whitespace, raw is prettier.\n subexit = state.enter('destinationRaw')\n value += tracker.move(\n state.safe(node.url, {\n before: value,\n after: node.title ? ' ' : '\\n',\n ...tracker.current()\n })\n )\n }\n\n subexit()\n\n if (node.title) {\n subexit = state.enter(`title${suffix}`)\n value += tracker.move(' ' + quote)\n value += tracker.move(\n state.safe(node.title, {\n before: value,\n after: quote,\n ...tracker.current()\n })\n )\n value += tracker.move(quote)\n subexit()\n }\n\n exit()\n\n return value\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['emphasis'], null | undefined>}\n */\nexport function checkEmphasis(state) {\n const marker = state.options.emphasis || '*'\n\n if (marker !== '*' && marker !== '_') {\n throw new Error(\n 'Cannot serialize emphasis with `' +\n marker +\n '` for `options.emphasis`, expected `*`, or `_`'\n )\n }\n\n return marker\n}\n","/**\n * Encode a code point as a character reference.\n *\n * @param {number} code\n * Code point to encode.\n * @returns {string}\n * Encoded character reference.\n */\nexport function encodeCharacterReference(code) {\n return '&#x' + code.toString(16).toUpperCase() + ';'\n}\n","/**\n * @import {Code} from 'micromark-util-types'\n */\n\nimport { markdownLineEndingOrSpace, unicodePunctuation, unicodeWhitespace } from 'micromark-util-character';\n/**\n * Classify whether a code represents whitespace, punctuation, or something\n * else.\n *\n * Used for attention (emphasis, strong), whose sequences can open or close\n * based on the class of surrounding characters.\n *\n * > 👉 **Note**: eof (`null`) is seen as whitespace.\n *\n * @param {Code} code\n * Code.\n * @returns {typeof constants.characterGroupWhitespace | typeof constants.characterGroupPunctuation | undefined}\n * Group.\n */\nexport function classifyCharacter(code) {\n if (code === null || markdownLineEndingOrSpace(code) || unicodeWhitespace(code)) {\n return 1;\n }\n if (unicodePunctuation(code)) {\n return 2;\n }\n}","/**\n * @import {EncodeSides} from '../types.js'\n */\n\nimport {classifyCharacter} from 'micromark-util-classify-character'\n\n/**\n * Check whether to encode (as a character reference) the characters\n * surrounding an attention run.\n *\n * Which characters are around an attention run influence whether it works or\n * not.\n *\n * See <https://github.com/orgs/syntax-tree/discussions/60> for more info.\n * See this markdown in a particular renderer to see what works:\n *\n * ```markdown\n * | | A (letter inside) | B (punctuation inside) | C (whitespace inside) | D (nothing inside) |\n * | ----------------------- | ----------------- | ---------------------- | --------------------- | ------------------ |\n * | 1 (letter outside) | x*y*z | x*.*z | x* *z | x**z |\n * | 2 (punctuation outside) | .*y*. | .*.*. | .* *. | .**. |\n * | 3 (whitespace outside) | x *y* z | x *.* z | x * * z | x ** z |\n * | 4 (nothing outside) | *x* | *.* | * * | ** |\n * ```\n *\n * @param {number} outside\n * Code point on the outer side of the run.\n * @param {number} inside\n * Code point on the inner side of the run.\n * @param {'*' | '_'} marker\n * Marker of the run.\n * Underscores are handled more strictly (they form less often) than\n * asterisks.\n * @returns {EncodeSides}\n * Whether to encode characters.\n */\n// Important: punctuation must never be encoded.\n// Punctuation is solely used by markdown constructs.\n// And by encoding itself.\n// Encoding them will break constructs or double encode things.\nexport function encodeInfo(outside, inside, marker) {\n const outsideKind = classifyCharacter(outside)\n const insideKind = classifyCharacter(inside)\n\n // Letter outside:\n if (outsideKind === undefined) {\n return insideKind === undefined\n ? // Letter inside:\n // we have to encode *both* letters for `_` as it is looser.\n // it already forms for `*` (and GFMs `~`).\n marker === '_'\n ? {inside: true, outside: true}\n : {inside: false, outside: false}\n : insideKind === 1\n ? // Whitespace inside: encode both (letter, whitespace).\n {inside: true, outside: true}\n : // Punctuation inside: encode outer (letter)\n {inside: false, outside: true}\n }\n\n // Whitespace outside:\n if (outsideKind === 1) {\n return insideKind === undefined\n ? // Letter inside: already forms.\n {inside: false, outside: false}\n : insideKind === 1\n ? // Whitespace inside: encode both (whitespace).\n {inside: true, outside: true}\n : // Punctuation inside: already forms.\n {inside: false, outside: false}\n }\n\n // Punctuation outside:\n return insideKind === undefined\n ? // Letter inside: already forms.\n {inside: false, outside: false}\n : insideKind === 1\n ? // Whitespace inside: encode inner (whitespace).\n {inside: true, outside: false}\n : // Punctuation inside: already forms.\n {inside: false, outside: false}\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Emphasis, Parents} from 'mdast'\n */\n\nimport {checkEmphasis} from '../util/check-emphasis.js'\nimport {encodeCharacterReference} from '../util/encode-character-reference.js'\nimport {encodeInfo} from '../util/encode-info.js'\n\nemphasis.peek = emphasisPeek\n\n/**\n * @param {Emphasis} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function emphasis(node, _, state, info) {\n const marker = checkEmphasis(state)\n const exit = state.enter('emphasis')\n const tracker = state.createTracker(info)\n const before = tracker.move(marker)\n\n let between = tracker.move(\n state.containerPhrasing(node, {\n after: marker,\n before,\n ...tracker.current()\n })\n )\n const betweenHead = between.charCodeAt(0)\n const open = encodeInfo(\n info.before.charCodeAt(info.before.length - 1),\n betweenHead,\n marker\n )\n\n if (open.inside) {\n between = encodeCharacterReference(betweenHead) + between.slice(1)\n }\n\n const betweenTail = between.charCodeAt(between.length - 1)\n const close = encodeInfo(info.after.charCodeAt(0), betweenTail, marker)\n\n if (close.inside) {\n between = between.slice(0, -1) + encodeCharacterReference(betweenTail)\n }\n\n const after = tracker.move(marker)\n\n exit()\n\n state.attentionEncodeSurroundingInfo = {\n after: close.outside,\n before: open.outside\n }\n return before + between + after\n}\n\n/**\n * @param {Emphasis} _\n * @param {Parents | undefined} _1\n * @param {State} state\n * @returns {string}\n */\nfunction emphasisPeek(_, _1, state) {\n return state.options.emphasis || '*'\n}\n","/**\n * @typedef {import('unist').Node} UnistNode\n * @typedef {import('unist').Parent} UnistParent\n * @typedef {import('unist-util-visit-parents').VisitorResult} VisitorResult\n */\n\n/**\n * @typedef {Exclude<import('unist-util-is').Test, undefined> | undefined} Test\n * Test from `unist-util-is`.\n *\n * Note: we have remove and add `undefined`, because otherwise when generating\n * automatic `.d.ts` files, TS tries to flatten paths from a local perspective,\n * which doesn’t work when publishing on npm.\n */\n\n// To do: use types from `unist-util-visit-parents` when it’s released.\n\n/**\n * @typedef {(\n * Fn extends (value: any) => value is infer Thing\n * ? Thing\n * : Fallback\n * )} Predicate\n * Get the value of a type guard `Fn`.\n * @template Fn\n * Value; typically function that is a type guard (such as `(x): x is Y`).\n * @template Fallback\n * Value to yield if `Fn` is not a type guard.\n */\n\n/**\n * @typedef {(\n * Check extends null | undefined // No test.\n * ? Value\n * : Value extends {type: Check} // String (type) test.\n * ? Value\n * : Value extends Check // Partial test.\n * ? Value\n * : Check extends Function // Function test.\n * ? Predicate<Check, Value> extends Value\n * ? Predicate<Check, Value>\n * : never\n * : never // Some other test?\n * )} MatchesOne\n * Check whether a node matches a primitive check in the type system.\n * @template Value\n * Value; typically unist `Node`.\n * @template Check\n * Value; typically `unist-util-is`-compatible test, but not arrays.\n */\n\n/**\n * @typedef {(\n * Check extends Array<any>\n * ? MatchesOne<Value, Check[keyof Check]>\n * : MatchesOne<Value, Check>\n * )} Matches\n * Check whether a node matches a check in the type system.\n * @template Value\n * Value; typically unist `Node`.\n * @template Check\n * Value; typically `unist-util-is`-compatible test.\n */\n\n/**\n * @typedef {0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10} Uint\n * Number; capped reasonably.\n */\n\n/**\n * @typedef {I extends 0 ? 1 : I extends 1 ? 2 : I extends 2 ? 3 : I extends 3 ? 4 : I extends 4 ? 5 : I extends 5 ? 6 : I extends 6 ? 7 : I extends 7 ? 8 : I extends 8 ? 9 : 10} Increment\n * Increment a number in the type system.\n * @template {Uint} [I=0]\n * Index.\n */\n\n/**\n * @typedef {(\n * Node extends UnistParent\n * ? Node extends {children: Array<infer Children>}\n * ? Child extends Children ? Node : never\n * : never\n * : never\n * )} InternalParent\n * Collect nodes that can be parents of `Child`.\n * @template {UnistNode} Node\n * All node types in a tree.\n * @template {UnistNode} Child\n * Node to search for.\n */\n\n/**\n * @typedef {InternalParent<InclusiveDescendant<Tree>, Child>} Parent\n * Collect nodes in `Tree` that can be parents of `Child`.\n * @template {UnistNode} Tree\n * All node types in a tree.\n * @template {UnistNode} Child\n * Node to search for.\n */\n\n/**\n * @typedef {(\n * Depth extends Max\n * ? never\n * :\n * | InternalParent<Node, Child>\n * | InternalAncestor<Node, InternalParent<Node, Child>, Max, Increment<Depth>>\n * )} InternalAncestor\n * Collect nodes in `Tree` that can be ancestors of `Child`.\n * @template {UnistNode} Node\n * All node types in a tree.\n * @template {UnistNode} Child\n * Node to search for.\n * @template {Uint} [Max=10]\n * Max; searches up to this depth.\n * @template {Uint} [Depth=0]\n * Current depth.\n */\n\n/**\n * @typedef {(\n * Tree extends UnistParent\n * ? Depth extends Max\n * ? Tree\n * : Tree | InclusiveDescendant<Tree['children'][number], Max, Increment<Depth>>\n * : Tree\n * )} InclusiveDescendant\n * Collect all (inclusive) descendants of `Tree`.\n *\n * > 👉 **Note**: for performance reasons, this seems to be the fastest way to\n * > recurse without actually running into an infinite loop, which the\n * > previous version did.\n * >\n * > Practically, a max of `2` is typically enough assuming a `Root` is\n * > passed, but it doesn’t improve performance.\n * > It gets higher with `List > ListItem > Table > TableRow > TableCell`.\n * > Using up to `10` doesn’t hurt or help either.\n * @template {UnistNode} Tree\n * Tree type.\n * @template {Uint} [Max=10]\n * Max; searches up to this depth.\n * @template {Uint} [Depth=0]\n * Current depth.\n */\n\n/**\n * @callback Visitor\n * Handle a node (matching `test`, if given).\n *\n * Visitors are free to transform `node`.\n * They can also transform `parent`.\n *\n * Replacing `node` itself, if `SKIP` is not returned, still causes its\n * descendants to be walked (which is a bug).\n *\n * When adding or removing previous siblings of `node` (or next siblings, in\n * case of reverse), the `Visitor` should return a new `Index` to specify the\n * sibling to traverse after `node` is traversed.\n * Adding or removing next siblings of `node` (or previous siblings, in case\n * of reverse) is handled as expected without needing to return a new `Index`.\n *\n * Removing the children property of `parent` still results in them being\n * traversed.\n * @param {Visited} node\n * Found node.\n * @param {Visited extends UnistNode ? number | undefined : never} index\n * Index of `node` in `parent`.\n * @param {Ancestor extends UnistParent ? Ancestor | undefined : never} parent\n * Parent of `node`.\n * @returns {VisitorResult}\n * What to do next.\n *\n * An `Index` is treated as a tuple of `[CONTINUE, Index]`.\n * An `Action` is treated as a tuple of `[Action]`.\n *\n * Passing a tuple back only makes sense if the `Action` is `SKIP`.\n * When the `Action` is `EXIT`, that action can be returned.\n * When the `Action` is `CONTINUE`, `Index` can be returned.\n * @template {UnistNode} [Visited=UnistNode]\n * Visited node type.\n * @template {UnistParent} [Ancestor=UnistParent]\n * Ancestor type.\n */\n\n/**\n * @typedef {Visitor<Visited, Parent<Ancestor, Visited>>} BuildVisitorFromMatch\n * Build a typed `Visitor` function from a node and all possible parents.\n *\n * It will infer which values are passed as `node` and which as `parent`.\n * @template {UnistNode} Visited\n * Node type.\n * @template {UnistParent} Ancestor\n * Parent type.\n */\n\n/**\n * @typedef {(\n * BuildVisitorFromMatch<\n * Matches<Descendant, Check>,\n * Extract<Descendant, UnistParent>\n * >\n * )} BuildVisitorFromDescendants\n * Build a typed `Visitor` function from a list of descendants and a test.\n *\n * It will infer which values are passed as `node` and which as `parent`.\n * @template {UnistNode} Descendant\n * Node type.\n * @template {Test} Check\n * Test type.\n */\n\n/**\n * @typedef {(\n * BuildVisitorFromDescendants<\n * InclusiveDescendant<Tree>,\n * Check\n * >\n * )} BuildVisitor\n * Build a typed `Visitor` function from a tree and a test.\n *\n * It will infer which values are passed as `node` and which as `parent`.\n * @template {UnistNode} [Tree=UnistNode]\n * Node type.\n * @template {Test} [Check=Test]\n * Test type.\n */\n\nimport {visitParents} from 'unist-util-visit-parents'\n\nexport {CONTINUE, EXIT, SKIP} from 'unist-util-visit-parents'\n\n/**\n * Visit nodes.\n *\n * This algorithm performs *depth-first* *tree traversal* in *preorder*\n * (**NLR**) or if `reverse` is given, in *reverse preorder* (**NRL**).\n *\n * You can choose for which nodes `visitor` is called by passing a `test`.\n * For complex tests, you should test yourself in `visitor`, as it will be\n * faster and will have improved type information.\n *\n * Walking the tree is an intensive task.\n * Make use of the return values of the visitor when possible.\n * Instead of walking a tree multiple times, walk it once, use `unist-util-is`\n * to check if a node matches, and then perform different operations.\n *\n * You can change the tree.\n * See `Visitor` for more info.\n *\n * @overload\n * @param {Tree} tree\n * @param {Check} check\n * @param {BuildVisitor<Tree, Check>} visitor\n * @param {boolean | null | undefined} [reverse]\n * @returns {undefined}\n *\n * @overload\n * @param {Tree} tree\n * @param {BuildVisitor<Tree>} visitor\n * @param {boolean | null | undefined} [reverse]\n * @returns {undefined}\n *\n * @param {UnistNode} tree\n * Tree to traverse.\n * @param {Visitor | Test} testOrVisitor\n * `unist-util-is`-compatible test (optional, omit to pass a visitor).\n * @param {Visitor | boolean | null | undefined} [visitorOrReverse]\n * Handle each node (when test is omitted, pass `reverse`).\n * @param {boolean | null | undefined} [maybeReverse=false]\n * Traverse in reverse preorder (NRL) instead of the default preorder (NLR).\n * @returns {undefined}\n * Nothing.\n *\n * @template {UnistNode} Tree\n * Node type.\n * @template {Test} Check\n * `unist-util-is`-compatible test.\n */\nexport function visit(tree, testOrVisitor, visitorOrReverse, maybeReverse) {\n /** @type {boolean | null | undefined} */\n let reverse\n /** @type {Test} */\n let test\n /** @type {Visitor} */\n let visitor\n\n if (\n typeof testOrVisitor === 'function' &&\n typeof visitorOrReverse !== 'function'\n ) {\n test = undefined\n visitor = testOrVisitor\n reverse = visitorOrReverse\n } else {\n // @ts-expect-error: assume the overload with test was given.\n test = testOrVisitor\n // @ts-expect-error: assume the overload with test was given.\n visitor = visitorOrReverse\n reverse = maybeReverse\n }\n\n visitParents(tree, test, overload, reverse)\n\n /**\n * @param {UnistNode} node\n * @param {Array<UnistParent>} parents\n */\n function overload(node, parents) {\n const parent = parents[parents.length - 1]\n const index = parent ? parent.children.indexOf(node) : undefined\n return visitor(node, index, parent)\n }\n}\n","/**\n * @typedef {import('mdast').Nodes} Nodes\n *\n * @typedef Options\n * Configuration (optional).\n * @property {boolean | null | undefined} [includeImageAlt=true]\n * Whether to use `alt` for `image`s (default: `true`).\n * @property {boolean | null | undefined} [includeHtml=true]\n * Whether to use `value` of HTML (default: `true`).\n */\n\n/** @type {Options} */\nconst emptyOptions = {}\n\n/**\n * Get the text content of a node or list of nodes.\n *\n * Prefers the node’s plain-text fields, otherwise serializes its children,\n * and if the given value is an array, serialize the nodes in it.\n *\n * @param {unknown} [value]\n * Thing to serialize, typically `Node`.\n * @param {Options | null | undefined} [options]\n * Configuration (optional).\n * @returns {string}\n * Serialized `value`.\n */\nexport function toString(value, options) {\n const settings = options || emptyOptions\n const includeImageAlt =\n typeof settings.includeImageAlt === 'boolean'\n ? settings.includeImageAlt\n : true\n const includeHtml =\n typeof settings.includeHtml === 'boolean' ? settings.includeHtml : true\n\n return one(value, includeImageAlt, includeHtml)\n}\n\n/**\n * One node or several nodes.\n *\n * @param {unknown} value\n * Thing to serialize.\n * @param {boolean} includeImageAlt\n * Include image `alt`s.\n * @param {boolean} includeHtml\n * Include HTML.\n * @returns {string}\n * Serialized node.\n */\nfunction one(value, includeImageAlt, includeHtml) {\n if (node(value)) {\n if ('value' in value) {\n return value.type === 'html' && !includeHtml ? '' : value.value\n }\n\n if (includeImageAlt && 'alt' in value && value.alt) {\n return value.alt\n }\n\n if ('children' in value) {\n return all(value.children, includeImageAlt, includeHtml)\n }\n }\n\n if (Array.isArray(value)) {\n return all(value, includeImageAlt, includeHtml)\n }\n\n return ''\n}\n\n/**\n * Serialize a list of nodes.\n *\n * @param {Array<unknown>} values\n * Thing to serialize.\n * @param {boolean} includeImageAlt\n * Include image `alt`s.\n * @param {boolean} includeHtml\n * Include HTML.\n * @returns {string}\n * Serialized nodes.\n */\nfunction all(values, includeImageAlt, includeHtml) {\n /** @type {Array<string>} */\n const result = []\n let index = -1\n\n while (++index < values.length) {\n result[index] = one(values[index], includeImageAlt, includeHtml)\n }\n\n return result.join('')\n}\n\n/**\n * Check if `value` looks like a node.\n *\n * @param {unknown} value\n * Thing.\n * @returns {value is Nodes}\n * Whether `value` is a node.\n */\nfunction node(value) {\n return Boolean(value && typeof value === 'object')\n}\n","/**\n * @import {State} from 'mdast-util-to-markdown'\n * @import {Heading} from 'mdast'\n */\n\nimport {EXIT, visit} from 'unist-util-visit'\nimport {toString} from 'mdast-util-to-string'\n\n/**\n * @param {Heading} node\n * @param {State} state\n * @returns {boolean}\n */\nexport function formatHeadingAsSetext(node, state) {\n let literalWithBreak = false\n\n // Look for literals with a line break.\n // Note that this also\n visit(node, function (node) {\n if (\n ('value' in node && /\\r?\\n|\\r/.test(node.value)) ||\n node.type === 'break'\n ) {\n literalWithBreak = true\n return EXIT\n }\n })\n\n return Boolean(\n (!node.depth || node.depth < 3) &&\n toString(node) &&\n (state.options.setext || literalWithBreak)\n )\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Heading, Parents} from 'mdast'\n */\n\nimport {encodeCharacterReference} from '../util/encode-character-reference.js'\nimport {formatHeadingAsSetext} from '../util/format-heading-as-setext.js'\n\n/**\n * @param {Heading} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function heading(node, _, state, info) {\n const rank = Math.max(Math.min(6, node.depth || 1), 1)\n const tracker = state.createTracker(info)\n\n if (formatHeadingAsSetext(node, state)) {\n const exit = state.enter('headingSetext')\n const subexit = state.enter('phrasing')\n const value = state.containerPhrasing(node, {\n ...tracker.current(),\n before: '\\n',\n after: '\\n'\n })\n subexit()\n exit()\n\n return (\n value +\n '\\n' +\n (rank === 1 ? '=' : '-').repeat(\n // The whole size…\n value.length -\n // Minus the position of the character after the last EOL (or\n // 0 if there is none)…\n (Math.max(value.lastIndexOf('\\r'), value.lastIndexOf('\\n')) + 1)\n )\n )\n }\n\n const sequence = '#'.repeat(rank)\n const exit = state.enter('headingAtx')\n const subexit = state.enter('phrasing')\n\n // Note: for proper tracking, we should reset the output positions when there\n // is no content returned, because then the space is not output.\n // Practically, in that case, there is no content, so it doesn’t matter that\n // we’ve tracked one too many characters.\n tracker.move(sequence + ' ')\n\n let value = state.containerPhrasing(node, {\n before: '# ',\n after: '\\n',\n ...tracker.current()\n })\n\n if (/^[\\t ]/.test(value)) {\n // To do: what effect has the character reference on tracking?\n value = encodeCharacterReference(value.charCodeAt(0)) + value.slice(1)\n }\n\n value = value ? sequence + ' ' + value : sequence\n\n if (state.options.closeAtx) {\n value += ' ' + sequence\n }\n\n subexit()\n exit()\n\n return value\n}\n","/**\n * @import {Html} from 'mdast'\n */\n\nhtml.peek = htmlPeek\n\n/**\n * @param {Html} node\n * @returns {string}\n */\nexport function html(node) {\n return node.value || ''\n}\n\n/**\n * @returns {string}\n */\nfunction htmlPeek() {\n return '<'\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Image, Parents} from 'mdast'\n */\n\nimport {checkQuote} from '../util/check-quote.js'\n\nimage.peek = imagePeek\n\n/**\n * @param {Image} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function image(node, _, state, info) {\n const quote = checkQuote(state)\n const suffix = quote === '\"' ? 'Quote' : 'Apostrophe'\n const exit = state.enter('image')\n let subexit = state.enter('label')\n const tracker = state.createTracker(info)\n let value = tracker.move('![')\n value += tracker.move(\n state.safe(node.alt, {before: value, after: ']', ...tracker.current()})\n )\n value += tracker.move('](')\n\n subexit()\n\n if (\n // If there’s no url but there is a title…\n (!node.url && node.title) ||\n // If there are control characters or whitespace.\n /[\\0- \\u007F]/.test(node.url)\n ) {\n subexit = state.enter('destinationLiteral')\n value += tracker.move('<')\n value += tracker.move(\n state.safe(node.url, {before: value, after: '>', ...tracker.current()})\n )\n value += tracker.move('>')\n } else {\n // No whitespace, raw is prettier.\n subexit = state.enter('destinationRaw')\n value += tracker.move(\n state.safe(node.url, {\n before: value,\n after: node.title ? ' ' : ')',\n ...tracker.current()\n })\n )\n }\n\n subexit()\n\n if (node.title) {\n subexit = state.enter(`title${suffix}`)\n value += tracker.move(' ' + quote)\n value += tracker.move(\n state.safe(node.title, {\n before: value,\n after: quote,\n ...tracker.current()\n })\n )\n value += tracker.move(quote)\n subexit()\n }\n\n value += tracker.move(')')\n exit()\n\n return value\n}\n\n/**\n * @returns {string}\n */\nfunction imagePeek() {\n return '!'\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {ImageReference, Parents} from 'mdast'\n */\n\nimageReference.peek = imageReferencePeek\n\n/**\n * @param {ImageReference} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function imageReference(node, _, state, info) {\n const type = node.referenceType\n const exit = state.enter('imageReference')\n let subexit = state.enter('label')\n const tracker = state.createTracker(info)\n let value = tracker.move('![')\n const alt = state.safe(node.alt, {\n before: value,\n after: ']',\n ...tracker.current()\n })\n value += tracker.move(alt + '][')\n\n subexit()\n // Hide the fact that we’re in phrasing, because escapes don’t work.\n const stack = state.stack\n state.stack = []\n subexit = state.enter('reference')\n // Note: for proper tracking, we should reset the output positions when we end\n // up making a `shortcut` reference, because then there is no brace output.\n // Practically, in that case, there is no content, so it doesn’t matter that\n // we’ve tracked one too many characters.\n const reference = state.safe(state.associationId(node), {\n before: value,\n after: ']',\n ...tracker.current()\n })\n subexit()\n state.stack = stack\n exit()\n\n if (type === 'full' || !alt || alt !== reference) {\n value += tracker.move(reference + ']')\n } else if (type === 'shortcut') {\n // Remove the unwanted `[`.\n value = value.slice(0, -1)\n } else {\n value += tracker.move(']')\n }\n\n return value\n}\n\n/**\n * @returns {string}\n */\nfunction imageReferencePeek() {\n return '!'\n}\n","/**\n * @import {State} from 'mdast-util-to-markdown'\n * @import {InlineCode, Parents} from 'mdast'\n */\n\ninlineCode.peek = inlineCodePeek\n\n/**\n * @param {InlineCode} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @returns {string}\n */\nexport function inlineCode(node, _, state) {\n let value = node.value || ''\n let sequence = '`'\n let index = -1\n\n // If there is a single grave accent on its own in the code, use a fence of\n // two.\n // If there are two in a row, use one.\n while (new RegExp('(^|[^`])' + sequence + '([^`]|$)').test(value)) {\n sequence += '`'\n }\n\n // If this is not just spaces or eols (tabs don’t count), and either the\n // first or last character are a space, eol, or tick, then pad with spaces.\n if (\n /[^ \\r\\n]/.test(value) &&\n ((/^[ \\r\\n]/.test(value) && /[ \\r\\n]$/.test(value)) || /^`|`$/.test(value))\n ) {\n value = ' ' + value + ' '\n }\n\n // We have a potential problem: certain characters after eols could result in\n // blocks being seen.\n // For example, if someone injected the string `'\\n# b'`, then that would\n // result in an ATX heading.\n // We can’t escape characters in `inlineCode`, but because eols are\n // transformed to spaces when going from markdown to HTML anyway, we can swap\n // them out.\n while (++index < state.unsafe.length) {\n const pattern = state.unsafe[index]\n const expression = state.compilePattern(pattern)\n /** @type {RegExpExecArray | null} */\n let match\n\n // Only look for `atBreak`s.\n // Btw: note that `atBreak` patterns will always start the regex at LF or\n // CR.\n if (!pattern.atBreak) continue\n\n while ((match = expression.exec(value))) {\n let position = match.index\n\n // Support CRLF (patterns only look for one of the characters).\n if (\n value.charCodeAt(position) === 10 /* `\\n` */ &&\n value.charCodeAt(position - 1) === 13 /* `\\r` */\n ) {\n position--\n }\n\n value = value.slice(0, position) + ' ' + value.slice(match.index + 1)\n }\n }\n\n return sequence + value + sequence\n}\n\n/**\n * @returns {string}\n */\nfunction inlineCodePeek() {\n return '`'\n}\n","/**\n * @import {State} from 'mdast-util-to-markdown'\n * @import {Link} from 'mdast'\n */\n\nimport {toString} from 'mdast-util-to-string'\n\n/**\n * @param {Link} node\n * @param {State} state\n * @returns {boolean}\n */\nexport function formatLinkAsAutolink(node, state) {\n const raw = toString(node)\n\n return Boolean(\n !state.options.resourceLink &&\n // If there’s a url…\n node.url &&\n // And there’s a no title…\n !node.title &&\n // And the content of `node` is a single text node…\n node.children &&\n node.children.length === 1 &&\n node.children[0].type === 'text' &&\n // And if the url is the same as the content…\n (raw === node.url || 'mailto:' + raw === node.url) &&\n // And that starts w/ a protocol…\n /^[a-z][a-z+.-]+:/i.test(node.url) &&\n // And that doesn’t contain ASCII control codes (character escapes and\n // references don’t work), space, or angle brackets…\n !/[\\0- <>\\u007F]/.test(node.url)\n )\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Link, Parents} from 'mdast'\n * @import {Exit} from '../types.js'\n */\n\nimport {checkQuote} from '../util/check-quote.js'\nimport {formatLinkAsAutolink} from '../util/format-link-as-autolink.js'\n\nlink.peek = linkPeek\n\n/**\n * @param {Link} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function link(node, _, state, info) {\n const quote = checkQuote(state)\n const suffix = quote === '\"' ? 'Quote' : 'Apostrophe'\n const tracker = state.createTracker(info)\n /** @type {Exit} */\n let exit\n /** @type {Exit} */\n let subexit\n\n if (formatLinkAsAutolink(node, state)) {\n // Hide the fact that we’re in phrasing, because escapes don’t work.\n const stack = state.stack\n state.stack = []\n exit = state.enter('autolink')\n let value = tracker.move('<')\n value += tracker.move(\n state.containerPhrasing(node, {\n before: value,\n after: '>',\n ...tracker.current()\n })\n )\n value += tracker.move('>')\n exit()\n state.stack = stack\n return value\n }\n\n exit = state.enter('link')\n subexit = state.enter('label')\n let value = tracker.move('[')\n value += tracker.move(\n state.containerPhrasing(node, {\n before: value,\n after: '](',\n ...tracker.current()\n })\n )\n value += tracker.move('](')\n subexit()\n\n if (\n // If there’s no url but there is a title…\n (!node.url && node.title) ||\n // If there are control characters or whitespace.\n /[\\0- \\u007F]/.test(node.url)\n ) {\n subexit = state.enter('destinationLiteral')\n value += tracker.move('<')\n value += tracker.move(\n state.safe(node.url, {before: value, after: '>', ...tracker.current()})\n )\n value += tracker.move('>')\n } else {\n // No whitespace, raw is prettier.\n subexit = state.enter('destinationRaw')\n value += tracker.move(\n state.safe(node.url, {\n before: value,\n after: node.title ? ' ' : ')',\n ...tracker.current()\n })\n )\n }\n\n subexit()\n\n if (node.title) {\n subexit = state.enter(`title${suffix}`)\n value += tracker.move(' ' + quote)\n value += tracker.move(\n state.safe(node.title, {\n before: value,\n after: quote,\n ...tracker.current()\n })\n )\n value += tracker.move(quote)\n subexit()\n }\n\n value += tracker.move(')')\n\n exit()\n return value\n}\n\n/**\n * @param {Link} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @returns {string}\n */\nfunction linkPeek(node, _, state) {\n return formatLinkAsAutolink(node, state) ? '<' : '['\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {LinkReference, Parents} from 'mdast'\n */\n\nlinkReference.peek = linkReferencePeek\n\n/**\n * @param {LinkReference} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function linkReference(node, _, state, info) {\n const type = node.referenceType\n const exit = state.enter('linkReference')\n let subexit = state.enter('label')\n const tracker = state.createTracker(info)\n let value = tracker.move('[')\n const text = state.containerPhrasing(node, {\n before: value,\n after: ']',\n ...tracker.current()\n })\n value += tracker.move(text + '][')\n\n subexit()\n // Hide the fact that we’re in phrasing, because escapes don’t work.\n const stack = state.stack\n state.stack = []\n subexit = state.enter('reference')\n // Note: for proper tracking, we should reset the output positions when we end\n // up making a `shortcut` reference, because then there is no brace output.\n // Practically, in that case, there is no content, so it doesn’t matter that\n // we’ve tracked one too many characters.\n const reference = state.safe(state.associationId(node), {\n before: value,\n after: ']',\n ...tracker.current()\n })\n subexit()\n state.stack = stack\n exit()\n\n if (type === 'full' || !text || text !== reference) {\n value += tracker.move(reference + ']')\n } else if (type === 'shortcut') {\n // Remove the unwanted `[`.\n value = value.slice(0, -1)\n } else {\n value += tracker.move(']')\n }\n\n return value\n}\n\n/**\n * @returns {string}\n */\nfunction linkReferencePeek() {\n return '['\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['bullet'], null | undefined>}\n */\nexport function checkBullet(state) {\n const marker = state.options.bullet || '*'\n\n if (marker !== '*' && marker !== '+' && marker !== '-') {\n throw new Error(\n 'Cannot serialize items with `' +\n marker +\n '` for `options.bullet`, expected `*`, `+`, or `-`'\n )\n }\n\n return marker\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\nimport {checkBullet} from './check-bullet.js'\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['bullet'], null | undefined>}\n */\nexport function checkBulletOther(state) {\n const bullet = checkBullet(state)\n const bulletOther = state.options.bulletOther\n\n if (!bulletOther) {\n return bullet === '*' ? '-' : '*'\n }\n\n if (bulletOther !== '*' && bulletOther !== '+' && bulletOther !== '-') {\n throw new Error(\n 'Cannot serialize items with `' +\n bulletOther +\n '` for `options.bulletOther`, expected `*`, `+`, or `-`'\n )\n }\n\n if (bulletOther === bullet) {\n throw new Error(\n 'Expected `bullet` (`' +\n bullet +\n '`) and `bulletOther` (`' +\n bulletOther +\n '`) to be different'\n )\n }\n\n return bulletOther\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['bulletOrdered'], null | undefined>}\n */\nexport function checkBulletOrdered(state) {\n const marker = state.options.bulletOrdered || '.'\n\n if (marker !== '.' && marker !== ')') {\n throw new Error(\n 'Cannot serialize items with `' +\n marker +\n '` for `options.bulletOrdered`, expected `.` or `)`'\n )\n }\n\n return marker\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['rule'], null | undefined>}\n */\nexport function checkRule(state) {\n const marker = state.options.rule || '*'\n\n if (marker !== '*' && marker !== '-' && marker !== '_') {\n throw new Error(\n 'Cannot serialize rules with `' +\n marker +\n '` for `options.rule`, expected `*`, `-`, or `_`'\n )\n }\n\n return marker\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {List, Parents} from 'mdast'\n */\n\nimport {checkBullet} from '../util/check-bullet.js'\nimport {checkBulletOther} from '../util/check-bullet-other.js'\nimport {checkBulletOrdered} from '../util/check-bullet-ordered.js'\nimport {checkRule} from '../util/check-rule.js'\n\n/**\n * @param {List} node\n * @param {Parents | undefined} parent\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function list(node, parent, state, info) {\n const exit = state.enter('list')\n const bulletCurrent = state.bulletCurrent\n /** @type {string} */\n let bullet = node.ordered ? checkBulletOrdered(state) : checkBullet(state)\n /** @type {string} */\n const bulletOther = node.ordered\n ? bullet === '.'\n ? ')'\n : '.'\n : checkBulletOther(state)\n let useDifferentMarker =\n parent && state.bulletLastUsed ? bullet === state.bulletLastUsed : false\n\n if (!node.ordered) {\n const firstListItem = node.children ? node.children[0] : undefined\n\n // If there’s an empty first list item directly in two list items,\n // we have to use a different bullet:\n //\n // ```markdown\n // * - *\n // ```\n //\n // …because otherwise it would become one big thematic break.\n if (\n // Bullet could be used as a thematic break marker:\n (bullet === '*' || bullet === '-') &&\n // Empty first list item:\n firstListItem &&\n (!firstListItem.children || !firstListItem.children[0]) &&\n // Directly in two other list items:\n state.stack[state.stack.length - 1] === 'list' &&\n state.stack[state.stack.length - 2] === 'listItem' &&\n state.stack[state.stack.length - 3] === 'list' &&\n state.stack[state.stack.length - 4] === 'listItem' &&\n // That are each the first child.\n state.indexStack[state.indexStack.length - 1] === 0 &&\n state.indexStack[state.indexStack.length - 2] === 0 &&\n state.indexStack[state.indexStack.length - 3] === 0\n ) {\n useDifferentMarker = true\n }\n\n // If there’s a thematic break at the start of the first list item,\n // we have to use a different bullet:\n //\n // ```markdown\n // * ---\n // ```\n //\n // …because otherwise it would become one big thematic break.\n if (checkRule(state) === bullet && firstListItem) {\n let index = -1\n\n while (++index < node.children.length) {\n const item = node.children[index]\n\n if (\n item &&\n item.type === 'listItem' &&\n item.children &&\n item.children[0] &&\n item.children[0].type === 'thematicBreak'\n ) {\n useDifferentMarker = true\n break\n }\n }\n }\n }\n\n if (useDifferentMarker) {\n bullet = bulletOther\n }\n\n state.bulletCurrent = bullet\n const value = state.containerFlow(node, info)\n state.bulletLastUsed = bullet\n state.bulletCurrent = bulletCurrent\n exit()\n return value\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['listItemIndent'], null | undefined>}\n */\nexport function checkListItemIndent(state) {\n const style = state.options.listItemIndent || 'one'\n\n if (style !== 'tab' && style !== 'one' && style !== 'mixed') {\n throw new Error(\n 'Cannot serialize items with `' +\n style +\n '` for `options.listItemIndent`, expected `tab`, `one`, or `mixed`'\n )\n }\n\n return style\n}\n","/**\n * @import {Info, Map, State} from 'mdast-util-to-markdown'\n * @import {ListItem, Parents} from 'mdast'\n */\n\nimport {checkBullet} from '../util/check-bullet.js'\nimport {checkListItemIndent} from '../util/check-list-item-indent.js'\n\n/**\n * @param {ListItem} node\n * @param {Parents | undefined} parent\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function listItem(node, parent, state, info) {\n const listItemIndent = checkListItemIndent(state)\n let bullet = state.bulletCurrent || checkBullet(state)\n\n // Add the marker value for ordered lists.\n if (parent && parent.type === 'list' && parent.ordered) {\n bullet =\n (typeof parent.start === 'number' && parent.start > -1\n ? parent.start\n : 1) +\n (state.options.incrementListMarker === false\n ? 0\n : parent.children.indexOf(node)) +\n bullet\n }\n\n let size = bullet.length + 1\n\n if (\n listItemIndent === 'tab' ||\n (listItemIndent === 'mixed' &&\n ((parent && parent.type === 'list' && parent.spread) || node.spread))\n ) {\n size = Math.ceil(size / 4) * 4\n }\n\n const tracker = state.createTracker(info)\n tracker.move(bullet + ' '.repeat(size - bullet.length))\n tracker.shift(size)\n const exit = state.enter('listItem')\n const value = state.indentLines(\n state.containerFlow(node, tracker.current()),\n map\n )\n exit()\n\n return value\n\n /** @type {Map} */\n function map(line, index, blank) {\n if (index) {\n return (blank ? '' : ' '.repeat(size)) + line\n }\n\n return (blank ? bullet : bullet + ' '.repeat(size - bullet.length)) + line\n }\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Paragraph, Parents} from 'mdast'\n */\n\n/**\n * @param {Paragraph} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function paragraph(node, _, state, info) {\n const exit = state.enter('paragraph')\n const subexit = state.enter('phrasing')\n const value = state.containerPhrasing(node, info)\n subexit()\n exit()\n return value\n}\n","/**\n * @typedef {import('mdast').Html} Html\n * @typedef {import('mdast').PhrasingContent} PhrasingContent\n */\n\nimport {convert} from 'unist-util-is'\n\n/**\n * Check if the given value is *phrasing content*.\n *\n * > 👉 **Note**: Excludes `html`, which can be both phrasing or flow.\n *\n * @param node\n * Thing to check, typically `Node`.\n * @returns\n * Whether `value` is phrasing content.\n */\n\nexport const phrasing =\n /** @type {(node?: unknown) => node is Exclude<PhrasingContent, Html>} */\n (\n convert([\n 'break',\n 'delete',\n 'emphasis',\n // To do: next major: removed since footnotes were added to GFM.\n 'footnote',\n 'footnoteReference',\n 'image',\n 'imageReference',\n 'inlineCode',\n // Enabled by `mdast-util-math`:\n 'inlineMath',\n 'link',\n 'linkReference',\n // Enabled by `mdast-util-mdx`:\n 'mdxJsxTextElement',\n // Enabled by `mdast-util-mdx`:\n 'mdxTextExpression',\n 'strong',\n 'text',\n // Enabled by `mdast-util-directive`:\n 'textDirective'\n ])\n )\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Parents, Root} from 'mdast'\n */\n\nimport {phrasing} from 'mdast-util-phrasing'\n\n/**\n * @param {Root} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function root(node, _, state, info) {\n // Note: `html` nodes are ambiguous.\n const hasPhrasing = node.children.some(function (d) {\n return phrasing(d)\n })\n\n const container = hasPhrasing ? state.containerPhrasing : state.containerFlow\n return container.call(state, node, info)\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['strong'], null | undefined>}\n */\nexport function checkStrong(state) {\n const marker = state.options.strong || '*'\n\n if (marker !== '*' && marker !== '_') {\n throw new Error(\n 'Cannot serialize strong with `' +\n marker +\n '` for `options.strong`, expected `*`, or `_`'\n )\n }\n\n return marker\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Parents, Strong} from 'mdast'\n */\n\nimport {checkStrong} from '../util/check-strong.js'\nimport {encodeCharacterReference} from '../util/encode-character-reference.js'\nimport {encodeInfo} from '../util/encode-info.js'\n\nstrong.peek = strongPeek\n\n/**\n * @param {Strong} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function strong(node, _, state, info) {\n const marker = checkStrong(state)\n const exit = state.enter('strong')\n const tracker = state.createTracker(info)\n const before = tracker.move(marker + marker)\n\n let between = tracker.move(\n state.containerPhrasing(node, {\n after: marker,\n before,\n ...tracker.current()\n })\n )\n const betweenHead = between.charCodeAt(0)\n const open = encodeInfo(\n info.before.charCodeAt(info.before.length - 1),\n betweenHead,\n marker\n )\n\n if (open.inside) {\n between = encodeCharacterReference(betweenHead) + between.slice(1)\n }\n\n const betweenTail = between.charCodeAt(between.length - 1)\n const close = encodeInfo(info.after.charCodeAt(0), betweenTail, marker)\n\n if (close.inside) {\n between = between.slice(0, -1) + encodeCharacterReference(betweenTail)\n }\n\n const after = tracker.move(marker + marker)\n\n exit()\n\n state.attentionEncodeSurroundingInfo = {\n after: close.outside,\n before: open.outside\n }\n return before + between + after\n}\n\n/**\n * @param {Strong} _\n * @param {Parents | undefined} _1\n * @param {State} state\n * @returns {string}\n */\nfunction strongPeek(_, _1, state) {\n return state.options.strong || '*'\n}\n","/**\n * @import {Info, State} from 'mdast-util-to-markdown'\n * @import {Parents, Text} from 'mdast'\n */\n\n/**\n * @param {Text} node\n * @param {Parents | undefined} _\n * @param {State} state\n * @param {Info} info\n * @returns {string}\n */\nexport function text(node, _, state, info) {\n return state.safe(node.value, info)\n}\n","/**\n * @import {Options, State} from 'mdast-util-to-markdown'\n */\n\n/**\n * @param {State} state\n * @returns {Exclude<Options['ruleRepetition'], null | undefined>}\n */\nexport function checkRuleRepetition(state) {\n const repetition = state.options.ruleRepetition || 3\n\n if (repetition < 3) {\n throw new Error(\n 'Cannot serialize rules with repetition `' +\n repetition +\n '` for `options.ruleRepetition`, expected `3` or more'\n )\n }\n\n return repetition\n}\n","/**\n * @import {State} from 'mdast-util-to-markdown'\n * @import {Parents, ThematicBreak} from 'mdast'\n */\n\nimport {checkRuleRepetition} from '../util/check-rule-repetition.js'\nimport {checkRule} from '../util/check-rule.js'\n\n/**\n * @param {ThematicBreak} _\n * @param {Parents | undefined} _1\n * @param {State} state\n * @returns {string}\n */\nexport function thematicBreak(_, _1, state) {\n const value = (\n checkRule(state) + (state.options.ruleSpaces ? ' ' : '')\n ).repeat(checkRuleRepetition(state))\n\n return state.options.ruleSpaces ? value.slice(0, -1) : value\n}\n","import {blockquote} from './blockquote.js'\nimport {hardBreak} from './break.js'\nimport {code} from './code.js'\nimport {definition} from './definition.js'\nimport {emphasis} from './emphasis.js'\nimport {heading} from './heading.js'\nimport {html} from './html.js'\nimport {image} from './image.js'\nimport {imageReference} from './image-reference.js'\nimport {inlineCode} from './inline-code.js'\nimport {link} from './link.js'\nimport {linkReference} from './link-reference.js'\nimport {list} from './list.js'\nimport {listItem} from './list-item.js'\nimport {paragraph} from './paragraph.js'\nimport {root} from './root.js'\nimport {strong} from './strong.js'\nimport {text} from './text.js'\nimport {thematicBreak} from './thematic-break.js'\n\n/**\n * Default (CommonMark) handlers.\n */\nexport const handle = {\n blockquote,\n break: hardBreak,\n code,\n definition,\n emphasis,\n hardBreak,\n heading,\n html,\n image,\n imageReference,\n inlineCode,\n link,\n linkReference,\n list,\n listItem,\n paragraph,\n root,\n strong,\n text,\n thematicBreak\n}\n","/**\n * @typedef {import('mdast').InlineCode} InlineCode\n * @typedef {import('mdast').Table} Table\n * @typedef {import('mdast').TableCell} TableCell\n * @typedef {import('mdast').TableRow} TableRow\n *\n * @typedef {import('markdown-table').Options} MarkdownTableOptions\n *\n * @typedef {import('mdast-util-from-markdown').CompileContext} CompileContext\n * @typedef {import('mdast-util-from-markdown').Extension} FromMarkdownExtension\n * @typedef {import('mdast-util-from-markdown').Handle} FromMarkdownHandle\n *\n * @typedef {import('mdast-util-to-markdown').Options} ToMarkdownExtension\n * @typedef {import('mdast-util-to-markdown').Handle} ToMarkdownHandle\n * @typedef {import('mdast-util-to-markdown').State} State\n * @typedef {import('mdast-util-to-markdown').Info} Info\n */\n\n/**\n * @typedef Options\n * Configuration.\n * @property {boolean | null | undefined} [tableCellPadding=true]\n * Whether to add a space of padding between delimiters and cells (default:\n * `true`).\n * @property {boolean | null | undefined} [tablePipeAlign=true]\n * Whether to align the delimiters (default: `true`).\n * @property {MarkdownTableOptions['stringLength'] | null | undefined} [stringLength]\n * Function to detect the length of table cell content, used when aligning\n * the delimiters between cells (optional).\n */\n\nimport {ok as assert} from 'devlop'\nimport {markdownTable} from 'markdown-table'\nimport {defaultHandlers} from 'mdast-util-to-markdown'\n\n/**\n * Create an extension for `mdast-util-from-markdown` to enable GFM tables in\n * markdown.\n *\n * @returns {FromMarkdownExtension}\n * Extension for `mdast-util-from-markdown` to enable GFM tables.\n */\nexport function gfmTableFromMarkdown() {\n return {\n enter: {\n table: enterTable,\n tableData: enterCell,\n tableHeader: enterCell,\n tableRow: enterRow\n },\n exit: {\n codeText: exitCodeText,\n table: exitTable,\n tableData: exit,\n tableHeader: exit,\n tableRow: exit\n }\n }\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterTable(token) {\n const align = token._align\n assert(align, 'expected `_align` on table')\n this.enter(\n {\n type: 'table',\n align: align.map(function (d) {\n return d === 'none' ? null : d\n }),\n children: []\n },\n token\n )\n this.data.inTable = true\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitTable(token) {\n this.exit(token)\n this.data.inTable = undefined\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterRow(token) {\n this.enter({type: 'tableRow', children: []}, token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exit(token) {\n this.exit(token)\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction enterCell(token) {\n this.enter({type: 'tableCell', children: []}, token)\n}\n\n// Overwrite the default code text data handler to unescape escaped pipes when\n// they are in tables.\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitCodeText(token) {\n let value = this.resume()\n\n if (this.data.inTable) {\n value = value.replace(/\\\\([\\\\|])/g, replace)\n }\n\n const node = this.stack[this.stack.length - 1]\n assert(node.type === 'inlineCode')\n node.value = value\n this.exit(token)\n}\n\n/**\n * @param {string} $0\n * @param {string} $1\n * @returns {string}\n */\nfunction replace($0, $1) {\n // Pipes work, backslashes don’t (but can’t escape pipes).\n return $1 === '|' ? $1 : $0\n}\n\n/**\n * Create an extension for `mdast-util-to-markdown` to enable GFM tables in\n * markdown.\n *\n * @param {Options | null | undefined} [options]\n * Configuration.\n * @returns {ToMarkdownExtension}\n * Extension for `mdast-util-to-markdown` to enable GFM tables.\n */\nexport function gfmTableToMarkdown(options) {\n const settings = options || {}\n const padding = settings.tableCellPadding\n const alignDelimiters = settings.tablePipeAlign\n const stringLength = settings.stringLength\n const around = padding ? ' ' : '|'\n\n return {\n unsafe: [\n {character: '\\r', inConstruct: 'tableCell'},\n {character: '\\n', inConstruct: 'tableCell'},\n // A pipe, when followed by a tab or space (padding), or a dash or colon\n // (unpadded delimiter row), could result in a table.\n {atBreak: true, character: '|', after: '[\\t :-]'},\n // A pipe in a cell must be encoded.\n {character: '|', inConstruct: 'tableCell'},\n // A colon must be followed by a dash, in which case it could start a\n // delimiter row.\n {atBreak: true, character: ':', after: '-'},\n // A delimiter row can also start with a dash, when followed by more\n // dashes, a colon, or a pipe.\n // This is a stricter version than the built in check for lists, thematic\n // breaks, and setex heading underlines though:\n // <https://github.com/syntax-tree/mdast-util-to-markdown/blob/51a2038/lib/unsafe.js#L57>\n {atBreak: true, character: '-', after: '[:|-]'}\n ],\n handlers: {\n inlineCode: inlineCodeWithTable,\n table: handleTable,\n tableCell: handleTableCell,\n tableRow: handleTableRow\n }\n }\n\n /**\n * @type {ToMarkdownHandle}\n * @param {Table} node\n */\n function handleTable(node, _, state, info) {\n return serializeData(handleTableAsData(node, state, info), node.align)\n }\n\n /**\n * This function isn’t really used normally, because we handle rows at the\n * table level.\n * But, if someone passes in a table row, this ensures we make somewhat sense.\n *\n * @type {ToMarkdownHandle}\n * @param {TableRow} node\n */\n function handleTableRow(node, _, state, info) {\n const row = handleTableRowAsData(node, state, info)\n const value = serializeData([row])\n // `markdown-table` will always add an align row\n return value.slice(0, value.indexOf('\\n'))\n }\n\n /**\n * @type {ToMarkdownHandle}\n * @param {TableCell} node\n */\n function handleTableCell(node, _, state, info) {\n const exit = state.enter('tableCell')\n const subexit = state.enter('phrasing')\n const value = state.containerPhrasing(node, {\n ...info,\n before: around,\n after: around\n })\n subexit()\n exit()\n return value\n }\n\n /**\n * @param {Array<Array<string>>} matrix\n * @param {Array<string | null | undefined> | null | undefined} [align]\n */\n function serializeData(matrix, align) {\n return markdownTable(matrix, {\n align,\n // @ts-expect-error: `markdown-table` types should support `null`.\n alignDelimiters,\n // @ts-expect-error: `markdown-table` types should support `null`.\n padding,\n // @ts-expect-error: `markdown-table` types should support `null`.\n stringLength\n })\n }\n\n /**\n * @param {Table} node\n * @param {State} state\n * @param {Info} info\n */\n function handleTableAsData(node, state, info) {\n const children = node.children\n let index = -1\n /** @type {Array<Array<string>>} */\n const result = []\n const subexit = state.enter('table')\n\n while (++index < children.length) {\n result[index] = handleTableRowAsData(children[index], state, info)\n }\n\n subexit()\n\n return result\n }\n\n /**\n * @param {TableRow} node\n * @param {State} state\n * @param {Info} info\n */\n function handleTableRowAsData(node, state, info) {\n const children = node.children\n let index = -1\n /** @type {Array<string>} */\n const result = []\n const subexit = state.enter('tableRow')\n\n while (++index < children.length) {\n // Note: the positional info as used here is incorrect.\n // Making it correct would be impossible due to aligning cells?\n // And it would need copy/pasting `markdown-table` into this project.\n result[index] = handleTableCell(children[index], node, state, info)\n }\n\n subexit()\n\n return result\n }\n\n /**\n * @type {ToMarkdownHandle}\n * @param {InlineCode} node\n */\n function inlineCodeWithTable(node, parent, state) {\n let value = defaultHandlers.inlineCode(node, parent, state)\n\n if (state.stack.includes('tableCell')) {\n value = value.replace(/\\|/g, '\\\\$&')\n }\n\n return value\n }\n}\n","/**\n * @typedef {import('mdast').ListItem} ListItem\n * @typedef {import('mdast').Paragraph} Paragraph\n * @typedef {import('mdast-util-from-markdown').CompileContext} CompileContext\n * @typedef {import('mdast-util-from-markdown').Extension} FromMarkdownExtension\n * @typedef {import('mdast-util-from-markdown').Handle} FromMarkdownHandle\n * @typedef {import('mdast-util-to-markdown').Options} ToMarkdownExtension\n * @typedef {import('mdast-util-to-markdown').Handle} ToMarkdownHandle\n */\n\nimport {ok as assert} from 'devlop'\nimport {defaultHandlers} from 'mdast-util-to-markdown'\n\n/**\n * Create an extension for `mdast-util-from-markdown` to enable GFM task\n * list items in markdown.\n *\n * @returns {FromMarkdownExtension}\n * Extension for `mdast-util-from-markdown` to enable GFM task list items.\n */\nexport function gfmTaskListItemFromMarkdown() {\n return {\n exit: {\n taskListCheckValueChecked: exitCheck,\n taskListCheckValueUnchecked: exitCheck,\n paragraph: exitParagraphWithTaskListItem\n }\n }\n}\n\n/**\n * Create an extension for `mdast-util-to-markdown` to enable GFM task list\n * items in markdown.\n *\n * @returns {ToMarkdownExtension}\n * Extension for `mdast-util-to-markdown` to enable GFM task list items.\n */\nexport function gfmTaskListItemToMarkdown() {\n return {\n unsafe: [{atBreak: true, character: '-', after: '[:|-]'}],\n handlers: {listItem: listItemWithTaskListItem}\n }\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitCheck(token) {\n // We’re always in a paragraph, in a list item.\n const node = this.stack[this.stack.length - 2]\n assert(node.type === 'listItem')\n node.checked = token.type === 'taskListCheckValueChecked'\n}\n\n/**\n * @this {CompileContext}\n * @type {FromMarkdownHandle}\n */\nfunction exitParagraphWithTaskListItem(token) {\n const parent = this.stack[this.stack.length - 2]\n\n if (\n parent &&\n parent.type === 'listItem' &&\n typeof parent.checked === 'boolean'\n ) {\n const node = this.stack[this.stack.length - 1]\n assert(node.type === 'paragraph')\n const head = node.children[0]\n\n if (head && head.type === 'text') {\n const siblings = parent.children\n let index = -1\n /** @type {Paragraph | undefined} */\n let firstParaghraph\n\n while (++index < siblings.length) {\n const sibling = siblings[index]\n if (sibling.type === 'paragraph') {\n firstParaghraph = sibling\n break\n }\n }\n\n if (firstParaghraph === node) {\n // Must start with a space or a tab.\n head.value = head.value.slice(1)\n\n if (head.value.length === 0) {\n node.children.shift()\n } else if (\n node.position &&\n head.position &&\n typeof head.position.start.offset === 'number'\n ) {\n head.position.start.column++\n head.position.start.offset++\n node.position.start = Object.assign({}, head.position.start)\n }\n }\n }\n }\n\n this.exit(token)\n}\n\n/**\n * @type {ToMarkdownHandle}\n * @param {ListItem} node\n */\nfunction listItemWithTaskListItem(node, parent, state, info) {\n const head = node.children[0]\n const checkable =\n typeof node.checked === 'boolean' && head && head.type === 'paragraph'\n const checkbox = '[' + (node.checked ? 'x' : ' ') + '] '\n const tracker = state.createTracker(info)\n\n if (checkable) {\n tracker.move(checkbox)\n }\n\n let value = defaultHandlers.listItem(node, parent, state, {\n ...info,\n ...tracker.current()\n })\n\n if (checkable) {\n value = value.replace(/^(?:[*+-]|\\d+\\.)([\\r\\n]| {1,3})/, check)\n }\n\n return value\n\n /**\n * @param {string} $0\n * @returns {string}\n */\n function check($0) {\n return $0 + checkbox\n }\n}\n","/**\n * @import {Extension as FromMarkdownExtension} from 'mdast-util-from-markdown'\n * @import {Options} from 'mdast-util-gfm'\n * @import {Options as ToMarkdownExtension} from 'mdast-util-to-markdown'\n */\n\nimport {\n gfmAutolinkLiteralFromMarkdown,\n gfmAutolinkLiteralToMarkdown\n} from 'mdast-util-gfm-autolink-literal'\nimport {\n gfmFootnoteFromMarkdown,\n gfmFootnoteToMarkdown\n} from 'mdast-util-gfm-footnote'\nimport {\n gfmStrikethroughFromMarkdown,\n gfmStrikethroughToMarkdown\n} from 'mdast-util-gfm-strikethrough'\nimport {gfmTableFromMarkdown, gfmTableToMarkdown} from 'mdast-util-gfm-table'\nimport {\n gfmTaskListItemFromMarkdown,\n gfmTaskListItemToMarkdown\n} from 'mdast-util-gfm-task-list-item'\n\n/**\n * Create an extension for `mdast-util-from-markdown` to enable GFM (autolink\n * literals, footnotes, strikethrough, tables, tasklists).\n *\n * @returns {Array<FromMarkdownExtension>}\n * Extension for `mdast-util-from-markdown` to enable GFM (autolink literals,\n * footnotes, strikethrough, tables, tasklists).\n */\nexport function gfmFromMarkdown() {\n return [\n gfmAutolinkLiteralFromMarkdown(),\n gfmFootnoteFromMarkdown(),\n gfmStrikethroughFromMarkdown(),\n gfmTableFromMarkdown(),\n gfmTaskListItemFromMarkdown()\n ]\n}\n\n/**\n * Create an extension for `mdast-util-to-markdown` to enable GFM (autolink\n * literals, footnotes, strikethrough, tables, tasklists).\n *\n * @param {Options | null | undefined} [options]\n * Configuration (optional).\n * @returns {ToMarkdownExtension}\n * Extension for `mdast-util-to-markdown` to enable GFM (autolink literals,\n * footnotes, strikethrough, tables, tasklists).\n */\nexport function gfmToMarkdown(options) {\n return {\n extensions: [\n gfmAutolinkLiteralToMarkdown(),\n gfmFootnoteToMarkdown(options),\n gfmStrikethroughToMarkdown(),\n gfmTableToMarkdown(options),\n gfmTaskListItemToMarkdown()\n ]\n }\n}\n","/**\n * Like `Array#splice`, but smarter for giant arrays.\n *\n * `Array#splice` takes all items to be inserted as individual argument which\n * causes a stack overflow in V8 when trying to insert 100k items for instance.\n *\n * Otherwise, this does not return the removed items, and takes `items` as an\n * array instead of rest parameters.\n *\n * @template {unknown} T\n * Item type.\n * @param {Array<T>} list\n * List to operate on.\n * @param {number} start\n * Index to remove/insert at (can be negative).\n * @param {number} remove\n * Number of items to remove.\n * @param {Array<T>} items\n * Items to inject into `list`.\n * @returns {undefined}\n * Nothing.\n */\nexport function splice(list, start, remove, items) {\n const end = list.length;\n let chunkStart = 0;\n /** @type {Array<unknown>} */\n let parameters;\n\n // Make start between zero and `end` (included).\n if (start < 0) {\n start = -start > end ? 0 : end + start;\n } else {\n start = start > end ? end : start;\n }\n remove = remove > 0 ? remove : 0;\n\n // No need to chunk the items if there’s only a couple (10k) items.\n if (items.length < 10000) {\n parameters = Array.from(items);\n parameters.unshift(start, remove);\n // @ts-expect-error Hush, it’s fine.\n list.splice(...parameters);\n } else {\n // Delete `remove` items starting from `start`\n if (remove) list.splice(start, remove);\n\n // Insert the items in chunks to not cause stack overflows.\n while (chunkStart < items.length) {\n parameters = items.slice(chunkStart, chunkStart + 10000);\n parameters.unshift(start, 0);\n // @ts-expect-error Hush, it’s fine.\n list.splice(...parameters);\n chunkStart += 10000;\n start += 10000;\n }\n }\n}\n\n/**\n * Append `items` (an array) at the end of `list` (another array).\n * When `list` was empty, returns `items` instead.\n *\n * This prevents a potentially expensive operation when `list` is empty,\n * and adds items in batches to prevent V8 from hanging.\n *\n * @template {unknown} T\n * Item type.\n * @param {Array<T>} list\n * List to operate on.\n * @param {Array<T>} items\n * Items to add to `list`.\n * @returns {Array<T>}\n * Either `list` or `items`.\n */\nexport function push(list, items) {\n if (list.length > 0) {\n splice(list, list.length, 0, items);\n return list;\n }\n return items;\n}","/**\n * @import {\n * Extension,\n * Handles,\n * HtmlExtension,\n * NormalizedExtension\n * } from 'micromark-util-types'\n */\n\nimport {splice} from 'micromark-util-chunked'\n\nconst hasOwnProperty = {}.hasOwnProperty\n\n/**\n * Combine multiple syntax extensions into one.\n *\n * @param {ReadonlyArray<Extension>} extensions\n * List of syntax extensions.\n * @returns {NormalizedExtension}\n * A single combined extension.\n */\nexport function combineExtensions(extensions) {\n /** @type {NormalizedExtension} */\n const all = {}\n let index = -1\n\n while (++index < extensions.length) {\n syntaxExtension(all, extensions[index])\n }\n\n return all\n}\n\n/**\n * Merge `extension` into `all`.\n *\n * @param {NormalizedExtension} all\n * Extension to merge into.\n * @param {Extension} extension\n * Extension to merge.\n * @returns {undefined}\n * Nothing.\n */\nfunction syntaxExtension(all, extension) {\n /** @type {keyof Extension} */\n let hook\n\n for (hook in extension) {\n const maybe = hasOwnProperty.call(all, hook) ? all[hook] : undefined\n /** @type {Record<string, unknown>} */\n const left = maybe || (all[hook] = {})\n /** @type {Record<string, unknown> | undefined} */\n const right = extension[hook]\n /** @type {string} */\n let code\n\n if (right) {\n for (code in right) {\n if (!hasOwnProperty.call(left, code)) left[code] = []\n const value = right[code]\n constructs(\n // @ts-expect-error Looks like a list.\n left[code],\n Array.isArray(value) ? value : value ? [value] : []\n )\n }\n }\n }\n}\n\n/**\n * Merge `list` into `existing` (both lists of constructs).\n * Mutates `existing`.\n *\n * @param {Array<unknown>} existing\n * List of constructs to merge into.\n * @param {Array<unknown>} list\n * List of constructs to merge.\n * @returns {undefined}\n * Nothing.\n */\nfunction constructs(existing, list) {\n let index = -1\n /** @type {Array<unknown>} */\n const before = []\n\n while (++index < list.length) {\n // @ts-expect-error Looks like an object.\n ;(list[index].add === 'after' ? existing : before).push(list[index])\n }\n\n splice(existing, 0, 0, before)\n}\n\n/**\n * Combine multiple HTML extensions into one.\n *\n * @param {ReadonlyArray<HtmlExtension>} htmlExtensions\n * List of HTML extensions.\n * @returns {HtmlExtension}\n * Single combined HTML extension.\n */\nexport function combineHtmlExtensions(htmlExtensions) {\n /** @type {HtmlExtension} */\n const handlers = {}\n let index = -1\n\n while (++index < htmlExtensions.length) {\n htmlExtension(handlers, htmlExtensions[index])\n }\n\n return handlers\n}\n\n/**\n * Merge `extension` into `all`.\n *\n * @param {HtmlExtension} all\n * Extension to merge into.\n * @param {HtmlExtension} extension\n * Extension to merge.\n * @returns {undefined}\n * Nothing.\n */\nfunction htmlExtension(all, extension) {\n /** @type {keyof HtmlExtension} */\n let hook\n\n for (hook in extension) {\n const maybe = hasOwnProperty.call(all, hook) ? all[hook] : undefined\n const left = maybe || (all[hook] = {})\n const right = extension[hook]\n /** @type {keyof Handles} */\n let type\n\n if (right) {\n for (type in right) {\n // @ts-expect-error assume document vs regular handler are managed correctly.\n left[type] = right[type]\n }\n }\n }\n}\n","/**\n * @import {Code, ConstructRecord, Event, Extension, Previous, State, TokenizeContext, Tokenizer} from 'micromark-util-types'\n */\n\nimport { asciiAlpha, asciiAlphanumeric, asciiControl, markdownLineEndingOrSpace, unicodePunctuation, unicodeWhitespace } from 'micromark-util-character';\nconst wwwPrefix = {\n tokenize: tokenizeWwwPrefix,\n partial: true\n};\nconst domain = {\n tokenize: tokenizeDomain,\n partial: true\n};\nconst path = {\n tokenize: tokenizePath,\n partial: true\n};\nconst trail = {\n tokenize: tokenizeTrail,\n partial: true\n};\nconst emailDomainDotTrail = {\n tokenize: tokenizeEmailDomainDotTrail,\n partial: true\n};\nconst wwwAutolink = {\n name: 'wwwAutolink',\n tokenize: tokenizeWwwAutolink,\n previous: previousWww\n};\nconst protocolAutolink = {\n name: 'protocolAutolink',\n tokenize: tokenizeProtocolAutolink,\n previous: previousProtocol\n};\nconst emailAutolink = {\n name: 'emailAutolink',\n tokenize: tokenizeEmailAutolink,\n previous: previousEmail\n};\n\n/** @type {ConstructRecord} */\nconst text = {};\n\n/**\n * Create an extension for `micromark` to support GitHub autolink literal\n * syntax.\n *\n * @returns {Extension}\n * Extension for `micromark` that can be passed in `extensions` to enable GFM\n * autolink literal syntax.\n */\nexport function gfmAutolinkLiteral() {\n return {\n text\n };\n}\n\n/** @type {Code} */\nlet code = 48;\n\n// Add alphanumerics.\nwhile (code < 123) {\n text[code] = emailAutolink;\n code++;\n if (code === 58) code = 65;else if (code === 91) code = 97;\n}\ntext[43] = emailAutolink;\ntext[45] = emailAutolink;\ntext[46] = emailAutolink;\ntext[95] = emailAutolink;\ntext[72] = [emailAutolink, protocolAutolink];\ntext[104] = [emailAutolink, protocolAutolink];\ntext[87] = [emailAutolink, wwwAutolink];\ntext[119] = [emailAutolink, wwwAutolink];\n\n// To do: perform email autolink literals on events, afterwards.\n// That’s where `markdown-rs` and `cmark-gfm` perform it.\n// It should look for `@`, then for atext backwards, and then for a label\n// forwards.\n// To do: `mailto:`, `xmpp:` protocol as prefix.\n\n/**\n * Email autolink literal.\n *\n * ```markdown\n * > | a contact@example.org b\n * ^^^^^^^^^^^^^^^^^^^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeEmailAutolink(effects, ok, nok) {\n const self = this;\n /** @type {boolean | undefined} */\n let dot;\n /** @type {boolean} */\n let data;\n return start;\n\n /**\n * Start of email autolink literal.\n *\n * ```markdown\n * > | a contact@example.org b\n * ^\n * ```\n *\n * @type {State}\n */\n function start(code) {\n if (!gfmAtext(code) || !previousEmail.call(self, self.previous) || previousUnbalanced(self.events)) {\n return nok(code);\n }\n effects.enter('literalAutolink');\n effects.enter('literalAutolinkEmail');\n return atext(code);\n }\n\n /**\n * In email atext.\n *\n * ```markdown\n * > | a contact@example.org b\n * ^\n * ```\n *\n * @type {State}\n */\n function atext(code) {\n if (gfmAtext(code)) {\n effects.consume(code);\n return atext;\n }\n if (code === 64) {\n effects.consume(code);\n return emailDomain;\n }\n return nok(code);\n }\n\n /**\n * In email domain.\n *\n * The reference code is a bit overly complex as it handles the `@`, of which\n * there may be just one.\n * Source: <https://github.com/github/cmark-gfm/blob/ef1cfcb/extensions/autolink.c#L318>\n *\n * ```markdown\n * > | a contact@example.org b\n * ^\n * ```\n *\n * @type {State}\n */\n function emailDomain(code) {\n // Dot followed by alphanumerical (not `-` or `_`).\n if (code === 46) {\n return effects.check(emailDomainDotTrail, emailDomainAfter, emailDomainDot)(code);\n }\n\n // Alphanumerical, `-`, and `_`.\n if (code === 45 || code === 95 || asciiAlphanumeric(code)) {\n data = true;\n effects.consume(code);\n return emailDomain;\n }\n\n // To do: `/` if xmpp.\n\n // Note: normally we’d truncate trailing punctuation from the link.\n // However, email autolink literals cannot contain any of those markers,\n // except for `.`, but that can only occur if it isn’t trailing.\n // So we can ignore truncating!\n return emailDomainAfter(code);\n }\n\n /**\n * In email domain, on dot that is not a trail.\n *\n * ```markdown\n * > | a contact@example.org b\n * ^\n * ```\n *\n * @type {State}\n */\n function emailDomainDot(code) {\n effects.consume(code);\n dot = true;\n return emailDomain;\n }\n\n /**\n * After email domain.\n *\n * ```markdown\n * > | a contact@example.org b\n * ^\n * ```\n *\n * @type {State}\n */\n function emailDomainAfter(code) {\n // Domain must not be empty, must include a dot, and must end in alphabetical.\n // Source: <https://github.com/github/cmark-gfm/blob/ef1cfcb/extensions/autolink.c#L332>.\n if (data && dot && asciiAlpha(self.previous)) {\n effects.exit('literalAutolinkEmail');\n effects.exit('literalAutolink');\n return ok(code);\n }\n return nok(code);\n }\n}\n\n/**\n * `www` autolink literal.\n *\n * ```markdown\n * > | a www.example.org b\n * ^^^^^^^^^^^^^^^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeWwwAutolink(effects, ok, nok) {\n const self = this;\n return wwwStart;\n\n /**\n * Start of www autolink literal.\n *\n * ```markdown\n * > | www.example.com/a?b#c\n * ^\n * ```\n *\n * @type {State}\n */\n function wwwStart(code) {\n if (code !== 87 && code !== 119 || !previousWww.call(self, self.previous) || previousUnbalanced(self.events)) {\n return nok(code);\n }\n effects.enter('literalAutolink');\n effects.enter('literalAutolinkWww');\n // Note: we *check*, so we can discard the `www.` we parsed.\n // If it worked, we consider it as a part of the domain.\n return effects.check(wwwPrefix, effects.attempt(domain, effects.attempt(path, wwwAfter), nok), nok)(code);\n }\n\n /**\n * After a www autolink literal.\n *\n * ```markdown\n * > | www.example.com/a?b#c\n * ^\n * ```\n *\n * @type {State}\n */\n function wwwAfter(code) {\n effects.exit('literalAutolinkWww');\n effects.exit('literalAutolink');\n return ok(code);\n }\n}\n\n/**\n * Protocol autolink literal.\n *\n * ```markdown\n * > | a https://example.org b\n * ^^^^^^^^^^^^^^^^^^^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeProtocolAutolink(effects, ok, nok) {\n const self = this;\n let buffer = '';\n let seen = false;\n return protocolStart;\n\n /**\n * Start of protocol autolink literal.\n *\n * ```markdown\n * > | https://example.com/a?b#c\n * ^\n * ```\n *\n * @type {State}\n */\n function protocolStart(code) {\n if ((code === 72 || code === 104) && previousProtocol.call(self, self.previous) && !previousUnbalanced(self.events)) {\n effects.enter('literalAutolink');\n effects.enter('literalAutolinkHttp');\n buffer += String.fromCodePoint(code);\n effects.consume(code);\n return protocolPrefixInside;\n }\n return nok(code);\n }\n\n /**\n * In protocol.\n *\n * ```markdown\n * > | https://example.com/a?b#c\n * ^^^^^\n * ```\n *\n * @type {State}\n */\n function protocolPrefixInside(code) {\n // `5` is size of `https`\n if (asciiAlpha(code) && buffer.length < 5) {\n // @ts-expect-error: definitely number.\n buffer += String.fromCodePoint(code);\n effects.consume(code);\n return protocolPrefixInside;\n }\n if (code === 58) {\n const protocol = buffer.toLowerCase();\n if (protocol === 'http' || protocol === 'https') {\n effects.consume(code);\n return protocolSlashesInside;\n }\n }\n return nok(code);\n }\n\n /**\n * In slashes.\n *\n * ```markdown\n * > | https://example.com/a?b#c\n * ^^\n * ```\n *\n * @type {State}\n */\n function protocolSlashesInside(code) {\n if (code === 47) {\n effects.consume(code);\n if (seen) {\n return afterProtocol;\n }\n seen = true;\n return protocolSlashesInside;\n }\n return nok(code);\n }\n\n /**\n * After protocol, before domain.\n *\n * ```markdown\n * > | https://example.com/a?b#c\n * ^\n * ```\n *\n * @type {State}\n */\n function afterProtocol(code) {\n // To do: this is different from `markdown-rs`:\n // https://github.com/wooorm/markdown-rs/blob/b3a921c761309ae00a51fe348d8a43adbc54b518/src/construct/gfm_autolink_literal.rs#L172-L182\n return code === null || asciiControl(code) || markdownLineEndingOrSpace(code) || unicodeWhitespace(code) || unicodePunctuation(code) ? nok(code) : effects.attempt(domain, effects.attempt(path, protocolAfter), nok)(code);\n }\n\n /**\n * After a protocol autolink literal.\n *\n * ```markdown\n * > | https://example.com/a?b#c\n * ^\n * ```\n *\n * @type {State}\n */\n function protocolAfter(code) {\n effects.exit('literalAutolinkHttp');\n effects.exit('literalAutolink');\n return ok(code);\n }\n}\n\n/**\n * `www` prefix.\n *\n * ```markdown\n * > | a www.example.org b\n * ^^^^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeWwwPrefix(effects, ok, nok) {\n let size = 0;\n return wwwPrefixInside;\n\n /**\n * In www prefix.\n *\n * ```markdown\n * > | www.example.com\n * ^^^^\n * ```\n *\n * @type {State}\n */\n function wwwPrefixInside(code) {\n if ((code === 87 || code === 119) && size < 3) {\n size++;\n effects.consume(code);\n return wwwPrefixInside;\n }\n if (code === 46 && size === 3) {\n effects.consume(code);\n return wwwPrefixAfter;\n }\n return nok(code);\n }\n\n /**\n * After www prefix.\n *\n * ```markdown\n * > | www.example.com\n * ^\n * ```\n *\n * @type {State}\n */\n function wwwPrefixAfter(code) {\n // If there is *anything*, we can link.\n return code === null ? nok(code) : ok(code);\n }\n}\n\n/**\n * Domain.\n *\n * ```markdown\n * > | a https://example.org b\n * ^^^^^^^^^^^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeDomain(effects, ok, nok) {\n /** @type {boolean | undefined} */\n let underscoreInLastSegment;\n /** @type {boolean | undefined} */\n let underscoreInLastLastSegment;\n /** @type {boolean | undefined} */\n let seen;\n return domainInside;\n\n /**\n * In domain.\n *\n * ```markdown\n * > | https://example.com/a\n * ^^^^^^^^^^^\n * ```\n *\n * @type {State}\n */\n function domainInside(code) {\n // Check whether this marker, which is a trailing punctuation\n // marker, optionally followed by more trailing markers, and then\n // followed by an end.\n if (code === 46 || code === 95) {\n return effects.check(trail, domainAfter, domainAtPunctuation)(code);\n }\n\n // GH documents that only alphanumerics (other than `-`, `.`, and `_`) can\n // occur, which sounds like ASCII only, but they also support `www.點看.com`,\n // so that’s Unicode.\n // Instead of some new production for Unicode alphanumerics, markdown\n // already has that for Unicode punctuation and whitespace, so use those.\n // Source: <https://github.com/github/cmark-gfm/blob/ef1cfcb/extensions/autolink.c#L12>.\n if (code === null || markdownLineEndingOrSpace(code) || unicodeWhitespace(code) || code !== 45 && unicodePunctuation(code)) {\n return domainAfter(code);\n }\n seen = true;\n effects.consume(code);\n return domainInside;\n }\n\n /**\n * In domain, at potential trailing punctuation, that was not trailing.\n *\n * ```markdown\n * > | https://example.com\n * ^\n * ```\n *\n * @type {State}\n */\n function domainAtPunctuation(code) {\n // There is an underscore in the last segment of the domain\n if (code === 95) {\n underscoreInLastSegment = true;\n }\n // Otherwise, it’s a `.`: save the last segment underscore in the\n // penultimate segment slot.\n else {\n underscoreInLastLastSegment = underscoreInLastSegment;\n underscoreInLastSegment = undefined;\n }\n effects.consume(code);\n return domainInside;\n }\n\n /**\n * After domain.\n *\n * ```markdown\n * > | https://example.com/a\n * ^\n * ```\n *\n * @type {State} */\n function domainAfter(code) {\n // Note: that’s GH says a dot is needed, but it’s not true:\n // <https://github.com/github/cmark-gfm/issues/279>\n if (underscoreInLastLastSegment || underscoreInLastSegment || !seen) {\n return nok(code);\n }\n return ok(code);\n }\n}\n\n/**\n * Path.\n *\n * ```markdown\n * > | a https://example.org/stuff b\n * ^^^^^^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizePath(effects, ok) {\n let sizeOpen = 0;\n let sizeClose = 0;\n return pathInside;\n\n /**\n * In path.\n *\n * ```markdown\n * > | https://example.com/a\n * ^^\n * ```\n *\n * @type {State}\n */\n function pathInside(code) {\n if (code === 40) {\n sizeOpen++;\n effects.consume(code);\n return pathInside;\n }\n\n // To do: `markdown-rs` also needs this.\n // If this is a paren, and there are less closings than openings,\n // we don’t check for a trail.\n if (code === 41 && sizeClose < sizeOpen) {\n return pathAtPunctuation(code);\n }\n\n // Check whether this trailing punctuation marker is optionally\n // followed by more trailing markers, and then followed\n // by an end.\n if (code === 33 || code === 34 || code === 38 || code === 39 || code === 41 || code === 42 || code === 44 || code === 46 || code === 58 || code === 59 || code === 60 || code === 63 || code === 93 || code === 95 || code === 126) {\n return effects.check(trail, ok, pathAtPunctuation)(code);\n }\n if (code === null || markdownLineEndingOrSpace(code) || unicodeWhitespace(code)) {\n return ok(code);\n }\n effects.consume(code);\n return pathInside;\n }\n\n /**\n * In path, at potential trailing punctuation, that was not trailing.\n *\n * ```markdown\n * > | https://example.com/a\"b\n * ^\n * ```\n *\n * @type {State}\n */\n function pathAtPunctuation(code) {\n // Count closing parens.\n if (code === 41) {\n sizeClose++;\n }\n effects.consume(code);\n return pathInside;\n }\n}\n\n/**\n * Trail.\n *\n * This calls `ok` if this *is* the trail, followed by an end, which means\n * the entire trail is not part of the link.\n * It calls `nok` if this *is* part of the link.\n *\n * ```markdown\n * > | https://example.com\").\n * ^^^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeTrail(effects, ok, nok) {\n return trail;\n\n /**\n * In trail of domain or path.\n *\n * ```markdown\n * > | https://example.com\").\n * ^\n * ```\n *\n * @type {State}\n */\n function trail(code) {\n // Regular trailing punctuation.\n if (code === 33 || code === 34 || code === 39 || code === 41 || code === 42 || code === 44 || code === 46 || code === 58 || code === 59 || code === 63 || code === 95 || code === 126) {\n effects.consume(code);\n return trail;\n }\n\n // `&` followed by one or more alphabeticals and then a `;`, is\n // as a whole considered as trailing punctuation.\n // In all other cases, it is considered as continuation of the URL.\n if (code === 38) {\n effects.consume(code);\n return trailCharacterReferenceStart;\n }\n\n // Needed because we allow literals after `[`, as we fix:\n // <https://github.com/github/cmark-gfm/issues/278>.\n // Check that it is not followed by `(` or `[`.\n if (code === 93) {\n effects.consume(code);\n return trailBracketAfter;\n }\n if (\n // `<` is an end.\n code === 60 ||\n // So is whitespace.\n code === null || markdownLineEndingOrSpace(code) || unicodeWhitespace(code)) {\n return ok(code);\n }\n return nok(code);\n }\n\n /**\n * In trail, after `]`.\n *\n * > 👉 **Note**: this deviates from `cmark-gfm` to fix a bug.\n * > See end of <https://github.com/github/cmark-gfm/issues/278> for more.\n *\n * ```markdown\n * > | https://example.com](\n * ^\n * ```\n *\n * @type {State}\n */\n function trailBracketAfter(code) {\n // Whitespace or something that could start a resource or reference is the end.\n // Switch back to trail otherwise.\n if (code === null || code === 40 || code === 91 || markdownLineEndingOrSpace(code) || unicodeWhitespace(code)) {\n return ok(code);\n }\n return trail(code);\n }\n\n /**\n * In character-reference like trail, after `&`.\n *\n * ```markdown\n * > | https://example.com&).\n * ^\n * ```\n *\n * @type {State}\n */\n function trailCharacterReferenceStart(code) {\n // When non-alpha, it’s not a trail.\n return asciiAlpha(code) ? trailCharacterReferenceInside(code) : nok(code);\n }\n\n /**\n * In character-reference like trail.\n *\n * ```markdown\n * > | https://example.com&).\n * ^\n * ```\n *\n * @type {State}\n */\n function trailCharacterReferenceInside(code) {\n // Switch back to trail if this is well-formed.\n if (code === 59) {\n effects.consume(code);\n return trail;\n }\n if (asciiAlpha(code)) {\n effects.consume(code);\n return trailCharacterReferenceInside;\n }\n\n // It’s not a trail.\n return nok(code);\n }\n}\n\n/**\n * Dot in email domain trail.\n *\n * This calls `ok` if this *is* the trail, followed by an end, which means\n * the trail is not part of the link.\n * It calls `nok` if this *is* part of the link.\n *\n * ```markdown\n * > | contact@example.org.\n * ^\n * ```\n *\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeEmailDomainDotTrail(effects, ok, nok) {\n return start;\n\n /**\n * Dot.\n *\n * ```markdown\n * > | contact@example.org.\n * ^ ^\n * ```\n *\n * @type {State}\n */\n function start(code) {\n // Must be dot.\n effects.consume(code);\n return after;\n }\n\n /**\n * After dot.\n *\n * ```markdown\n * > | contact@example.org.\n * ^ ^\n * ```\n *\n * @type {State}\n */\n function after(code) {\n // Not a trail if alphanumeric.\n return asciiAlphanumeric(code) ? nok(code) : ok(code);\n }\n}\n\n/**\n * See:\n * <https://github.com/github/cmark-gfm/blob/ef1cfcb/extensions/autolink.c#L156>.\n *\n * @type {Previous}\n */\nfunction previousWww(code) {\n return code === null || code === 40 || code === 42 || code === 95 || code === 91 || code === 93 || code === 126 || markdownLineEndingOrSpace(code);\n}\n\n/**\n * See:\n * <https://github.com/github/cmark-gfm/blob/ef1cfcb/extensions/autolink.c#L214>.\n *\n * @type {Previous}\n */\nfunction previousProtocol(code) {\n return !asciiAlpha(code);\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Previous}\n */\nfunction previousEmail(code) {\n // Do not allow a slash “inside” atext.\n // The reference code is a bit weird, but that’s what it results in.\n // Source: <https://github.com/github/cmark-gfm/blob/ef1cfcb/extensions/autolink.c#L307>.\n // Other than slash, every preceding character is allowed.\n return !(code === 47 || gfmAtext(code));\n}\n\n/**\n * @param {Code} code\n * @returns {boolean}\n */\nfunction gfmAtext(code) {\n return code === 43 || code === 45 || code === 46 || code === 95 || asciiAlphanumeric(code);\n}\n\n/**\n * @param {Array<Event>} events\n * @returns {boolean}\n */\nfunction previousUnbalanced(events) {\n let index = events.length;\n let result = false;\n while (index--) {\n const token = events[index][1];\n if ((token.type === 'labelLink' || token.type === 'labelImage') && !token._balanced) {\n result = true;\n break;\n }\n\n // If we’ve seen this token, and it was marked as not having any unbalanced\n // bracket before it, we can exit.\n if (token._gfmAutolinkLiteralWalkedInto) {\n result = false;\n break;\n }\n }\n if (events.length > 0 && !result) {\n // Mark the last token as “walked into” w/o finding\n // anything.\n events[events.length - 1][1]._gfmAutolinkLiteralWalkedInto = true;\n }\n return result;\n}","/**\n * @import {Event, Resolver, TokenizeContext} from 'micromark-util-types'\n */\n\n/**\n * Call all `resolveAll`s.\n *\n * @param {ReadonlyArray<{resolveAll?: Resolver | undefined}>} constructs\n * List of constructs, optionally with `resolveAll`s.\n * @param {Array<Event>} events\n * List of events.\n * @param {TokenizeContext} context\n * Context used by `tokenize`.\n * @returns {Array<Event>}\n * Changed events.\n */\nexport function resolveAll(constructs, events, context) {\n /** @type {Array<Resolver>} */\n const called = []\n let index = -1\n\n while (++index < constructs.length) {\n const resolve = constructs[index].resolveAll\n\n if (resolve && !called.includes(resolve)) {\n events = resolve(events, context)\n called.push(resolve)\n }\n }\n\n return events\n}\n","/**\n * @import {Effects, State, TokenType} from 'micromark-util-types'\n */\n\nimport { markdownSpace } from 'micromark-util-character';\n\n// To do: implement `spaceOrTab`, `spaceOrTabMinMax`, `spaceOrTabWithOptions`.\n\n/**\n * Parse spaces and tabs.\n *\n * There is no `nok` parameter:\n *\n * * spaces in markdown are often optional, in which case this factory can be\n * used and `ok` will be switched to whether spaces were found or not\n * * one line ending or space can be detected with `markdownSpace(code)` right\n * before using `factorySpace`\n *\n * ###### Examples\n *\n * Where `␉` represents a tab (plus how much it expands) and `␠` represents a\n * single space.\n *\n * ```markdown\n * ␉\n * ␠␠␠␠\n * ␉␠\n * ```\n *\n * @param {Effects} effects\n * Context.\n * @param {State} ok\n * State switched to when successful.\n * @param {TokenType} type\n * Type (`' \\t'`).\n * @param {number | undefined} [max=Infinity]\n * Max (exclusive).\n * @returns {State}\n * Start state.\n */\nexport function factorySpace(effects, ok, type, max) {\n const limit = max ? max - 1 : Number.POSITIVE_INFINITY;\n let size = 0;\n return start;\n\n /** @type {State} */\n function start(code) {\n if (markdownSpace(code)) {\n effects.enter(type);\n return prefix(code);\n }\n return ok(code);\n }\n\n /** @type {State} */\n function prefix(code) {\n if (markdownSpace(code) && size++ < limit) {\n effects.consume(code);\n return prefix;\n }\n effects.exit(type);\n return ok(code);\n }\n}","/**\n * @import {\n * Construct,\n * State,\n * TokenizeContext,\n * Tokenizer\n * } from 'micromark-util-types'\n */\n\nimport { factorySpace } from 'micromark-factory-space';\nimport { markdownLineEnding, markdownSpace } from 'micromark-util-character';\n/** @type {Construct} */\nexport const blankLine = {\n partial: true,\n tokenize: tokenizeBlankLine\n};\n\n/**\n * @this {TokenizeContext}\n * Context.\n * @type {Tokenizer}\n */\nfunction tokenizeBlankLine(effects, ok, nok) {\n return start;\n\n /**\n * Start of blank line.\n *\n * > 👉 **Note**: `␠` represents a space character.\n *\n * ```markdown\n * > | ␠␠␊\n * ^\n * > | ␊\n * ^\n * ```\n *\n * @type {State}\n */\n function start(code) {\n return markdownSpace(code) ? factorySpace(effects, after, \"linePrefix\")(code) : after(code);\n }\n\n /**\n * At eof/eol, after optional whitespace.\n *\n * > 👉 **Note**: `␠` represents a space character.\n *\n * ```markdown\n * > | ␠␠␊\n * ^\n * > | ␊\n * ^\n * ```\n *\n * @type {State}\n */\n function after(code) {\n return code === null || markdownLineEnding(code) ? ok(code) : nok(code);\n }\n}","/**\n * @import {Event, Exiter, Extension, Resolver, State, Token, TokenizeContext, Tokenizer} from 'micromark-util-types'\n */\n\nimport { blankLine } from 'micromark-core-commonmark';\nimport { factorySpace } from 'micromark-factory-space';\nimport { markdownLineEndingOrSpace } from 'micromark-util-character';\nimport { normalizeIdentifier } from 'micromark-util-normalize-identifier';\nconst indent = {\n tokenize: tokenizeIndent,\n partial: true\n};\n\n// To do: micromark should support a `_hiddenGfmFootnoteSupport`, which only\n// affects label start (image).\n// That will let us drop `tokenizePotentialGfmFootnote*`.\n// It currently has a `_hiddenFootnoteSupport`, which affects that and more.\n// That can be removed when `micromark-extension-footnote` is archived.\n\n/**\n * Create an extension for `micromark` to enable GFM footnote syntax.\n *\n * @returns {Extension}\n * Extension for `micromark` that can be passed in `extensions` to\n * enable GFM footnote syntax.\n */\nexport function gfmFootnote() {\n /** @type {Extension} */\n return {\n document: {\n [91]: {\n name: 'gfmFootnoteDefinition',\n tokenize: tokenizeDefinitionStart,\n continuation: {\n tokenize: tokenizeDefinitionContinuation\n },\n exit: gfmFootnoteDefinitionEnd\n }\n },\n text: {\n [91]: {\n name: 'gfmFootnoteCall',\n tokenize: tokenizeGfmFootnoteCall\n },\n [93]: {\n name: 'gfmPotentialFootnoteCall',\n add: 'after',\n tokenize: tokenizePotentialGfmFootnoteCall,\n resolveTo: resolveToPotentialGfmFootnoteCall\n }\n }\n };\n}\n\n// To do: remove after micromark update.\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizePotentialGfmFootnoteCall(effects, ok, nok) {\n const self = this;\n let index = self.events.length;\n const defined = self.parser.gfmFootnotes || (self.parser.gfmFootnotes = []);\n /** @type {Token} */\n let labelStart;\n\n // Find an opening.\n while (index--) {\n const token = self.events[index][1];\n if (token.type === \"labelImage\") {\n labelStart = token;\n break;\n }\n\n // Exit if we’ve walked far enough.\n if (token.type === 'gfmFootnoteCall' || token.type === \"labelLink\" || token.type === \"label\" || token.type === \"image\" || token.type === \"link\") {\n break;\n }\n }\n return start;\n\n /**\n * @type {State}\n */\n function start(code) {\n if (!labelStart || !labelStart._balanced) {\n return nok(code);\n }\n const id = normalizeIdentifier(self.sliceSerialize({\n start: labelStart.end,\n end: self.now()\n }));\n if (id.codePointAt(0) !== 94 || !defined.includes(id.slice(1))) {\n return nok(code);\n }\n effects.enter('gfmFootnoteCallLabelMarker');\n effects.consume(code);\n effects.exit('gfmFootnoteCallLabelMarker');\n return ok(code);\n }\n}\n\n// To do: remove after micromark update.\n/** @type {Resolver} */\nfunction resolveToPotentialGfmFootnoteCall(events, context) {\n let index = events.length;\n /** @type {Token | undefined} */\n let labelStart;\n\n // Find an opening.\n while (index--) {\n if (events[index][1].type === \"labelImage\" && events[index][0] === 'enter') {\n labelStart = events[index][1];\n break;\n }\n }\n // Change the `labelImageMarker` to a `data`.\n events[index + 1][1].type = \"data\";\n events[index + 3][1].type = 'gfmFootnoteCallLabelMarker';\n\n // The whole (without `!`):\n /** @type {Token} */\n const call = {\n type: 'gfmFootnoteCall',\n start: Object.assign({}, events[index + 3][1].start),\n end: Object.assign({}, events[events.length - 1][1].end)\n };\n // The `^` marker\n /** @type {Token} */\n const marker = {\n type: 'gfmFootnoteCallMarker',\n start: Object.assign({}, events[index + 3][1].end),\n end: Object.assign({}, events[index + 3][1].end)\n };\n // Increment the end 1 character.\n marker.end.column++;\n marker.end.offset++;\n marker.end._bufferIndex++;\n /** @type {Token} */\n const string = {\n type: 'gfmFootnoteCallString',\n start: Object.assign({}, marker.end),\n end: Object.assign({}, events[events.length - 1][1].start)\n };\n /** @type {Token} */\n const chunk = {\n type: \"chunkString\",\n contentType: 'string',\n start: Object.assign({}, string.start),\n end: Object.assign({}, string.end)\n };\n\n /** @type {Array<Event>} */\n const replacement = [\n // Take the `labelImageMarker` (now `data`, the `!`)\n events[index + 1], events[index + 2], ['enter', call, context],\n // The `[`\n events[index + 3], events[index + 4],\n // The `^`.\n ['enter', marker, context], ['exit', marker, context],\n // Everything in between.\n ['enter', string, context], ['enter', chunk, context], ['exit', chunk, context], ['exit', string, context],\n // The ending (`]`, properly parsed and labelled).\n events[events.length - 2], events[events.length - 1], ['exit', call, context]];\n events.splice(index, events.length - index + 1, ...replacement);\n return events;\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeGfmFootnoteCall(effects, ok, nok) {\n const self = this;\n const defined = self.parser.gfmFootnotes || (self.parser.gfmFootnotes = []);\n let size = 0;\n /** @type {boolean} */\n let data;\n\n // Note: the implementation of `markdown-rs` is different, because it houses\n // core *and* extensions in one project.\n // Therefore, it can include footnote logic inside `label-end`.\n // We can’t do that, but luckily, we can parse footnotes in a simpler way than\n // needed for labels.\n return start;\n\n /**\n * Start of footnote label.\n *\n * ```markdown\n * > | a [^b] c\n * ^\n * ```\n *\n * @type {State}\n */\n function start(code) {\n effects.enter('gfmFootnoteCall');\n effects.enter('gfmFootnoteCallLabelMarker');\n effects.consume(code);\n effects.exit('gfmFootnoteCallLabelMarker');\n return callStart;\n }\n\n /**\n * After `[`, at `^`.\n *\n * ```markdown\n * > | a [^b] c\n * ^\n * ```\n *\n * @type {State}\n */\n function callStart(code) {\n if (code !== 94) return nok(code);\n effects.enter('gfmFootnoteCallMarker');\n effects.consume(code);\n effects.exit('gfmFootnoteCallMarker');\n effects.enter('gfmFootnoteCallString');\n effects.enter('chunkString').contentType = 'string';\n return callData;\n }\n\n /**\n * In label.\n *\n * ```markdown\n * > | a [^b] c\n * ^\n * ```\n *\n * @type {State}\n */\n function callData(code) {\n if (\n // Too long.\n size > 999 ||\n // Closing brace with nothing.\n code === 93 && !data ||\n // Space or tab is not supported by GFM for some reason.\n // `\\n` and `[` not being supported makes sense.\n code === null || code === 91 || markdownLineEndingOrSpace(code)) {\n return nok(code);\n }\n if (code === 93) {\n effects.exit('chunkString');\n const token = effects.exit('gfmFootnoteCallString');\n if (!defined.includes(normalizeIdentifier(self.sliceSerialize(token)))) {\n return nok(code);\n }\n effects.enter('gfmFootnoteCallLabelMarker');\n effects.consume(code);\n effects.exit('gfmFootnoteCallLabelMarker');\n effects.exit('gfmFootnoteCall');\n return ok;\n }\n if (!markdownLineEndingOrSpace(code)) {\n data = true;\n }\n size++;\n effects.consume(code);\n return code === 92 ? callEscape : callData;\n }\n\n /**\n * On character after escape.\n *\n * ```markdown\n * > | a [^b\\c] d\n * ^\n * ```\n *\n * @type {State}\n */\n function callEscape(code) {\n if (code === 91 || code === 92 || code === 93) {\n effects.consume(code);\n size++;\n return callData;\n }\n return callData(code);\n }\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeDefinitionStart(effects, ok, nok) {\n const self = this;\n const defined = self.parser.gfmFootnotes || (self.parser.gfmFootnotes = []);\n /** @type {string} */\n let identifier;\n let size = 0;\n /** @type {boolean | undefined} */\n let data;\n return start;\n\n /**\n * Start of GFM footnote definition.\n *\n * ```markdown\n * > | [^a]: b\n * ^\n * ```\n *\n * @type {State}\n */\n function start(code) {\n effects.enter('gfmFootnoteDefinition')._container = true;\n effects.enter('gfmFootnoteDefinitionLabel');\n effects.enter('gfmFootnoteDefinitionLabelMarker');\n effects.consume(code);\n effects.exit('gfmFootnoteDefinitionLabelMarker');\n return labelAtMarker;\n }\n\n /**\n * In label, at caret.\n *\n * ```markdown\n * > | [^a]: b\n * ^\n * ```\n *\n * @type {State}\n */\n function labelAtMarker(code) {\n if (code === 94) {\n effects.enter('gfmFootnoteDefinitionMarker');\n effects.consume(code);\n effects.exit('gfmFootnoteDefinitionMarker');\n effects.enter('gfmFootnoteDefinitionLabelString');\n effects.enter('chunkString').contentType = 'string';\n return labelInside;\n }\n return nok(code);\n }\n\n /**\n * In label.\n *\n * > 👉 **Note**: `cmark-gfm` prevents whitespace from occurring in footnote\n * > definition labels.\n *\n * ```markdown\n * > | [^a]: b\n * ^\n * ```\n *\n * @type {State}\n */\n function labelInside(code) {\n if (\n // Too long.\n size > 999 ||\n // Closing brace with nothing.\n code === 93 && !data ||\n // Space or tab is not supported by GFM for some reason.\n // `\\n` and `[` not being supported makes sense.\n code === null || code === 91 || markdownLineEndingOrSpace(code)) {\n return nok(code);\n }\n if (code === 93) {\n effects.exit('chunkString');\n const token = effects.exit('gfmFootnoteDefinitionLabelString');\n identifier = normalizeIdentifier(self.sliceSerialize(token));\n effects.enter('gfmFootnoteDefinitionLabelMarker');\n effects.consume(code);\n effects.exit('gfmFootnoteDefinitionLabelMarker');\n effects.exit('gfmFootnoteDefinitionLabel');\n return labelAfter;\n }\n if (!markdownLineEndingOrSpace(code)) {\n data = true;\n }\n size++;\n effects.consume(code);\n return code === 92 ? labelEscape : labelInside;\n }\n\n /**\n * After `\\`, at a special character.\n *\n * > 👉 **Note**: `cmark-gfm` currently does not support escaped brackets:\n * > <https://github.com/github/cmark-gfm/issues/240>\n *\n * ```markdown\n * > | [^a\\*b]: c\n * ^\n * ```\n *\n * @type {State}\n */\n function labelEscape(code) {\n if (code === 91 || code === 92 || code === 93) {\n effects.consume(code);\n size++;\n return labelInside;\n }\n return labelInside(code);\n }\n\n /**\n * After definition label.\n *\n * ```markdown\n * > | [^a]: b\n * ^\n * ```\n *\n * @type {State}\n */\n function labelAfter(code) {\n if (code === 58) {\n effects.enter('definitionMarker');\n effects.consume(code);\n effects.exit('definitionMarker');\n if (!defined.includes(identifier)) {\n defined.push(identifier);\n }\n\n // Any whitespace after the marker is eaten, forming indented code\n // is not possible.\n // No space is also fine, just like a block quote marker.\n return factorySpace(effects, whitespaceAfter, 'gfmFootnoteDefinitionWhitespace');\n }\n return nok(code);\n }\n\n /**\n * After definition prefix.\n *\n * ```markdown\n * > | [^a]: b\n * ^\n * ```\n *\n * @type {State}\n */\n function whitespaceAfter(code) {\n // `markdown-rs` has a wrapping token for the prefix that is closed here.\n return ok(code);\n }\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeDefinitionContinuation(effects, ok, nok) {\n /// Start of footnote definition continuation.\n ///\n /// ```markdown\n /// | [^a]: b\n /// > | c\n /// ^\n /// ```\n //\n // Either a blank line, which is okay, or an indented thing.\n return effects.check(blankLine, ok, effects.attempt(indent, ok, nok));\n}\n\n/** @type {Exiter} */\nfunction gfmFootnoteDefinitionEnd(effects) {\n effects.exit('gfmFootnoteDefinition');\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeIndent(effects, ok, nok) {\n const self = this;\n return factorySpace(effects, afterPrefix, 'gfmFootnoteDefinitionIndent', 4 + 1);\n\n /**\n * @type {State}\n */\n function afterPrefix(code) {\n const tail = self.events[self.events.length - 1];\n return tail && tail[1].type === 'gfmFootnoteDefinitionIndent' && tail[2].sliceSerialize(tail[1], true).length === 4 ? ok(code) : nok(code);\n }\n}","/**\n * @import {Options} from 'micromark-extension-gfm-strikethrough'\n * @import {Event, Extension, Resolver, State, Token, TokenizeContext, Tokenizer} from 'micromark-util-types'\n */\n\nimport { splice } from 'micromark-util-chunked';\nimport { classifyCharacter } from 'micromark-util-classify-character';\nimport { resolveAll } from 'micromark-util-resolve-all';\n/**\n * Create an extension for `micromark` to enable GFM strikethrough syntax.\n *\n * @param {Options | null | undefined} [options={}]\n * Configuration.\n * @returns {Extension}\n * Extension for `micromark` that can be passed in `extensions`, to\n * enable GFM strikethrough syntax.\n */\nexport function gfmStrikethrough(options) {\n const options_ = options || {};\n let single = options_.singleTilde;\n const tokenizer = {\n name: 'strikethrough',\n tokenize: tokenizeStrikethrough,\n resolveAll: resolveAllStrikethrough\n };\n if (single === null || single === undefined) {\n single = true;\n }\n return {\n text: {\n [126]: tokenizer\n },\n insideSpan: {\n null: [tokenizer]\n },\n attentionMarkers: {\n null: [126]\n }\n };\n\n /**\n * Take events and resolve strikethrough.\n *\n * @type {Resolver}\n */\n function resolveAllStrikethrough(events, context) {\n let index = -1;\n\n // Walk through all events.\n while (++index < events.length) {\n // Find a token that can close.\n if (events[index][0] === 'enter' && events[index][1].type === 'strikethroughSequenceTemporary' && events[index][1]._close) {\n let open = index;\n\n // Now walk back to find an opener.\n while (open--) {\n // Find a token that can open the closer.\n if (events[open][0] === 'exit' && events[open][1].type === 'strikethroughSequenceTemporary' && events[open][1]._open &&\n // If the sizes are the same:\n events[index][1].end.offset - events[index][1].start.offset === events[open][1].end.offset - events[open][1].start.offset) {\n events[index][1].type = 'strikethroughSequence';\n events[open][1].type = 'strikethroughSequence';\n\n /** @type {Token} */\n const strikethrough = {\n type: 'strikethrough',\n start: Object.assign({}, events[open][1].start),\n end: Object.assign({}, events[index][1].end)\n };\n\n /** @type {Token} */\n const text = {\n type: 'strikethroughText',\n start: Object.assign({}, events[open][1].end),\n end: Object.assign({}, events[index][1].start)\n };\n\n // Opening.\n /** @type {Array<Event>} */\n const nextEvents = [['enter', strikethrough, context], ['enter', events[open][1], context], ['exit', events[open][1], context], ['enter', text, context]];\n const insideSpan = context.parser.constructs.insideSpan.null;\n if (insideSpan) {\n // Between.\n splice(nextEvents, nextEvents.length, 0, resolveAll(insideSpan, events.slice(open + 1, index), context));\n }\n\n // Closing.\n splice(nextEvents, nextEvents.length, 0, [['exit', text, context], ['enter', events[index][1], context], ['exit', events[index][1], context], ['exit', strikethrough, context]]);\n splice(events, open - 1, index - open + 3, nextEvents);\n index = open + nextEvents.length - 2;\n break;\n }\n }\n }\n }\n index = -1;\n while (++index < events.length) {\n if (events[index][1].type === 'strikethroughSequenceTemporary') {\n events[index][1].type = \"data\";\n }\n }\n return events;\n }\n\n /**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\n function tokenizeStrikethrough(effects, ok, nok) {\n const previous = this.previous;\n const events = this.events;\n let size = 0;\n return start;\n\n /** @type {State} */\n function start(code) {\n if (previous === 126 && events[events.length - 1][1].type !== \"characterEscape\") {\n return nok(code);\n }\n effects.enter('strikethroughSequenceTemporary');\n return more(code);\n }\n\n /** @type {State} */\n function more(code) {\n const before = classifyCharacter(previous);\n if (code === 126) {\n // If this is the third marker, exit.\n if (size > 1) return nok(code);\n effects.consume(code);\n size++;\n return more;\n }\n if (size < 2 && !single) return nok(code);\n const token = effects.exit('strikethroughSequenceTemporary');\n const after = classifyCharacter(code);\n token._open = !after || after === 2 && Boolean(before);\n token._close = !before || before === 2 && Boolean(after);\n return ok(code);\n }\n }\n}","/**\n * @import {Event} from 'micromark-util-types'\n */\n\n// Port of `edit_map.rs` from `markdown-rs`.\n// This should move to `markdown-js` later.\n\n// Deal with several changes in events, batching them together.\n//\n// Preferably, changes should be kept to a minimum.\n// Sometimes, it’s needed to change the list of events, because parsing can be\n// messy, and it helps to expose a cleaner interface of events to the compiler\n// and other users.\n// It can also help to merge many adjacent similar events.\n// And, in other cases, it’s needed to parse subcontent: pass some events\n// through another tokenizer and inject the result.\n\n/**\n * @typedef {[number, number, Array<Event>]} Change\n * @typedef {[number, number, number]} Jump\n */\n\n/**\n * Tracks a bunch of edits.\n */\nexport class EditMap {\n /**\n * Create a new edit map.\n */\n constructor() {\n /**\n * Record of changes.\n *\n * @type {Array<Change>}\n */\n this.map = [];\n }\n\n /**\n * Create an edit: a remove and/or add at a certain place.\n *\n * @param {number} index\n * @param {number} remove\n * @param {Array<Event>} add\n * @returns {undefined}\n */\n add(index, remove, add) {\n addImplementation(this, index, remove, add);\n }\n\n // To do: add this when moving to `micromark`.\n // /**\n // * Create an edit: but insert `add` before existing additions.\n // *\n // * @param {number} index\n // * @param {number} remove\n // * @param {Array<Event>} add\n // * @returns {undefined}\n // */\n // addBefore(index, remove, add) {\n // addImplementation(this, index, remove, add, true)\n // }\n\n /**\n * Done, change the events.\n *\n * @param {Array<Event>} events\n * @returns {undefined}\n */\n consume(events) {\n this.map.sort(function (a, b) {\n return a[0] - b[0];\n });\n\n /* c8 ignore next 3 -- `resolve` is never called without tables, so without edits. */\n if (this.map.length === 0) {\n return;\n }\n\n // To do: if links are added in events, like they are in `markdown-rs`,\n // this is needed.\n // // Calculate jumps: where items in the current list move to.\n // /** @type {Array<Jump>} */\n // const jumps = []\n // let index = 0\n // let addAcc = 0\n // let removeAcc = 0\n // while (index < this.map.length) {\n // const [at, remove, add] = this.map[index]\n // removeAcc += remove\n // addAcc += add.length\n // jumps.push([at, removeAcc, addAcc])\n // index += 1\n // }\n //\n // . shiftLinks(events, jumps)\n\n let index = this.map.length;\n /** @type {Array<Array<Event>>} */\n const vecs = [];\n while (index > 0) {\n index -= 1;\n vecs.push(events.slice(this.map[index][0] + this.map[index][1]), this.map[index][2]);\n\n // Truncate rest.\n events.length = this.map[index][0];\n }\n vecs.push(events.slice());\n events.length = 0;\n let slice = vecs.pop();\n while (slice) {\n for (const element of slice) {\n events.push(element);\n }\n slice = vecs.pop();\n }\n\n // Truncate everything.\n this.map.length = 0;\n }\n}\n\n/**\n * Create an edit.\n *\n * @param {EditMap} editMap\n * @param {number} at\n * @param {number} remove\n * @param {Array<Event>} add\n * @returns {undefined}\n */\nfunction addImplementation(editMap, at, remove, add) {\n let index = 0;\n\n /* c8 ignore next 3 -- `resolve` is never called without tables, so without edits. */\n if (remove === 0 && add.length === 0) {\n return;\n }\n while (index < editMap.map.length) {\n if (editMap.map[index][0] === at) {\n editMap.map[index][1] += remove;\n\n // To do: before not used by tables, use when moving to micromark.\n // if (before) {\n // add.push(...editMap.map[index][2])\n // editMap.map[index][2] = add\n // } else {\n editMap.map[index][2].push(...add);\n // }\n\n return;\n }\n index += 1;\n }\n editMap.map.push([at, remove, add]);\n}\n\n// /**\n// * Shift `previous` and `next` links according to `jumps`.\n// *\n// * This fixes links in case there are events removed or added between them.\n// *\n// * @param {Array<Event>} events\n// * @param {Array<Jump>} jumps\n// */\n// function shiftLinks(events, jumps) {\n// let jumpIndex = 0\n// let index = 0\n// let add = 0\n// let rm = 0\n\n// while (index < events.length) {\n// const rmCurr = rm\n\n// while (jumpIndex < jumps.length && jumps[jumpIndex][0] <= index) {\n// add = jumps[jumpIndex][2]\n// rm = jumps[jumpIndex][1]\n// jumpIndex += 1\n// }\n\n// // Ignore items that will be removed.\n// if (rm > rmCurr) {\n// index += rm - rmCurr\n// } else {\n// // ?\n// // if let Some(link) = &events[index].link {\n// // if let Some(next) = link.next {\n// // events[next].link.as_mut().unwrap().previous = Some(index + add - rm);\n// // while jumpIndex < jumps.len() && jumps[jumpIndex].0 <= next {\n// // add = jumps[jumpIndex].2;\n// // rm = jumps[jumpIndex].1;\n// // jumpIndex += 1;\n// // }\n// // events[index].link.as_mut().unwrap().next = Some(next + add - rm);\n// // index = next;\n// // continue;\n// // }\n// // }\n// index += 1\n// }\n// }\n// }","/**\n * @import {Event} from 'micromark-util-types'\n */\n\n/**\n * @typedef {'center' | 'left' | 'none' | 'right'} Align\n */\n\n/**\n * Figure out the alignment of a GFM table.\n *\n * @param {Readonly<Array<Event>>} events\n * List of events.\n * @param {number} index\n * Table enter event.\n * @returns {Array<Align>}\n * List of aligns.\n */\nexport function gfmTableAlign(events, index) {\n let inDelimiterRow = false;\n /** @type {Array<Align>} */\n const align = [];\n while (index < events.length) {\n const event = events[index];\n if (inDelimiterRow) {\n if (event[0] === 'enter') {\n // Start of alignment value: set a new column.\n // To do: `markdown-rs` uses `tableDelimiterCellValue`.\n if (event[1].type === 'tableContent') {\n align.push(events[index + 1][1].type === 'tableDelimiterMarker' ? 'left' : 'none');\n }\n }\n // Exits:\n // End of alignment value: change the column.\n // To do: `markdown-rs` uses `tableDelimiterCellValue`.\n else if (event[1].type === 'tableContent') {\n if (events[index - 1][1].type === 'tableDelimiterMarker') {\n const alignIndex = align.length - 1;\n align[alignIndex] = align[alignIndex] === 'left' ? 'center' : 'right';\n }\n }\n // Done!\n else if (event[1].type === 'tableDelimiterRow') {\n break;\n }\n } else if (event[0] === 'enter' && event[1].type === 'tableDelimiterRow') {\n inDelimiterRow = true;\n }\n index += 1;\n }\n return align;\n}","/**\n * @import {Event, Extension, Point, Resolver, State, Token, TokenizeContext, Tokenizer} from 'micromark-util-types'\n */\n\n/**\n * @typedef {[number, number, number, number]} Range\n * Cell info.\n *\n * @typedef {0 | 1 | 2 | 3} RowKind\n * Where we are: `1` for head row, `2` for delimiter row, `3` for body row.\n */\n\nimport { factorySpace } from 'micromark-factory-space';\nimport { markdownLineEnding, markdownLineEndingOrSpace, markdownSpace } from 'micromark-util-character';\nimport { EditMap } from './edit-map.js';\nimport { gfmTableAlign } from './infer.js';\n\n/**\n * Create an HTML extension for `micromark` to support GitHub tables syntax.\n *\n * @returns {Extension}\n * Extension for `micromark` that can be passed in `extensions` to enable GFM\n * table syntax.\n */\nexport function gfmTable() {\n return {\n flow: {\n null: {\n name: 'table',\n tokenize: tokenizeTable,\n resolveAll: resolveTable\n }\n }\n };\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeTable(effects, ok, nok) {\n const self = this;\n let size = 0;\n let sizeB = 0;\n /** @type {boolean | undefined} */\n let seen;\n return start;\n\n /**\n * Start of a GFM table.\n *\n * If there is a valid table row or table head before, then we try to parse\n * another row.\n * Otherwise, we try to parse a head.\n *\n * ```markdown\n * > | | a |\n * ^\n * | | - |\n * > | | b |\n * ^\n * ```\n * @type {State}\n */\n function start(code) {\n let index = self.events.length - 1;\n while (index > -1) {\n const type = self.events[index][1].type;\n if (type === \"lineEnding\" ||\n // Note: markdown-rs uses `whitespace` instead of `linePrefix`\n type === \"linePrefix\") index--;else break;\n }\n const tail = index > -1 ? self.events[index][1].type : null;\n const next = tail === 'tableHead' || tail === 'tableRow' ? bodyRowStart : headRowBefore;\n\n // Don’t allow lazy body rows.\n if (next === bodyRowStart && self.parser.lazy[self.now().line]) {\n return nok(code);\n }\n return next(code);\n }\n\n /**\n * Before table head row.\n *\n * ```markdown\n * > | | a |\n * ^\n * | | - |\n * | | b |\n * ```\n *\n * @type {State}\n */\n function headRowBefore(code) {\n effects.enter('tableHead');\n effects.enter('tableRow');\n return headRowStart(code);\n }\n\n /**\n * Before table head row, after whitespace.\n *\n * ```markdown\n * > | | a |\n * ^\n * | | - |\n * | | b |\n * ```\n *\n * @type {State}\n */\n function headRowStart(code) {\n if (code === 124) {\n return headRowBreak(code);\n }\n\n // To do: micromark-js should let us parse our own whitespace in extensions,\n // like `markdown-rs`:\n //\n // ```js\n // // 4+ spaces.\n // if (markdownSpace(code)) {\n // return nok(code)\n // }\n // ```\n\n seen = true;\n // Count the first character, that isn’t a pipe, double.\n sizeB += 1;\n return headRowBreak(code);\n }\n\n /**\n * At break in table head row.\n *\n * ```markdown\n * > | | a |\n * ^\n * ^\n * ^\n * | | - |\n * | | b |\n * ```\n *\n * @type {State}\n */\n function headRowBreak(code) {\n if (code === null) {\n // Note: in `markdown-rs`, we need to reset, in `micromark-js` we don‘t.\n return nok(code);\n }\n if (markdownLineEnding(code)) {\n // If anything other than one pipe (ignoring whitespace) was used, it’s fine.\n if (sizeB > 1) {\n sizeB = 0;\n // To do: check if this works.\n // Feel free to interrupt:\n self.interrupt = true;\n effects.exit('tableRow');\n effects.enter(\"lineEnding\");\n effects.consume(code);\n effects.exit(\"lineEnding\");\n return headDelimiterStart;\n }\n\n // Note: in `markdown-rs`, we need to reset, in `micromark-js` we don‘t.\n return nok(code);\n }\n if (markdownSpace(code)) {\n // To do: check if this is fine.\n // effects.attempt(State::Next(StateName::GfmTableHeadRowBreak), State::Nok)\n // State::Retry(space_or_tab(tokenizer))\n return factorySpace(effects, headRowBreak, \"whitespace\")(code);\n }\n sizeB += 1;\n if (seen) {\n seen = false;\n // Header cell count.\n size += 1;\n }\n if (code === 124) {\n effects.enter('tableCellDivider');\n effects.consume(code);\n effects.exit('tableCellDivider');\n // Whether a delimiter was seen.\n seen = true;\n return headRowBreak;\n }\n\n // Anything else is cell data.\n effects.enter(\"data\");\n return headRowData(code);\n }\n\n /**\n * In table head row data.\n *\n * ```markdown\n * > | | a |\n * ^\n * | | - |\n * | | b |\n * ```\n *\n * @type {State}\n */\n function headRowData(code) {\n if (code === null || code === 124 || markdownLineEndingOrSpace(code)) {\n effects.exit(\"data\");\n return headRowBreak(code);\n }\n effects.consume(code);\n return code === 92 ? headRowEscape : headRowData;\n }\n\n /**\n * In table head row escape.\n *\n * ```markdown\n * > | | a\\-b |\n * ^\n * | | ---- |\n * | | c |\n * ```\n *\n * @type {State}\n */\n function headRowEscape(code) {\n if (code === 92 || code === 124) {\n effects.consume(code);\n return headRowData;\n }\n return headRowData(code);\n }\n\n /**\n * Before delimiter row.\n *\n * ```markdown\n * | | a |\n * > | | - |\n * ^\n * | | b |\n * ```\n *\n * @type {State}\n */\n function headDelimiterStart(code) {\n // Reset `interrupt`.\n self.interrupt = false;\n\n // Note: in `markdown-rs`, we need to handle piercing here too.\n if (self.parser.lazy[self.now().line]) {\n return nok(code);\n }\n effects.enter('tableDelimiterRow');\n // Track if we’ve seen a `:` or `|`.\n seen = false;\n if (markdownSpace(code)) {\n return factorySpace(effects, headDelimiterBefore, \"linePrefix\", self.parser.constructs.disable.null.includes('codeIndented') ? undefined : 4)(code);\n }\n return headDelimiterBefore(code);\n }\n\n /**\n * Before delimiter row, after optional whitespace.\n *\n * Reused when a `|` is found later, to parse another cell.\n *\n * ```markdown\n * | | a |\n * > | | - |\n * ^\n * | | b |\n * ```\n *\n * @type {State}\n */\n function headDelimiterBefore(code) {\n if (code === 45 || code === 58) {\n return headDelimiterValueBefore(code);\n }\n if (code === 124) {\n seen = true;\n // If we start with a pipe, we open a cell marker.\n effects.enter('tableCellDivider');\n effects.consume(code);\n effects.exit('tableCellDivider');\n return headDelimiterCellBefore;\n }\n\n // More whitespace / empty row not allowed at start.\n return headDelimiterNok(code);\n }\n\n /**\n * After `|`, before delimiter cell.\n *\n * ```markdown\n * | | a |\n * > | | - |\n * ^\n * ```\n *\n * @type {State}\n */\n function headDelimiterCellBefore(code) {\n if (markdownSpace(code)) {\n return factorySpace(effects, headDelimiterValueBefore, \"whitespace\")(code);\n }\n return headDelimiterValueBefore(code);\n }\n\n /**\n * Before delimiter cell value.\n *\n * ```markdown\n * | | a |\n * > | | - |\n * ^\n * ```\n *\n * @type {State}\n */\n function headDelimiterValueBefore(code) {\n // Align: left.\n if (code === 58) {\n sizeB += 1;\n seen = true;\n effects.enter('tableDelimiterMarker');\n effects.consume(code);\n effects.exit('tableDelimiterMarker');\n return headDelimiterLeftAlignmentAfter;\n }\n\n // Align: none.\n if (code === 45) {\n sizeB += 1;\n // To do: seems weird that this *isn’t* left aligned, but that state is used?\n return headDelimiterLeftAlignmentAfter(code);\n }\n if (code === null || markdownLineEnding(code)) {\n return headDelimiterCellAfter(code);\n }\n return headDelimiterNok(code);\n }\n\n /**\n * After delimiter cell left alignment marker.\n *\n * ```markdown\n * | | a |\n * > | | :- |\n * ^\n * ```\n *\n * @type {State}\n */\n function headDelimiterLeftAlignmentAfter(code) {\n if (code === 45) {\n effects.enter('tableDelimiterFiller');\n return headDelimiterFiller(code);\n }\n\n // Anything else is not ok after the left-align colon.\n return headDelimiterNok(code);\n }\n\n /**\n * In delimiter cell filler.\n *\n * ```markdown\n * | | a |\n * > | | - |\n * ^\n * ```\n *\n * @type {State}\n */\n function headDelimiterFiller(code) {\n if (code === 45) {\n effects.consume(code);\n return headDelimiterFiller;\n }\n\n // Align is `center` if it was `left`, `right` otherwise.\n if (code === 58) {\n seen = true;\n effects.exit('tableDelimiterFiller');\n effects.enter('tableDelimiterMarker');\n effects.consume(code);\n effects.exit('tableDelimiterMarker');\n return headDelimiterRightAlignmentAfter;\n }\n effects.exit('tableDelimiterFiller');\n return headDelimiterRightAlignmentAfter(code);\n }\n\n /**\n * After delimiter cell right alignment marker.\n *\n * ```markdown\n * | | a |\n * > | | -: |\n * ^\n * ```\n *\n * @type {State}\n */\n function headDelimiterRightAlignmentAfter(code) {\n if (markdownSpace(code)) {\n return factorySpace(effects, headDelimiterCellAfter, \"whitespace\")(code);\n }\n return headDelimiterCellAfter(code);\n }\n\n /**\n * After delimiter cell.\n *\n * ```markdown\n * | | a |\n * > | | -: |\n * ^\n * ```\n *\n * @type {State}\n */\n function headDelimiterCellAfter(code) {\n if (code === 124) {\n return headDelimiterBefore(code);\n }\n if (code === null || markdownLineEnding(code)) {\n // Exit when:\n // * there was no `:` or `|` at all (it’s a thematic break or setext\n // underline instead)\n // * the header cell count is not the delimiter cell count\n if (!seen || size !== sizeB) {\n return headDelimiterNok(code);\n }\n\n // Note: in markdown-rs`, a reset is needed here.\n effects.exit('tableDelimiterRow');\n effects.exit('tableHead');\n // To do: in `markdown-rs`, resolvers need to be registered manually.\n // effects.register_resolver(ResolveName::GfmTable)\n return ok(code);\n }\n return headDelimiterNok(code);\n }\n\n /**\n * In delimiter row, at a disallowed byte.\n *\n * ```markdown\n * | | a |\n * > | | x |\n * ^\n * ```\n *\n * @type {State}\n */\n function headDelimiterNok(code) {\n // Note: in `markdown-rs`, we need to reset, in `micromark-js` we don‘t.\n return nok(code);\n }\n\n /**\n * Before table body row.\n *\n * ```markdown\n * | | a |\n * | | - |\n * > | | b |\n * ^\n * ```\n *\n * @type {State}\n */\n function bodyRowStart(code) {\n // Note: in `markdown-rs` we need to manually take care of a prefix,\n // but in `micromark-js` that is done for us, so if we’re here, we’re\n // never at whitespace.\n effects.enter('tableRow');\n return bodyRowBreak(code);\n }\n\n /**\n * At break in table body row.\n *\n * ```markdown\n * | | a |\n * | | - |\n * > | | b |\n * ^\n * ^\n * ^\n * ```\n *\n * @type {State}\n */\n function bodyRowBreak(code) {\n if (code === 124) {\n effects.enter('tableCellDivider');\n effects.consume(code);\n effects.exit('tableCellDivider');\n return bodyRowBreak;\n }\n if (code === null || markdownLineEnding(code)) {\n effects.exit('tableRow');\n return ok(code);\n }\n if (markdownSpace(code)) {\n return factorySpace(effects, bodyRowBreak, \"whitespace\")(code);\n }\n\n // Anything else is cell content.\n effects.enter(\"data\");\n return bodyRowData(code);\n }\n\n /**\n * In table body row data.\n *\n * ```markdown\n * | | a |\n * | | - |\n * > | | b |\n * ^\n * ```\n *\n * @type {State}\n */\n function bodyRowData(code) {\n if (code === null || code === 124 || markdownLineEndingOrSpace(code)) {\n effects.exit(\"data\");\n return bodyRowBreak(code);\n }\n effects.consume(code);\n return code === 92 ? bodyRowEscape : bodyRowData;\n }\n\n /**\n * In table body row escape.\n *\n * ```markdown\n * | | a |\n * | | ---- |\n * > | | b\\-c |\n * ^\n * ```\n *\n * @type {State}\n */\n function bodyRowEscape(code) {\n if (code === 92 || code === 124) {\n effects.consume(code);\n return bodyRowData;\n }\n return bodyRowData(code);\n }\n}\n\n/** @type {Resolver} */\n\nfunction resolveTable(events, context) {\n let index = -1;\n let inFirstCellAwaitingPipe = true;\n /** @type {RowKind} */\n let rowKind = 0;\n /** @type {Range} */\n let lastCell = [0, 0, 0, 0];\n /** @type {Range} */\n let cell = [0, 0, 0, 0];\n let afterHeadAwaitingFirstBodyRow = false;\n let lastTableEnd = 0;\n /** @type {Token | undefined} */\n let currentTable;\n /** @type {Token | undefined} */\n let currentBody;\n /** @type {Token | undefined} */\n let currentCell;\n const map = new EditMap();\n while (++index < events.length) {\n const event = events[index];\n const token = event[1];\n if (event[0] === 'enter') {\n // Start of head.\n if (token.type === 'tableHead') {\n afterHeadAwaitingFirstBodyRow = false;\n\n // Inject previous (body end and) table end.\n if (lastTableEnd !== 0) {\n flushTableEnd(map, context, lastTableEnd, currentTable, currentBody);\n currentBody = undefined;\n lastTableEnd = 0;\n }\n\n // Inject table start.\n currentTable = {\n type: 'table',\n start: Object.assign({}, token.start),\n // Note: correct end is set later.\n end: Object.assign({}, token.end)\n };\n map.add(index, 0, [['enter', currentTable, context]]);\n } else if (token.type === 'tableRow' || token.type === 'tableDelimiterRow') {\n inFirstCellAwaitingPipe = true;\n currentCell = undefined;\n lastCell = [0, 0, 0, 0];\n cell = [0, index + 1, 0, 0];\n\n // Inject table body start.\n if (afterHeadAwaitingFirstBodyRow) {\n afterHeadAwaitingFirstBodyRow = false;\n currentBody = {\n type: 'tableBody',\n start: Object.assign({}, token.start),\n // Note: correct end is set later.\n end: Object.assign({}, token.end)\n };\n map.add(index, 0, [['enter', currentBody, context]]);\n }\n rowKind = token.type === 'tableDelimiterRow' ? 2 : currentBody ? 3 : 1;\n }\n // Cell data.\n else if (rowKind && (token.type === \"data\" || token.type === 'tableDelimiterMarker' || token.type === 'tableDelimiterFiller')) {\n inFirstCellAwaitingPipe = false;\n\n // First value in cell.\n if (cell[2] === 0) {\n if (lastCell[1] !== 0) {\n cell[0] = cell[1];\n currentCell = flushCell(map, context, lastCell, rowKind, undefined, currentCell);\n lastCell = [0, 0, 0, 0];\n }\n cell[2] = index;\n }\n } else if (token.type === 'tableCellDivider') {\n if (inFirstCellAwaitingPipe) {\n inFirstCellAwaitingPipe = false;\n } else {\n if (lastCell[1] !== 0) {\n cell[0] = cell[1];\n currentCell = flushCell(map, context, lastCell, rowKind, undefined, currentCell);\n }\n lastCell = cell;\n cell = [lastCell[1], index, 0, 0];\n }\n }\n }\n // Exit events.\n else if (token.type === 'tableHead') {\n afterHeadAwaitingFirstBodyRow = true;\n lastTableEnd = index;\n } else if (token.type === 'tableRow' || token.type === 'tableDelimiterRow') {\n lastTableEnd = index;\n if (lastCell[1] !== 0) {\n cell[0] = cell[1];\n currentCell = flushCell(map, context, lastCell, rowKind, index, currentCell);\n } else if (cell[1] !== 0) {\n currentCell = flushCell(map, context, cell, rowKind, index, currentCell);\n }\n rowKind = 0;\n } else if (rowKind && (token.type === \"data\" || token.type === 'tableDelimiterMarker' || token.type === 'tableDelimiterFiller')) {\n cell[3] = index;\n }\n }\n if (lastTableEnd !== 0) {\n flushTableEnd(map, context, lastTableEnd, currentTable, currentBody);\n }\n map.consume(context.events);\n\n // To do: move this into `html`, when events are exposed there.\n // That’s what `markdown-rs` does.\n // That needs updates to `mdast-util-gfm-table`.\n index = -1;\n while (++index < context.events.length) {\n const event = context.events[index];\n if (event[0] === 'enter' && event[1].type === 'table') {\n event[1]._align = gfmTableAlign(context.events, index);\n }\n }\n return events;\n}\n\n/**\n * Generate a cell.\n *\n * @param {EditMap} map\n * @param {Readonly<TokenizeContext>} context\n * @param {Readonly<Range>} range\n * @param {RowKind} rowKind\n * @param {number | undefined} rowEnd\n * @param {Token | undefined} previousCell\n * @returns {Token | undefined}\n */\n// eslint-disable-next-line max-params\nfunction flushCell(map, context, range, rowKind, rowEnd, previousCell) {\n // `markdown-rs` uses:\n // rowKind === 2 ? 'tableDelimiterCell' : 'tableCell'\n const groupName = rowKind === 1 ? 'tableHeader' : rowKind === 2 ? 'tableDelimiter' : 'tableData';\n // `markdown-rs` uses:\n // rowKind === 2 ? 'tableDelimiterCellValue' : 'tableCellText'\n const valueName = 'tableContent';\n\n // Insert an exit for the previous cell, if there is one.\n //\n // ```markdown\n // > | | aa | bb | cc |\n // ^-- exit\n // ^^^^-- this cell\n // ```\n if (range[0] !== 0) {\n previousCell.end = Object.assign({}, getPoint(context.events, range[0]));\n map.add(range[0], 0, [['exit', previousCell, context]]);\n }\n\n // Insert enter of this cell.\n //\n // ```markdown\n // > | | aa | bb | cc |\n // ^-- enter\n // ^^^^-- this cell\n // ```\n const now = getPoint(context.events, range[1]);\n previousCell = {\n type: groupName,\n start: Object.assign({}, now),\n // Note: correct end is set later.\n end: Object.assign({}, now)\n };\n map.add(range[1], 0, [['enter', previousCell, context]]);\n\n // Insert text start at first data start and end at last data end, and\n // remove events between.\n //\n // ```markdown\n // > | | aa | bb | cc |\n // ^-- enter\n // ^-- exit\n // ^^^^-- this cell\n // ```\n if (range[2] !== 0) {\n const relatedStart = getPoint(context.events, range[2]);\n const relatedEnd = getPoint(context.events, range[3]);\n /** @type {Token} */\n const valueToken = {\n type: valueName,\n start: Object.assign({}, relatedStart),\n end: Object.assign({}, relatedEnd)\n };\n map.add(range[2], 0, [['enter', valueToken, context]]);\n if (rowKind !== 2) {\n // Fix positional info on remaining events\n const start = context.events[range[2]];\n const end = context.events[range[3]];\n start[1].end = Object.assign({}, end[1].end);\n start[1].type = \"chunkText\";\n start[1].contentType = \"text\";\n\n // Remove if needed.\n if (range[3] > range[2] + 1) {\n const a = range[2] + 1;\n const b = range[3] - range[2] - 1;\n map.add(a, b, []);\n }\n }\n map.add(range[3] + 1, 0, [['exit', valueToken, context]]);\n }\n\n // Insert an exit for the last cell, if at the row end.\n //\n // ```markdown\n // > | | aa | bb | cc |\n // ^-- exit\n // ^^^^^^-- this cell (the last one contains two “between” parts)\n // ```\n if (rowEnd !== undefined) {\n previousCell.end = Object.assign({}, getPoint(context.events, rowEnd));\n map.add(rowEnd, 0, [['exit', previousCell, context]]);\n previousCell = undefined;\n }\n return previousCell;\n}\n\n/**\n * Generate table end (and table body end).\n *\n * @param {Readonly<EditMap>} map\n * @param {Readonly<TokenizeContext>} context\n * @param {number} index\n * @param {Token} table\n * @param {Token | undefined} tableBody\n */\n// eslint-disable-next-line max-params\nfunction flushTableEnd(map, context, index, table, tableBody) {\n /** @type {Array<Event>} */\n const exits = [];\n const related = getPoint(context.events, index);\n if (tableBody) {\n tableBody.end = Object.assign({}, related);\n exits.push(['exit', tableBody, context]);\n }\n table.end = Object.assign({}, related);\n exits.push(['exit', table, context]);\n map.add(index + 1, 0, exits);\n}\n\n/**\n * @param {Readonly<Array<Event>>} events\n * @param {number} index\n * @returns {Readonly<Point>}\n */\nfunction getPoint(events, index) {\n const event = events[index];\n const side = event[0] === 'enter' ? 'start' : 'end';\n return event[1][side];\n}","/**\n * @import {Extension, State, TokenizeContext, Tokenizer} from 'micromark-util-types'\n */\n\nimport { factorySpace } from 'micromark-factory-space';\nimport { markdownLineEnding, markdownLineEndingOrSpace, markdownSpace } from 'micromark-util-character';\nconst tasklistCheck = {\n name: 'tasklistCheck',\n tokenize: tokenizeTasklistCheck\n};\n\n/**\n * Create an HTML extension for `micromark` to support GFM task list items\n * syntax.\n *\n * @returns {Extension}\n * Extension for `micromark` that can be passed in `htmlExtensions` to\n * support GFM task list items when serializing to HTML.\n */\nexport function gfmTaskListItem() {\n return {\n text: {\n [91]: tasklistCheck\n }\n };\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction tokenizeTasklistCheck(effects, ok, nok) {\n const self = this;\n return open;\n\n /**\n * At start of task list item check.\n *\n * ```markdown\n * > | * [x] y.\n * ^\n * ```\n *\n * @type {State}\n */\n function open(code) {\n if (\n // Exit if there’s stuff before.\n self.previous !== null ||\n // Exit if not in the first content that is the first child of a list\n // item.\n !self._gfmTasklistFirstContentOfListItem) {\n return nok(code);\n }\n effects.enter('taskListCheck');\n effects.enter('taskListCheckMarker');\n effects.consume(code);\n effects.exit('taskListCheckMarker');\n return inside;\n }\n\n /**\n * In task list item check.\n *\n * ```markdown\n * > | * [x] y.\n * ^\n * ```\n *\n * @type {State}\n */\n function inside(code) {\n // Currently we match how GH works in files.\n // To match how GH works in comments, use `markdownSpace` (`[\\t ]`) instead\n // of `markdownLineEndingOrSpace` (`[\\t\\n\\r ]`).\n if (markdownLineEndingOrSpace(code)) {\n effects.enter('taskListCheckValueUnchecked');\n effects.consume(code);\n effects.exit('taskListCheckValueUnchecked');\n return close;\n }\n if (code === 88 || code === 120) {\n effects.enter('taskListCheckValueChecked');\n effects.consume(code);\n effects.exit('taskListCheckValueChecked');\n return close;\n }\n return nok(code);\n }\n\n /**\n * At close of task list item check.\n *\n * ```markdown\n * > | * [x] y.\n * ^\n * ```\n *\n * @type {State}\n */\n function close(code) {\n if (code === 93) {\n effects.enter('taskListCheckMarker');\n effects.consume(code);\n effects.exit('taskListCheckMarker');\n effects.exit('taskListCheck');\n return after;\n }\n return nok(code);\n }\n\n /**\n * @type {State}\n */\n function after(code) {\n // EOL in paragraph means there must be something else after it.\n if (markdownLineEnding(code)) {\n return ok(code);\n }\n\n // Space or tab?\n // Check what comes after.\n if (markdownSpace(code)) {\n return effects.check({\n tokenize: spaceThenNonSpace\n }, ok, nok)(code);\n }\n\n // EOF, or non-whitespace, both wrong.\n return nok(code);\n }\n}\n\n/**\n * @this {TokenizeContext}\n * @type {Tokenizer}\n */\nfunction spaceThenNonSpace(effects, ok, nok) {\n return factorySpace(effects, after, \"whitespace\");\n\n /**\n * After whitespace, after task list item check.\n *\n * ```markdown\n * > | * [x] y.\n * ^\n * ```\n *\n * @type {State}\n */\n function after(code) {\n // EOF means there was nothing, so bad.\n // EOL means there’s content after it, so good.\n // Impossible to have more spaces.\n // Anything else is good.\n return code === null ? nok(code) : ok(code);\n }\n}","/**\n * @typedef {import('micromark-extension-gfm-footnote').HtmlOptions} HtmlOptions\n * @typedef {import('micromark-extension-gfm-strikethrough').Options} Options\n * @typedef {import('micromark-util-types').Extension} Extension\n * @typedef {import('micromark-util-types').HtmlExtension} HtmlExtension\n */\n\nimport {\n combineExtensions,\n combineHtmlExtensions\n} from 'micromark-util-combine-extensions'\nimport {\n gfmAutolinkLiteral,\n gfmAutolinkLiteralHtml\n} from 'micromark-extension-gfm-autolink-literal'\nimport {gfmFootnote, gfmFootnoteHtml} from 'micromark-extension-gfm-footnote'\nimport {\n gfmStrikethrough,\n gfmStrikethroughHtml\n} from 'micromark-extension-gfm-strikethrough'\nimport {gfmTable, gfmTableHtml} from 'micromark-extension-gfm-table'\nimport {gfmTagfilterHtml} from 'micromark-extension-gfm-tagfilter'\nimport {\n gfmTaskListItem,\n gfmTaskListItemHtml\n} from 'micromark-extension-gfm-task-list-item'\n\n/**\n * Create an extension for `micromark` to enable GFM syntax.\n *\n * @param {Options | null | undefined} [options]\n * Configuration (optional).\n *\n * Passed to `micromark-extens-gfm-strikethrough`.\n * @returns {Extension}\n * Extension for `micromark` that can be passed in `extensions` to enable GFM\n * syntax.\n */\nexport function gfm(options) {\n return combineExtensions([\n gfmAutolinkLiteral(),\n gfmFootnote(),\n gfmStrikethrough(options),\n gfmTable(),\n gfmTaskListItem()\n ])\n}\n\n/**\n * Create an extension for `micromark` to support GFM when serializing to HTML.\n *\n * @param {HtmlOptions | null | undefined} [options]\n * Configuration (optional).\n *\n * Passed to `micromark-extens-gfm-footnote`.\n * @returns {HtmlExtension}\n * Extension for `micromark` that can be passed in `htmlExtensions` to\n * support GFM when serializing to HTML.\n */\nexport function gfmHtml(options) {\n return combineHtmlExtensions([\n gfmAutolinkLiteralHtml(),\n gfmFootnoteHtml(options),\n gfmStrikethroughHtml(),\n gfmTableHtml(),\n gfmTagfilterHtml(),\n gfmTaskListItemHtml()\n ])\n}\n","/**\n * @import {Root} from 'mdast'\n * @import {Options} from 'remark-gfm'\n * @import {} from 'remark-parse'\n * @import {} from 'remark-stringify'\n * @import {Processor} from 'unified'\n */\n\nimport {gfmFromMarkdown, gfmToMarkdown} from 'mdast-util-gfm'\nimport {gfm} from 'micromark-extension-gfm'\n\n/** @type {Options} */\nconst emptyOptions = {}\n\n/**\n * Add support GFM (autolink literals, footnotes, strikethrough, tables,\n * tasklists).\n *\n * @param {Options | null | undefined} [options]\n * Configuration (optional).\n * @returns {undefined}\n * Nothing.\n */\nexport default function remarkGfm(options) {\n // @ts-expect-error: TS is wrong about `this`.\n // eslint-disable-next-line unicorn/no-this-assignment\n const self = /** @type {Processor<Root>} */ (this)\n const settings = options || emptyOptions\n const data = self.data()\n\n const micromarkExtensions =\n data.micromarkExtensions || (data.micromarkExtensions = [])\n const fromMarkdownExtensions =\n data.fromMarkdownExtensions || (data.fromMarkdownExtensions = [])\n const toMarkdownExtensions =\n data.toMarkdownExtensions || (data.toMarkdownExtensions = [])\n\n micromarkExtensions.push(gfm(settings))\n fromMarkdownExtensions.push(gfmFromMarkdown())\n toMarkdownExtensions.push(gfmToMarkdown(settings))\n}\n","export function cn(...classes: Array<string | false | null | undefined>): string {\n return classes.filter(Boolean).join(' ');\n}\n\n","'use client';\n\nimport '@assistant-ui/react-markdown/styles/dot.css';\n\nimport {\n type CodeHeaderProps,\n MarkdownTextPrimitive,\n unstable_memoizeMarkdownComponents as memoizeMarkdownComponents,\n useIsMarkdownCodeBlock,\n} from '@assistant-ui/react-markdown';\nimport remarkGfm from 'remark-gfm';\nimport { type FC, memo, useState } from 'react';\nimport { CheckIcon, CopyIcon } from 'lucide-react';\nimport { cn } from '../utils/cn';\n\ninterface MarkdownTextProps {\n content: string;\n}\n\nconst MarkdownTextImpl: FC<MarkdownTextProps> = ({ content }) => {\n return (\n <MarkdownTextPrimitive\n remarkPlugins={[remarkGfm]}\n className=\"cuadra-aui-md\"\n components={defaultComponents}\n {...({ children: content } as { children: string })}\n />\n );\n};\n\nexport const MarkdownText = memo(MarkdownTextImpl);\n\nconst CodeHeader: FC<CodeHeaderProps> = ({ language, code }) => {\n const { isCopied, copyToClipboard } = useCopyToClipboard();\n const onCopy = () => {\n if (!code || isCopied) return;\n copyToClipboard(code);\n };\n\n return (\n <div className=\"cuadra-flex cuadra-items-center cuadra-justify-between cuadra-gap-4 cuadra-mt-4 cuadra-rounded-t-lg cuadra-bg-muted/50 cuadra-border-b cuadra-border-border cuadra-px-4 cuadra-py-2 cuadra-text-sm cuadra-font-normal cuadra-text-foreground cuadra-font-brand\">\n <span className=\"cuadra-lowercase [&>span]:cuadra-text-xs cuadra-font-brand\">{language}</span>\n <button\n onClick={onCopy}\n className=\"cuadra-p-1 cuadra-rounded hover:cuadra-bg-muted cuadra-transition-colors\"\n aria-label=\"Copy code\"\n >\n {!isCopied && <CopyIcon className=\"cuadra-h-4 cuadra-w-4\" />}\n {isCopied && <CheckIcon className=\"cuadra-h-4 cuadra-w-4\" />}\n </button>\n </div>\n );\n};\n\nconst useCopyToClipboard = ({\n copiedDuration = 3000,\n}: {\n copiedDuration?: number;\n} = {}) => {\n const [isCopied, setIsCopied] = useState<boolean>(false);\n\n const copyToClipboard = (value: string) => {\n if (!value) return;\n\n navigator.clipboard.writeText(value).then(() => {\n setIsCopied(true);\n setTimeout(() => setIsCopied(false), copiedDuration);\n });\n };\n\n return { isCopied, copyToClipboard };\n};\n\nconst defaultComponents = memoizeMarkdownComponents({\n h1: ({ className, ...props }) => (\n <h1\n className={cn('cuadra-mb-6 cuadra-scroll-m-20 cuadra-text-2xl cuadra-font-normal cuadra-tracking-tight last:cuadra-mb-0 font-brand', className)}\n {...props}\n />\n ),\n h2: ({ className, ...props }) => (\n <h2\n className={cn(\n 'cuadra-mb-4 cuadra-mt-6 cuadra-scroll-m-20 cuadra-text-xl cuadra-font-normal cuadra-tracking-tight first:cuadra-mt-0 last:cuadra-mb-0 font-brand',\n className,\n )}\n {...props}\n />\n ),\n h3: ({ className, ...props }) => (\n <h3\n className={cn(\n 'cuadra-mb-3 cuadra-mt-5 cuadra-scroll-m-20 cuadra-text-lg cuadra-font-normal cuadra-tracking-tight first:cuadra-mt-0 last:cuadra-mb-0 font-brand',\n className,\n )}\n {...props}\n />\n ),\n h4: ({ className, ...props }) => (\n <h4\n className={cn(\n 'cuadra-mb-4 cuadra-mt-6 cuadra-scroll-m-20 cuadra-text-xl cuadra-font-normal cuadra-tracking-tight first:cuadra-mt-0 last:cuadra-mb-0 font-brand',\n className,\n )}\n {...props}\n />\n ),\n h5: ({ className, ...props }) => (\n <h5 className={cn('cuadra-my-4 cuadra-text-lg cuadra-font-normal first:cuadra-mt-0 last:cuadra-mb-0 font-brand', className)} {...props} />\n ),\n h6: ({ className, ...props }) => (\n <h6 className={cn('cuadra-my-4 cuadra-font-normal first:cuadra-mb-0 font-brand', className)} {...props} />\n ),\n p: ({ className, ...props }) => (\n <p className={cn('cuadra-mb-5 cuadra-mt-5 cuadra-leading-7 first:cuadra-mt-0 last:cuadra-mb-0 font-brand', className)} {...props} />\n ),\n a: ({ className, ...props }) => (\n <a\n className={cn('cuadra-text-primary cuadra-font-medium cuadra-underline cuadra-underline-offset-4 font-brand', className)}\n {...props}\n />\n ),\n blockquote: ({ className, ...props }) => (\n <blockquote className={cn('cuadra-border-l-2 cuadra-pl-6 cuadra-italic font-brand', className)} {...props} />\n ),\n ul: ({ className, ...props }) => (\n <ul className={cn('cuadra-my-5 cuadra-ml-6 cuadra-list-disc [&>li]:cuadra-mt-2 font-brand', className)} {...props} />\n ),\n ol: ({ className, ...props }) => (\n <ol className={cn('cuadra-my-5 cuadra-ml-6 cuadra-list-decimal [&>li]:cuadra-mt-2 font-brand', className)} {...props} />\n ),\n hr: ({ className, ...props }) => <hr className={cn('cuadra-my-5 cuadra-border-b', className)} {...props} />,\n table: ({ className, ...props }) => (\n <table\n className={cn('cuadra-my-5 cuadra-w-full cuadra-border-separate cuadra-border-spacing-0 cuadra-overflow-y-auto font-brand', className)}\n {...props}\n />\n ),\n th: ({ className, ...props }) => (\n <th\n className={cn(\n 'cuadra-bg-muted cuadra-px-4 cuadra-py-2 cuadra-text-left cuadra-font-normal first:cuadra-rounded-tl-lg last:cuadra-rounded-tr-lg [&[align=center]]:cuadra-text-center [&[align=right]]:cuadra-text-right font-brand',\n className,\n )}\n {...props}\n />\n ),\n td: ({ className, ...props }) => (\n <td\n className={cn(\n 'cuadra-border-b cuadra-border-l cuadra-px-4 cuadra-py-2 cuadra-text-left last:cuadra-border-r [&[align=center]]:cuadra-text-center [&[align=right]]:cuadra-text-right font-brand',\n className,\n )}\n {...props}\n />\n ),\n tr: ({ className, ...props }) => (\n <tr\n className={cn(\n 'cuadra-m-0 cuadra-border-b cuadra-p-0 first:cuadra-border-t [&:last-child>td:first-child]:cuadra-rounded-bl-lg [&:last-child>td:last-child]:cuadra-rounded-br-lg',\n className,\n )}\n {...props}\n />\n ),\n sup: ({ className, ...props }) => (\n <sup className={cn('[&>a]:cuadra-text-xs [&>a]:cuadra-no-underline font-brand', className)} {...props} />\n ),\n pre: ({ className, ...props }) => (\n <pre\n className={cn(\n 'cuadra-overflow-x-auto cuadra-whitespace-pre-wrap cuadra-break-words cuadra-rounded-b-lg !cuadra-rounded-t-none cuadra-bg-muted cuadra-p-4 cuadra-text-foreground font-brand',\n className,\n )}\n {...props}\n />\n ),\n code: function Code({ className, ...props }) {\n const isCodeBlock = useIsMarkdownCodeBlock();\n return (\n <code\n className={cn(!isCodeBlock && 'cuadra-bg-muted cuadra-rounded cuadra-border cuadra-font-normal font-brand', className)}\n {...props}\n />\n );\n },\n CodeHeader,\n});\n","import * as React from \"react\";\n\nexport interface BadgeProps extends React.HTMLAttributes<HTMLDivElement> {\n variant?: \"default\" | \"subtle\" | \"outline\" | \"ghost\";\n}\n\n/**\n * Badge component - simple, subtle styling\n */\nfunction Badge({ className = \"\", variant = \"default\", ...props }: BadgeProps) {\n const baseClasses = \n \"cuadra-inline-flex cuadra-items-center cuadra-rounded-md cuadra-px-3 cuadra-py-1.5 cuadra-text-xs cuadra-font-medium font-brand\";\n\n const variantClasses: Record<string, string> = {\n default:\n \"cuadra-border cuadra-border-border cuadra-bg-muted cuadra-text-muted-foreground\",\n subtle:\n \"cuadra-border cuadra-border-border/50 cuadra-bg-muted/50 cuadra-text-muted-foreground\",\n outline: \n \"cuadra-border cuadra-border-border cuadra-bg-background cuadra-text-foreground\",\n ghost:\n \"cuadra-text-muted-foreground\",\n };\n\n return (\n <div\n className={`${baseClasses} ${variantClasses[variant] || variantClasses.default} ${className}`}\n {...props}\n />\n );\n}\n\nexport { Badge };\n\n","import { FileSpreadsheet, FileText, FileType } from 'lucide-react';\nimport type { Source } from '../types/cuadra';\nimport { Badge } from './ui/badge';\n\nexport interface SourceCitationsProps {\n /** Array of source citations */\n sources: Source[];\n /** Maximum number of sources to show before collapsing */\n maxVisible?: number;\n}\n\n/**\n * Deduplicate sources by filename, keeping the highest score for each file\n */\nfunction deduplicateSources(sources: Source[]): Source[] {\n const byFilename = new Map<string, Source>();\n \n for (const source of sources) {\n const existing = byFilename.get(source.filename);\n if (!existing || source.score > existing.score) {\n byFilename.set(source.filename, source);\n }\n }\n \n // Sort by score descending\n return Array.from(byFilename.values()).sort((a, b) => b.score - a.score);\n}\n\n/**\n * Component to display RAG source citations\n * Shows document references that were used to generate the response\n * Deduplicates by filename, showing highest relevance score per file\n */\nexport function SourceCitations({ \n sources, \n maxVisible = 3,\n}: SourceCitationsProps) {\n if (!sources || sources.length === 0) return null;\n\n // Deduplicate by filename\n const uniqueSources = deduplicateSources(sources);\n const visibleSources = uniqueSources.slice(0, maxVisible);\n const hiddenCount = uniqueSources.length - maxVisible;\n\n return (\n <div className=\"cuadra-mt-4 cuadra-pt-3 cuadra-border-t cuadra-border-border\">\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-1.5 cuadra-mb-2.5\">\n <FileText className=\"cuadra-h-3.5 cuadra-w-3.5 cuadra-text-muted-foreground\" />\n <span className=\"cuadra-text-xs cuadra-font-medium cuadra-text-muted-foreground font-brand\">\n Sources\n </span>\n </div>\n \n <div className=\"cuadra-flex cuadra-flex-wrap cuadra-gap-2\">\n {visibleSources.map((source, index) => (\n <SourceBadge key={source.sourceId || index} source={source} />\n ))}\n {hiddenCount > 0 && (\n <Badge variant=\"subtle\">\n +{hiddenCount} more\n </Badge>\n )}\n </div>\n </div>\n );\n}\n\ninterface SourceBadgeProps {\n source: Source;\n}\n\n/**\n * Get icon based on file extension\n */\nfunction getFileIcon(filename: string): typeof FileText {\n const ext = filename.split('.').pop()?.toLowerCase();\n \n switch (ext) {\n case 'csv':\n case 'xlsx':\n case 'xls':\n return FileSpreadsheet;\n case 'md':\n case 'txt':\n return FileType;\n default:\n return FileText;\n }\n}\n\n/**\n * Individual source badge showing filename and optional relevance score\n */\nfunction SourceBadge({ source }: SourceBadgeProps) {\n const { filename, score } = source;\n const Icon = getFileIcon(filename);\n\n return (\n <Badge \n className=\"cuadra-gap-2 cuadra-cursor-default\"\n title={score < 1 ? `${Math.round(score * 100)}% relevant` : undefined}\n >\n <Icon className=\"cuadra-h-3.5 cuadra-w-3.5 cuadra-shrink-0 cuadra-opacity-60\" />\n <span className=\"cuadra-truncate cuadra-max-w-[180px]\">\n {filename}\n </span>\n {score < 1 && score > 0 && (\n <span className=\"cuadra-opacity-50 cuadra-text-[10px] cuadra-tabular-nums\">\n {Math.round(score * 100)}%\n </span>\n )}\n </Badge>\n );\n}\n\n","import { AttachmentPrimitive, ComposerPrimitive, MessagePrimitive, ThreadListItemPrimitive, ThreadListPrimitive, ThreadPrimitive, useAssistantRuntime, useAui, useAuiState, useThread, useThreadListItem, useThreadListItemRuntime, useThreadRuntime } from '@assistant-ui/react';\nimport { AlertCircle, ArrowUpIcon, Brain, Check, ChevronDown, EllipsisVertical, Eye, Loader2, MessageSquare, Paperclip, Pencil, Plus, Trash2, Upload, X } from 'lucide-react';\nimport { forwardRef, useCallback, useEffect, useImperativeHandle, useRef, useState, useSyncExternalStore } from 'react';\nimport { MarkdownText } from './MarkdownText';\nimport { SourceCitations } from './SourceCitations';\n// ThemeToggle import removed — theme is now controlled via the `theme` prop only\nimport { queuePreMadeResponse, streamingMetadataStore } from '../adapters/chatModelAdapter';\nimport { attachmentErrorStore } from '../adapters/attachmentErrorStore';\nimport '@assistant-ui/react-markdown/styles/dot.css';\n\n/**\n * Handle for controlling SimpleThread externally\n */\nexport interface SimpleThreadHandle {\n /** Send a user message programmatically (triggers normal API call) */\n sendMessage: (message: string) => void;\n /** Send a pre-made Q&A pair (instant, simulated streaming, no API call) */\n sendPreMadeQA: (question: string, answer: string, options?: { delay?: number; streamingSpeed?: number }) => void;\n /** Clear the current chat and start fresh */\n clearChat: () => void;\n}\n\n/** Suggestion with optional pre-made response */\nexport interface Suggestion {\n /** Suggestion prompt text */\n prompt: string;\n /** Pre-made response (optional) - if provided, response appears instantly with simulated streaming */\n response?: string;\n /** Delay before showing response in ms (overrides global preMadeResponseDelay) */\n responseDelay?: number;\n}\n\n/** Model option for the composer model selector */\nexport interface ModelOption {\n /** Unique model identifier */\n id: string;\n /** Display name shown in selector (e.g. \"GPT-4o\", \"Claude 3.5\") */\n name: string;\n /** Optional short description */\n description?: string;\n /** Whether the model supports vision/image inputs */\n supportsVision?: boolean;\n /** Whether the model supports reasoning/thinking */\n supportsReasoning?: boolean;\n}\n\nexport interface SimpleThreadProps {\n /** Welcome screen title */\n welcomeTitle?: string;\n /** Welcome screen subtitle */\n welcomeSubtitle?: string;\n /** Extra top padding for thread viewport (e.g., '1rem', '2rem') */\n extraTopPadding?: string;\n /** Suggestions to show in welcome screen */\n suggestions?: Suggestion[];\n /** Placeholder text for the input field */\n inputPlaceholder?: string;\n /** Enable file attachments in the chat input */\n enableAttachments?: boolean;\n /** Default delay before pre-made responses in ms (default: 1000) */\n preMadeResponseDelay?: number;\n /** Speed of simulated streaming in ms per word (default: 50) */\n streamingSpeed?: number;\n /** Apply safe area insets for mobile devices */\n safeArea?: boolean;\n\n // --- Claude-like features ---\n\n /** Show a visible border on the composer (default: false — uses shadow only) */\n composerBorder?: boolean;\n /** Available models for the composer model selector */\n models?: ModelOption[];\n /** Currently selected model ID */\n selectedModelId?: string;\n /** Callback when user selects a different model */\n onModelChange?: (modelId: string) => void;\n /** Show the chat title header at the top of the thread area */\n showChatHeader?: boolean;\n /** Show the theme toggle in the chat header */\n showThemeToggle?: boolean;\n /** Initial theme preference */\n theme?: 'light' | 'dark' | 'system';\n}\n\n/**\n * Simple Thread component wrapper\n * Provides a basic chat interface using assistant-ui primitives\n */\nexport const SimpleThread = forwardRef<SimpleThreadHandle, SimpleThreadProps>(\n function SimpleThread({\n welcomeTitle = 'What should we work on?',\n welcomeSubtitle,\n extraTopPadding,\n suggestions,\n inputPlaceholder = 'Reply...',\n enableAttachments = false,\n preMadeResponseDelay = 1000,\n streamingSpeed = 50,\n safeArea = false,\n composerBorder = false,\n models,\n selectedModelId,\n onModelChange,\n showChatHeader = false,\n // theme and showThemeToggle kept in interface for consumer API but not used internally\n }, ref) {\n const thread = useThreadRuntime();\n const aui = useAui();\n const [isLoadingMessages, setIsLoadingMessages] = useState(false);\n const [currentThreadId, setCurrentThreadId] = useState<string | null>(null);\n const [isProcessingPreMade, _setIsProcessingPreMade] = useState(false);\n\n // Subscribe to attachment validation errors (auto-dismiss after 4s)\n const attachmentError = useSyncExternalStore(\n attachmentErrorStore.subscribe,\n attachmentErrorStore.getSnapshot,\n );\n\n useEffect(() => {\n if (attachmentError) {\n const timer = setTimeout(() => attachmentErrorStore.clear(), 4000);\n return () => clearTimeout(timer);\n }\n }, [attachmentError]);\n\n // Track whether thread is empty for centered welcome layout\n const threadIsEmpty = useThread((state) => {\n const messages = (state as unknown as { messages?: readonly unknown[] }).messages;\n return !messages || messages.length === 0;\n });\n const showCenteredWelcome = threadIsEmpty && !isLoadingMessages;\n\n // Switch to a new thread when the model changes (multiChat mode).\n // This runs inside the AssistantRuntimeProvider so it has direct\n // hook access to the runtime — no fragile window globals needed.\n // We use a ref for the runtime + a short delay so the switch happens\n // after all cascading state updates from the model change have settled\n // (CuadraRuntimeProvider syncs selectedModelId via useEffect → extra render).\n const assistantRuntime = useAssistantRuntime();\n const assistantRuntimeRef = useRef(assistantRuntime);\n assistantRuntimeRef.current = assistantRuntime;\n const prevModelIdRef = useRef(selectedModelId);\n useEffect(() => {\n const prev = prevModelIdRef.current;\n prevModelIdRef.current = selectedModelId;\n\n if (prev && selectedModelId && prev !== selectedModelId) {\n const timer = setTimeout(() => {\n try {\n void assistantRuntimeRef.current.switchToNewThread();\n } catch (_error) {\n // Silently handle — user can manually create a new thread\n }\n }, 50);\n return () => clearTimeout(timer);\n }\n }, [selectedModelId]); // only depends on selectedModelId — not assistantRuntime\n\n /**\n * Send a pre-made Q&A with simulated streaming\n * Queues the response and triggers normal send flow - adapter intercepts and returns fake response\n */\n const sendPreMadeQAInternal = useCallback((\n question: string, \n answer: string, \n options?: { delay?: number; streamingSpeed?: number }\n ) => {\n if (isProcessingPreMade) return;\n \n try {\n // Queue the pre-made response - the adapter will intercept and return it\n // Also stores the Q&A for context in follow-up API calls\n const speed = options?.streamingSpeed ?? streamingSpeed;\n const delay = options?.delay ?? preMadeResponseDelay;\n queuePreMadeResponse(question, answer, speed, delay);\n \n // Trigger normal send flow with the question\n // The adapter will see the queued response and return it instead of calling API\n aui.composer().setText(question);\n setTimeout(() => {\n aui.composer().send();\n }, 10);\n } catch (_err) {\n // Pre-made QA error - silently ignore\n }\n }, [isProcessingPreMade, streamingSpeed, preMadeResponseDelay, aui]);\n\n /**\n * Send a message normally (triggers API call)\n */\n const sendMessageInternal = useCallback((message: string) => {\n try {\n aui.composer().setText(message);\n setTimeout(() => {\n aui.composer().send();\n }, 10);\n } catch {\n // Silently handle\n }\n }, [aui]);\n\n /**\n * Handle suggestion click - lifted to parent so it doesn't unmount mid-execution\n */\n const handleSuggestionClick = useCallback(async (suggestion: Suggestion) => {\n if (isProcessingPreMade || !thread) return;\n\n try {\n if (suggestion.response) {\n // Pre-made response mode\n const delay = suggestion.responseDelay ?? preMadeResponseDelay;\n await sendPreMadeQAInternal(suggestion.prompt, suggestion.response, { \n delay, \n streamingSpeed \n });\n } else {\n // Normal mode: use composer API\n sendMessageInternal(suggestion.prompt);\n }\n } catch {\n // Silently handle\n }\n }, [thread, isProcessingPreMade, preMadeResponseDelay, streamingSpeed, sendPreMadeQAInternal, sendMessageInternal]);\n\n /**\n * Clear the chat\n */\n const clearChatInternal = useCallback(async () => {\n try {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const runtime = (window as any).__cuadraThreadListRuntime;\n const threadListRuntime = runtime?.threadList;\n\n if (threadListRuntime?.switchToNewThread) {\n await threadListRuntime.switchToNewThread();\n }\n } catch {\n // Silently handle\n }\n }, []);\n\n // Expose methods via ref\n useImperativeHandle(ref, () => ({\n sendMessage: sendMessageInternal,\n sendPreMadeQA: sendPreMadeQAInternal,\n clearChat: clearChatInternal,\n }), [sendMessageInternal, sendPreMadeQAInternal, clearChatInternal]);\n\n // Track thread changes and loading state by subscribing to store changes\n useEffect(() => {\n if (!thread) {\n setIsLoadingMessages(false);\n return;\n }\n\n try {\n // Access runtime store to track loading state\n interface RuntimeStore {\n getState?: () => {\n threadId?: string;\n messages?: unknown[];\n isLoadingHistory?: boolean;\n isRunning?: boolean;\n };\n subscribe?: (callback: (state: { threadId?: string; messages?: unknown[]; isLoadingHistory?: boolean; isRunning?: boolean }) => void) => () => void;\n }\n const runtimeStore = (thread as { store?: RuntimeStore })?.store;\n if (!runtimeStore) {\n setIsLoadingMessages(false);\n return;\n }\n\n // Subscribe to store changes\n const unsubscribe = runtimeStore.subscribe?.((state) => {\n const threadId = state?.threadId;\n const messages = (state?.messages || []) as Array<{\n role?: string;\n content?: Array<{ type?: string; text?: string }> | string;\n }>;\n const isLoadingHistory = state?.isLoadingHistory || false;\n const messageCount = messages.length;\n\n // Check if thread changed\n if (threadId && threadId !== currentThreadId) {\n setCurrentThreadId(threadId);\n }\n\n // Only show loading if:\n // 1. We're actively loading history\n // 2. We have no messages yet\n const shouldShowLoading = isLoadingHistory && messageCount === 0;\n setIsLoadingMessages(shouldShowLoading);\n });\n\n // Initial check\n const state = runtimeStore.getState?.();\n if (state) {\n const threadId = state?.threadId;\n const messages = (state?.messages || []) as Array<{\n role?: string;\n content?: Array<{ type?: string; text?: string }> | string;\n }>;\n const isLoadingHistory = state?.isLoadingHistory || false;\n const messageCount = messages.length;\n\n if (threadId && threadId !== currentThreadId) {\n setCurrentThreadId(threadId);\n }\n\n const shouldShowLoading = isLoadingHistory && messageCount === 0;\n setIsLoadingMessages(shouldShowLoading);\n }\n\n return () => {\n if (unsubscribe) {\n unsubscribe();\n }\n };\n } catch {\n setIsLoadingMessages(false);\n }\n }, [thread, currentThreadId]);\n\n return (\n <ThreadPrimitive.Root className=\"cuadra-bg-inherit cuadra-flex cuadra-flex-col cuadra-w-full cuadra-h-full cuadra-relative\">\n {/* Top bar — centered model selector + optional thread selector */}\n {(showChatHeader || (models && models.length > 0)) && (\n <div className=\"cuadra-relative cuadra-flex cuadra-items-center cuadra-justify-center cuadra-px-3 cuadra-py-2 cuadra-min-h-[48px]\">\n {/* Thread selector: right on mobile, left on desktop (absolute so model stays centered) */}\n {showChatHeader && (\n <div className=\"cuadra-absolute cuadra-right-3 md:cuadra-right-auto md:cuadra-left-3 cuadra-top-0 cuadra-bottom-0 cuadra-flex cuadra-items-center\">\n <ChatHeader />\n </div>\n )}\n {/* Center: Model selector */}\n {models && models.length > 0 && (\n <ModelSelector\n models={models}\n selectedModelId={selectedModelId}\n onModelChange={onModelChange}\n />\n )}\n </div>\n )}\n {/* Drag-and-drop + content wrapper */}\n <ComposerPrimitive.AttachmentDropzone\n disabled={!enableAttachments}\n className={`cuadra-flex-1 cuadra-flex cuadra-flex-col cuadra-min-h-0 cuadra-relative ${showCenteredWelcome ? 'cuadra-justify-center' : ''}`}\n >\n {/* Drag overlay */}\n <div className=\"cuadra-dropzone-overlay cuadra-pointer-events-none cuadra-absolute cuadra-inset-0 cuadra-z-50 cuadra-flex cuadra-items-center cuadra-justify-center cuadra-rounded-lg cuadra-transition-opacity cuadra-duration-200\" style={{ opacity: 0 }}>\n <div className=\"cuadra-flex cuadra-flex-col cuadra-items-center cuadra-gap-3\">\n <div className=\"cuadra-flex cuadra-items-center cuadra-justify-center cuadra-size-14 cuadra-rounded-2xl\" style={{ backgroundColor: 'var(--cuadra-primary)', color: 'white' }}>\n <Upload className=\"cuadra-h-6 cuadra-w-6\" />\n </div>\n <p className=\"cuadra-text-sm cuadra-font-medium cuadra-text-foreground font-brand\">Drop files here</p>\n </div>\n </div>\n\n <ThreadPrimitive.Viewport \n className={`cuadra-w-full cuadra-bg-inherit cuadra-relative scrollbar-thin ${\n showCenteredWelcome \n ? 'cuadra-flex-none' \n : 'cuadra-flex-1 cuadra-min-h-0 cuadra-overflow-y-auto'\n }`}\n style={{ paddingTop: showCenteredWelcome ? '0' : (extraTopPadding || ((showChatHeader || (models && models.length > 0)) ? '0.5rem' : '2rem')) }}\n >\n {isLoadingMessages ? (\n <div className=\"cuadra-absolute cuadra-inset-0 cuadra-flex cuadra-items-center cuadra-justify-center\">\n <div className=\"cuadra-flex cuadra-flex-col cuadra-items-center cuadra-gap-3\">\n <Loader2 className=\"cuadra-h-5 cuadra-w-5 cuadra-animate-spin cuadra-text-muted-foreground\" />\n <p className=\"cuadra-text-sm cuadra-text-muted-foreground font-brand\">Loading messages...</p>\n </div>\n </div>\n ) : (\n <div className=\"cuadra-max-w-3xl cuadra-mx-auto cuadra-w-full cuadra-px-4\">\n <ThreadPrimitive.Empty>\n <WelcomeScreen title={welcomeTitle} subtitle={welcomeSubtitle} />\n </ThreadPrimitive.Empty>\n <ThreadPrimitive.Messages\n components={{\n UserMessage: UserMessage,\n AssistantMessage: AssistantMessage,\n }}\n />\n </div>\n )}\n </ThreadPrimitive.Viewport>\n\n {/* Composer area — Claude-like elevated surface with toolbar */}\n <div \n className={`cuadra-w-full cuadra-px-4 cuadra-pt-2 ${showCenteredWelcome ? 'cuadra-pb-2' : 'cuadra-pb-4'}`}\n style={safeArea ? { paddingBottom: 'calc(env(safe-area-inset-bottom) + 1rem)' } : undefined}\n >\n <div className=\"cuadra-max-w-3xl cuadra-mx-auto cuadra-w-full\">\n {/* Attachment validation error toast */}\n {attachmentError && (\n <div\n className=\"cuadra-flex cuadra-items-center cuadra-gap-2 cuadra-mb-2 cuadra-px-3 cuadra-py-2 cuadra-rounded-lg cuadra-text-xs font-brand cuadra-animate-in cuadra-fade-in cuadra-slide-in-from-bottom-2\"\n style={{\n backgroundColor: 'var(--cuadra-muted)',\n color: 'var(--cuadra-foreground)',\n opacity: 0.85,\n }}\n >\n <AlertCircle className=\"cuadra-h-3.5 cuadra-w-3.5 cuadra-flex-shrink-0 cuadra-text-muted-foreground\" />\n <span className=\"cuadra-flex-1 cuadra-text-muted-foreground\">{attachmentError.message}</span>\n <button\n type=\"button\"\n onClick={() => attachmentErrorStore.clear()}\n className=\"cuadra-flex-shrink-0 cuadra-p-0.5 cuadra-rounded hover:cuadra-bg-black/5 cuadra-transition-colors\"\n aria-label=\"Dismiss\"\n >\n <X className=\"cuadra-h-3 cuadra-w-3 cuadra-text-muted-foreground\" />\n </button>\n </div>\n )}\n {/* Composer attachments preview */}\n {enableAttachments && (\n <ComposerPrimitive.Attachments\n components={{\n Attachment: () => {\n const attachment = useAuiState(({ attachment }) => attachment);\n if (!attachment) return null;\n \n return (\n <div className=\"cuadra-relative cuadra-inline-flex cuadra-items-center cuadra-gap-2 cuadra-mb-2 cuadra-px-3 cuadra-py-1.5 cuadra-rounded-lg cuadra-transition-colors\"\n style={{ \n backgroundColor: 'var(--cuadra-muted)',\n border: '1px solid rgba(0,0,0,0.06)',\n }}\n >\n <Paperclip className=\"cuadra-h-3.5 cuadra-w-3.5 cuadra-text-muted-foreground\" strokeWidth={1.5} />\n <span className=\"cuadra-text-xs cuadra-text-foreground font-brand\">\n {attachment.name || 'Attachment'}\n </span>\n <AttachmentPrimitive.Remove asChild>\n <button\n type=\"button\"\n className=\"cuadra-flex cuadra-items-center cuadra-justify-center cuadra-size-4 cuadra-rounded-full hover:cuadra-bg-background/60 cuadra-transition-colors cuadra-ml-1\"\n aria-label=\"Remove attachment\"\n >\n <X className=\"cuadra-h-3 cuadra-w-3 cuadra-text-muted-foreground\" />\n </button>\n </AttachmentPrimitive.Remove>\n </div>\n );\n },\n }}\n />\n )}\n <ComposerPrimitive.Root \n className=\"cuadra-relative cuadra-w-full cuadra-rounded-2xl cuadra-transition-shadow cuadra-duration-300 cuadra-flex cuadra-flex-col\"\n style={{\n backgroundColor: 'var(--cuadra-composer)',\n boxShadow: 'var(--cuadra-composer-shadow)',\n border: composerBorder ? '1px solid hsl(var(--cuadra-border))' : 'none',\n }}\n >\n {/* Input area */}\n <ComposerPrimitive.Input\n asChild\n >\n <textarea\n rows={1}\n placeholder={inputPlaceholder}\n className=\"cuadra-flex cuadra-min-h-[44px] cuadra-w-full cuadra-rounded-t-2xl cuadra-border-0 cuadra-bg-transparent cuadra-px-4 cuadra-pt-3.5 cuadra-pb-1 cuadra-text-base cuadra-text-foreground placeholder:cuadra-text-muted-foreground focus-visible:cuadra-outline-none disabled:cuadra-cursor-not-allowed disabled:cuadra-opacity-50 cuadra-resize-none font-brand\"\n style={{ \n fontSize: '1rem',\n lineHeight: '1.5',\n }}\n />\n </ComposerPrimitive.Input>\n {/* Bottom toolbar — attach, model selector, send */}\n <div className=\"cuadra-flex cuadra-items-center cuadra-justify-between cuadra-px-3 cuadra-pb-2.5 cuadra-pt-1\">\n {/* Left side: attach button */}\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-1\">\n {enableAttachments && (\n <ComposerPrimitive.AddAttachment\n multiple\n asChild\n >\n <button\n type=\"button\"\n className=\"cuadra-group cuadra-relative cuadra-flex cuadra-items-center cuadra-justify-center cuadra-size-8 cuadra-rounded-lg hover:cuadra-bg-muted/60 cuadra-transition-colors\"\n aria-label=\"Attach files\"\n >\n <Plus className=\"cuadra-h-4 cuadra-w-4 cuadra-text-muted-foreground\" strokeWidth={1.5} />\n <span className=\"cuadra-pointer-events-none cuadra-absolute cuadra-bottom-full cuadra-left-1/2 cuadra--translate-x-1/2 cuadra-mb-1.5 cuadra-whitespace-nowrap cuadra-rounded-md cuadra-px-2.5 cuadra-py-1 cuadra-text-xs cuadra-text-foreground cuadra-opacity-0 group-hover:cuadra-opacity-100 cuadra-transition-opacity cuadra-duration-150 cuadra-delay-300 font-brand\" style={{ backgroundColor: 'var(--cuadra-popover)', boxShadow: '0 2px 8px rgba(0,0,0,0.15)' }}>\n Attach files\n </span>\n </button>\n </ComposerPrimitive.AddAttachment>\n )}\n </div>\n {/* Right side: send button */}\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-2\">\n <ComposerPrimitive.Send asChild>\n <button\n className=\"cuadra-flex cuadra-items-center cuadra-justify-center cuadra-size-8 cuadra-rounded-full cuadra-transition-all cuadra-duration-200 hover:cuadra-opacity-90 disabled:cuadra-opacity-30 disabled:cuadra-cursor-not-allowed\"\n style={{ \n backgroundColor: 'var(--cuadra-primary)',\n color: 'white',\n }}\n aria-label=\"Send message\"\n type=\"button\"\n >\n <ArrowUpIcon className=\"cuadra-h-4 cuadra-w-4\" strokeWidth={2} />\n </button>\n </ComposerPrimitive.Send>\n </div>\n </div>\n </ComposerPrimitive.Root>\n </div>\n </div>\n\n {/* Suggestion pills — below composer, only visible on empty thread */}\n {showCenteredWelcome && suggestions && suggestions.length > 0 && (\n <div className=\"cuadra-flex cuadra-flex-wrap cuadra-justify-center cuadra-gap-2 cuadra-px-4 cuadra-pb-6 cuadra-pt-1 cuadra-max-w-3xl cuadra-mx-auto cuadra-w-full\">\n {suggestions.map((suggestion, index) => (\n <button\n key={`suggestion-${index}`}\n type=\"button\"\n onClick={() => handleSuggestionClick(suggestion)}\n disabled={isProcessingPreMade}\n className=\"cuadra-px-3.5 cuadra-py-2 cuadra-rounded-full cuadra-text-sm cuadra-text-muted-foreground hover:cuadra-text-foreground cuadra-transition-all cuadra-duration-200 font-brand cuadra-cursor-pointer disabled:cuadra-opacity-50 disabled:cuadra-cursor-not-allowed\"\n style={{ \n backgroundColor: 'var(--cuadra-composer)',\n boxShadow: '0 0 0 1px rgba(0,0,0,0.06)',\n }}\n >\n {suggestion.prompt}\n </button>\n ))}\n </div>\n )}\n </ComposerPrimitive.AttachmentDropzone>\n </ThreadPrimitive.Root>\n );\n});\n\n/**\n * Welcome screen component — title + optional subtitle (suggestions are below the composer)\n */\nfunction WelcomeScreen({ title, subtitle }: { title: string; subtitle?: string }) {\n return (\n <div className=\"cuadra-flex cuadra-flex-col cuadra-items-center cuadra-text-center cuadra-py-6\">\n <h2\n className=\"cuadra-text-2xl md:cuadra-text-4xl cuadra-font-normal cuadra-text-foreground/80 cuadra-tracking-tight\"\n style={{ fontFamily: \"var(--cuadra-font-logo), Outfit, -apple-system, sans-serif\" }}\n >\n {title}\n </h2>\n {subtitle && (\n <p className=\"cuadra-text-sm md:cuadra-text-base cuadra-text-muted-foreground cuadra-mt-2 md:cuadra-mt-3 cuadra-max-w-lg cuadra-whitespace-pre-line\">\n {subtitle}\n </p>\n )}\n </div>\n );\n}\n\n/**\n * User message component — Claude-like right-aligned bubble\n */\nfunction UserMessage() {\n return (\n <MessagePrimitive.Root className=\"cuadra-flex cuadra-w-full cuadra-justify-end cuadra-py-2.5\">\n <div className=\"cuadra-flex cuadra-flex-col cuadra-items-end cuadra-gap-2 cuadra-max-w-[80%]\">\n {/* Display attachments */}\n <MessagePrimitive.Attachments\n components={{\n Attachment: () => {\n const attachment = useAuiState(({ attachment }) => attachment);\n if (!attachment) return null;\n \n const isImage = attachment.type === 'image';\n const file = attachment.file;\n const imageContent = attachment.content?.find((c): c is { type: 'image'; image: string } => c.type === 'image');\n const src = file ? URL.createObjectURL(file) : imageContent?.image;\n \n return (\n <div className=\"cuadra-mb-1\">\n {isImage && src ? (\n <img\n src={src}\n alt={attachment.name || 'Attachment'}\n className=\"cuadra-rounded-2xl cuadra-max-h-64 cuadra-max-w-full cuadra-object-contain\"\n style={{ border: '1px solid rgba(0,0,0,0.06)' }}\n />\n ) : (\n <div \n className=\"cuadra-flex cuadra-items-center cuadra-gap-2 cuadra-rounded-xl cuadra-px-3 cuadra-py-2\"\n style={{ backgroundColor: 'var(--cuadra-muted)', border: '1px solid rgba(0,0,0,0.06)' }}\n >\n <Paperclip className=\"cuadra-h-3.5 cuadra-w-3.5 cuadra-text-muted-foreground\" />\n <span className=\"cuadra-text-sm cuadra-text-foreground font-brand\">\n {attachment.name || 'Attachment'}\n </span>\n </div>\n )}\n </div>\n );\n },\n }}\n />\n <div \n className=\"cuadra-break-words cuadra-text-foreground font-brand\"\n style={{ \n borderRadius: '1.25rem',\n backgroundColor: 'var(--cuadra-muted)',\n paddingLeft: '1rem',\n paddingRight: '1rem',\n paddingTop: '0.625rem',\n paddingBottom: '0.625rem',\n wordBreak: 'break-word',\n fontSize: '0.9375rem',\n lineHeight: '1.625',\n }}\n >\n <MessagePrimitive.Parts\n components={{\n Text: ({ text }) => <MarkdownText content={text || ''} />,\n }}\n />\n </div>\n </div>\n </MessagePrimitive.Root>\n );\n}\n\n/**\n * Inline Reasoning component for MessagePrimitive.Parts\n * - While streaming: Shows actual reasoning content with shimmer effect\n * - After completion: Collapses into a minimal \"Reasoning\" toggle (no borders, no icons)\n */\nfunction Reasoning({ text }: { text: string }) {\n const [isOpen, setIsOpen] = useState(false);\n const isStreaming = useAuiState(({ message }) => message?.status?.type === \"running\");\n \n if (!text) return null;\n \n // While streaming: show actual reasoning content with shimmer\n if (isStreaming) {\n return (\n <div className=\"cuadra-mb-3 cuadra-py-1\">\n <div className=\"cuadra-text-sm cuadra-text-muted-foreground cuadra-leading-relaxed font-brand cuadra-animate-text-shimmer\">\n {text}\n </div>\n </div>\n );\n }\n \n // After completion: minimal collapsible\n return (\n <div className=\"cuadra-mb-3\">\n <button\n type=\"button\"\n onClick={() => setIsOpen(!isOpen)}\n className=\"cuadra-flex cuadra-items-center cuadra-gap-1.5 cuadra-py-1 cuadra-text-sm cuadra-text-muted-foreground hover:cuadra-text-foreground cuadra-transition-colors font-brand\"\n >\n <ChevronDown \n className={`cuadra-h-3.5 cuadra-w-3.5 cuadra-shrink-0 cuadra-transition-transform cuadra-duration-200 ${isOpen ? '' : 'cuadra--rotate-90'}`} \n />\n <span>Reasoning</span>\n </button>\n \n <div \n className={`cuadra-overflow-hidden cuadra-transition-all cuadra-duration-200 cuadra-ease-out ${\n isOpen ? 'cuadra-max-h-[2000px] cuadra-opacity-100' : 'cuadra-max-h-0 cuadra-opacity-0'\n }`}\n >\n <div className=\"cuadra-pl-5 cuadra-pt-2 cuadra-text-sm cuadra-text-muted-foreground cuadra-leading-relaxed font-brand\">\n <MarkdownText content={text} />\n </div>\n </div>\n </div>\n );\n}\n\n/**\n * Thinking indicator shown while assistant is generating response\n */\nfunction ThinkingIndicator() {\n return (\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-2.5 cuadra-text-muted-foreground cuadra-text-sm font-brand cuadra-py-3\">\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-1\">\n <span className=\"cuadra-inline-block cuadra-w-1.5 cuadra-h-1.5 cuadra-rounded-full cuadra-animate-bounce\" style={{ animationDelay: '0ms', backgroundColor: 'var(--cuadra-primary)', opacity: 0.7 }} />\n <span className=\"cuadra-inline-block cuadra-w-1.5 cuadra-h-1.5 cuadra-rounded-full cuadra-animate-bounce\" style={{ animationDelay: '150ms', backgroundColor: 'var(--cuadra-primary)', opacity: 0.5 }} />\n <span className=\"cuadra-inline-block cuadra-w-1.5 cuadra-h-1.5 cuadra-rounded-full cuadra-animate-bounce\" style={{ animationDelay: '300ms', backgroundColor: 'var(--cuadra-primary)', opacity: 0.3 }} />\n </div>\n <span>Thinking...</span>\n </div>\n );\n}\n\n/**\n * Assistant message component with native reasoning and custom sources\n */\nfunction AssistantMessage() {\n // Check if message is currently being generated (status.type === \"running\")\n // Also check if there's no content yet to show the thinking indicator\n const isInProgress = useAuiState(({ message }) => message?.status?.type === \"running\");\n const hasContent = useAuiState(({ message }) => {\n const content = message?.content;\n if (!content || !Array.isArray(content)) return false;\n // Check if any content part has actual text\n return content.some((part) => {\n if (part.type === 'text' && part.text && part.text.length > 0) return true;\n if (part.type === 'reasoning' && part.text && part.text.length > 0) return true;\n return false;\n });\n });\n\n // Capture the message ID when streaming actually starts (has content),\n // not at mount time (which is too early - shows thinking indicator before API call)\n // Use a ref to track if we've captured the ID and state to store it\n const hasCapturedRef = useRef(false);\n const [capturedMessageId, setCapturedMessageId] = useState<string | null>(null);\n \n // Capture the message ID when streaming starts (when we have content)\n useEffect(() => {\n if (isInProgress && hasContent && !hasCapturedRef.current) {\n const currentId = streamingMetadataStore.getCurrentMessageId();\n if (currentId) {\n hasCapturedRef.current = true;\n setCapturedMessageId(currentId);\n }\n }\n }, [isInProgress, hasContent]);\n \n // Subscribe to store changes and look up this message's sources\n // Only show sources if they were included in this specific message's response\n const sources = useSyncExternalStore(\n (callback) => streamingMetadataStore.subscribe(callback),\n () => {\n // If we have a captured message ID, use it\n if (capturedMessageId) {\n const msgSources = streamingMetadataStore.getForMessage(capturedMessageId);\n return msgSources.sources?.length ? msgSources.sources : null;\n }\n // Only check current streaming message if THIS message is currently in progress\n // This prevents completed messages from showing sources from new streaming messages\n if (isInProgress && hasContent) {\n const currentId = streamingMetadataStore.getCurrentMessageId();\n if (currentId) {\n const currentSources = streamingMetadataStore.getForMessage(currentId);\n return currentSources.sources?.length ? currentSources.sources : null;\n }\n }\n return null;\n },\n () => null\n );\n\n return (\n <MessagePrimitive.Root className=\"cuadra-flex cuadra-w-full cuadra-justify-start cuadra-py-2.5\">\n <div \n className=\"cuadra-text-foreground cuadra-max-w-full cuadra-break-words font-brand\"\n style={{ \n fontSize: '0.9375rem',\n lineHeight: '1.75',\n }}\n >\n {/* Content parts - includes native reasoning and text */}\n <MessagePrimitive.Parts\n components={{\n Text: ({ text }) => <MarkdownText content={text || ''} />,\n Reasoning: ({ text }) => <Reasoning text={text || ''} />,\n }}\n />\n \n {/* Show thinking indicator while message is being generated (no content yet) */}\n {isInProgress && !hasContent && <ThinkingIndicator />}\n \n {/* Source citations - custom component for RAG sources */}\n {sources && sources.length > 0 && (\n <SourceCitations sources={sources} />\n )}\n </div>\n </MessagePrimitive.Root>\n );\n}\n\n// ─── Icon Tooltip ────────────────────────────────────────────────\n\nfunction IconTooltip({ label, children }: { label: string; children: React.ReactNode }) {\n return (\n <span className=\"cuadra-relative cuadra-inline-flex cuadra-group/tip\">\n {children}\n <span\n className=\"cuadra-absolute cuadra-bottom-full cuadra-left-1/2 cuadra-mb-1.5 cuadra-hidden group-hover/tip:cuadra-flex cuadra-items-center cuadra-whitespace-nowrap cuadra-rounded-md cuadra-px-2 cuadra-py-1 cuadra-text-[10px] cuadra-pointer-events-none cuadra-z-50 font-brand\"\n style={{\n backgroundColor: 'var(--cuadra-popover)',\n color: 'var(--cuadra-foreground)',\n boxShadow: '0 2px 8px rgba(0,0,0,0.12), 0 0 0 1px rgba(0,0,0,0.04)',\n transform: 'translateX(-50%)',\n }}\n >\n {label}\n </span>\n </span>\n );\n}\n\n// ─── Model Selector ──────────────────────────────────────────────\n\ninterface ModelSelectorProps {\n models: ModelOption[];\n selectedModelId?: string;\n onModelChange?: (modelId: string) => void;\n}\n\nfunction ModelSelector({ models, selectedModelId, onModelChange }: ModelSelectorProps) {\n const [isOpen, setIsOpen] = useState(false);\n const selectedModel = models.find((m) => m.id === selectedModelId) || models[0];\n\n if (!selectedModel) return null;\n\n const handleModelSelect = (modelId: string) => {\n setIsOpen(false);\n if (modelId !== selectedModelId) {\n onModelChange?.(modelId);\n }\n };\n\n return (\n <div className=\"cuadra-relative\">\n <button\n type=\"button\"\n onClick={() => setIsOpen(!isOpen)}\n className=\"cuadra-flex cuadra-items-center cuadra-gap-1.5 cuadra-px-2.5 cuadra-py-1.5 cuadra-rounded-lg cuadra-text-sm md:cuadra-text-base cuadra-text-muted-foreground hover:cuadra-text-foreground cuadra-transition-colors font-brand\"\n >\n <span>{selectedModel.name}</span>\n <ChevronDown className=\"cuadra-h-3.5 cuadra-w-3.5\" />\n </button>\n\n {isOpen && (\n <>\n {/* Backdrop — desktop only */}\n <div \n className=\"cuadra-hidden md:cuadra-block cuadra-fixed cuadra-inset-0 cuadra-z-40\" \n onClick={() => setIsOpen(false)} \n />\n\n {/* Mobile: fullscreen overlay */}\n <div\n className=\"cuadra-fixed cuadra-inset-0 cuadra-z-50 cuadra-flex cuadra-flex-col md:cuadra-hidden\"\n style={{ backgroundColor: 'var(--cuadra-popover)' }}\n >\n {/* Mobile close button */}\n <div className=\"cuadra-flex cuadra-items-center cuadra-justify-end cuadra-px-4 cuadra-py-3 cuadra-min-h-[48px]\">\n <button\n type=\"button\"\n onClick={() => setIsOpen(false)}\n className=\"cuadra-flex cuadra-items-center cuadra-justify-center cuadra-p-1.5 cuadra-rounded-lg cuadra-text-muted-foreground hover:cuadra-bg-muted/60 cuadra-transition-colors\"\n >\n <X className=\"cuadra-h-5 cuadra-w-5\" />\n </button>\n </div>\n\n <div className=\"cuadra-flex-1 cuadra-min-h-0 cuadra-overflow-y-auto cuadra-py-1\">\n {models.map((model) => (\n <button\n key={model.id}\n type=\"button\"\n className=\"cuadra-w-full cuadra-flex cuadra-items-center cuadra-justify-between cuadra-gap-3 cuadra-px-4 cuadra-py-3 cuadra-text-sm cuadra-text-left hover:cuadra-bg-muted/60 cuadra-transition-colors cuadra-border-b cuadra-border-border/40 last:cuadra-border-b-0 font-brand\"\n onClick={() => handleModelSelect(model.id)}\n >\n <div className=\"cuadra-flex cuadra-flex-col cuadra-min-w-0\">\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-1.5\">\n <span className=\"cuadra-text-foreground cuadra-truncate\">{model.name}</span>\n {model.supportsReasoning && (\n <IconTooltip label=\"Reasoning\"><Brain className=\"cuadra-h-3 cuadra-w-3 cuadra-text-muted-foreground cuadra-flex-shrink-0\" /></IconTooltip>\n )}\n {model.supportsVision && (\n <IconTooltip label=\"Vision\"><Eye className=\"cuadra-h-3 cuadra-w-3 cuadra-text-muted-foreground cuadra-flex-shrink-0\" /></IconTooltip>\n )}\n </div>\n {model.description && (\n <span className=\"cuadra-text-xs cuadra-text-muted-foreground\">{model.description}</span>\n )}\n </div>\n {model.id === selectedModelId && (\n <Check className=\"cuadra-h-4 cuadra-w-4 cuadra-text-primary cuadra-flex-shrink-0\" />\n )}\n </button>\n ))}\n </div>\n </div>\n\n {/* Desktop: positioned dropdown (opens downward, centered) */}\n <div \n className=\"cuadra-hidden md:cuadra-block cuadra-absolute cuadra-top-full cuadra-left-1/2 cuadra--translate-x-1/2 cuadra-mt-2 cuadra-z-50 cuadra-rounded-xl cuadra-min-w-[200px] cuadra-py-1\"\n style={{ \n backgroundColor: 'var(--cuadra-popover)', \n boxShadow: '0 4px 24px rgba(0,0,0,0.12), 0 0 0 1px rgba(0,0,0,0.04)' \n }}\n >\n {models.map((model) => (\n <button\n key={model.id}\n type=\"button\"\n className=\"cuadra-w-full cuadra-flex cuadra-items-center cuadra-justify-between cuadra-gap-3 cuadra-px-3 cuadra-py-2 cuadra-text-sm cuadra-text-left hover:cuadra-bg-muted/60 cuadra-transition-colors font-brand\"\n onClick={() => handleModelSelect(model.id)}\n >\n <div className=\"cuadra-flex cuadra-flex-col cuadra-min-w-0\">\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-1.5\">\n <span className=\"cuadra-text-foreground cuadra-truncate\">{model.name}</span>\n {model.supportsReasoning && (\n <IconTooltip label=\"Reasoning\"><Brain className=\"cuadra-h-3 cuadra-w-3 cuadra-text-muted-foreground cuadra-flex-shrink-0\" /></IconTooltip>\n )}\n {model.supportsVision && (\n <IconTooltip label=\"Vision\"><Eye className=\"cuadra-h-3 cuadra-w-3 cuadra-text-muted-foreground cuadra-flex-shrink-0\" /></IconTooltip>\n )}\n </div>\n {model.description && (\n <span className=\"cuadra-text-xs cuadra-text-muted-foreground\">{model.description}</span>\n )}\n </div>\n {model.id === selectedModelId && (\n <Check className=\"cuadra-h-4 cuadra-w-4 cuadra-text-primary cuadra-flex-shrink-0\" />\n )}\n </button>\n ))}\n </div>\n </>\n )}\n </div>\n );\n}\n\n// ─── Chat Header with Thread List Dropdown ───────────────────────\n\nfunction ChatHeader() {\n const threadTitle = useAuiState(({ threadListItem }) => threadListItem?.title ?? null);\n const [isDropdownOpen, setIsDropdownOpen] = useState(false);\n const displayTitle = threadTitle || 'Untitled chat';\n\n return (\n <div className=\"cuadra-relative\">\n {/* Desktop: title + chevron */}\n <button\n type=\"button\"\n onClick={() => setIsDropdownOpen(!isDropdownOpen)}\n className=\"cuadra-hidden md:cuadra-flex cuadra-items-center cuadra-gap-1 cuadra-text-sm cuadra-text-foreground hover:cuadra-bg-muted/60 cuadra-rounded-lg cuadra-px-2 cuadra-py-1.5 cuadra-transition-colors font-brand\"\n >\n <span className=\"cuadra-font-medium cuadra-truncate cuadra-max-w-[200px]\">{displayTitle}</span>\n <ChevronDown className={`cuadra-h-3.5 cuadra-w-3.5 cuadra-text-muted-foreground cuadra-flex-shrink-0 cuadra-transition-transform cuadra-duration-200 ${isDropdownOpen ? 'cuadra-rotate-180' : ''}`} />\n </button>\n\n {/* Mobile: icon only */}\n <button\n type=\"button\"\n onClick={() => setIsDropdownOpen(!isDropdownOpen)}\n className=\"cuadra-flex md:cuadra-hidden cuadra-items-center cuadra-justify-center cuadra-text-foreground hover:cuadra-bg-muted/60 cuadra-rounded-lg cuadra-p-2 cuadra-transition-colors\"\n >\n <MessageSquare className=\"cuadra-h-5 cuadra-w-5\" />\n </button>\n\n {isDropdownOpen && (\n <ThreadListDropdown onClose={() => setIsDropdownOpen(false)} />\n )}\n </div>\n );\n}\n\n// ─── Thread List Dropdown ────────────────────────────────────────\n\nfunction ThreadListDropdown({ onClose }: { onClose: () => void }) {\n return (\n <>\n {/* Backdrop — visible on desktop only (mobile uses fullscreen panel) */}\n <div className=\"cuadra-hidden md:cuadra-block cuadra-fixed cuadra-inset-0 cuadra-z-40\" onClick={onClose} />\n\n {/* Mobile: fullscreen overlay */}\n <div\n className=\"cuadra-fixed cuadra-inset-0 cuadra-z-50 cuadra-flex cuadra-flex-col md:cuadra-hidden\"\n style={{ backgroundColor: 'var(--cuadra-popover)' }}\n >\n {/* Mobile close button */}\n <div className=\"cuadra-flex cuadra-items-center cuadra-justify-end cuadra-px-4 cuadra-py-3 cuadra-min-h-[48px]\">\n <button\n type=\"button\"\n onClick={onClose}\n className=\"cuadra-flex cuadra-items-center cuadra-justify-center cuadra-p-1.5 cuadra-rounded-lg cuadra-text-muted-foreground hover:cuadra-bg-muted/60 cuadra-transition-colors\"\n >\n <X className=\"cuadra-h-5 cuadra-w-5\" />\n </button>\n </div>\n\n <ThreadListPrimitive.Root className=\"cuadra-flex cuadra-flex-col cuadra-flex-1 cuadra-min-h-0\">\n {/* New Chat option */}\n <div className=\"cuadra-px-3 cuadra-py-2\">\n <ThreadListPrimitive.New\n className=\"cuadra-w-full cuadra-flex cuadra-items-center cuadra-gap-2.5 cuadra-px-3 cuadra-py-2.5 cuadra-rounded-lg cuadra-text-sm cuadra-text-foreground hover:cuadra-bg-muted cuadra-transition-colors cuadra-cursor-pointer font-brand\"\n onClick={onClose}\n >\n <Plus className=\"cuadra-h-4 cuadra-w-4 cuadra-text-muted-foreground\" />\n New Chat\n </ThreadListPrimitive.New>\n </div>\n\n {/* Thread list — scrollable, fills remaining space */}\n <DropdownThreadList onSelect={onClose} fullHeight />\n </ThreadListPrimitive.Root>\n </div>\n\n {/* Desktop: positioned dropdown (unchanged) */}\n <div\n className=\"cuadra-hidden md:cuadra-flex cuadra-absolute cuadra-left-0 cuadra-top-full cuadra-mt-1 cuadra-z-50 cuadra-w-[300px] cuadra-rounded-xl cuadra-overflow-hidden cuadra-flex-col\"\n style={{\n backgroundColor: 'var(--cuadra-popover)',\n boxShadow: '0 4px 24px rgba(0,0,0,0.12), 0 0 0 1px rgba(0,0,0,0.04)',\n }}\n >\n <ThreadListPrimitive.Root className=\"cuadra-flex cuadra-flex-col\">\n {/* New Chat option */}\n <div className=\"cuadra-px-2 cuadra-py-1.5\">\n <ThreadListPrimitive.New\n className=\"cuadra-w-full cuadra-flex cuadra-items-center cuadra-gap-2.5 cuadra-px-3 cuadra-py-2 cuadra-rounded-lg cuadra-text-sm cuadra-text-foreground hover:cuadra-bg-muted cuadra-transition-colors cuadra-cursor-pointer font-brand\"\n onClick={onClose}\n >\n <Plus className=\"cuadra-h-4 cuadra-w-4 cuadra-text-muted-foreground\" />\n New Chat\n </ThreadListPrimitive.New>\n </div>\n\n {/* Thread list — separator + scrollable area only when threads exist */}\n <DropdownThreadList onSelect={onClose} />\n </ThreadListPrimitive.Root>\n </div>\n </>\n );\n}\n\n// ─── Dropdown Thread List (separator + scrollable area, hidden when empty) ───\n\nfunction DropdownThreadList({ onSelect, fullHeight }: { onSelect?: () => void; fullHeight?: boolean }) {\n const threadIds = useAuiState(({ threads }) => threads?.threadIds ?? []);\n const newThreadId = useAuiState(({ threads }) => threads?.newThreadId);\n const hasChats = threadIds.some((id: string) => id !== newThreadId);\n\n if (!hasChats) return null;\n\n return (\n <>\n <div className=\"cuadra-mx-3 cuadra-my-0.5\" style={{ borderTop: '1px solid rgba(0,0,0,0.06)' }} />\n <div\n className={`cuadra-overflow-y-auto scrollbar-thin ${\n fullHeight\n ? 'cuadra-flex-1 cuadra-min-h-0 cuadra-px-3 cuadra-pb-3 cuadra-flex cuadra-flex-col cuadra-gap-2'\n : 'cuadra-px-2 cuadra-pb-1.5'\n }`}\n style={fullHeight ? undefined : { maxHeight: '340px' }}\n >\n <ThreadListPrimitive.Items\n components={{\n ThreadListItem: (props) => <DropdownThreadItem {...props} onSelect={onSelect} fullHeight={fullHeight} />,\n }}\n />\n </div>\n </>\n );\n}\n\n// ─── Dropdown Thread Item ────────────────────────────────────────\n\nfunction DropdownThreadItem({ onSelect, fullHeight }: { onSelect?: () => void; fullHeight?: boolean }) {\n const threadItem = useThreadListItem();\n const itemRuntime = useThreadListItemRuntime();\n const assistantRuntime = useAssistantRuntime();\n const currentThread = useThread((state) => state.threadId);\n const isActive = threadItem?.id === currentThread;\n const [isHovered, setIsHovered] = useState(false);\n const [isRenaming, setIsRenaming] = useState(false);\n const [nextTitle, setNextTitle] = useState(threadItem?.title || '');\n const [busy, setBusy] = useState(false);\n\n const handleRename = async () => {\n if (!threadItem?.remoteId || !nextTitle || nextTitle === threadItem.title) {\n setIsRenaming(false);\n return;\n }\n try {\n setBusy(true);\n // Use the runtime's rename so internal state updates and the UI\n // reflects the new title without needing a page refresh.\n await itemRuntime.rename(nextTitle);\n } catch {\n // Silently handle\n } finally {\n setBusy(false);\n setIsRenaming(false);\n }\n };\n\n const handleDelete = async (e: React.MouseEvent) => {\n e.stopPropagation();\n\n try {\n // Delete via the item runtime first — this calls the adapter and\n // removes the thread from the runtime's internal list.\n await itemRuntime.delete();\n\n // If this was the thread we were viewing, switch away now that\n // the runtime state is clean (avoids stale resource lookup).\n if (isActive) {\n await assistantRuntime.switchToNewThread();\n }\n } catch (err) {\n // eslint-disable-next-line no-console -- Intentional: surface delete failures for debugging\n console.error('[CuadraChat] Failed to delete thread:', err);\n }\n };\n\n useEffect(() => {\n if (threadItem?.title && !isRenaming) setNextTitle(threadItem.title);\n }, [threadItem?.title, isRenaming]);\n\n if (!threadItem) return null;\n\n // Mobile fullscreen: card style with three-dot menu\n if (fullHeight) {\n return (\n <ThreadListItemPrimitive.Root\n className={`cuadra-relative cuadra-flex cuadra-items-center cuadra-rounded-xl cuadra-transition-colors cuadra-cursor-pointer ${\n isActive\n ? 'cuadra-bg-primary/10 cuadra-ring-1 cuadra-ring-primary/20'\n : 'cuadra-bg-muted/40'\n }`}\n >\n {isRenaming ? (\n <div\n className=\"cuadra-flex-1 cuadra-px-4 cuadra-py-3.5 cuadra-min-w-0\"\n onClick={(e) => e.stopPropagation()}\n onMouseDown={(e) => e.stopPropagation()}\n onKeyDown={(e) => e.stopPropagation()}\n onKeyUp={(e) => e.stopPropagation()}\n onKeyPress={(e) => e.stopPropagation()}\n >\n <input\n className=\"cuadra-w-full cuadra-bg-transparent cuadra-border-0 cuadra-border-b cuadra-text-sm cuadra-text-foreground cuadra-py-0.5 cuadra-px-0 cuadra-outline-none font-brand\"\n style={{ borderBottomColor: 'var(--cuadra-primary)', borderBottomWidth: '1.5px' }}\n value={nextTitle}\n onChange={(e) => setNextTitle(e.target.value)}\n onKeyDown={(e) => {\n e.stopPropagation();\n if (e.key === 'Enter') void handleRename();\n if (e.key === 'Escape') { setIsRenaming(false); setNextTitle(threadItem.title || ''); }\n }}\n onBlur={() => void handleRename()}\n disabled={busy}\n autoFocus\n />\n </div>\n ) : (\n <ThreadListItemPrimitive.Trigger\n className=\"cuadra-flex-1 cuadra-px-4 cuadra-py-3.5 cuadra-min-w-0 cuadra-cursor-pointer\"\n onClick={() => onSelect?.()}\n >\n <span className=\"cuadra-text-sm cuadra-text-foreground cuadra-truncate cuadra-block cuadra-text-left font-brand\">\n {threadItem.title || 'Chat'}\n </span>\n </ThreadListItemPrimitive.Trigger>\n )}\n\n {/* Three-dot menu on mobile */}\n {!isRenaming && (\n <MobileThreadActions\n onRename={() => {\n setIsRenaming(true);\n setNextTitle(threadItem.title || 'Chat');\n }}\n onDelete={handleDelete}\n />\n )}\n </ThreadListItemPrimitive.Root>\n );\n }\n\n // Desktop: compact list with hover-to-reveal actions\n return (\n <ThreadListItemPrimitive.Root\n className={`cuadra-relative cuadra-flex cuadra-items-center cuadra-rounded-lg cuadra-transition-colors cuadra-cursor-pointer cuadra-border-b cuadra-border-border/40 last:cuadra-border-b-0 ${\n isActive ? 'cuadra-bg-muted' : 'hover:cuadra-bg-muted/60'\n }`}\n onMouseEnter={() => setIsHovered(true)}\n onMouseLeave={() => setIsHovered(false)}\n >\n {isRenaming ? (\n <div\n className=\"cuadra-flex-1 cuadra-px-3 cuadra-py-2 cuadra-min-w-0\"\n onClick={(e) => e.stopPropagation()}\n onMouseDown={(e) => e.stopPropagation()}\n onKeyDown={(e) => e.stopPropagation()}\n onKeyUp={(e) => e.stopPropagation()}\n onKeyPress={(e) => e.stopPropagation()}\n >\n <input\n className=\"cuadra-w-full cuadra-bg-transparent cuadra-border-0 cuadra-border-b cuadra-text-sm cuadra-text-foreground cuadra-py-0.5 cuadra-px-0 cuadra-outline-none font-brand\"\n style={{ borderBottomColor: 'var(--cuadra-primary)', borderBottomWidth: '1.5px' }}\n value={nextTitle}\n onChange={(e) => setNextTitle(e.target.value)}\n onKeyDown={(e) => {\n e.stopPropagation();\n if (e.key === 'Enter') void handleRename();\n if (e.key === 'Escape') { setIsRenaming(false); setNextTitle(threadItem.title || ''); }\n }}\n onBlur={() => void handleRename()}\n disabled={busy}\n autoFocus\n />\n </div>\n ) : (\n <ThreadListItemPrimitive.Trigger\n className=\"cuadra-flex-1 cuadra-px-3 cuadra-py-2 cuadra-min-w-0 cuadra-cursor-pointer\"\n onClick={() => onSelect?.()}\n >\n <span className=\"cuadra-text-sm cuadra-text-foreground cuadra-truncate cuadra-block cuadra-text-left font-brand\">\n {threadItem.title || 'Chat'}\n </span>\n </ThreadListItemPrimitive.Trigger>\n )}\n\n {/* Hover actions: rename + delete */}\n {(isHovered || isActive) && !isRenaming && (\n <div className=\"cuadra-flex cuadra-items-center cuadra-gap-0.5 cuadra-mr-1.5 cuadra-flex-shrink-0\">\n <button\n type=\"button\"\n className=\"cuadra-p-1 cuadra-rounded hover:cuadra-bg-background/60 cuadra-transition-colors\"\n onClick={(e) => {\n e.stopPropagation();\n setIsRenaming(true);\n setNextTitle(threadItem.title || 'Chat');\n }}\n title=\"Rename\"\n >\n <Pencil className=\"cuadra-h-3 cuadra-w-3 cuadra-text-muted-foreground\" />\n </button>\n <button\n type=\"button\"\n className=\"cuadra-p-1 cuadra-rounded hover:cuadra-bg-destructive/10 cuadra-transition-colors\"\n onClick={handleDelete}\n title=\"Delete\"\n >\n <Trash2 className=\"cuadra-h-3 cuadra-w-3 cuadra-text-destructive\" />\n </button>\n </div>\n )}\n </ThreadListItemPrimitive.Root>\n );\n}\n\n// ─── Mobile Thread Actions (three-dot menu) ─────────────────────\n\nfunction MobileThreadActions({ onRename, onDelete }: { onRename: () => void; onDelete: (e: React.MouseEvent) => void }) {\n const [isOpen, setIsOpen] = useState(false);\n\n return (\n <div className=\"cuadra-relative cuadra-flex-shrink-0 cuadra-mr-2\">\n <button\n type=\"button\"\n className=\"cuadra-p-2 cuadra-rounded-lg cuadra-text-muted-foreground hover:cuadra-bg-background/60 cuadra-transition-colors\"\n onClick={(e) => {\n e.stopPropagation();\n setIsOpen(!isOpen);\n }}\n >\n <EllipsisVertical className=\"cuadra-h-4 cuadra-w-4\" />\n </button>\n\n {isOpen && (\n <>\n <div className=\"cuadra-fixed cuadra-inset-0 cuadra-z-50\" onClick={(e) => { e.stopPropagation(); setIsOpen(false); }} />\n <div\n className=\"cuadra-absolute cuadra-right-0 cuadra-top-full cuadra-mt-1 cuadra-z-[60] cuadra-w-36 cuadra-rounded-xl cuadra-py-1 cuadra-overflow-hidden\"\n style={{\n backgroundColor: 'var(--cuadra-popover)',\n boxShadow: '0 4px 24px rgba(0,0,0,0.12), 0 0 0 1px rgba(0,0,0,0.04)',\n }}\n >\n <button\n type=\"button\"\n className=\"cuadra-w-full cuadra-flex cuadra-items-center cuadra-gap-2.5 cuadra-px-3 cuadra-py-2.5 cuadra-text-sm cuadra-text-foreground hover:cuadra-bg-muted cuadra-transition-colors font-brand\"\n onClick={(e) => {\n e.stopPropagation();\n setIsOpen(false);\n onRename();\n }}\n >\n <Pencil className=\"cuadra-h-4 cuadra-w-4 cuadra-text-muted-foreground\" />\n Rename\n </button>\n <button\n type=\"button\"\n className=\"cuadra-w-full cuadra-flex cuadra-items-center cuadra-gap-2.5 cuadra-px-3 cuadra-py-2.5 cuadra-text-sm cuadra-text-destructive hover:cuadra-bg-destructive/10 cuadra-transition-colors font-brand\"\n onClick={(e) => {\n e.stopPropagation();\n setIsOpen(false);\n onDelete(e);\n }}\n >\n <Trash2 className=\"cuadra-h-4 cuadra-w-4\" />\n Delete\n </button>\n </div>\n </>\n )}\n </div>\n );\n}\n\n","import React from 'react';\n\ninterface TexturedCardProps {\n children: React.ReactNode;\n className?: string;\n paddingX?: string;\n paddingY?: string;\n /** Border width in pixels. Set to 0 for borderless. */\n borderSizePx?: number;\n style?: React.CSSProperties;\n contentStyle?: React.CSSProperties;\n /** @deprecated No longer used — kept for backward compatibility */\n rotating?: boolean;\n}\n\nexport const TexturedCard: React.FC<TexturedCardProps> = ({\n children,\n className = '',\n paddingX = 'px-0',\n paddingY = 'py-0',\n borderSizePx = 2,\n style,\n contentStyle,\n}) => {\n return (\n <div\n className={`cuadra-relative cuadra-bg-background md:cuadra-rounded-lg cuadra-overflow-hidden cuadra-flex cuadra-flex-col ${paddingX} ${paddingY} ${className}`}\n style={{\n border: borderSizePx > 0 ? `${borderSizePx}px solid var(--cuadra-primary)` : 'none',\n ...style,\n ...contentStyle,\n }}\n >\n {children}\n </div>\n );\n};\n","import React, { createContext, useContext } from 'react';\n\ninterface CuadraWidgetContextType {\n currentModelId: string | null;\n models: Array<{ id: string; name: string }>;\n modelsLoading: boolean;\n modelsError: Error | null;\n onModelChange: (modelId: string) => void;\n refetchModels: () => void;\n}\n\nconst CuadraWidgetContext = createContext<CuadraWidgetContextType | undefined>(undefined);\n\nexport const useCuadraWidgetContext = () => {\n const context = useContext(CuadraWidgetContext);\n if (!context) {\n throw new Error('useCuadraWidgetContext must be used within CuadraWidgetProvider');\n }\n return context;\n};\n\ninterface CuadraWidgetProviderProps {\n children: React.ReactNode;\n value: CuadraWidgetContextType;\n}\n\nexport const CuadraWidgetProvider: React.FC<CuadraWidgetProviderProps> = ({ children, value }) => {\n return (\n <CuadraWidgetContext.Provider value={value}>{children}</CuadraWidgetContext.Provider>\n );\n};\n\n","import { useEffect } from 'react';\n\n/**\n * Hook to inject uikit CSS styles into the DOM\n * \n * IMPORTANT: For the CSS to work, consumers must import it:\n * import '@cuadra-ai/uikit/styles';\n * \n * This hook provides a fallback that tries to inject styles if they're not already loaded.\n */\nexport function useInjectStyles(): void {\n useEffect(() => {\n // Check if styles already injected or if CSS is already loaded\n const existingStyle = document.querySelector('style[data-cuadra-uikit]');\n if (existingStyle) {\n return;\n }\n\n // Check if CSS classes are already available (styles might be loaded via import)\n const testEl = document.createElement('div');\n testEl.className = 'cuadra-flex';\n document.body.appendChild(testEl);\n const stylesLoaded = window.getComputedStyle(testEl).display === 'flex';\n document.body.removeChild(testEl);\n\n if (stylesLoaded) {\n // Styles are already loaded, no need to inject\n return;\n }\n\n // Styles not loaded - try to inject them\n // Note: This is a fallback. The proper way is to import '@cuadra-ai/uikit/styles'\n // Silently attempt to load styles - user should import '@cuadra-ai/uikit/styles'\n\n // Create a link element to load the CSS\n const link = document.createElement('link');\n link.setAttribute('data-cuadra-uikit', 'true');\n link.rel = 'stylesheet';\n link.type = 'text/css';\n \n // Try to find the CSS file\n // In development with file: protocol, try relative path\n // In production, it should be in node_modules\n const possiblePaths = [\n '/node_modules/@cuadra-ai/uikit/dist/uikit.css',\n new URL('../dist/uikit.css', import.meta.url).href,\n ];\n\n let loaded = false;\n for (const path of possiblePaths) {\n link.href = path;\n link.onload = () => {\n loaded = true;\n };\n link.onerror = () => {\n // Try next path\n };\n document.head.appendChild(link);\n if (loaded) break;\n }\n }, []);\n}\n\n","import { forwardRef, useCallback, useEffect, useImperativeHandle, useMemo, useRef, useState } from 'react';\nimport { Loader2 } from 'lucide-react';\nimport { CuadraRuntimeProvider } from './CuadraRuntimeProvider';\nimport { SimpleThread, type SimpleThreadHandle } from './SimpleThread';\nimport { TexturedCard } from '../widget/components/TexturedCard';\nimport { CuadraWidgetProvider } from '../widget/context/CuadraWidgetContext';\nimport { CuadraChatClient } from '../lib/cuadraChatClient';\nimport type { ModelPage } from '../types/cuadra';\nimport { useInjectStyles } from '../utils/useInjectStyles';\n\n/**\n * Handle for controlling WidgetContent externally\n */\nexport interface WidgetContentHandle {\n /** Send a user message programmatically (triggers normal API call) */\n sendMessage: (message: string) => void;\n /** Send a pre-made Q&A pair (instant, simulated streaming, no API call) */\n sendPreMadeQA: (question: string, answer: string, options?: { delay?: number; streamingSpeed?: number }) => void;\n /** Clear the current chat and start fresh */\n clearChat: () => void;\n}\n\n// Export the config type for use in React components\nexport interface CuadraUIKitConfig {\n // API Configuration (choose one)\n /** Cuadra API base URL (e.g., 'https://api.cuadra.ai') */\n baseUrl?: string;\n /** Proxy URL for backend-handled authentication (e.g., '/api/chat') */\n proxyUrl?: string;\n \n // Authentication (optional if using proxyUrl)\n /** Bearer token for authentication (optional if backend handles auth) */\n sessionToken?: string | null;\n \n // Chat configuration\n /** Chat mode: single thread or multi-thread */\n mode?: 'singleChat' | 'multiChat';\n /** Model selection mode */\n modelMode?: 'selector' | 'fixed';\n /** Model ID (required if modelMode=\"fixed\") */\n modelId?: string;\n /** Callback when model changes (selector mode) */\n onModelChange?: (modelId: string) => void;\n /** Create temporary chats that auto-delete */\n ephemeral?: boolean;\n /** System prompt for the assistant */\n systemPrompt?: string;\n /** Enable reasoning/thinking output for supported models (e.g., o1, o3) */\n enableReasoning?: boolean;\n \n // Thread configuration\n /** Load existing thread (multiChat mode) */\n initialThreadId?: string;\n \n // UI Configuration\n /** Language/locale for i18n (e.g., 'en', 'es', 'fr') */\n language?: string;\n /** Welcome screen title */\n welcomeTitle?: string;\n /** Welcome screen subtitle */\n welcomeSubtitle?: string;\n /** Extra top padding for thread viewport (e.g., '1rem', '2rem') */\n extraTopPadding?: string;\n /** Suggestions to show in welcome screen (with optional pre-made responses) */\n suggestions?: Array<{\n /** Suggestion prompt text */\n prompt: string;\n /** Pre-made response (optional) - if provided, response appears instantly with simulated streaming */\n response?: string;\n /** Delay before showing response in ms (overrides preMadeResponseDelay) */\n responseDelay?: number;\n }>;\n /** Placeholder text for the input field */\n inputPlaceholder?: string;\n /** Default delay before pre-made responses in ms (default: 1000) */\n preMadeResponseDelay?: number;\n /** Speed of simulated streaming in ms per word (default: 50) */\n streamingSpeed?: number;\n \n // Theme configuration\n /** Show theme toggle button */\n showThemeToggle?: boolean;\n /** Initial theme ('light' | 'dark' | 'system') */\n theme?: 'light' | 'dark' | 'system';\n \n // Container\n /** Container element ID (defaults to 'cuadra-chat') */\n containerId?: string;\n /** CSS classes to add to container element */\n containerClass?: string;\n \n // Callbacks\n /** Called when an error occurs */\n onError?: (error: Error) => void;\n /** Called when chat is created */\n onChatCreated?: (chatId: string) => void;\n /** Called when a user message is sent */\n onUserMessage?: () => void;\n /** Called when thread ID updates */\n onThreadIdUpdate?: (oldId: string, newId: string) => void;\n /** Called when logout button is clicked */\n onLogout?: () => void;\n /** Enable file attachments in the chat input */\n enableAttachments?: boolean;\n /** Whether the current model supports vision/image inputs. When false, image attachments are rejected with an error. */\n supportsVision?: boolean;\n /** Initial message to send automatically when chat loads */\n initialMessage?: string;\n /** Initial pre-made Q&A to display when chat loads */\n initialPreMadeQA?: {\n question: string;\n answer: string;\n };\n /**\n * Interceptor called before each outgoing API request.\n * Return `true` to allow the request, `false` to block it.\n * Can be async (e.g., to show a login modal).\n */\n onBeforeRequest?: () => boolean | Promise<boolean>;\n /**\n * Mock response to show when onBeforeRequest returns false.\n * If set, instead of throwing an error, this message will be streamed as a response.\n * Can be a string or a function that returns a string (called each time a request is blocked).\n * Useful for showing helpful messages like \"Complete setup before chatting\".\n */\n mockResponseOnBlocked?: string | (() => string);\n /**\n * Hide the decorative border around the chat container\n * Useful for mobile apps or when embedding in a full-screen context\n * @default false\n */\n borderless?: boolean;\n /**\n * Apply safe area insets for mobile devices (notch, home indicator)\n * When true, adds padding to header, sidebar, and input areas\n * @default false\n */\n safeArea?: boolean;\n /**\n * Organization name to display in the sidebar header\n * Used for workspace branding in mobile apps\n */\n orgName?: string;\n /**\n * Primary color for branding (hex format)\n * Overrides the default primary color CSS variable\n * @example '#3B82F6'\n */\n primaryColor?: string;\n}\n\n/**\n * Main widget content component - shared between library and widget builds\n * This component does NOT use createRoot - it's a regular React component\n */\nexport const WidgetContent = forwardRef<WidgetContentHandle, { config: CuadraUIKitConfig }>(\n function WidgetContent({ config }, ref) {\n // Inject styles when component mounts\n useInjectStyles();\n \n // Ref to SimpleThread for external control\n const simpleThreadRef = useRef<SimpleThreadHandle>(null);\n \n // Track if initial message has been sent\n const initialMessageSentRef = useRef(false);\n\n const {\n baseUrl,\n proxyUrl,\n sessionToken,\n mode = 'multiChat',\n modelMode = 'fixed',\n modelId,\n onModelChange,\n ephemeral = false,\n systemPrompt,\n enableReasoning = false,\n initialThreadId,\n welcomeTitle,\n welcomeSubtitle,\n extraTopPadding,\n suggestions,\n inputPlaceholder,\n preMadeResponseDelay = 1000,\n streamingSpeed = 50,\n // showThemeToggle kept in interface for consumer API but not used internally\n initialMessage,\n initialPreMadeQA,\n theme: initialTheme = 'system',\n onError,\n onChatCreated,\n onUserMessage,\n onThreadIdUpdate,\n enableAttachments = false,\n supportsVision,\n onBeforeRequest,\n mockResponseOnBlocked,\n } = config;\n\n // Expose methods via ref\n useImperativeHandle(ref, () => ({\n sendMessage: (message: string) => {\n simpleThreadRef.current?.sendMessage(message);\n },\n sendPreMadeQA: (question: string, answer: string, options?: { delay?: number; streamingSpeed?: number }) => {\n simpleThreadRef.current?.sendPreMadeQA(question, answer, options);\n },\n clearChat: () => {\n simpleThreadRef.current?.clearChat();\n },\n }), []);\n\n // Handle initial message or pre-made Q&A on mount\n useEffect(() => {\n if (initialMessageSentRef.current) return;\n \n // Wait for chat to be ready before sending initial message\n const timer = setTimeout(() => {\n if (initialPreMadeQA) {\n initialMessageSentRef.current = true;\n simpleThreadRef.current?.sendPreMadeQA(\n initialPreMadeQA.question, \n initialPreMadeQA.answer,\n { delay: preMadeResponseDelay, streamingSpeed }\n );\n } else if (initialMessage) {\n initialMessageSentRef.current = true;\n simpleThreadRef.current?.sendMessage(initialMessage);\n }\n }, 500); // Wait 500ms for runtime to be ready\n \n return () => clearTimeout(timer);\n }, [initialMessage, initialPreMadeQA, preMadeResponseDelay, streamingSpeed]);\n\n // Determine the actual baseUrl and proxy mode\n const actualBaseUrl = proxyUrl || baseUrl || '';\n const isProxyMode = !!proxyUrl;\n\n // Create client for fetching models\n const client = useMemo(\n () => new CuadraChatClient(actualBaseUrl, sessionToken || undefined, isProxyMode),\n [actualBaseUrl, sessionToken, isProxyMode],\n );\n\n // Models state (replacing react-query)\n const [modelsData, setModelsData] = useState<ModelPage | null>(null);\n const [modelsLoading, setModelsLoading] = useState(false);\n const [modelsError, setModelsError] = useState<Error | null>(null);\n \n // Chats loading state (for multiChat mode)\n const [chatsLoading, setChatsLoading] = useState(false);\n \n // Set chatsLoading to true initially for multiChat mode\n useEffect(() => {\n if (mode === 'multiChat') {\n setChatsLoading(true);\n }\n }, [mode]);\n\n // Fetch models function\n const fetchModels = useCallback(async () => {\n if (modelMode !== 'selector') return;\n \n setModelsLoading(true);\n setModelsError(null);\n try {\n const data = await client.listModels();\n setModelsData(data);\n } catch (error) {\n setModelsError(error instanceof Error ? error : new Error('Failed to load models'));\n } finally {\n setModelsLoading(false);\n }\n }, [client, modelMode]);\n\n // Initial fetch and refetch function\n useEffect(() => {\n if (modelMode === 'selector') {\n fetchModels();\n }\n }, [modelMode, fetchModels]);\n\n // Expose refetch function on window for external access\n useEffect(() => {\n if (typeof window !== 'undefined') {\n (window as Window & { __cuadraUIKitRefetchModels?: typeof fetchModels }).__cuadraUIKitRefetchModels = fetchModels;\n }\n return () => {\n if (typeof window !== 'undefined') {\n delete (window as Window & { __cuadraUIKitRefetchModels?: typeof fetchModels }).__cuadraUIKitRefetchModels;\n }\n };\n }, [fetchModels]);\n\n // Track current model ID\n const [currentModelId, setCurrentModelId] = useState<string | null>(\n modelMode === 'fixed' ? modelId || null : null,\n );\n \n // Track previous model ID to detect changes\n const prevModelIdRef = useRef<string | null>(currentModelId);\n\n // Handle model change — only updates state. Thread switching is handled\n // inside SimpleThread (which lives inside the AssistantRuntimeProvider\n // and has direct hook access to the runtime).\n const handleModelChange = useCallback(\n (newModelId: string) => {\n setCurrentModelId(newModelId);\n prevModelIdRef.current = newModelId;\n onModelChange?.(newModelId);\n },\n [onModelChange, mode],\n );\n\n // Auto-select first model when models are loaded\n useEffect(() => {\n if (\n modelMode === 'selector' &&\n modelsData?.items &&\n modelsData.items.length > 0 &&\n !currentModelId\n ) {\n const firstModel = modelsData.items[0];\n if (firstModel?.id) {\n const modelId = firstModel.id as string;\n setCurrentModelId(modelId);\n onModelChange?.(modelId);\n }\n }\n }, [modelMode, modelsData?.items, currentModelId, onModelChange]);\n\n // Don't render chat UI until we have a model selected (for selector mode)\n const hasModel = modelMode === 'fixed' ? !!modelId : !!currentModelId;\n\n // Derive supportsVision from selected model data (selector mode) or use the explicit prop\n const resolvedSupportsVision = useMemo(() => {\n // If explicitly set by consumer, respect it\n if (supportsVision !== undefined) return supportsVision;\n // In selector mode, derive from the currently selected model's data\n if (modelMode === 'selector' && currentModelId && modelsData?.items) {\n const selectedModel = modelsData.items.find((m) => m.id === currentModelId);\n return selectedModel?.supportsVision ?? true;\n }\n // Default to true (allow images)\n return true;\n }, [supportsVision, modelMode, currentModelId, modelsData?.items]);\n\n // Prepare models for ChatHeader\n const models = useMemo(\n () =>\n (modelsData?.items || [])\n .filter((m) => m?.id && m?.displayName)\n .map((m) => ({\n id: m.id as string,\n name: m.displayName as string,\n supportsVision: m.supportsVision ?? false,\n supportsReasoning: m.supportsReasoning ?? false,\n })),\n [modelsData],\n );\n\n // Create widget context value\n const widgetContextValue = useMemo(\n () => ({\n currentModelId: currentModelId || '',\n models,\n modelsLoading,\n modelsError,\n onModelChange: handleModelChange,\n refetchModels: fetchModels,\n }),\n [currentModelId, models, modelsLoading, modelsError, handleModelChange, fetchModels],\n );\n\n // Stable callback for when chats finish loading (avoids recreating wrappedAdapter on every render)\n const handleChatsLoaded = useCallback(() => {\n if (mode === 'multiChat') {\n setChatsLoading(false);\n }\n }, [mode]);\n\n // Determine if we're still loading (models and/or chats)\n const isLoading = \n (modelMode === 'selector' && modelsLoading) || \n (mode === 'multiChat' && chatsLoading);\n\n // Show loading state if we're still loading models or chats\n if (isLoading) {\n return (\n <CuadraWidgetProvider value={widgetContextValue}>\n <TexturedCard \n className=\"cuadra-h-full\" \n paddingX=\"cuadra-px-0\" \n paddingY=\"cuadra-py-0\"\n borderSizePx={config.borderless ? 0 : 2}\n >\n <div className=\"cuadra-flex cuadra-items-center cuadra-justify-center cuadra-h-full\">\n <Loader2 className=\"cuadra-h-6 cuadra-w-6 cuadra-animate-spin cuadra-text-muted-foreground\" />\n </div>\n </TexturedCard>\n </CuadraWidgetProvider>\n );\n }\n\n // Show error state if we need a model but don't have one\n if (modelMode === 'selector' && !hasModel && modelsError) {\n return (\n <CuadraWidgetProvider value={widgetContextValue}>\n <TexturedCard className=\"cuadra-h-full\" paddingX=\"cuadra-px-0\" paddingY=\"cuadra-py-0\">\n <div className=\"cuadra-flex cuadra-items-center cuadra-justify-center cuadra-h-full\">\n <div className=\"cuadra-flex cuadra-flex-col cuadra-items-center cuadra-gap-2\">\n <p className=\"cuadra-text-sm cuadra-text-destructive font-brand\">Failed to load models</p>\n </div>\n </div>\n </TexturedCard>\n </CuadraWidgetProvider>\n );\n }\n\n return (\n <div className=\"cuadra-uikit cuadra-h-full cuadra-w-full\" style={{ height: '100%', width: '100%' }}>\n <CuadraWidgetProvider value={widgetContextValue}>\n <CuadraRuntimeProvider\n baseUrl={isProxyMode ? actualBaseUrl : (baseUrl || actualBaseUrl)}\n sessionToken={sessionToken}\n isProxyMode={isProxyMode}\n mode={mode}\n modelMode={modelMode}\n modelId={currentModelId || modelId || undefined}\n onModelChange={handleModelChange}\n ephemeral={ephemeral}\n systemPrompt={systemPrompt}\n enableReasoning={enableReasoning}\n initialThreadId={initialThreadId}\n enableAttachments={enableAttachments}\n supportsVision={resolvedSupportsVision}\n onError={onError}\n onChatCreated={onChatCreated}\n onUserMessage={onUserMessage}\n onThreadIdUpdate={onThreadIdUpdate}\n onBeforeRequest={onBeforeRequest}\n mockResponseOnBlocked={mockResponseOnBlocked}\n onChatsLoaded={handleChatsLoaded}\n >\n <TexturedCard\n paddingX=\"cuadra-px-0\"\n paddingY=\"cuadra-py-0\"\n className={`cuadra-h-full ${config.containerClass || ''}`}\n borderSizePx={config.borderless ? 0 : 2}\n style={{ height: '100%' }}\n >\n <div \n className=\"cuadra-h-full cuadra-flex cuadra-relative\" \n style={{ height: '100%' }}\n >\n <div className=\"cuadra-flex-1 cuadra-flex cuadra-flex-col\">\n <div className=\"cuadra-flex-1 cuadra-relative cuadra-min-h-0\">\n <SimpleThread \n ref={simpleThreadRef}\n welcomeTitle={welcomeTitle}\n welcomeSubtitle={welcomeSubtitle}\n extraTopPadding={extraTopPadding}\n suggestions={suggestions}\n inputPlaceholder={inputPlaceholder}\n enableAttachments={enableAttachments}\n preMadeResponseDelay={preMadeResponseDelay}\n streamingSpeed={streamingSpeed}\n safeArea={config.safeArea}\n showChatHeader={mode === 'multiChat'}\n models={modelMode === 'selector' ? models : undefined}\n selectedModelId={currentModelId || undefined}\n onModelChange={modelMode === 'selector' ? handleModelChange : undefined}\n theme={initialTheme}\n />\n </div>\n </div>\n </div>\n </TexturedCard>\n </CuadraRuntimeProvider>\n </CuadraWidgetProvider>\n </div>\n );\n});\n\n","import type { CuadraTheme } from '../types/theme';\n\n/**\n * Apply custom theme CSS variables.\n *\n * Variables are injected via a `<style>` element that targets `.cuadra-uikit`\n * (and `.dark .cuadra-uikit` for dark mode) so they override the scoped\n * defaults declared in the UIKit stylesheet.\n *\n * The variables are also set on `:root` as a fallback for any elements that\n * live outside the `.cuadra-uikit` wrapper (e.g. portalled menus).\n *\n * @param theme - Theme configuration object with light and/or dark mode colors\n * @param _scope - Unused, kept for backwards-compatible call sites\n */\nexport function applyTheme(theme: CuadraTheme, _scope: string = ':root'): void {\n if (typeof document === 'undefined') {\n return;\n }\n\n const root = document.documentElement;\n const parts: string[] = [];\n\n // Apply light theme\n if (theme.light) {\n // Inline styles on :root for portalled elements\n applyThemeColors(theme.light as Record<string, string | undefined>, root);\n\n // Style rule for .cuadra-uikit scope (overrides the scoped defaults)\n const lightVars = buildCSSVariables(theme.light as Record<string, string | undefined>);\n if (lightVars) {\n parts.push(`.cuadra-uikit { ${lightVars} }`);\n }\n }\n\n // Apply dark theme\n if (theme.dark) {\n const darkVars = buildCSSVariables(theme.dark as Record<string, string | undefined>);\n if (darkVars) {\n parts.push(`.dark { ${darkVars} }`);\n parts.push(`.dark .cuadra-uikit { ${darkVars} }`);\n }\n }\n\n // Create or update a single style element for all custom theme rules\n if (parts.length > 0) {\n let styleEl = document.getElementById('cuadra-theme-custom') as HTMLStyleElement;\n if (!styleEl) {\n styleEl = document.createElement('style');\n styleEl.id = 'cuadra-theme-custom';\n document.head.appendChild(styleEl);\n }\n styleEl.textContent = parts.join('\\n');\n }\n}\n\n/**\n * Map theme property names to CSS variable names\n */\nconst CSS_VAR_MAP: Record<string, string> = {\n background: 'background',\n foreground: 'foreground',\n card: 'card',\n cardForeground: 'card-foreground',\n popover: 'popover',\n popoverForeground: 'popover-foreground',\n primary: 'primary',\n primaryForeground: 'primary-foreground',\n primaryLight: 'primary-light',\n primaryLighter: 'primary-lighter',\n secondary: 'secondary',\n secondaryForeground: 'secondary-foreground',\n muted: 'muted',\n mutedForeground: 'muted-foreground',\n accent: 'accent',\n accentForeground: 'accent-foreground',\n border: 'border',\n input: 'input',\n ring: 'ring',\n destructive: 'destructive',\n destructiveForeground: 'destructive-foreground',\n success: 'success',\n successForeground: 'success-foreground',\n info: 'info',\n infoForeground: 'info-foreground',\n warning: 'warning',\n warningForeground: 'warning-foreground',\n};\n\n/**\n * Convert camelCase property name to CSS variable name\n */\nfunction getCSSVarName(key: string): string {\n const mapped = CSS_VAR_MAP[key];\n if (mapped) {\n return `--cuadra-${mapped}`;\n }\n // Fallback: convert camelCase to kebab-case\n return `--cuadra-${key.replace(/([A-Z])/g, '-$1').toLowerCase()}`;\n}\n\n/**\n * Apply theme colors to an element\n */\nfunction applyThemeColors(colors: Record<string, string | undefined>, element: HTMLElement): void {\n Object.entries(colors).forEach(([key, value]) => {\n if (value) {\n const cssVarName = getCSSVarName(key);\n element.style.setProperty(cssVarName, value);\n }\n });\n}\n\n/**\n * Build CSS variables string from theme colors\n */\nfunction buildCSSVariables(colors: Record<string, string | undefined>): string {\n return Object.entries(colors)\n .map(([key, value]) => {\n if (value) {\n const cssVarName = getCSSVarName(key);\n return `${cssVarName}: ${value};`;\n }\n return '';\n })\n .filter(Boolean)\n .join(' ');\n}\n\n/**\n * Remove custom theme (restore defaults)\n */\nexport function removeTheme(): void {\n if (typeof document === 'undefined') {\n return;\n }\n\n const root = document.documentElement;\n \n // Remove all custom CSS variables from root\n const customVars = Array.from(root.style).filter(prop => prop.startsWith('--cuadra-'));\n customVars.forEach(prop => root.style.removeProperty(prop));\n\n // Remove custom theme style element\n const customStyle = document.getElementById('cuadra-theme-custom');\n if (customStyle) {\n customStyle.remove();\n }\n\n // Remove legacy dark theme style element (from older versions)\n const darkStyle = document.getElementById('cuadra-theme-dark');\n if (darkStyle) {\n darkStyle.remove();\n }\n}\n\n","import { Component, type ErrorInfo, type ReactNode } from 'react';\n\n/**\n * Props for CuadraErrorBoundary\n */\nexport interface CuadraErrorBoundaryProps {\n /** Child components to wrap */\n children: ReactNode;\n /** \n * Custom fallback UI or render function\n * If not provided, DefaultErrorFallback is used\n */\n fallback?: ReactNode | ((error: Error, reset: () => void) => ReactNode);\n /** \n * Callback when an error is caught\n * Use for logging, analytics, etc.\n */\n onError?: (error: Error, errorInfo: ErrorInfo) => void;\n}\n\ninterface ErrorBoundaryState {\n hasError: boolean;\n error: Error | null;\n}\n\n/**\n * Default fallback UI shown when an error occurs\n */\nfunction DefaultErrorFallback({ \n onReset, \n error \n}: { \n onReset: () => void; \n error: Error | null;\n}) {\n return (\n <div className=\"cuadra-flex cuadra-flex-col cuadra-items-center cuadra-justify-center cuadra-h-full cuadra-p-6 cuadra-text-center\">\n <div className=\"cuadra-mb-4\">\n <svg \n className=\"cuadra-w-12 cuadra-h-12 cuadra-text-destructive cuadra-mx-auto\"\n fill=\"none\" \n viewBox=\"0 0 24 24\" \n stroke=\"currentColor\"\n >\n <path \n strokeLinecap=\"round\" \n strokeLinejoin=\"round\" \n strokeWidth={1.5} \n d=\"M12 9v2m0 4h.01m-6.938 4h13.856c1.54 0 2.502-1.667 1.732-3L13.732 4c-.77-1.333-2.694-1.333-3.464 0L3.34 16c-.77 1.333.192 3 1.732 3z\" \n />\n </svg>\n </div>\n <h3 className=\"cuadra-text-lg cuadra-font-medium cuadra-text-foreground cuadra-mb-2\">\n Something went wrong\n </h3>\n <p className=\"cuadra-text-sm cuadra-text-muted-foreground cuadra-mb-4 cuadra-max-w-sm\">\n {error?.message || 'An unexpected error occurred. Please try again.'}\n </p>\n <button\n onClick={onReset}\n className=\"cuadra-px-4 cuadra-py-2 cuadra-text-sm cuadra-font-medium cuadra-rounded-md cuadra-transition-colors\"\n style={{\n backgroundColor: 'var(--cuadra-primary)',\n color: 'hsl(var(--cuadra-primary-foreground))',\n }}\n >\n Try Again\n </button>\n </div>\n );\n}\n\n/**\n * Error boundary component for CuadraChat\n * \n * Catches JavaScript errors in child component tree and displays\n * a fallback UI instead of crashing the entire application.\n * \n * @example Basic usage (uses default fallback)\n * ```tsx\n * <CuadraErrorBoundary>\n * <CuadraChat {...props} />\n * </CuadraErrorBoundary>\n * ```\n * \n * @example With custom fallback\n * ```tsx\n * <CuadraErrorBoundary\n * fallback={<div>Oops! Something went wrong.</div>}\n * >\n * <CuadraChat {...props} />\n * </CuadraErrorBoundary>\n * ```\n * \n * @example With render function fallback\n * ```tsx\n * <CuadraErrorBoundary\n * fallback={(error, reset) => (\n * <div>\n * <p>Error: {error.message}</p>\n * <button onClick={reset}>Retry</button>\n * </div>\n * )}\n * onError={(error, info) => {\n * console.error('Chat error:', error);\n * analytics.track('chat_error', { message: error.message });\n * }}\n * >\n * <CuadraChat {...props} />\n * </CuadraErrorBoundary>\n * ```\n */\nexport class CuadraErrorBoundary extends Component<CuadraErrorBoundaryProps, ErrorBoundaryState> {\n constructor(props: CuadraErrorBoundaryProps) {\n super(props);\n this.state = { hasError: false, error: null };\n }\n\n static getDerivedStateFromError(error: Error): ErrorBoundaryState {\n return { hasError: true, error };\n }\n\n componentDidCatch(error: Error, errorInfo: ErrorInfo): void {\n this.props.onError?.(error, errorInfo);\n }\n\n /**\n * Reset the error state and attempt to re-render children\n */\n reset = (): void => {\n this.setState({ hasError: false, error: null });\n };\n\n render(): ReactNode {\n if (this.state.hasError) {\n const { fallback } = this.props;\n const { error } = this.state;\n\n // Render function fallback\n if (typeof fallback === 'function') {\n return fallback(error!, this.reset);\n }\n\n // Static ReactNode fallback\n if (fallback) {\n return fallback;\n }\n\n // Default fallback\n return <DefaultErrorFallback error={error} onReset={this.reset} />;\n }\n\n return this.props.children;\n }\n}\n\n/**\n * Hook-friendly wrapper for error boundary\n * Use when you need to reset error boundary from a child component\n */\nexport function withErrorBoundary<P extends object>(\n WrappedComponent: React.ComponentType<P>,\n errorBoundaryProps?: Omit<CuadraErrorBoundaryProps, 'children'>\n): React.FC<P> {\n const WithErrorBoundary: React.FC<P> = (props) => (\n <CuadraErrorBoundary {...errorBoundaryProps}>\n <WrappedComponent {...props} />\n </CuadraErrorBoundary>\n );\n\n WithErrorBoundary.displayName = `withErrorBoundary(${WrappedComponent.displayName || WrappedComponent.name || 'Component'})`;\n\n return WithErrorBoundary;\n}\n","import { createContext, type ReactNode, type RefObject, useCallback, useContext, useMemo, useState } from 'react';\nimport type { CuadraChatHandle } from './CuadraChat';\n\n/**\n * Context for accessing CuadraChat controls from any child component\n * \n * This provides a React-idiomatic way to control the chat component\n * from anywhere in your component tree without prop drilling.\n * \n * @example Using in a parent component\n * ```tsx\n * import { CuadraChat, CuadraChatProvider, useCuadraChat } from '@cuadra-ai/uikit';\n * \n * function App() {\n * return (\n * <CuadraChatProvider>\n * <CuadraChat connection={{ baseUrl: '...' }} />\n * <MyCustomControls />\n * </CuadraChatProvider>\n * );\n * }\n * \n * function MyCustomControls() {\n * const { controls } = useCuadraChat();\n * \n * return (\n * <button onClick={() => controls?.sendMessage('Hello!')}>\n * Send Hello\n * </button>\n * );\n * }\n * ```\n */\nexport interface CuadraChatContextValue {\n /**\n * Reference to the CuadraChat controls\n * May be null before the CuadraChat component mounts\n */\n controls: CuadraChatHandle | null;\n \n /**\n * Internal: Reference setter for CuadraChat to register itself\n * @internal\n */\n registerControls: (controls: CuadraChatHandle) => void;\n \n /**\n * Internal: Reference clearer for CuadraChat to unregister on unmount\n * @internal\n */\n unregisterControls: () => void;\n}\n\n/**\n * Context for CuadraChat controls\n * \n * Default value is null - use CuadraChatProvider to provide an actual value.\n * Consumers can check if context is null to detect missing provider.\n */\nexport const CuadraChatContext = createContext<CuadraChatContextValue | null>(null);\n\nCuadraChatContext.displayName = 'CuadraChatContext';\n\n/**\n * Hook to access CuadraChat controls from any child component\n * \n * @returns Object containing `controls` (CuadraChatHandle | null) and `isReady` boolean\n * @throws Error if used outside of CuadraChatProvider\n * \n * @example\n * ```tsx\n * function MyComponent() {\n * const { controls, isReady } = useCuadraChat();\n * \n * if (!isReady) {\n * return <span>Chat is loading...</span>;\n * }\n * \n * return (\n * <button onClick={() => controls.sendMessage('Hi!')}>\n * Send Message\n * </button>\n * );\n * }\n * ```\n */\nexport function useCuadraChat(): { controls: CuadraChatHandle | null; isReady: boolean } {\n const context = useContext(CuadraChatContext);\n \n if (context === null) {\n throw new Error(\n 'useCuadraChat must be used within a CuadraChatProvider. ' +\n 'Wrap your component tree with <CuadraChatProvider>...</CuadraChatProvider>'\n );\n }\n \n return {\n controls: context.controls,\n isReady: context.controls !== null,\n };\n}\n\n/**\n * Hook to optionally access CuadraChat controls\n * \n * Unlike useCuadraChat, this doesn't throw if the provider is missing.\n * Returns null if CuadraChatProvider is not present.\n * \n * @returns Object with controls and isReady, or null if no provider\n * \n * @example\n * ```tsx\n * function OptionalChatControl() {\n * const chat = useCuadraChatOptional();\n * \n * if (!chat) {\n * // No CuadraChatProvider in tree - gracefully handle\n * return null;\n * }\n * \n * return chat.isReady ? (\n * <button onClick={() => chat.controls?.sendMessage('Hi!')}>Send</button>\n * ) : null;\n * }\n * ```\n */\nexport function useCuadraChatOptional(): { controls: CuadraChatHandle | null; isReady: boolean } | null {\n const context = useContext(CuadraChatContext);\n \n if (context === null) {\n return null;\n }\n \n return {\n controls: context.controls,\n isReady: context.controls !== null,\n };\n}\n\n/**\n * Provider component for CuadraChat context\n * \n * Wrap your component tree with this to enable useCuadraChat hook\n * access from any descendant component.\n * \n * @example\n * ```tsx\n * function App() {\n * return (\n * <CuadraChatProvider>\n * <Header />\n * <CuadraChat connection={{ baseUrl: '...' }} />\n * <Sidebar />\n * </CuadraChatProvider>\n * );\n * }\n * ```\n */\nexport interface CuadraChatProviderProps {\n children: ReactNode;\n}\n\nexport function CuadraChatProvider({ children }: CuadraChatProviderProps) {\n const [controls, setControls] = useState<CuadraChatHandle | null>(null);\n \n const registerControls = useCallback((newControls: CuadraChatHandle) => {\n setControls(newControls);\n }, []);\n \n const unregisterControls = useCallback(() => {\n setControls(null);\n }, []);\n \n const value = useMemo<CuadraChatContextValue>(() => ({\n controls,\n registerControls,\n unregisterControls,\n }), [controls, registerControls, unregisterControls]);\n \n return (\n <CuadraChatContext.Provider value={value}>\n {children}\n </CuadraChatContext.Provider>\n );\n}\n\n/**\n * Helper to create a ref object that syncs with context\n * Used internally by CuadraChat to register itself with the context\n * @internal\n */\nexport function createContextualRef(\n registerControls: (controls: CuadraChatHandle) => void,\n unregisterControls: () => void\n): RefObject<CuadraChatHandle | null> {\n let currentValue: CuadraChatHandle | null = null;\n \n return {\n get current() {\n return currentValue;\n },\n set current(value: CuadraChatHandle | null) {\n currentValue = value;\n if (value) {\n registerControls(value);\n } else {\n unregisterControls();\n }\n },\n };\n}\n","/**\n * Props type definitions for CuadraChat component\n * \n * This module provides both the new grouped props interface (V2)\n * and backwards compatibility with the legacy flat props interface.\n */\n\nimport type { ReactNode } from 'react';\nimport type { CuadraTheme } from './theme';\n\n// ============================================\n// V2 PROPS - NEW GROUPED INTERFACE\n// ============================================\n\n/**\n * Connection configuration for the Cuadra API\n * \n * Either `baseUrl` (direct API access) or `proxyUrl` (backend-handled auth)\n * should be provided.\n * \n * @example Direct API access\n * ```ts\n * connection: {\n * baseUrl: 'https://api.cuadra.ai',\n * sessionToken: 'stytch_session_xxx',\n * }\n * ```\n * \n * @example Proxy mode (recommended for production)\n * ```ts\n * connection: {\n * proxyUrl: '/api/chat',\n * }\n * ```\n */\nexport interface CuadraConnectionConfig {\n /**\n * Cuadra API base URL for direct API access\n * @example 'https://api.cuadra.ai'\n */\n baseUrl?: string;\n\n /**\n * Proxy URL for backend-handled authentication\n * When set, requests go through your backend which adds auth headers\n * @example '/api/chat' or 'https://your-backend.com/api/chat'\n */\n proxyUrl?: string;\n\n /**\n * Bearer token for direct API authentication\n * Only needed when using `baseUrl` (not with `proxyUrl`)\n */\n sessionToken?: string | null;\n}\n\n/**\n * Chat behavior and model configuration\n */\nexport interface CuadraChatConfig {\n /**\n * Chat mode\n * - `singleChat`: Single conversation, no sidebar, simple UX\n * - `multiChat`: Multiple conversations with thread list sidebar\n * @default 'multiChat'\n */\n mode?: 'singleChat' | 'multiChat';\n\n /**\n * Model selection mode\n * - `fixed`: Use the specified `modelId`\n * - `selector`: Show dropdown to choose from available models\n * @default 'fixed'\n */\n modelMode?: 'fixed' | 'selector';\n\n /**\n * Model ID from your Cuadra Dashboard\n * Required when `modelMode` is `'fixed'`\n * @see https://dashboard.cuadra.ai\n */\n modelId?: string;\n\n /**\n * System prompt to prepend to all conversations\n * Use for setting assistant behavior, persona, or context\n */\n systemPrompt?: string;\n\n /**\n * Create ephemeral chats that auto-delete when browser closes\n * Useful for demo/preview scenarios\n * @default false\n */\n ephemeral?: boolean;\n\n /**\n * Enable reasoning/thinking output for supported models\n * Models like o1, o3 can show their \"thinking\" process\n * @default false\n */\n enableReasoning?: boolean;\n\n /**\n * Enable file attachments in the chat input\n * @default false\n */\n enableAttachments?: boolean;\n\n /**\n * Whether the current model supports vision/image inputs.\n * When false, image attachments will be rejected with a clear error message.\n * Only relevant when `enableAttachments` is true.\n * @default true\n */\n supportsVision?: boolean;\n\n /**\n * Load an existing thread on mount (multiChat mode only)\n * Useful for deep-linking to specific conversations\n */\n initialThreadId?: string;\n}\n\n/**\n * Suggestion shown in the welcome screen\n */\nexport interface CuadraSuggestion {\n /**\n * Suggestion prompt text shown as a button\n */\n prompt: string;\n\n /**\n * Pre-made response (optional)\n * If provided, the response appears instantly with simulated streaming\n * instead of making an API call. Great for demos and onboarding.\n */\n response?: string;\n\n /**\n * Delay before showing the pre-made response in milliseconds\n * Overrides the global `preMade.responseDelay` setting\n */\n responseDelay?: number;\n}\n\n/**\n * UI customization options\n */\nexport interface CuadraUIConfig {\n /**\n * Color theme mode\n * - `light`: Light theme\n * - `dark`: Dark theme\n * - `system`: Follow system preference\n * @default 'system'\n */\n theme?: 'light' | 'dark' | 'system';\n\n /**\n * Custom theme colors to override defaults\n * @see CuadraTheme for available color tokens\n */\n customTheme?: CuadraTheme;\n\n /**\n * Show the theme toggle button in the header\n * @default true\n */\n showThemeToggle?: boolean;\n\n /**\n * Language/locale for i18n\n * @example 'en', 'es', 'fr', 'de', 'pt'\n */\n language?: string;\n\n /**\n * Welcome screen title\n * @default \"Hi, how can I help you today?\"\n */\n welcomeTitle?: string;\n\n /**\n * Welcome screen subtitle\n * @default \"Start exploring our platform and discover what you can build.\"\n */\n welcomeSubtitle?: string;\n\n /**\n * Suggestion buttons shown in the welcome screen\n */\n suggestions?: CuadraSuggestion[];\n\n /**\n * Placeholder text for the chat input field\n * @default \"Type your message...\"\n */\n inputPlaceholder?: string;\n\n /**\n * Extra top padding for the thread viewport\n * Useful when embedding under a fixed header\n * @example '1rem', '2rem', '64px'\n */\n extraTopPadding?: string;\n\n /**\n * Hide the decorative border around the chat container\n * Useful for mobile apps or when embedding in a full-screen context\n * @default false\n */\n borderless?: boolean;\n\n /**\n * Apply safe area insets for mobile devices (notch, home indicator)\n * When true, adds padding to header, sidebar, and input areas\n * @default false\n */\n safeArea?: boolean;\n\n /**\n * Organization name to display in the sidebar header\n * Used for workspace branding in mobile apps\n */\n orgName?: string;\n\n /**\n * Primary color for branding (hex format)\n * Overrides the default primary color CSS variable\n * @example '#3B82F6'\n */\n primaryColor?: string;\n}\n\n/**\n * Pre-made response configuration\n * Controls timing for simulated streaming of pre-made responses\n */\nexport interface CuadraPreMadeConfig {\n /**\n * Default delay before pre-made responses start streaming\n * Simulates \"thinking\" time\n * @default 1000\n */\n responseDelay?: number;\n\n /**\n * Speed of simulated streaming in milliseconds per word\n * Lower = faster streaming\n * @default 50\n */\n streamingSpeed?: number;\n}\n\n/**\n * Initial state for the chat component\n */\nexport interface CuadraInitialState {\n /**\n * Message to send automatically when chat loads\n * Triggers a real API call\n */\n message?: string;\n\n /**\n * Pre-made Q&A to display when chat loads\n * Shows instantly with simulated streaming, no API call\n * Great for demos and onboarding\n */\n preMadeQA?: {\n question: string;\n answer: string;\n };\n}\n\n/**\n * Request interceptors for advanced control\n */\nexport interface CuadraInterceptors {\n /**\n * Called before each outgoing API request\n * Return `true` to allow the request, `false` to block it\n * Can be async (e.g., to show a login modal first)\n * \n * @example Block requests until user is logged in\n * ```ts\n * onBeforeRequest: async () => {\n * if (!isLoggedIn) {\n * showLoginModal();\n * return false; // Block the request\n * }\n * return true;\n * }\n * ```\n */\n onBeforeRequest?: () => boolean | Promise<boolean>;\n\n /**\n * Response to show when `onBeforeRequest` returns false\n * Instead of showing an error, this message will be streamed as a response\n * Can be a string or function returning a string\n * \n * @example\n * ```ts\n * mockResponseOnBlocked: \"Please complete setup before chatting!\"\n * ```\n */\n mockResponseOnBlocked?: string | (() => string);\n}\n\n/**\n * Event callbacks for monitoring and integration\n */\nexport interface CuadraCallbacks {\n /**\n * Called when a new chat is created\n * @param chatId - The server-assigned chat ID\n */\n onChatCreated?: (chatId: string) => void;\n\n /**\n * Called when an error occurs\n * Use for error tracking and user notification\n */\n onError?: (error: Error) => void;\n\n /**\n * Called when a user message is sent\n * Useful for analytics or rate limiting\n */\n onUserMessage?: () => void;\n\n /**\n * Called when the selected model changes (selector mode)\n * @param modelId - The newly selected model ID\n */\n onModelChange?: (modelId: string) => void;\n\n /**\n * Called when a local thread ID is mapped to a server chat ID\n * Useful for URL synchronization and deep linking\n */\n onThreadIdUpdate?: (oldId: string, newId: string) => void;\n\n /**\n * Called when the logout button is clicked\n * Use to trigger your app's logout flow\n */\n onLogout?: () => void;\n}\n\n/**\n * CuadraChat props with grouped configuration\n * \n * This is the recommended props format for new implementations.\n * It provides better discoverability and organization of options.\n * \n * @example Basic usage\n * ```tsx\n * <CuadraChat\n * connection={{ baseUrl: 'https://api.cuadra.ai', sessionToken: 'xxx' }}\n * chat={{ modelId: 'model-123' }}\n * />\n * ```\n * \n * @example Full configuration\n * ```tsx\n * <CuadraChat\n * connection={{ proxyUrl: '/api/chat' }}\n * chat={{\n * mode: 'multiChat',\n * modelMode: 'fixed',\n * modelId: 'model-123',\n * enableReasoning: true,\n * }}\n * ui={{\n * theme: 'dark',\n * welcomeTitle: 'Hello!',\n * suggestions: [{ prompt: 'What can you do?' }],\n * }}\n * callbacks={{\n * onChatCreated: (id) => console.log('Chat created:', id),\n * onError: (err) => console.error(err),\n * }}\n * className=\"my-chat\"\n * />\n * ```\n */\nexport interface CuadraChatPropsV2 {\n /**\n * API connection configuration\n * Either `baseUrl` or `proxyUrl` is required\n */\n connection: CuadraConnectionConfig;\n\n /**\n * Chat behavior configuration\n */\n chat?: CuadraChatConfig;\n\n /**\n * UI customization options\n */\n ui?: CuadraUIConfig;\n\n /**\n * Pre-made response timing configuration\n */\n preMade?: CuadraPreMadeConfig;\n\n /**\n * Initial state for auto-starting chat\n */\n initial?: CuadraInitialState;\n\n /**\n * Request interceptors\n */\n interceptors?: CuadraInterceptors;\n\n /**\n * Event callbacks\n */\n callbacks?: CuadraCallbacks;\n\n /**\n * CSS class name for the container element\n */\n className?: string;\n\n /**\n * Custom error fallback UI\n * Can be a ReactNode or render function with error and reset\n */\n errorFallback?: ReactNode | ((error: Error, reset: () => void) => ReactNode);\n}\n\n// ============================================\n// LEGACY PROPS - BACKWARDS COMPATIBILITY\n// ============================================\n\n/**\n * Legacy flat props interface\n * @deprecated Use CuadraChatPropsV2 with grouped props instead\n * \n * @example Migration\n * ```tsx\n * // Before (legacy)\n * <CuadraChat\n * baseUrl=\"https://api.cuadra.ai\"\n * sessionToken=\"xxx\"\n * modelId=\"model-123\"\n * welcomeTitle=\"Hi!\"\n * />\n * \n * // After (recommended)\n * <CuadraChat\n * connection={{ baseUrl: 'https://api.cuadra.ai', sessionToken: 'xxx' }}\n * chat={{ modelId: 'model-123' }}\n * ui={{ welcomeTitle: 'Hi!' }}\n * />\n * ```\n */\nexport interface CuadraChatPropsLegacy {\n /** @deprecated Use `connection.baseUrl` */\n baseUrl?: string;\n /** @deprecated Use `connection.proxyUrl` */\n proxyUrl?: string;\n /** @deprecated Use `connection.sessionToken` */\n sessionToken?: string | null;\n /** @deprecated Use `chat.mode` */\n mode?: 'singleChat' | 'multiChat';\n /** @deprecated Use `chat.modelMode` */\n modelMode?: 'fixed' | 'selector';\n /** @deprecated Use `chat.modelId` */\n modelId?: string;\n /** @deprecated Use `chat.systemPrompt` */\n systemPrompt?: string;\n /** @deprecated Use `chat.ephemeral` */\n ephemeral?: boolean;\n /** @deprecated Use `chat.enableReasoning` */\n enableReasoning?: boolean;\n /** @deprecated Use `chat.enableAttachments` */\n enableAttachments?: boolean;\n /** @deprecated Use `chat.supportsVision` */\n supportsVision?: boolean;\n /** @deprecated Use `chat.initialThreadId` */\n initialThreadId?: string;\n /** @deprecated Use `ui.theme` */\n theme?: 'light' | 'dark' | 'system';\n /** @deprecated Use `ui.customTheme` */\n customTheme?: CuadraTheme;\n /** @deprecated Use `ui.showThemeToggle` */\n showThemeToggle?: boolean;\n /** @deprecated Use `ui.language` */\n language?: string;\n /** @deprecated Use `ui.welcomeTitle` */\n welcomeTitle?: string;\n /** @deprecated Use `ui.welcomeSubtitle` */\n welcomeSubtitle?: string;\n /** @deprecated Use `ui.suggestions` */\n suggestions?: CuadraSuggestion[];\n /** @deprecated Use `ui.inputPlaceholder` */\n inputPlaceholder?: string;\n /** @deprecated Use `ui.extraTopPadding` */\n extraTopPadding?: string;\n /** @deprecated Use `preMade.responseDelay` */\n preMadeResponseDelay?: number;\n /** @deprecated Use `preMade.streamingSpeed` */\n streamingSpeed?: number;\n /** @deprecated Use `initial.message` */\n initialMessage?: string;\n /** @deprecated Use `initial.preMadeQA` */\n initialPreMadeQA?: { question: string; answer: string };\n /** @deprecated Use `interceptors.onBeforeRequest` */\n onBeforeRequest?: () => boolean | Promise<boolean>;\n /** @deprecated Use `interceptors.mockResponseOnBlocked` */\n mockResponseOnBlocked?: string | (() => string);\n /** @deprecated Use `callbacks.onChatCreated` */\n onChatCreated?: (chatId: string) => void;\n /** @deprecated Use `callbacks.onError` */\n onError?: (error: Error) => void;\n /** @deprecated Use `callbacks.onUserMessage` */\n onUserMessage?: () => void;\n /** @deprecated Use `callbacks.onModelChange` */\n onModelChange?: (modelId: string) => void;\n /** @deprecated Use `callbacks.onThreadIdUpdate` */\n onThreadIdUpdate?: (oldId: string, newId: string) => void;\n /** @deprecated Use `callbacks.onLogout` */\n onLogout?: () => void;\n /** Keep at top level */\n className?: string;\n /** Keep at top level */\n errorFallback?: ReactNode | ((error: Error, reset: () => void) => ReactNode);\n}\n\n// ============================================\n// TYPE UTILITIES\n// ============================================\n\n/**\n * Union type supporting both V2 (grouped) and legacy (flat) props\n * The component will automatically detect which format is used\n */\nexport type CuadraChatProps = CuadraChatPropsV2 | CuadraChatPropsLegacy;\n\n/**\n * Type guard to detect V2 props format\n * V2 props have a `connection` object at the top level\n * \n * @example\n * ```ts\n * if (isV2Props(props)) {\n * // TypeScript knows props is CuadraChatPropsV2\n * const { baseUrl } = props.connection;\n * }\n * ```\n */\nexport function isV2Props(props: CuadraChatProps): props is CuadraChatPropsV2 {\n return 'connection' in props && typeof props.connection === 'object';\n}\n\n/**\n * Normalized internal config used by CuadraChat components\n * Always uses the V2 structure internally\n */\nexport interface NormalizedCuadraChatConfig {\n connection: Required<Pick<CuadraConnectionConfig, 'baseUrl' | 'proxyUrl'>> & {\n sessionToken: string | null;\n };\n chat: Required<Omit<CuadraChatConfig, 'initialThreadId' | 'supportsVision'>> & {\n initialThreadId?: string;\n supportsVision?: boolean;\n };\n ui: Required<Omit<CuadraUIConfig, 'customTheme' | 'language' | 'suggestions' | 'extraTopPadding' | 'borderless' | 'safeArea' | 'orgName' | 'primaryColor'>> & {\n customTheme?: CuadraTheme;\n language?: string;\n suggestions?: CuadraSuggestion[];\n extraTopPadding?: string;\n borderless?: boolean;\n safeArea?: boolean;\n orgName?: string;\n primaryColor?: string;\n };\n preMade: Required<CuadraPreMadeConfig>;\n initial: CuadraInitialState;\n interceptors: CuadraInterceptors;\n callbacks: CuadraCallbacks;\n className?: string;\n errorFallback?: ReactNode | ((error: Error, reset: () => void) => ReactNode);\n}\n\n/**\n * Normalize CuadraChat props to internal config format\n * Handles both V2 (grouped) and legacy (flat) props\n * \n * @param props - Either V2 or legacy props\n * @returns Normalized configuration with defaults applied\n */\nexport function normalizeCuadraChatProps(props: CuadraChatProps): NormalizedCuadraChatConfig {\n if (isV2Props(props)) {\n return normalizeV2Props(props);\n }\n return normalizeLegacyProps(props);\n}\n\n/**\n * Normalize V2 props with defaults\n */\nfunction normalizeV2Props(props: CuadraChatPropsV2): NormalizedCuadraChatConfig {\n return {\n connection: {\n baseUrl: props.connection.baseUrl || '',\n proxyUrl: props.connection.proxyUrl || '',\n sessionToken: props.connection.sessionToken ?? null,\n },\n chat: {\n mode: props.chat?.mode ?? 'multiChat',\n modelMode: props.chat?.modelMode ?? 'fixed',\n modelId: props.chat?.modelId ?? '',\n systemPrompt: props.chat?.systemPrompt ?? '',\n ephemeral: props.chat?.ephemeral ?? false,\n enableReasoning: props.chat?.enableReasoning ?? false,\n enableAttachments: props.chat?.enableAttachments ?? false,\n supportsVision: props.chat?.supportsVision,\n initialThreadId: props.chat?.initialThreadId,\n },\n ui: {\n theme: props.ui?.theme ?? 'system',\n customTheme: props.ui?.customTheme,\n showThemeToggle: props.ui?.showThemeToggle ?? true,\n language: props.ui?.language,\n welcomeTitle: props.ui?.welcomeTitle ?? 'Hi, how can I help you today?',\n welcomeSubtitle: props.ui?.welcomeSubtitle ?? 'Start exploring our platform and discover what you can build.',\n suggestions: props.ui?.suggestions,\n inputPlaceholder: props.ui?.inputPlaceholder ?? 'Type your message...',\n extraTopPadding: props.ui?.extraTopPadding,\n borderless: props.ui?.borderless,\n safeArea: props.ui?.safeArea,\n orgName: props.ui?.orgName,\n primaryColor: props.ui?.primaryColor,\n },\n preMade: {\n responseDelay: props.preMade?.responseDelay ?? 1000,\n streamingSpeed: props.preMade?.streamingSpeed ?? 50,\n },\n initial: {\n message: props.initial?.message,\n preMadeQA: props.initial?.preMadeQA,\n },\n interceptors: {\n onBeforeRequest: props.interceptors?.onBeforeRequest,\n mockResponseOnBlocked: props.interceptors?.mockResponseOnBlocked,\n },\n callbacks: {\n onChatCreated: props.callbacks?.onChatCreated,\n onError: props.callbacks?.onError,\n onUserMessage: props.callbacks?.onUserMessage,\n onModelChange: props.callbacks?.onModelChange,\n onThreadIdUpdate: props.callbacks?.onThreadIdUpdate,\n onLogout: props.callbacks?.onLogout,\n },\n className: props.className,\n errorFallback: props.errorFallback,\n };\n}\n\n/**\n * Normalize legacy flat props to internal config format\n */\nfunction normalizeLegacyProps(props: CuadraChatPropsLegacy): NormalizedCuadraChatConfig {\n // Log deprecation warning in development\n if (typeof process !== 'undefined' && process.env?.NODE_ENV === 'development') {\n // eslint-disable-next-line no-console\n console.warn(\n '[CuadraUIKit] Using legacy flat props format. Consider migrating to grouped props for better organization.\\n' +\n 'See: https://docs.cuadra.ai/guides/uikit-migration'\n );\n }\n\n return {\n connection: {\n baseUrl: props.baseUrl || '',\n proxyUrl: props.proxyUrl || '',\n sessionToken: props.sessionToken ?? null,\n },\n chat: {\n mode: props.mode ?? 'multiChat',\n modelMode: props.modelMode ?? 'fixed',\n modelId: props.modelId ?? '',\n systemPrompt: props.systemPrompt ?? '',\n ephemeral: props.ephemeral ?? false,\n enableReasoning: props.enableReasoning ?? false,\n enableAttachments: props.enableAttachments ?? false,\n supportsVision: props.supportsVision,\n initialThreadId: props.initialThreadId,\n },\n ui: {\n theme: props.theme ?? 'system',\n customTheme: props.customTheme,\n showThemeToggle: props.showThemeToggle ?? true,\n language: props.language,\n welcomeTitle: props.welcomeTitle ?? 'Hi, how can I help you today?',\n welcomeSubtitle: props.welcomeSubtitle ?? 'Start exploring our platform and discover what you can build.',\n suggestions: props.suggestions,\n inputPlaceholder: props.inputPlaceholder ?? 'Type your message...',\n extraTopPadding: props.extraTopPadding,\n borderless: false, // Legacy props don't support borderless\n safeArea: false, // Legacy props don't support safeArea\n orgName: undefined, // Legacy props don't support orgName\n primaryColor: undefined, // Legacy props don't support primaryColor\n },\n preMade: {\n responseDelay: props.preMadeResponseDelay ?? 1000,\n streamingSpeed: props.streamingSpeed ?? 50,\n },\n initial: {\n message: props.initialMessage,\n preMadeQA: props.initialPreMadeQA,\n },\n interceptors: {\n onBeforeRequest: props.onBeforeRequest,\n mockResponseOnBlocked: props.mockResponseOnBlocked,\n },\n callbacks: {\n onChatCreated: props.onChatCreated,\n onError: props.onError,\n onUserMessage: props.onUserMessage,\n onModelChange: props.onModelChange,\n onThreadIdUpdate: props.onThreadIdUpdate,\n onLogout: props.onLogout,\n },\n className: props.className,\n errorFallback: props.errorFallback,\n };\n}\n","import { forwardRef, useContext, useEffect, useImperativeHandle, useMemo, useRef } from 'react';\nimport { type CuadraUIKitConfig, WidgetContent, type WidgetContentHandle } from './WidgetContent';\nimport { applyTheme } from '../utils/applyTheme';\nimport { CuadraErrorBoundary } from './ErrorBoundary';\nimport { CuadraChatContext } from './CuadraChatContext';\nimport { type CuadraChatProps, normalizeCuadraChatProps, type NormalizedCuadraChatConfig } from '../types/props';\n\n/**\n * Handle for controlling CuadraChat externally\n */\nexport interface CuadraChatHandle {\n /** Send a user message programmatically (triggers normal API call) */\n sendMessage: (message: string) => void;\n /** \n * Send a pre-made Q&A pair (instant, simulated streaming, no API call) \n * User can continue chatting normally after.\n */\n sendPreMadeQA: (question: string, answer: string, options?: { delay?: number; streamingSpeed?: number }) => void;\n /** Clear the current chat and start fresh */\n clearChat: () => void;\n}\n\n// Re-export the props types for convenience\nexport type { \n CuadraChatProps, \n CuadraChatPropsV2, \n CuadraChatPropsLegacy,\n CuadraConnectionConfig,\n CuadraChatConfig,\n CuadraUIConfig,\n CuadraCallbacks,\n CuadraInterceptors,\n CuadraPreMadeConfig,\n CuadraInitialState,\n CuadraSuggestion,\n} from '../types/props';\n\n// Module-level flag so the deprecation warning only logs once per session\nlet deprecationWarned = false;\n\n/**\n * Internal component that receives normalized config\n */\ninterface CuadraChatInnerProps {\n config: NormalizedCuadraChatConfig;\n}\n\nconst CuadraChatInner = forwardRef<CuadraChatHandle, CuadraChatInnerProps>(\n function CuadraChatInner({ config }, ref) {\n const widgetRef = useRef<WidgetContentHandle>(null);\n const chatContext = useContext(CuadraChatContext);\n\n // Apply custom theme if provided\n useEffect(() => {\n if (config.ui.customTheme) {\n applyTheme(config.ui.customTheme);\n }\n }, [config.ui.customTheme]);\n\n // Apply primary color override from branding\n useEffect(() => {\n if (config.ui.primaryColor && typeof document !== 'undefined') {\n const root = document.documentElement;\n // Convert hex to HSL for CSS variable compatibility\n const hexToHsl = (hex: string): string => {\n // Remove # if present\n const cleanHex = hex.replace('#', '');\n const r = parseInt(cleanHex.slice(0, 2), 16) / 255;\n const g = parseInt(cleanHex.slice(2, 4), 16) / 255;\n const b = parseInt(cleanHex.slice(4, 6), 16) / 255;\n\n const max = Math.max(r, g, b);\n const min = Math.min(r, g, b);\n let h = 0;\n let s = 0;\n const l = (max + min) / 2;\n\n if (max !== min) {\n const d = max - min;\n s = l > 0.5 ? d / (2 - max - min) : d / (max + min);\n switch (max) {\n case r: h = ((g - b) / d + (g < b ? 6 : 0)) / 6; break;\n case g: h = ((b - r) / d + 2) / 6; break;\n case b: h = ((r - g) / d + 4) / 6; break;\n }\n }\n\n return `${Math.round(h * 360)} ${Math.round(s * 100)}% ${Math.round(l * 100)}%`;\n };\n\n try {\n const hslValue = hexToHsl(config.ui.primaryColor);\n // --primary uses HSL components (for Tailwind: hsl(var(--primary)))\n root.style.setProperty('--primary', hslValue);\n // --cuadra-primary needs full color value (for inline styles)\n root.style.setProperty('--cuadra-primary', `hsl(${hslValue})`);\n } catch {\n // Invalid hex color, ignore\n }\n\n return () => {\n // Cleanup: remove the custom property on unmount\n root.style.removeProperty('--primary');\n root.style.removeProperty('--cuadra-primary');\n };\n }\n }, [config.ui.primaryColor]);\n\n // Create the controls object (memoized — methods delegate to widgetRef which is always current)\n const controls: CuadraChatHandle = useMemo(() => ({\n sendMessage: (message: string) => widgetRef.current?.sendMessage(message),\n sendPreMadeQA: (question: string, answer: string, options?: { delay?: number; streamingSpeed?: number }) => \n widgetRef.current?.sendPreMadeQA(question, answer, options),\n clearChat: () => widgetRef.current?.clearChat(),\n }), []);\n\n // Expose methods via ref\n useImperativeHandle(ref, () => controls, []);\n\n // Register with CuadraChatContext if available\n useEffect(() => {\n if (chatContext) {\n chatContext.registerControls(controls);\n return () => chatContext.unregisterControls();\n }\n }, [chatContext, controls]);\n\n // Also expose on window for non-React consumers (deprecated)\n useEffect(() => {\n if (typeof window !== 'undefined') {\n // Log deprecation warning once per session in development\n if (process.env.NODE_ENV === 'development' && !deprecationWarned) {\n deprecationWarned = true;\n // eslint-disable-next-line no-console\n console.warn(\n '[CuadraUIKit] window.__cuadraChatControls is deprecated. ' +\n 'Use CuadraChatProvider and useCuadraChat() hook instead. ' +\n 'See: https://docs.cuadra.ai/guides/uikit-migration'\n );\n }\n (window as Window & { __cuadraChatControls?: CuadraChatHandle }).__cuadraChatControls = controls;\n }\n return () => {\n if (typeof window !== 'undefined') {\n delete (window as Window & { __cuadraChatControls?: CuadraChatHandle }).__cuadraChatControls;\n }\n };\n }, [controls]);\n\n // Build WidgetContent config from normalized props\n const widgetConfig: CuadraUIKitConfig = {\n // Connection\n baseUrl: config.connection.baseUrl || undefined,\n proxyUrl: config.connection.proxyUrl || undefined,\n sessionToken: config.connection.sessionToken,\n \n // Chat behavior\n mode: config.chat.mode,\n modelMode: config.chat.modelMode,\n modelId: config.chat.modelId || undefined,\n systemPrompt: config.chat.systemPrompt || undefined,\n ephemeral: config.chat.ephemeral,\n enableReasoning: config.chat.enableReasoning,\n enableAttachments: config.chat.enableAttachments,\n supportsVision: config.chat.supportsVision,\n initialThreadId: config.chat.initialThreadId,\n \n // UI\n theme: config.ui.theme,\n showThemeToggle: config.ui.showThemeToggle,\n language: config.ui.language,\n welcomeTitle: config.ui.welcomeTitle,\n welcomeSubtitle: config.ui.welcomeSubtitle,\n suggestions: config.ui.suggestions,\n inputPlaceholder: config.ui.inputPlaceholder,\n extraTopPadding: config.ui.extraTopPadding,\n containerClass: config.className,\n borderless: config.ui.borderless,\n safeArea: config.ui.safeArea,\n orgName: config.ui.orgName,\n primaryColor: config.ui.primaryColor,\n \n // Pre-made\n preMadeResponseDelay: config.preMade.responseDelay,\n streamingSpeed: config.preMade.streamingSpeed,\n \n // Initial state\n initialMessage: config.initial.message,\n initialPreMadeQA: config.initial.preMadeQA,\n \n // Interceptors\n onBeforeRequest: config.interceptors.onBeforeRequest,\n mockResponseOnBlocked: config.interceptors.mockResponseOnBlocked,\n \n // Callbacks\n onChatCreated: config.callbacks.onChatCreated,\n onError: config.callbacks.onError,\n onUserMessage: config.callbacks.onUserMessage,\n onModelChange: config.callbacks.onModelChange,\n onThreadIdUpdate: config.callbacks.onThreadIdUpdate,\n onLogout: config.callbacks.onLogout,\n };\n\n return <WidgetContent ref={widgetRef} config={widgetConfig} />;\n }\n);\n\n/**\n * All-in-one Cuadra Chat component\n * \n * Supports both V2 grouped props (recommended) and legacy flat props.\n * \n * ## V2 Props (Recommended)\n * \n * ```tsx\n * <CuadraChat\n * connection={{\n * baseUrl: 'https://api.cuadra.ai',\n * sessionToken: 'your-token',\n * }}\n * chat={{\n * mode: 'multiChat',\n * modelId: 'your-model-id',\n * enableReasoning: true,\n * }}\n * ui={{\n * theme: 'dark',\n * welcomeTitle: 'Welcome!',\n * suggestions: [{ prompt: 'What can you do?' }],\n * }}\n * callbacks={{\n * onChatCreated: (id) => console.log('Chat:', id),\n * }}\n * />\n * ```\n * \n * ## Legacy Props (Still Supported)\n * \n * ```tsx\n * <CuadraChat\n * baseUrl=\"https://api.cuadra.ai\"\n * sessionToken=\"your-token\"\n * mode=\"multiChat\"\n * modelId=\"your-model-id\"\n * enableReasoning={true}\n * theme=\"dark\"\n * welcomeTitle=\"Welcome!\"\n * />\n * ```\n * \n * ## External Control\n * \n * Use `CuadraChatProvider` and `useCuadraChat()` hook for context-based control:\n * \n * ```tsx\n * <CuadraChatProvider>\n * <CuadraChat connection={{ baseUrl: '...' }} />\n * <MyControlsComponent /> {/* Can use useCuadraChat() *\\/}\n * </CuadraChatProvider>\n * ```\n * \n * Or use a ref for direct control:\n * \n * ```tsx\n * const chatRef = useRef<CuadraChatHandle>(null);\n * <CuadraChat ref={chatRef} connection={{ baseUrl: '...' }} />\n * chatRef.current?.sendMessage('Hello!');\n * ```\n */\nexport const CuadraChat = forwardRef<CuadraChatHandle, CuadraChatProps>(\n function CuadraChat(props, ref) {\n // Normalize props (handles both V2 and legacy formats)\n const config = normalizeCuadraChatProps(props);\n\n // Determine fallback - could be ReactNode or function\n const fallback = config.errorFallback;\n\n return (\n <CuadraErrorBoundary \n fallback={fallback}\n onError={(error, errorInfo) => {\n // Call user's onError callback if provided\n config.callbacks.onError?.(error);\n // Log in development\n if (process.env.NODE_ENV === 'development') {\n // eslint-disable-next-line no-console\n console.error('[CuadraChat] Error caught by boundary:', error, errorInfo);\n }\n }}\n >\n <CuadraChatInner ref={ref} config={config} />\n </CuadraErrorBoundary>\n );\n }\n);\n"],"names":["stylesInjected","injectionAttempted","autoInjectStyles","existingLink","existingStyle","testEl","stylesLoaded","injectStylesLink","link","currentUrl","cssUrl","alternatives","attemptIndex","tryNext","toKebabCase","string","toCamelCase","match","p1","p2","toPascalCase","camelCase","mergeClasses","classes","className","index","array","hasA11yProp","props","prop","defaultAttributes","Icon","forwardRef","color","size","strokeWidth","absoluteStrokeWidth","children","iconNode","rest","ref","createElement","tag","attrs","createLucideIcon","iconName","Component","__iconNode","ArrowUp","Brain","Check","ChevronDown","CircleAlert","Copy","EllipsisVertical","Eye","FileSpreadsheet","FileText","FileType","LoaderCircle","MessageSquare","Paperclip","Pencil","Plus","Trash2","Upload","X","INJECTION_PATTERNS","sanitizeSystemPrompt","prompt","sanitized","pattern","isValidUrl","url","parsed","validateBaseUrl","isLocalhostUrl","escapeHtml","text","htmlEntities","char","generateSecureUUID","bytes","hex","b","CuadraChatClient","baseUrl","sessionToken","isProxyMode","__publicField","token","endpoint","path","request","abortSignal","headers","body","response","errorData","chatId","params","expand","update","errorText","modelId","file","resourceType","resourceId","idempotencyKey","formData","parseSSEStream","stream","reader","decoder","buffer","done","value","lines","line","data","convertToCuadraMessages","messages","msg","content","textContent","c","convertFromCuadraMessage","promiseWithResolvers","resolve","reject","promise","res","rej","createMergeStream","list","sealed","controller","currentPull","handlePull","item","e","chunk","TextStreamControllerImpl","textDelta","createTextStream","readable","createTextStreamController","ToolCallStreamControllerImpl","_controller","hasArgsText","createToolCallStream","createToolCallStreamController","Counter","PathAppendEncoder","idx","PathAppendDecoder","idx2","PathMergeEncoder","counter","innerCounter","mapping","mappedIdx","customAlphabet","alphabet","defaultSize","id","i","generateId","AssistantStreamControllerImpl","state","callback","part","options","opt","toolName","toolCallId","parentId","createAssistantStream","localToRemoteIdMap","messageIdToChatIdMap","activeThreadRemoteId","resolveServerId","remoteId","mapped","createThreadListAdapter","client","onChatIdReceived","all","cursor","safety","items","chat","t","threadId","newTitle","actualId","_remoteId","_messages","title","error","rawText","m","p","localThreadId","serverChatId","threadListItem","useThreadListItem","isMain","React","history","converted","message","_message","adapters","jsx","RuntimeAdapterProvider","StreamingMetadataStore","messageId","existing","newSources","listener","streamingMetadataStore","preMadeResponseQueue","preMadeConversationHistory","queuePreMadeResponse","question","streamingSpeed","initialDelay","createChatModelAdapter","staticModelId","getModelId","systemPrompt","ephemeral","enableReasoning","onChatCreated","onUserMessage","onThreadIdUpdate","onBeforeRequest","mockResponseOnBlocked","resolveModelId","pendingPreMade","words","accumulatedText","speed","fakeChatId","messagesToSend","lastAssistantIndex","apiMessages","preMadeApiMessages","fileIds","lastUserMessage","attachments","attachment","uploaded","currentModelId","responseText","userMessages","userQuestion","pendingMock","accumulatedReasoning","sources","lastSourcesCount","hasAddedReasoningToTextDelay","streamMessageId","updateStoreIfNeeded","createYieldObject","chunkId","threadIdForStorage","chunkSources","source","s","chunkReasoning","delta","AttachmentValidationError","code","DEFAULT_ALLOWED_MIME_TYPES","DEFAULT_ALLOWED_EXTENSIONS","DEFAULT_MAX_FILE_SIZE","formatFileSize","k","sizes","getFileExtension","filename","mimeMatchesExtension","mimeType","extension","validExtensions","generateAttachmentId","createAttachmentAdapter","validation","supportsVision","onValidationError","onAttachmentAdded","onAttachmentRemoved","maxFileSizeBytes","allowedMimeTypes","allowedExtensions","validateContentType","acceptString","validateFile","ext","mime","prefix","type","dataURL","AttachmentErrorStore","l","attachmentErrorStore","CuadraRuntimeProvider","mode","_onModelChange","initialThreadId","onError","onChatsLoaded","enableAttachments","useMemo","selectedModelId","setSelectedModelId","useState","useEffect","handleChatCreated","useCallback","handleThreadIdUpdate","oldId","newId","modelAdapter","SingleChatProvider","MultiChatProvider","initialMessages","setInitialMessages","isLoadingThread","setIsLoadingThread","runtimeKey","setRuntimeKey","onChatCreatedRef","useRef","cancelled","loadThread","SingleChatRuntime","ThreadLoadingIndicator","attachmentAdapter","localRuntime","useLocalRuntime","AssistantRuntimeProvider","modelAdapterOptions","_initialThreadId","_onError","threadListAdapter","handleThreadIdUpdateFromAdapter","onChatsLoadedRef","wrappedAdapter","result","modelIdRef","stableAdapterOptions","_modelId","runtimeHook","runtime","unstable_useRemoteThreadListRuntime","ccount","character","count","asciiAlpha","regexCheck","asciiAlphanumeric","asciiControl","markdownLineEnding","markdownLineEndingOrSpace","markdownSpace","unicodePunctuation","unicodeWhitespace","regex","check","escapeStringRegexp","convert","test","ok","castFactory","anyFactory","propertiesFactory","typeFactory","tests","checks","any","parameters","checkAsRecord","node","nodeAsRecord","key","testFunction","parent","looksLikeANode","empty","CONTINUE","EXIT","SKIP","visitParents","tree","visitor","reverse","is","step","factory","parents","name","visit","subresult","offset","grandparents","toResult","nodeAsParent","child","findAndReplace","ignored","pairs","toPairs","pairIndex","grandparent","siblings","handler","find","replace","start","change","nodes","position","matchObject","tupleOrList","tuple","toExpression","toFunction","escape","inConstruct","notInConstruct","gfmAutolinkLiteralFromMarkdown","transformGfmAutolinkLiterals","enterLiteralAutolink","enterLiteralAutolinkValue","exitLiteralAutolink","exitLiteralAutolinkEmail","exitLiteralAutolinkHttp","exitLiteralAutolinkWww","gfmAutolinkLiteralToMarkdown","findUrl","findEmail","_","protocol","domain","previous","isCorrectDomain","parts","splitUrl","atext","label","trailExec","trail","closingParenIndex","openingParens","closingParens","email","normalizeIdentifier","footnoteReference","footnoteReferencePeek","enterFootnoteCallString","enterFootnoteCall","enterFootnoteDefinitionLabelString","enterFootnoteDefinition","exitFootnoteCallString","exitFootnoteCall","exitFootnoteDefinitionLabelString","exitFootnoteDefinition","info","tracker","exit","subexit","gfmFootnoteFromMarkdown","gfmFootnoteToMarkdown","firstLineBlank","footnoteDefinition","mapAll","mapExceptFirst","blank","constructsWithoutStrikethrough","handleDelete","peekDelete","gfmStrikethroughFromMarkdown","enterStrikethrough","exitStrikethrough","gfmStrikethroughToMarkdown","defaultStringLength","markdownTable","table","settings","align","stringLength","alignments","cellMatrix","sizeMatrix","longestCellByColumn","mostCellsPerRow","rowIndex","row","columnIndex","cell","serialize","toAlignment","before","after","blockquote","map","patternInScope","stack","listInScope","none","hardBreak","_1","longestStreak","substring","expected","max","formatCodeAsIndented","checkFence","marker","raw","suffix","sequence","checkQuote","definition","quote","checkEmphasis","encodeCharacterReference","classifyCharacter","encodeInfo","outside","inside","outsideKind","insideKind","emphasis","emphasisPeek","between","betweenHead","open","betweenTail","close","testOrVisitor","visitorOrReverse","maybeReverse","overload","emptyOptions","toString","includeImageAlt","includeHtml","one","values","formatHeadingAsSetext","literalWithBreak","heading","rank","html","htmlPeek","image","imagePeek","imageReference","imageReferencePeek","alt","reference","inlineCode","inlineCodePeek","expression","formatLinkAsAutolink","linkPeek","linkReference","linkReferencePeek","checkBullet","checkBulletOther","bullet","bulletOther","checkBulletOrdered","checkRule","bulletCurrent","useDifferentMarker","firstListItem","checkListItemIndent","style","listItem","listItemIndent","paragraph","phrasing","root","d","checkStrong","strong","strongPeek","checkRuleRepetition","repetition","thematicBreak","handle","gfmTableFromMarkdown","enterTable","enterCell","enterRow","exitCodeText","exitTable","$0","$1","gfmTableToMarkdown","padding","alignDelimiters","around","inlineCodeWithTable","handleTable","handleTableCell","handleTableRow","serializeData","handleTableAsData","handleTableRowAsData","matrix","defaultHandlers","gfmTaskListItemFromMarkdown","exitCheck","exitParagraphWithTaskListItem","gfmTaskListItemToMarkdown","listItemWithTaskListItem","head","firstParaghraph","sibling","checkable","checkbox","gfmFromMarkdown","gfmToMarkdown","splice","remove","end","chunkStart","hasOwnProperty","combineExtensions","extensions","syntaxExtension","hook","left","right","constructs","wwwPrefix","tokenizeWwwPrefix","tokenizeDomain","tokenizePath","tokenizeTrail","emailDomainDotTrail","tokenizeEmailDomainDotTrail","wwwAutolink","tokenizeWwwAutolink","previousWww","protocolAutolink","tokenizeProtocolAutolink","previousProtocol","emailAutolink","tokenizeEmailAutolink","previousEmail","gfmAutolinkLiteral","effects","nok","self","dot","gfmAtext","previousUnbalanced","emailDomain","emailDomainAfter","emailDomainDot","wwwStart","wwwAfter","seen","protocolStart","protocolPrefixInside","protocolSlashesInside","afterProtocol","protocolAfter","wwwPrefixInside","wwwPrefixAfter","underscoreInLastSegment","underscoreInLastLastSegment","domainInside","domainAfter","domainAtPunctuation","sizeOpen","sizeClose","pathInside","pathAtPunctuation","trailCharacterReferenceStart","trailBracketAfter","trailCharacterReferenceInside","events","resolveAll","context","called","factorySpace","limit","blankLine","tokenizeBlankLine","indent","tokenizeIndent","gfmFootnote","tokenizeDefinitionStart","tokenizeDefinitionContinuation","gfmFootnoteDefinitionEnd","tokenizeGfmFootnoteCall","tokenizePotentialGfmFootnoteCall","resolveToPotentialGfmFootnoteCall","defined","labelStart","call","replacement","callStart","callData","callEscape","identifier","labelAtMarker","labelInside","labelAfter","labelEscape","whitespaceAfter","afterPrefix","tail","gfmStrikethrough","single","tokenizer","tokenizeStrikethrough","resolveAllStrikethrough","strikethrough","nextEvents","insideSpan","more","EditMap","add","addImplementation","a","vecs","slice","element","editMap","at","gfmTableAlign","inDelimiterRow","event","alignIndex","gfmTable","tokenizeTable","resolveTable","sizeB","next","bodyRowStart","headRowBefore","headRowStart","headRowBreak","headDelimiterStart","headRowData","headRowEscape","headDelimiterBefore","headDelimiterValueBefore","headDelimiterCellBefore","headDelimiterNok","headDelimiterLeftAlignmentAfter","headDelimiterCellAfter","headDelimiterFiller","headDelimiterRightAlignmentAfter","bodyRowBreak","bodyRowData","bodyRowEscape","inFirstCellAwaitingPipe","rowKind","lastCell","afterHeadAwaitingFirstBodyRow","lastTableEnd","currentTable","currentBody","currentCell","flushTableEnd","flushCell","range","rowEnd","previousCell","groupName","valueName","getPoint","now","relatedStart","relatedEnd","valueToken","tableBody","exits","related","side","tasklistCheck","tokenizeTasklistCheck","gfmTaskListItem","spaceThenNonSpace","gfm","remarkGfm","micromarkExtensions","fromMarkdownExtensions","toMarkdownExtensions","cn","MarkdownTextImpl","MarkdownTextPrimitive","defaultComponents","MarkdownText","memo","CodeHeader","language","isCopied","copyToClipboard","useCopyToClipboard","onCopy","jsxs","CopyIcon","CheckIcon","copiedDuration","setIsCopied","memoizeMarkdownComponents","isCodeBlock","useIsMarkdownCodeBlock","Badge","variant","baseClasses","variantClasses","deduplicateSources","byFilename","SourceCitations","maxVisible","uniqueSources","visibleSources","hiddenCount","SourceBadge","getFileIcon","score","SimpleThread","welcomeTitle","welcomeSubtitle","extraTopPadding","suggestions","inputPlaceholder","preMadeResponseDelay","safeArea","composerBorder","models","onModelChange","showChatHeader","thread","useThreadRuntime","aui","useAui","isLoadingMessages","setIsLoadingMessages","currentThreadId","setCurrentThreadId","isProcessingPreMade","_setIsProcessingPreMade","attachmentError","useSyncExternalStore","timer","showCenteredWelcome","useThread","assistantRuntime","useAssistantRuntime","assistantRuntimeRef","prevModelIdRef","prev","sendPreMadeQAInternal","answer","delay","sendMessageInternal","handleSuggestionClick","suggestion","clearChatInternal","threadListRuntime","useImperativeHandle","runtimeStore","unsubscribe","isLoadingHistory","messageCount","ThreadPrimitive","ChatHeader","ModelSelector","ComposerPrimitive","Loader2","WelcomeScreen","UserMessage","AssistantMessage","AlertCircle","useAuiState","AttachmentPrimitive","ArrowUpIcon","subtitle","MessagePrimitive","isImage","imageContent","src","Reasoning","isOpen","setIsOpen","isStreaming","ThinkingIndicator","isInProgress","hasContent","hasCapturedRef","capturedMessageId","setCapturedMessageId","currentId","msgSources","currentSources","IconTooltip","selectedModel","handleModelSelect","Fragment","model","threadTitle","isDropdownOpen","setIsDropdownOpen","displayTitle","ThreadListDropdown","onClose","ThreadListPrimitive","DropdownThreadList","onSelect","fullHeight","threadIds","threads","newThreadId","DropdownThreadItem","threadItem","itemRuntime","useThreadListItemRuntime","currentThread","isActive","isHovered","setIsHovered","isRenaming","setIsRenaming","nextTitle","setNextTitle","busy","setBusy","handleRename","err","ThreadListItemPrimitive","MobileThreadActions","onRename","onDelete","TexturedCard","paddingX","paddingY","borderSizePx","contentStyle","CuadraWidgetContext","createContext","CuadraWidgetProvider","useInjectStyles","possiblePaths","loaded","WidgetContent","config","simpleThreadRef","initialMessageSentRef","proxyUrl","modelMode","initialMessage","initialPreMadeQA","initialTheme","actualBaseUrl","modelsData","setModelsData","modelsLoading","setModelsLoading","modelsError","setModelsError","chatsLoading","setChatsLoading","fetchModels","setCurrentModelId","handleModelChange","newModelId","firstModel","hasModel","resolvedSupportsVision","widgetContextValue","handleChatsLoaded","applyTheme","theme","_scope","applyThemeColors","lightVars","buildCSSVariables","darkVars","styleEl","CSS_VAR_MAP","getCSSVarName","colors","cssVarName","removeTheme","customStyle","darkStyle","DefaultErrorFallback","onReset","CuadraErrorBoundary","errorInfo","fallback","withErrorBoundary","WrappedComponent","errorBoundaryProps","WithErrorBoundary","CuadraChatContext","useCuadraChat","useContext","useCuadraChatOptional","CuadraChatProvider","controls","setControls","registerControls","newControls","unregisterControls","isV2Props","normalizeCuadraChatProps","normalizeV2Props","normalizeLegacyProps","deprecationWarned","CuadraChatInner","widgetRef","chatContext","hexToHsl","cleanHex","r","g","min","h","hslValue","widgetConfig","CuadraChat"],"mappings":"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","x_google_ignoreList":[1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,27,28,29,30,31,32,33,34,35,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60,61,62,63,64,65,66,67,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,84,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,101,102,103,104,105,106,107,108,109,110]}
|