@maizzle/framework 6.0.0-rc.24 → 6.0.0-rc.26

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (143) hide show
  1. package/dist/build.d.ts +19 -1
  2. package/dist/build.d.ts.map +1 -1
  3. package/dist/build.js +139 -102
  4. package/dist/build.js.map +1 -1
  5. package/dist/components/Body.vue +12 -0
  6. package/dist/components/Button.vue +16 -29
  7. package/dist/components/CodeBlock.vue +5 -4
  8. package/dist/components/CodeInline.vue +9 -8
  9. package/dist/components/Column.vue +17 -4
  10. package/dist/components/Container.vue +7 -11
  11. package/dist/components/Hr.vue +1 -1
  12. package/dist/components/Img.vue +39 -22
  13. package/dist/components/Layout.vue +1 -1
  14. package/dist/components/Markdown.vue +9 -14
  15. package/dist/components/QrCode.vue +2 -2
  16. package/dist/components/Section.vue +9 -6
  17. package/dist/components/Text.vue +2 -2
  18. package/dist/components/utils.d.ts +25 -1
  19. package/dist/components/utils.d.ts.map +1 -1
  20. package/dist/components/utils.js +29 -1
  21. package/dist/components/utils.js.map +1 -1
  22. package/dist/components/utils.ts +46 -0
  23. package/dist/composables/useConfig.d.ts.map +1 -1
  24. package/dist/composables/useCurrentTemplate.d.ts.map +1 -1
  25. package/dist/composables/useEvent.d.ts.map +1 -1
  26. package/dist/composables/useFont.js.map +1 -1
  27. package/dist/config/index.js +1 -1
  28. package/dist/config/index.js.map +1 -1
  29. package/dist/events/index.d.ts.map +1 -1
  30. package/dist/events/index.js.map +1 -1
  31. package/dist/index.js +2 -2
  32. package/dist/plaintext.js.map +1 -1
  33. package/dist/plugins/postcss/mergeMediaQueries.js.map +1 -1
  34. package/dist/plugins/postcss/pruneVars.js.map +1 -1
  35. package/dist/plugins/postcss/quoteFontFamilies.d.ts.map +1 -1
  36. package/dist/plugins/postcss/quoteFontFamilies.js.map +1 -1
  37. package/dist/plugins/postcss/removeDeclarations.js.map +1 -1
  38. package/dist/plugins/postcss/resolveProps.d.ts.map +1 -1
  39. package/dist/plugins/postcss/resolveProps.js +0 -3
  40. package/dist/plugins/postcss/resolveProps.js.map +1 -1
  41. package/dist/plugins/postcss/tailwindCleanup.js.map +1 -1
  42. package/dist/prepare.js +1 -1
  43. package/dist/prepare.js.map +1 -1
  44. package/dist/render/active.d.ts.map +1 -1
  45. package/dist/render/buildTemplate.d.ts +49 -0
  46. package/dist/render/buildTemplate.d.ts.map +1 -0
  47. package/dist/render/buildTemplate.js +139 -0
  48. package/dist/render/buildTemplate.js.map +1 -0
  49. package/dist/render/createRenderer.d.ts +3 -1
  50. package/dist/render/createRenderer.d.ts.map +1 -1
  51. package/dist/render/createRenderer.js +43 -10
  52. package/dist/render/createRenderer.js.map +1 -1
  53. package/dist/render/index.js +1 -1
  54. package/dist/render/index.js.map +1 -1
  55. package/dist/render/injectFonts.js.map +1 -1
  56. package/dist/render/parallel/buildWorker.d.ts +31 -0
  57. package/dist/render/parallel/buildWorker.d.ts.map +1 -0
  58. package/dist/render/parallel/buildWorker.js +66 -0
  59. package/dist/render/parallel/buildWorker.js.map +1 -0
  60. package/dist/render/parallel/worker.mjs +28 -0
  61. package/dist/render/plugins/codeBlockExtract.d.ts.map +1 -1
  62. package/dist/render/plugins/codeBlockExtract.js.map +1 -1
  63. package/dist/render/plugins/markdownExtract.d.ts.map +1 -1
  64. package/dist/render/plugins/markdownExtract.js.map +1 -1
  65. package/dist/render/plugins/rawExtract.d.ts.map +1 -1
  66. package/dist/render/plugins/rawExtract.js.map +1 -1
  67. package/dist/render/plugins/rowSourceLocation.d.ts.map +1 -1
  68. package/dist/render/plugins/rowSourceLocation.js.map +1 -1
  69. package/dist/serve.d.ts.map +1 -1
  70. package/dist/serve.js +73 -53
  71. package/dist/serve.js.map +1 -1
  72. package/dist/server/compatibility.d.ts.map +1 -1
  73. package/dist/server/compatibility.js.map +1 -1
  74. package/dist/server/linter.js.map +1 -1
  75. package/dist/server/sfc-utils.js +1 -1
  76. package/dist/server/sfc-utils.js.map +1 -1
  77. package/dist/server/ui/pages/Preview.vue +34 -11
  78. package/dist/server/ui/vite-env.d.ts +1 -0
  79. package/dist/tests/render/_helpers.d.ts.map +1 -1
  80. package/dist/tests/render/_helpers.js.map +1 -1
  81. package/dist/transformers/addAttributes.js +2 -3
  82. package/dist/transformers/addAttributes.js.map +1 -1
  83. package/dist/transformers/base.d.ts +1 -1
  84. package/dist/transformers/base.d.ts.map +1 -1
  85. package/dist/transformers/base.js +5 -10
  86. package/dist/transformers/base.js.map +1 -1
  87. package/dist/transformers/columnWidth.d.ts.map +1 -1
  88. package/dist/transformers/columnWidth.js +2 -7
  89. package/dist/transformers/columnWidth.js.map +1 -1
  90. package/dist/transformers/entities.js.map +1 -1
  91. package/dist/transformers/filters/defaults.js.map +1 -1
  92. package/dist/transformers/filters/index.js.map +1 -1
  93. package/dist/transformers/format.js.map +1 -1
  94. package/dist/transformers/imgWidth.d.ts +20 -0
  95. package/dist/transformers/imgWidth.d.ts.map +1 -0
  96. package/dist/transformers/imgWidth.js +76 -0
  97. package/dist/transformers/imgWidth.js.map +1 -0
  98. package/dist/transformers/index.d.ts.map +1 -1
  99. package/dist/transformers/index.js +2 -0
  100. package/dist/transformers/index.js.map +1 -1
  101. package/dist/transformers/inlineCss.d.ts +3 -2
  102. package/dist/transformers/inlineCss.d.ts.map +1 -1
  103. package/dist/transformers/inlineCss.js.map +1 -1
  104. package/dist/transformers/inlineLink.js +1 -1
  105. package/dist/transformers/inlineLink.js.map +1 -1
  106. package/dist/transformers/minify.js.map +1 -1
  107. package/dist/transformers/minifyCodeInline.js.map +1 -1
  108. package/dist/transformers/msoPlaceholders.d.ts.map +1 -1
  109. package/dist/transformers/msoPlaceholders.js +2 -7
  110. package/dist/transformers/msoPlaceholders.js.map +1 -1
  111. package/dist/transformers/purgeCss.js.map +1 -1
  112. package/dist/transformers/replaceStrings.js.map +1 -1
  113. package/dist/transformers/safeSelectors.js.map +1 -1
  114. package/dist/transformers/shorthandCss.js.map +1 -1
  115. package/dist/transformers/tailwindComponent.js.map +1 -1
  116. package/dist/transformers/tailwindcss.js +1 -1
  117. package/dist/transformers/tailwindcss.js.map +1 -1
  118. package/dist/transformers/urlQuery.js.map +1 -1
  119. package/dist/types/config.d.ts +26 -4
  120. package/dist/types/config.d.ts.map +1 -1
  121. package/dist/utils/ast/serializer.js.map +1 -1
  122. package/dist/utils/compileTailwindCss.js.map +1 -1
  123. package/dist/utils/componentSources.js.map +1 -1
  124. package/dist/utils/cssBox.d.ts.map +1 -1
  125. package/dist/utils/cssBox.js +2 -7
  126. package/dist/utils/cssBox.js.map +1 -1
  127. package/dist/utils/decodeStyleEntities.js.map +1 -1
  128. package/dist/utils/url.js.map +1 -1
  129. package/dist/utils/watchPaths.js.map +1 -1
  130. package/node_modules/@clack/core/CHANGELOG.md +30 -0
  131. package/node_modules/@clack/core/dist/index.d.mts +109 -3
  132. package/node_modules/@clack/core/dist/index.mjs +972 -17
  133. package/node_modules/@clack/core/package.json +2 -1
  134. package/node_modules/@clack/prompts/CHANGELOG.md +42 -0
  135. package/node_modules/@clack/prompts/README.md +30 -9
  136. package/node_modules/@clack/prompts/dist/index.d.mts +458 -27
  137. package/node_modules/@clack/prompts/dist/index.mjs +1378 -141
  138. package/node_modules/@clack/prompts/package.json +2 -2
  139. package/node_modules/tinyexec/package.json +4 -4
  140. package/package.json +13 -11
  141. package/dist/components/Overlap.vue +0 -156
  142. package/node_modules/@clack/core/dist/index.mjs.map +0 -1
  143. package/node_modules/@clack/prompts/dist/index.mjs.map +0 -1
@@ -1 +1 @@
1
- {"version":3,"file":"createRenderer.js","names":["relPath","merge"],"sources":["../../src/render/createRenderer.ts"],"sourcesContent":["import { dirname, relative as relPath, resolve } from 'node:path'\nimport { mkdirSync, writeFileSync, existsSync, rmSync } from 'node:fs'\nimport { fileURLToPath } from 'node:url'\nimport { isLaravel } from '../utils/detect.ts'\nimport { rowSourceLocation } from './plugins/rowSourceLocation.ts'\nimport { rawExtract } from './plugins/rawExtract.ts'\nimport { codeBlockExtract } from './plugins/codeBlockExtract.ts'\nimport { markdownExtract } from './plugins/markdownExtract.ts'\nimport { createServer, mergeConfig, type InlineConfig, type Plugin } from 'vite'\nimport vue from '@vitejs/plugin-vue'\nimport Markdown from 'unplugin-vue-markdown/vite'\nimport AutoImport from 'unplugin-auto-import/vite'\nimport Components from 'unplugin-vue-components/vite'\nimport { unheadVueComposablesImports } from '@unhead/vue'\nimport { defu as merge } from 'defu'\nimport { glob, globSync } from 'tinyglobby'\nimport { createSSRApp } from 'vue'\nimport { renderToString } from 'vue/server-renderer'\nimport { createHead } from '@unhead/vue/server'\nimport { MaizzleConfigKey } from '../composables/useConfig.ts'\nimport { RenderContextKey } from '../composables/renderContext.ts'\nimport { componentNameFromPath, type NormalizedComponentSource } from '../utils/componentSources.ts'\nimport type { Component, InjectionKey } from 'vue'\nimport type { MaizzleConfig, MarkdownConfig } from '../types/index.ts'\nimport type { MarkdownExit } from 'markdown-exit'\nimport type { RenderContext } from '../composables/renderContext.ts'\n\nconst __dirname = dirname(fileURLToPath(import.meta.url))\n\nconst vuePkgDir = dirname(fileURLToPath(import.meta.resolve('vue/package.json')))\nconst vueServerRendererPkgDir = dirname(fileURLToPath(import.meta.resolve('@vue/server-renderer/package.json')))\nconst unheadVuePkgDir = resolve(dirname(fileURLToPath(import.meta.resolve('@unhead/vue'))), '..')\nconst vueRouterPkgDir = dirname(fileURLToPath(import.meta.resolve('vue-router/package.json')))\n\nexport interface RenderedTemplate {\n html: string\n doctype?: string\n templateConfig: MaizzleConfig\n sfcEventHandlers: RenderContext['sfcEventHandlers']\n plaintext?: RenderContext['plaintext']\n outputPath?: RenderContext['outputPath']\n tailwindBlocks?: RenderContext['tailwindBlocks']\n}\n\nexport interface Renderer {\n render(input: string | Component, config: MaizzleConfig): Promise<RenderedTemplate>\n invalidate(filePath: string): Promise<void>\n invalidateAll(): Promise<void>\n close(): Promise<void>\n}\n\nexport interface CreateRendererOptions {\n /** Generate .d.ts files for auto-imports and components (default: false) */\n dts?: boolean\n /** Options passed to unplugin-vue-markdown */\n markdown?: MarkdownConfig\n /** Root directory for resolving user component dirs and .d.ts output */\n root?: string\n /**\n * Additional component sources to register for auto-import. Already\n * normalized — pass through `normalizeComponentSources()` first.\n */\n componentDirs?: NormalizedComponentSource[]\n /** User Vite config options to merge into the internal SSR server */\n vite?: InlineConfig\n}\n\n/**\n * Lightweight Vite SSR loader for rendering Vue SFC email templates.\n *\n * Uses only Vue + unplugin for component/auto-import resolution.\n * Tailwind CSS compilation is handled by the transformer pipeline.\n */\nexport async function createRenderer(\n options: CreateRendererOptions = {},\n): Promise<Renderer> {\n const { dts = false, markdown: markdownOptionsRaw, root = process.cwd(), componentDirs = [], vite: userViteConfig } = options\n const { shikiTheme = 'github-light', ...markdownOptions } = markdownOptionsRaw ?? {}\n\n /**\n * Sources without an explicit prefix get registered via unplugin's `dirs`\n * (folder name auto-namespaces). Sources with an explicit `prefix` are\n * registered through a custom resolver below so we fully control naming.\n */\n const dirSources = componentDirs.filter(s => s.prefix === undefined)\n const prefixedSources = componentDirs.filter(s => s.prefix !== undefined)\n\n /**\n * Absolute component dirs — used to skip auto-wrapping `.md` files that\n * are imported as reusable components (vs. entry-point email templates).\n */\n const componentDirsAbs = [resolve(root, 'components'), ...componentDirs.map(s => s.path)]\n\n const dtsDir = isLaravel()\n ? resolve(process.cwd(), 'resources/js/types/maizzle')\n : resolve(root, '.maizzle')\n\n /**\n * Built-in framework components live at this path. When a user provides\n * a top-level file with the same (PascalCased) basename, drop the\n * built-in from unplugin's scan so the user's component is the only\n * candidate. This avoids the \"naming conflicts\" warning and the\n * alphabetical-glob ordering pitfall that decides who wins when\n * both are present in `dirs`.\n */\n const frameworkComponentsDir = resolve(__dirname, '../components')\n\n function topLevelBasenamesLower(dir: string): Set<string> {\n if (!existsSync(dir)) return new Set()\n const files = globSync(['*.vue', '*.md'], { cwd: dir, absolute: false })\n return new Set(files.map(f => f.replace(/\\.(vue|md)$/, '').toLowerCase()))\n }\n\n const frameworkFiles = globSync(['*.vue', '*.md'], { cwd: frameworkComponentsDir, absolute: false })\n const frameworkByLower = new Map(\n frameworkFiles.map(f => [f.replace(/\\.(vue|md)$/, '').toLowerCase(), f]),\n )\n\n const shadowedNames = new Set<string>()\n for (const dir of [resolve(root, 'components'), ...dirSources.map(s => s.path)]) {\n for (const lower of topLevelBasenamesLower(dir)) {\n if (frameworkByLower.has(lower)) shadowedNames.add(lower)\n }\n }\n\n const frameworkExcludes = [...shadowedNames]\n .map(lower => `${frameworkComponentsDir}/${frameworkByLower.get(lower)}`)\n\n /**\n * Pre-scanned name → absolute-path map for prefixed sources. Rebuilt\n * on file add/unlink via the watcher hook plugin further down. Drives\n * the runtime resolver and the d.ts we emit for IDE autocompletion.\n */\n const prefixedNameMap = new Map<string, string>()\n\n async function scanPrefixedSources(): Promise<void> {\n prefixedNameMap.clear()\n const seen = new Map<string, string>()\n for (const source of prefixedSources) {\n const files = await glob(['**/*.vue', '**/*.md'], { cwd: source.path, absolute: true })\n for (const file of files) {\n const name = componentNameFromPath({\n filePath: file,\n dirRoot: source.path,\n prefix: source.prefix,\n pathPrefix: source.pathPrefix,\n })\n const existing = seen.get(name)\n if (existing && existing !== file) {\n throw new Error(\n `[maizzle] Component name collision: \"${name}\" resolved from both \"${existing}\" and \"${file}\". `\n + 'Rename one of the files or split them into separate sources with distinct prefixes.',\n )\n }\n seen.set(name, file)\n prefixedNameMap.set(name, file)\n }\n }\n }\n\n await scanPrefixedSources()\n\n const prefixedResolver = (name: string) => prefixedNameMap.get(name)\n\n /**\n * unplugin-vue-components' own d.ts only covers components found via\n * `dirs`; its `types` option emits named-import entries which break\n * for SFC `default` exports. Write a sibling d.ts for prefixed\n * sources so editors get correct autocompletion via TypeScript\n * interface merging on `vue.GlobalComponents`.\n */\n const prefixedDtsPath = resolve(dtsDir, 'prefixed-components.d.ts')\n\n function writePrefixedDts(): void {\n if (!dts) return\n if (prefixedNameMap.size === 0) {\n if (existsSync(prefixedDtsPath)) rmSync(prefixedDtsPath)\n return\n }\n const dtsBase = dirname(prefixedDtsPath)\n mkdirSync(dtsBase, { recursive: true })\n const lines = Array.from(prefixedNameMap.entries())\n .sort(([a], [b]) => a.localeCompare(b))\n .map(([name, file]) => {\n const relativePath = relPath(dtsBase, file).replace(/\\\\/g, '/')\n const importPath = relativePath.startsWith('.') ? relativePath : `./${relativePath}`\n return ` ${name}: typeof import('${importPath}')['default']`\n })\n .join('\\n')\n writeFileSync(\n prefixedDtsPath,\n `/* eslint-disable */\\n// @ts-nocheck\\n// biome-ignore lint: disable\\n// oxlint-disable\\n// Generated by Maizzle for prefixed component sources\\n\\nexport {}\\n\\n/* prettier-ignore */\\ndeclare module 'vue' {\\n export interface GlobalComponents {\\n${lines}\\n }\\n}\\n`,\n )\n }\n\n writePrefixedDts()\n\n /**\n * Watches prefixed source dirs and rebuilds {@link prefixedNameMap} when\n * files are added/removed. Vite's watcher already covers `dirSources`\n * via unplugin-vue-components' own filesystem hooks.\n */\n const prefixedSourceWatcher: Plugin | null = prefixedSources.length > 0\n ? {\n name: 'maizzle:prefixed-component-watcher',\n configureServer(server) {\n for (const source of prefixedSources) {\n server.watcher.add(source.path)\n }\n const refresh = async (file: string) => {\n if (!prefixedSources.some(s => file.startsWith(`${s.path}/`))) return\n if (!/\\.(vue|md)$/.test(file)) return\n await scanPrefixedSources()\n writePrefixedDts()\n }\n server.watcher.on('add', refresh)\n server.watcher.on('unlink', refresh)\n },\n }\n : null\n\n const VIRTUAL_SFC_ID = 'virtual:maizzle-sfc.vue'\n let virtualSfcSource = ''\n\n /**\n * Never load the host project's vite.config.ts here. Doing so pulls\n * every host plugin (Nitro, TanStack Start, the Maizzle plugin\n * itself, …) into this isolated SSR pipeline, where they override\n * env factories, re-trigger configureServer hooks, and break\n * Vite's hot channel wiring. Users who need extra Vite plugins\n * for SSR pass them explicitly via the `vite` option.\n */\n const maizzleConfig: InlineConfig = {\n configFile: false,\n plugins: [\n rawExtract(),\n codeBlockExtract(),\n markdownExtract(),\n rowSourceLocation(),\n {\n name: 'maizzle:virtual-sfc',\n resolveId(id) {\n if (id === VIRTUAL_SFC_ID) return id\n },\n load(id) {\n if (id === VIRTUAL_SFC_ID) return virtualSfcSource\n },\n },\n vue({\n include: [/\\.vue$/, /\\.md$/],\n template: {\n transformAssetUrls: false,\n compilerOptions: {\n /**\n * AMP4Email tags (<amp-carousel>, <amp-img>, <amp-list> ...)\n * render verbatim — skip the component resolver. Users who\n * want to wrap an amp tag in a Vue component should register\n * it under a PascalCase name (e.g. `components/AmpCarousel.vue`\n * → `<AmpCarousel>`).\n */\n isCustomElement: (tag: string) => tag.startsWith('amp-'),\n },\n },\n }),\n Markdown(merge(markdownOptions ?? {}, {\n headEnabled: true,\n wrapperDiv: false,\n wrapperClasses: 'prose',\n wrapperComponent: (id: string, raw: string) => {\n const fm = raw.match(/^---\\r?\\n([\\s\\S]*?)\\r?\\n---/)?.[1]\n const layout = fm?.match(/^[ \\t]*layout[ \\t]*:[ \\t]*['\"]?([A-Za-z][\\w-]*|false|none)['\"]?[ \\t]*$/m)?.[1]\n if (layout === 'false' || layout === 'none') return null\n if (layout) return layout\n /**\n * No `layout:` set — default to the built-in `MarkdownLayout`\n * for entry-template `.md` files. Skip for `.md` files inside\n * component dirs, which are reusable fragments imported into\n * other templates.\n */\n const inComponentDir = componentDirsAbs.some(d => id === d || id.startsWith(`${d}/`))\n return inComponentDir ? null : 'MarkdownLayout'\n },\n markdownOptions: {\n async highlight(code: string, lang: string) {\n const { codeToHtml } = await import('shiki')\n return codeToHtml(code, { lang, theme: shikiTheme })\n },\n },\n markdownSetup(md: MarkdownExit) {\n const wrapPre = (html: string) =>\n `<table class=\"w-full\"><tr><td class=\"max-w-0 mso-padding-alt-4\">${html}</td></tr></table>\\n`\n\n const defaultFence = md.renderer.rules.fence!\n md.renderer.rules.fence = (...args) => {\n const result = defaultFence(...args)\n if (typeof result === 'string') return wrapPre(result)\n return result.then(wrapPre)\n }\n\n const defaultCodeBlock = md.renderer.rules.code_block!\n md.renderer.rules.code_block = (...args) => wrapPre(defaultCodeBlock(...args) as string)\n },\n })),\n AutoImport({\n dirs: [\n resolve(__dirname, '../composables'),\n resolve(__dirname, '../filters'),\n ],\n imports: ['vue', unheadVueComposablesImports],\n /**\n * unplugin-auto-import's default `include` doesn't match `.md`, so\n * auto-imports (Vue, unhead and Maizzle composables/filters) were\n * never injected into Markdown templates — `useConfig()` and friends\n * threw at runtime. Extend the default list with `.md` (and its\n * `?vue` script sub-requests) to mirror the `.md` coverage the\n * Components plugin already declares below.\n */\n include: [/\\.[jt]sx?$/, /\\.vue$/, /\\.vue\\?vue/, /\\.md$/, /\\.md\\?vue/],\n dts: dts ? resolve(dtsDir, 'auto-imports.d.ts') : false,\n }),\n Components({\n extensions: ['vue', 'md'],\n include: [/\\.vue$/, /\\.vue\\?vue/, /\\.md$/],\n dirs: [\n frameworkComponentsDir,\n resolve(root, 'components'),\n ...dirSources.map(s => s.path),\n ],\n /**\n * Drop built-in component files whose name the user has shadowed.\n * This makes the user's version the only match — no \"naming\n * conflicts\" warning, no glob-ordering games.\n */\n globsExclude: frameworkExcludes,\n directoryAsNamespace: true,\n collapseSamePrefixes: true,\n resolvers: prefixedSources.length > 0 ? [prefixedResolver] : undefined,\n dts: dts ? resolve(dtsDir, 'components.d.ts') : false,\n }),\n ...(prefixedSourceWatcher ? [prefixedSourceWatcher] : []),\n ],\n resolve: {\n alias: {\n 'vue/server-renderer': resolve(vueServerRendererPkgDir, 'dist/server-renderer.esm-bundler.js'),\n 'vue': resolve(vuePkgDir, 'dist/vue.runtime.esm-bundler.js'),\n 'vue-router': vueRouterPkgDir,\n '@unhead/vue/server': resolve(unheadVuePkgDir, 'dist/server.mjs'),\n '@unhead/vue': resolve(unheadVuePkgDir, 'dist/index.mjs'),\n },\n },\n server: {\n middlewareMode: true,\n hmr: false,\n /**\n * Watcher is required so unplugin-vue-components and unplugin-auto-import\n * detect added/removed component files and rewrite their .d.ts on the fly.\n * (We only render via SSR — HMR is off, but chokidar still drives plugins.)\n */\n fs: {\n allow: [process.cwd(), root, ...componentDirs.map(s => s.path), vuePkgDir, vueServerRendererPkgDir, unheadVuePkgDir, vueRouterPkgDir],\n },\n },\n appType: 'custom',\n logLevel: 'silent',\n optimizeDeps: {\n noDiscovery: true,\n },\n }\n\n /**\n * Merge user's vite config (from config.vite) under Maizzle's config.\n * mergeConfig(a, b) → b overrides a for scalars, arrays concatenate.\n * This ensures Maizzle's critical settings (middlewareMode, appType,\n * etc.) always win, while user plugins and other options remain.\n */\n const finalConfig = userViteConfig\n ? mergeConfig(userViteConfig, maizzleConfig)\n : maizzleConfig\n\n const server = await createServer(finalConfig)\n\n return {\n async render(input: string | Component, config: MaizzleConfig): Promise<RenderedTemplate> {\n let component: Component\n let configKey: InjectionKey<MaizzleConfig>\n let contextKey: InjectionKey<RenderContext>\n\n if (typeof input === 'string') {\n /**\n * String input goes through Vite — must use ssrLoadModule for\n * injection keys so they share the same module instance as SFC.\n */\n const configModule = await server.ssrLoadModule(resolve(__dirname, '../composables/useConfig'))\n const contextModule = await server.ssrLoadModule(resolve(__dirname, '../composables/renderContext'))\n configKey = configModule.MaizzleConfigKey\n contextKey = contextModule.RenderContextKey\n\n if (input.includes('<template') || input.includes('<script')) {\n virtualSfcSource = input\n const mod = server.moduleGraph.getModuleById(VIRTUAL_SFC_ID)\n if (mod) server.moduleGraph.invalidateModule(mod)\n component = (await server.ssrLoadModule(VIRTUAL_SFC_ID)).default\n } else {\n component = (await server.ssrLoadModule(input)).default\n }\n } else {\n // Pre-compiled component — use directly imported keys\n component = input\n configKey = MaizzleConfigKey\n contextKey = RenderContextKey\n }\n\n const renderContext: RenderContext = {\n doctype: undefined,\n sfcConfig: undefined,\n sfcEventHandlers: [],\n }\n\n const head = createHead({ disableDefaults: true })\n const app = createSSRApp(component)\n app.use(head)\n\n // Register user Vue plugins, directives, and global properties\n if (config.vue) {\n const plugins = typeof config.vue.plugins === 'function'\n ? config.vue.plugins()\n : config.vue.plugins ?? []\n for (const plugin of plugins) {\n app.use(plugin)\n }\n for (const [name, directive] of Object.entries(config.vue.directives ?? {})) {\n app.directive(name, directive)\n }\n Object.assign(app.config.globalProperties, config.vue.globalProperties)\n }\n\n app.provide(configKey, config)\n app.provide(contextKey, renderContext)\n\n const ssrContext: Record<string, any> = {}\n let html: string = await renderToString(app, ssrContext)\n\n const { headTags, bodyTags, bodyTagsOpen, htmlAttrs, bodyAttrs } = head.render()\n\n // Inject head entries into the rendered HTML\n if (htmlAttrs) {\n html = html.replace(/<html([^>]*)>/, `<html$1 ${htmlAttrs}>`)\n }\n if (headTags) {\n html = html.replace('</head>', `${headTags}\\n</head>`)\n }\n if (bodyAttrs) {\n html = html.replace(/<body([^>]*)>/, `<body$1 ${bodyAttrs}>`)\n }\n if (bodyTagsOpen) {\n html = html.replace(/<body([^>]*)>/, `<body$1>\\n${bodyTagsOpen}`)\n }\n if (bodyTags) {\n html = html.replace('</body>', `${bodyTags}\\n</body>`)\n }\n\n // Inject SSR teleport content into their target elements\n const hasTeleports = ssrContext.teleports && Object.keys(ssrContext.teleports).length > 0\n const hasFonts = (renderContext.fonts?.length ?? 0) > 0\n\n if (hasTeleports || hasFonts) {\n const { parse: parseDom, serialize: serializeDom, walk } = await import('../utils/ast/index.ts')\n let dom = parseDom(html)\n\n if (hasTeleports) {\n for (const [rawTarget, content] of Object.entries(ssrContext.teleports) as [string, string][]) {\n if (!content) continue\n\n const prepend = rawTarget.endsWith(':start')\n const target = prepend ? rawTarget.slice(0, -6) : rawTarget\n const targetChildren = parseDom(content)\n\n walk(dom, (node) => {\n const el = node as import('domhandler').Element\n\n if (!el.name) return\n\n const matched\n = target === el.name\n || (target.startsWith('#') && el.attribs?.id === target.slice(1))\n || (target.startsWith('.') && el.attribs?.class?.split(/\\s+/).includes(target.slice(1)))\n\n if (matched) {\n for (const child of targetChildren) {\n child.parent = el as any\n }\n\n el.children = prepend\n ? [...targetChildren, ...(el.children || [])] as any\n : [...(el.children || []), ...targetChildren] as any\n }\n })\n }\n }\n\n if (hasFonts) {\n const { injectFonts } = await import('./injectFonts.ts')\n injectFonts(dom, renderContext.fonts!, parseDom, walk)\n }\n\n html = serializeDom(dom)\n }\n\n // Inject preheader text from usePreheader() composable\n if (renderContext.preheader) {\n const { text, fillerCount } = renderContext.preheader\n const filler = '\\u2007\\uFEFF\\u034F '.repeat(fillerCount)\n const previewHtml = `<div style=\"display:none\">${text}${filler}\\u00A0</div>`\n html = html.replace(/<body([^>]*)>/, `<body$1>${previewHtml}`)\n }\n\n /**\n * Strip Vue SSR fragment markers + teleport anchor comments. These\n * are rendering hygiene, not transformer concerns — must run\n * regardless of `useTransformers` state. Fragment markers contain\n * `-->`, which would prematurely terminate MSO conditional\n * comments downstream.\n */\n html = html\n .replaceAll('<!--[-->', '')\n .replaceAll('<!--]-->', '')\n .replaceAll('<!--teleport start anchor-->', '')\n .replaceAll('<!--teleport anchor-->', '')\n .replaceAll('<!--teleport start-->', '')\n .replaceAll('<!--teleport end-->', '')\n\n return {\n html,\n doctype: renderContext.doctype,\n /**\n * Layer sfcConfig over config — sfcConfig is a partial override\n * emitted by composables (defineConfig, useTransformers, etc.).\n * A naive replacement (`sfcConfig ?? config`) drops defaults\n * from the resolved config when the SFC only sets a single\n * key, since the composables' inject() of globalConfig can\n * return `{}` in dev when ssrLoadModule and the SFC's\n * auto-imported module resolve to different module\n * instances (different Symbols).\n */\n templateConfig: renderContext.sfcConfig ? merge(renderContext.sfcConfig, config) : config,\n sfcEventHandlers: renderContext.sfcEventHandlers,\n plaintext: renderContext.plaintext,\n outputPath: renderContext.outputPath,\n tailwindBlocks: renderContext.tailwindBlocks,\n }\n },\n\n async invalidate(filePath: string): Promise<void> {\n const mod = await server.moduleGraph.getModuleByUrl(filePath)\n if (mod) {\n server.moduleGraph.invalidateModule(mod)\n }\n },\n\n async invalidateAll(): Promise<void> {\n for (const mod of server.moduleGraph.idToModuleMap.values()) {\n server.moduleGraph.invalidateModule(mod)\n }\n },\n\n async close(): Promise<void> {\n await server.close()\n /**\n * unplugin-auto-import schedules a 500ms-throttled, fire-and-forget\n * d.ts write on its first scan. server.close() doesn't drain that\n * pending write, so callers tearing down the working dir right\n * after close (tests, ephemeral build pipelines) can race the\n * mkdir against a missing parent directory. Wait one throttle\n * window past close so the lingering write resolves while\n * the dir still exists.\n */\n if (dts) {\n await new Promise(resolve => setTimeout(resolve, 600))\n }\n },\n }\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"createRenderer.js","names":["relPath","merge"],"sources":["../../src/render/createRenderer.ts"],"sourcesContent":["import { dirname, relative as relPath, resolve } from 'node:path'\nimport { mkdirSync, writeFileSync, existsSync, rmSync } from 'node:fs'\nimport { fileURLToPath } from 'node:url'\nimport { isLaravel } from '../utils/detect.ts'\nimport { rowSourceLocation } from './plugins/rowSourceLocation.ts'\nimport { rawExtract } from './plugins/rawExtract.ts'\nimport { codeBlockExtract } from './plugins/codeBlockExtract.ts'\nimport { markdownExtract } from './plugins/markdownExtract.ts'\nimport { createServer, mergeConfig, type InlineConfig, type Plugin } from 'vite'\nimport vue from '@vitejs/plugin-vue'\nimport Markdown from 'unplugin-vue-markdown/vite'\nimport AutoImport from 'unplugin-auto-import/vite'\nimport Components from 'unplugin-vue-components/vite'\nimport { unheadVueComposablesImports } from '@unhead/vue'\nimport { defu as merge } from 'defu'\nimport { glob, globSync } from 'tinyglobby'\nimport { createSSRApp } from 'vue'\nimport { renderToString } from 'vue/server-renderer'\nimport { createHead } from '@unhead/vue/server'\nimport { MaizzleConfigKey } from '../composables/useConfig.ts'\nimport { RenderContextKey } from '../composables/renderContext.ts'\nimport { componentNameFromPath, type NormalizedComponentSource } from '../utils/componentSources.ts'\nimport { shikiToCodeBlock } from '../components/utils.ts'\nimport type { Component, InjectionKey } from 'vue'\nimport type { MaizzleConfig, MarkdownConfig } from '../types/index.ts'\nimport type { MarkdownExit } from 'markdown-exit'\nimport type { RenderContext } from '../composables/renderContext.ts'\n\nconst __dirname = dirname(fileURLToPath(import.meta.url))\n\nconst vuePkgDir = dirname(fileURLToPath(import.meta.resolve('vue/package.json')))\nconst vueServerRendererPkgDir = dirname(fileURLToPath(import.meta.resolve('@vue/server-renderer/package.json')))\nconst unheadVuePkgDir = resolve(dirname(fileURLToPath(import.meta.resolve('@unhead/vue'))), '..')\nconst vueRouterPkgDir = dirname(fileURLToPath(import.meta.resolve('vue-router/package.json')))\n\nexport interface RenderedTemplate {\n html: string\n doctype?: string\n templateConfig: MaizzleConfig\n sfcEventHandlers: RenderContext['sfcEventHandlers']\n plaintext?: RenderContext['plaintext']\n outputPath?: RenderContext['outputPath']\n tailwindBlocks?: RenderContext['tailwindBlocks']\n}\n\nexport interface Renderer {\n render(input: string | Component, config: MaizzleConfig, opts?: { source?: string }): Promise<RenderedTemplate>\n invalidate(filePath: string): Promise<void>\n invalidateAll(): Promise<void>\n close(): Promise<void>\n}\n\nexport interface CreateRendererOptions {\n /** Generate .d.ts files for auto-imports and components (default: false) */\n dts?: boolean\n /** Options passed to unplugin-vue-markdown */\n markdown?: MarkdownConfig\n /** Root directory for resolving user component dirs and .d.ts output */\n root?: string\n /**\n * Additional component sources to register for auto-import. Already\n * normalized — pass through `normalizeComponentSources()` first.\n */\n componentDirs?: NormalizedComponentSource[]\n /** User Vite config options to merge into the internal SSR server */\n vite?: InlineConfig\n}\n\n/**\n * Lightweight Vite SSR loader for rendering Vue SFC email templates.\n *\n * Uses only Vue + unplugin for component/auto-import resolution.\n * Tailwind CSS compilation is handled by the transformer pipeline.\n */\nexport async function createRenderer(\n options: CreateRendererOptions = {},\n): Promise<Renderer> {\n const { dts = false, markdown: markdownOptionsRaw, root = process.cwd(), componentDirs = [], vite: userViteConfig } = options\n const { shikiTheme = 'github-light', ...markdownOptions } = markdownOptionsRaw ?? {}\n\n /**\n * Sources without an explicit prefix get registered via unplugin's `dirs`\n * (folder name auto-namespaces). Sources with an explicit `prefix` are\n * registered through a custom resolver below so we fully control naming.\n */\n const dirSources = componentDirs.filter(s => s.prefix === undefined)\n const prefixedSources = componentDirs.filter(s => s.prefix !== undefined)\n\n /**\n * Absolute component dirs — used to skip auto-wrapping `.md` files that\n * are imported as reusable components (vs. entry-point email templates).\n */\n const componentDirsAbs = [resolve(root, 'components'), ...componentDirs.map(s => s.path)]\n\n const dtsDir = isLaravel()\n ? resolve(process.cwd(), 'resources/js/types/maizzle')\n : resolve(root, '.maizzle')\n\n /**\n * Built-in framework components live at this path. When a user provides\n * a top-level file with the same (PascalCased) basename, drop the\n * built-in from unplugin's scan so the user's component is the only\n * candidate. This avoids the \"naming conflicts\" warning and the\n * alphabetical-glob ordering pitfall that decides who wins when\n * both are present in `dirs`.\n */\n const frameworkComponentsDir = resolve(__dirname, '../components')\n\n function topLevelBasenamesLower(dir: string): Set<string> {\n if (!existsSync(dir)) return new Set()\n const files = globSync(['*.vue', '*.md'], { cwd: dir, absolute: false })\n return new Set(files.map(f => f.replace(/\\.(vue|md)$/, '').toLowerCase()))\n }\n\n const frameworkFiles = globSync(['*.vue', '*.md'], { cwd: frameworkComponentsDir, absolute: false })\n const frameworkByLower = new Map(\n frameworkFiles.map(f => [f.replace(/\\.(vue|md)$/, '').toLowerCase(), f]),\n )\n\n const shadowedNames = new Set<string>()\n for (const dir of [resolve(root, 'components'), ...dirSources.map(s => s.path)]) {\n for (const lower of topLevelBasenamesLower(dir)) {\n if (frameworkByLower.has(lower)) shadowedNames.add(lower)\n }\n }\n\n const frameworkExcludes = [...shadowedNames]\n .map(lower => `${frameworkComponentsDir}/${frameworkByLower.get(lower)}`)\n\n /**\n * Pre-scanned name → absolute-path map for prefixed sources. Rebuilt\n * on file add/unlink via the watcher hook plugin further down. Drives\n * the runtime resolver and the d.ts we emit for IDE autocompletion.\n */\n const prefixedNameMap = new Map<string, string>()\n\n async function scanPrefixedSources(): Promise<void> {\n prefixedNameMap.clear()\n const seen = new Map<string, string>()\n for (const source of prefixedSources) {\n const files = await glob(['**/*.vue', '**/*.md'], { cwd: source.path, absolute: true })\n for (const file of files) {\n const name = componentNameFromPath({\n filePath: file,\n dirRoot: source.path,\n prefix: source.prefix,\n pathPrefix: source.pathPrefix,\n })\n const existing = seen.get(name)\n if (existing && existing !== file) {\n throw new Error(\n `[maizzle] Component name collision: \"${name}\" resolved from both \"${existing}\" and \"${file}\". `\n + 'Rename one of the files or split them into separate sources with distinct prefixes.',\n )\n }\n seen.set(name, file)\n prefixedNameMap.set(name, file)\n }\n }\n }\n\n await scanPrefixedSources()\n\n const prefixedResolver = (name: string) => prefixedNameMap.get(name)\n\n /**\n * unplugin-vue-components' own d.ts only covers components found via\n * `dirs`; its `types` option emits named-import entries which break\n * for SFC `default` exports. Write a sibling d.ts for prefixed\n * sources so editors get correct autocompletion via TypeScript\n * interface merging on `vue.GlobalComponents`.\n */\n const prefixedDtsPath = resolve(dtsDir, 'prefixed-components.d.ts')\n\n function writePrefixedDts(): void {\n if (!dts) return\n if (prefixedNameMap.size === 0) {\n if (existsSync(prefixedDtsPath)) rmSync(prefixedDtsPath)\n return\n }\n const dtsBase = dirname(prefixedDtsPath)\n mkdirSync(dtsBase, { recursive: true })\n const lines = Array.from(prefixedNameMap.entries())\n .sort(([a], [b]) => a.localeCompare(b))\n .map(([name, file]) => {\n const relativePath = relPath(dtsBase, file).replace(/\\\\/g, '/')\n const importPath = relativePath.startsWith('.') ? relativePath : `./${relativePath}`\n return ` ${name}: typeof import('${importPath}')['default']`\n })\n .join('\\n')\n writeFileSync(\n prefixedDtsPath,\n `/* eslint-disable */\\n// @ts-nocheck\\n// biome-ignore lint: disable\\n// oxlint-disable\\n// Generated by Maizzle for prefixed component sources\\n\\nexport {}\\n\\n/* prettier-ignore */\\ndeclare module 'vue' {\\n export interface GlobalComponents {\\n${lines}\\n }\\n}\\n`,\n )\n }\n\n writePrefixedDts()\n\n /**\n * Watches prefixed source dirs and rebuilds {@link prefixedNameMap} when\n * files are added/removed. Vite's watcher already covers `dirSources`\n * via unplugin-vue-components' own filesystem hooks.\n */\n const prefixedSourceWatcher: Plugin | null = prefixedSources.length > 0\n ? {\n name: 'maizzle:prefixed-component-watcher',\n configureServer(server) {\n for (const source of prefixedSources) {\n server.watcher.add(source.path)\n }\n const refresh = async (file: string) => {\n if (!prefixedSources.some(s => file.startsWith(`${s.path}/`))) return\n if (!/\\.(vue|md)$/.test(file)) return\n await scanPrefixedSources()\n writePrefixedDts()\n }\n server.watcher.on('add', refresh)\n server.watcher.on('unlink', refresh)\n },\n }\n : null\n\n const VIRTUAL_SFC_ID = 'virtual:maizzle-sfc.vue'\n let virtualSfcSource = ''\n\n /**\n * Per-render source overrides keyed by absolute template path. Lets the\n * build's beforeRender event rewrite a template's source before compile\n * while keeping the real file id — so relative imports, asset URLs and\n * component resolution still resolve against the actual file location\n * (which the virtual-SFC path can't do).\n */\n const sourceOverrides = new Map<string, string>()\n\n /**\n * Never load the host project's vite.config.ts here. Doing so pulls\n * every host plugin (Nitro, TanStack Start, the Maizzle plugin\n * itself, …) into this isolated SSR pipeline, where they override\n * env factories, re-trigger configureServer hooks, and break\n * Vite's hot channel wiring. Users who need extra Vite plugins\n * for SSR pass them explicitly via the `vite` option.\n */\n const maizzleConfig: InlineConfig = {\n configFile: false,\n plugins: [\n rawExtract(),\n codeBlockExtract(),\n markdownExtract(),\n rowSourceLocation(),\n {\n name: 'maizzle:virtual-sfc',\n resolveId(id) {\n if (id === VIRTUAL_SFC_ID) return id\n },\n load(id) {\n if (id === VIRTUAL_SFC_ID) return virtualSfcSource\n },\n },\n {\n name: 'maizzle:source-override',\n load(id) {\n const override = sourceOverrides.get(id.split('?')[0])\n if (override !== undefined) return override\n },\n },\n vue({\n include: [/\\.vue$/, /\\.md$/],\n template: {\n transformAssetUrls: false,\n compilerOptions: {\n /**\n * AMP4Email tags (<amp-carousel>, <amp-img>, <amp-list> ...)\n * render verbatim — skip the component resolver. Users who\n * want to wrap an amp tag in a Vue component should register\n * it under a PascalCase name (e.g. `components/AmpCarousel.vue`\n * → `<AmpCarousel>`).\n */\n isCustomElement: (tag: string) => tag.startsWith('amp-'),\n },\n },\n }),\n Markdown(merge(markdownOptions ?? {}, {\n headEnabled: true,\n wrapperDiv: false,\n wrapperClasses: 'prose',\n wrapperComponent: (id: string, raw: string) => {\n const fm = raw.match(/^---\\r?\\n([\\s\\S]*?)\\r?\\n---/)?.[1]\n const layout = fm?.match(/^[ \\t]*layout[ \\t]*:[ \\t]*['\"]?([A-Za-z][\\w-]*|false|none)['\"]?[ \\t]*$/m)?.[1]\n if (layout === 'false' || layout === 'none') return null\n if (layout) return layout\n /**\n * No `layout:` set — default to the built-in `MarkdownLayout`\n * for entry-template `.md` files. Skip for `.md` files inside\n * component dirs, which are reusable fragments imported into\n * other templates.\n */\n const inComponentDir = componentDirsAbs.some(d => id === d || id.startsWith(`${d}/`))\n return inComponentDir ? null : 'MarkdownLayout'\n },\n markdownOptions: {\n async highlight(code: string, lang: string) {\n const { codeToHtml } = await import('shiki')\n return codeToHtml(code, { lang, theme: shikiTheme })\n },\n },\n markdownSetup(md: MarkdownExit) {\n const defaultFence = md.renderer.rules.fence!\n md.renderer.rules.fence = (...args) =>\n Promise.resolve(defaultFence(...args)).then(shikiToCodeBlock)\n\n const defaultCodeBlock = md.renderer.rules.code_block!\n md.renderer.rules.code_block = (...args) => shikiToCodeBlock(defaultCodeBlock(...args) as string)\n },\n })),\n AutoImport({\n dirs: [\n resolve(__dirname, '../composables'),\n resolve(__dirname, '../filters'),\n ],\n imports: ['vue', unheadVueComposablesImports],\n /**\n * unplugin-auto-import's default `include` doesn't match `.md`, so\n * auto-imports (Vue, unhead and Maizzle composables/filters) were\n * never injected into Markdown templates — `useConfig()` and friends\n * threw at runtime. Extend the default list with `.md` (and its\n * `?vue` script sub-requests) to mirror the `.md` coverage the\n * Components plugin already declares below.\n */\n include: [/\\.[jt]sx?$/, /\\.vue$/, /\\.vue\\?vue/, /\\.md$/, /\\.md\\?vue/],\n dts: dts ? resolve(dtsDir, 'auto-imports.d.ts') : false,\n }),\n Components({\n extensions: ['vue', 'md'],\n include: [/\\.vue$/, /\\.vue\\?vue/, /\\.md$/],\n dirs: [\n frameworkComponentsDir,\n resolve(root, 'components'),\n ...dirSources.map(s => s.path),\n ],\n /**\n * Drop built-in component files whose name the user has shadowed.\n * This makes the user's version the only match — no \"naming\n * conflicts\" warning, no glob-ordering games.\n */\n globsExclude: frameworkExcludes,\n directoryAsNamespace: true,\n collapseSamePrefixes: true,\n resolvers: prefixedSources.length > 0 ? [prefixedResolver] : undefined,\n dts: dts ? resolve(dtsDir, 'components.d.ts') : false,\n }),\n ...(prefixedSourceWatcher ? [prefixedSourceWatcher] : []),\n ],\n resolve: {\n alias: {\n 'vue/server-renderer': resolve(vueServerRendererPkgDir, 'dist/server-renderer.esm-bundler.js'),\n 'vue': resolve(vuePkgDir, 'dist/vue.runtime.esm-bundler.js'),\n 'vue-router': vueRouterPkgDir,\n '@unhead/vue/server': resolve(unheadVuePkgDir, 'dist/server.mjs'),\n '@unhead/vue': resolve(unheadVuePkgDir, 'dist/index.mjs'),\n },\n },\n server: {\n middlewareMode: true,\n hmr: false,\n /**\n * Watcher is required so unplugin-vue-components and unplugin-auto-import\n * detect added/removed component files and rewrite their .d.ts on the fly.\n * (We only render via SSR — HMR is off, but chokidar still drives plugins.)\n */\n fs: {\n allow: [process.cwd(), root, ...componentDirs.map(s => s.path), vuePkgDir, vueServerRendererPkgDir, unheadVuePkgDir, vueRouterPkgDir],\n },\n },\n appType: 'custom',\n logLevel: 'silent',\n optimizeDeps: {\n noDiscovery: true,\n },\n }\n\n /**\n * Merge user's vite config (from config.vite) under Maizzle's config.\n * mergeConfig(a, b) → b overrides a for scalars, arrays concatenate.\n * This ensures Maizzle's critical settings (middlewareMode, appType,\n * etc.) always win, while user plugins and other options remain.\n */\n const finalConfig = userViteConfig\n ? mergeConfig(userViteConfig, maizzleConfig)\n : maizzleConfig\n\n const server = await createServer(finalConfig)\n\n return {\n async render(input: string | Component, config: MaizzleConfig, opts?: { source?: string }): Promise<RenderedTemplate> {\n let component: Component\n let configKey: InjectionKey<MaizzleConfig>\n let contextKey: InjectionKey<RenderContext>\n\n if (typeof input === 'string') {\n /**\n * String input goes through Vite — must use ssrLoadModule for\n * injection keys so they share the same module instance as SFC.\n */\n const configModule = await server.ssrLoadModule(resolve(__dirname, '../composables/useConfig'))\n const contextModule = await server.ssrLoadModule(resolve(__dirname, '../composables/renderContext'))\n configKey = configModule.MaizzleConfigKey\n contextKey = contextModule.RenderContextKey\n\n if (input.includes('<template') || input.includes('<script')) {\n virtualSfcSource = input\n const mod = server.moduleGraph.getModuleById(VIRTUAL_SFC_ID)\n if (mod) server.moduleGraph.invalidateModule(mod)\n component = (await server.ssrLoadModule(VIRTUAL_SFC_ID)).default\n } else {\n /**\n * A beforeRender handler may have rewritten the source. Register it\n * under the real path id and invalidate so ssrLoadModule compiles\n * the override; clear + invalidate afterwards so the override never\n * leaks into a later render of the same path.\n */\n const hasOverride = opts?.source !== undefined\n if (hasOverride) {\n sourceOverrides.set(input, opts!.source!)\n const mod = await server.moduleGraph.getModuleByUrl(input)\n if (mod) server.moduleGraph.invalidateModule(mod)\n }\n try {\n component = (await server.ssrLoadModule(input)).default\n } finally {\n if (hasOverride) {\n sourceOverrides.delete(input)\n const mod = await server.moduleGraph.getModuleByUrl(input)\n if (mod) server.moduleGraph.invalidateModule(mod)\n }\n }\n }\n } else {\n // Pre-compiled component — use directly imported keys\n component = input\n configKey = MaizzleConfigKey\n contextKey = RenderContextKey\n }\n\n const renderContext: RenderContext = {\n doctype: undefined,\n sfcConfig: undefined,\n sfcEventHandlers: [],\n }\n\n const head = createHead({ disableDefaults: true })\n const app = createSSRApp(component)\n app.use(head)\n\n // Register user Vue plugins, directives, and global properties\n if (config.vue) {\n const plugins = typeof config.vue.plugins === 'function'\n ? config.vue.plugins()\n : config.vue.plugins ?? []\n for (const plugin of plugins) {\n app.use(plugin)\n }\n for (const [name, directive] of Object.entries(config.vue.directives ?? {})) {\n app.directive(name, directive)\n }\n Object.assign(app.config.globalProperties, config.vue.globalProperties)\n }\n\n app.provide(configKey, config)\n app.provide(contextKey, renderContext)\n\n const ssrContext: Record<string, any> = {}\n let html: string = await renderToString(app, ssrContext)\n\n const { headTags, bodyTags, bodyTagsOpen, htmlAttrs, bodyAttrs } = head.render()\n\n // Inject head entries into the rendered HTML\n if (htmlAttrs) {\n html = html.replace(/<html([^>]*)>/, `<html$1 ${htmlAttrs}>`)\n }\n if (headTags) {\n html = html.replace('</head>', `${headTags}\\n</head>`)\n }\n if (bodyAttrs) {\n html = html.replace(/<body([^>]*)>/, `<body$1 ${bodyAttrs}>`)\n }\n if (bodyTagsOpen) {\n html = html.replace(/<body([^>]*)>/, `<body$1>\\n${bodyTagsOpen}`)\n }\n if (bodyTags) {\n html = html.replace('</body>', `${bodyTags}\\n</body>`)\n }\n\n // Inject SSR teleport content into their target elements\n const hasTeleports = ssrContext.teleports && Object.keys(ssrContext.teleports).length > 0\n const hasFonts = (renderContext.fonts?.length ?? 0) > 0\n\n if (hasTeleports || hasFonts) {\n const { parse: parseDom, serialize: serializeDom, walk } = await import('../utils/ast/index.ts')\n let dom = parseDom(html)\n\n if (hasTeleports) {\n for (const [rawTarget, content] of Object.entries(ssrContext.teleports) as [string, string][]) {\n if (!content) continue\n\n const prepend = rawTarget.endsWith(':start')\n const target = prepend ? rawTarget.slice(0, -6) : rawTarget\n const targetChildren = parseDom(content)\n\n walk(dom, (node) => {\n const el = node as import('domhandler').Element\n\n if (!el.name) return\n\n const matched\n = target === el.name\n || (target.startsWith('#') && el.attribs?.id === target.slice(1))\n || (target.startsWith('.') && el.attribs?.class?.split(/\\s+/).includes(target.slice(1)))\n\n if (matched) {\n for (const child of targetChildren) {\n child.parent = el as any\n }\n\n el.children = prepend\n ? [...targetChildren, ...(el.children || [])] as any\n : [...(el.children || []), ...targetChildren] as any\n }\n })\n }\n }\n\n if (hasFonts) {\n const { injectFonts } = await import('./injectFonts.ts')\n injectFonts(dom, renderContext.fonts!, parseDom, walk)\n }\n\n html = serializeDom(dom)\n }\n\n // Inject preheader text from usePreheader() composable\n if (renderContext.preheader) {\n const { text, fillerCount } = renderContext.preheader\n const filler = '\\u2007\\uFEFF\\u034F '.repeat(fillerCount)\n const previewHtml = `<div style=\"display:none\">${text}${filler}\\u00A0</div>`\n html = html.replace(/<body([^>]*)>/, `<body$1>${previewHtml}`)\n }\n\n /**\n * Strip Vue SSR fragment markers + teleport anchor comments. These\n * are rendering hygiene, not transformer concerns — must run\n * regardless of `useTransformers` state. Fragment markers contain\n * `-->`, which would prematurely terminate MSO conditional\n * comments downstream.\n */\n html = html\n .replaceAll('<!--[-->', '')\n .replaceAll('<!--]-->', '')\n .replaceAll('<!--teleport start anchor-->', '')\n .replaceAll('<!--teleport anchor-->', '')\n .replaceAll('<!--teleport start-->', '')\n .replaceAll('<!--teleport end-->', '')\n\n return {\n html,\n doctype: renderContext.doctype,\n /**\n * Layer sfcConfig over config — sfcConfig is a partial override\n * emitted by composables (defineConfig, useTransformers, etc.).\n * A naive replacement (`sfcConfig ?? config`) drops defaults\n * from the resolved config when the SFC only sets a single\n * key, since the composables' inject() of globalConfig can\n * return `{}` in dev when ssrLoadModule and the SFC's\n * auto-imported module resolve to different module\n * instances (different Symbols).\n */\n templateConfig: renderContext.sfcConfig ? merge(renderContext.sfcConfig, config) : config,\n sfcEventHandlers: renderContext.sfcEventHandlers,\n plaintext: renderContext.plaintext,\n outputPath: renderContext.outputPath,\n tailwindBlocks: renderContext.tailwindBlocks,\n }\n },\n\n async invalidate(filePath: string): Promise<void> {\n const mod = await server.moduleGraph.getModuleByUrl(filePath)\n if (mod) {\n server.moduleGraph.invalidateModule(mod)\n }\n },\n\n async invalidateAll(): Promise<void> {\n for (const mod of server.moduleGraph.idToModuleMap.values()) {\n server.moduleGraph.invalidateModule(mod)\n }\n },\n\n async close(): Promise<void> {\n await server.close()\n /**\n * unplugin-auto-import schedules a 500ms-throttled, fire-and-forget\n * d.ts write on its first scan. server.close() doesn't drain that\n * pending write, so callers tearing down the working dir right\n * after close (tests, ephemeral build pipelines) can race the\n * mkdir against a missing parent directory. Wait one throttle\n * window past close so the lingering write resolves while\n * the dir still exists.\n */\n if (dts) {\n await new Promise(resolve => setTimeout(resolve, 600))\n }\n },\n }\n}\n"],"mappings":"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"}
@@ -1,7 +1,7 @@
1
1
  import { resolveConfig } from "../config/index.js";
2
- import { runTransformers } from "../transformers/index.js";
3
2
  import { normalizeComponentSources } from "../utils/componentSources.js";
4
3
  import { createRenderer } from "./createRenderer.js";
4
+ import { runTransformers } from "../transformers/index.js";
5
5
  import { createPlaintext } from "../plaintext.js";
6
6
  import { stripForHtml, stripForPlaintext } from "../utils/output-markers.js";
7
7
  import { getActiveRenderer } from "./active.js";
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","names":[],"sources":["../../src/render/index.ts"],"sourcesContent":["import { resolve, extname } from 'node:path'\nimport { resolveConfig } from '../config/index.ts'\nimport { runTransformers } from '../transformers/index.ts'\nimport { createPlaintext } from '../plaintext.ts'\nimport { stripForHtml, stripForPlaintext } from '../utils/output-markers.ts'\nimport defu from 'defu'\nimport type { Component } from 'vue'\nimport type { MaizzleConfig } from '../types/index.ts'\nimport { createRenderer } from './createRenderer.ts'\nimport { getActiveRenderer } from './active.ts'\nimport { normalizeComponentSources } from '../utils/componentSources.ts'\n\nexport type { Renderer, RenderedTemplate, CreateRendererOptions } from './createRenderer.ts'\nexport { createRenderer } from './createRenderer.ts'\n\nexport interface RenderResult {\n html: string\n config: MaizzleConfig\n plaintext?: string\n}\n\n/**\n * Render a Vue SFC email template to a fully-transformed HTML string.\n * Accepts a file path or a raw SFC source string.\n */\nexport async function render(\n template: string | Component,\n config?: Partial<MaizzleConfig>,\n): Promise<RenderResult> {\n if (template == null) {\n throw new Error(\n `render() received ${template}. If you used \\`import X from './x.vue'\\`, Node cannot load .vue files natively — pass the path string instead: render('./x.vue').`,\n )\n }\n if (typeof template !== 'string' && typeof template !== 'object' && typeof template !== 'function') {\n throw new TypeError(\n `render() expected a file path or SFC source string, got ${typeof template}.`,\n )\n }\n\n const resolvedConfig = await resolveConfig(config)\n\n /**\n * Reuse a renderer started by the Vite plugin when one is active.\n * Spinning up a fresh Vite SSR server inside a host Vite dev process\n * (e.g. TanStack Start) collides on env wiring and throws\n * \"outsideEmitter undefined\".\n */\n const active = getActiveRenderer()\n const renderer = active ?? await createRenderer({\n markdown: resolvedConfig.markdown,\n root: resolvedConfig.root,\n componentDirs: normalizeComponentSources(resolvedConfig.components?.source, process.cwd()),\n vite: resolvedConfig.vite,\n })\n\n try {\n const isFile = typeof template === 'string'\n && ['.vue', '.md'].includes(extname(template))\n && !template.includes('\\n')\n\n const rendered = await renderer.render(isFile ? resolve(template) : template, resolvedConfig)\n let html = rendered.html\n\n const doctype = rendered.doctype ?? rendered.templateConfig.doctype ?? '<!DOCTYPE html>'\n\n if (rendered.templateConfig.useTransformers !== false) {\n html = await runTransformers(html, rendered.templateConfig, isFile ? resolve(template) : undefined, doctype, rendered.tailwindBlocks)\n }\n if (doctype) html = `${doctype}\\n${html}`\n\n const globalPlaintext = rendered.templateConfig.plaintext\n const sfcPlaintext = rendered.plaintext\n\n let plaintextResult: string | undefined\n\n if (globalPlaintext || sfcPlaintext) {\n const globalCfg = typeof globalPlaintext === 'object' ? globalPlaintext : {}\n const stripOptions = defu(sfcPlaintext?.options, globalCfg.options)\n plaintextResult = createPlaintext(stripForPlaintext(html), stripOptions)\n }\n\n return { html: stripForHtml(html), config: rendered.templateConfig, plaintext: plaintextResult }\n } finally {\n if (!active) await renderer.close()\n }\n}\n"],"mappings":";;;;;;;;;;;;;;AAyBA,eAAsB,OACpB,UACA,QACuB;CACvB,IAAI,YAAY,MACd,MAAM,IAAI,MACR,qBAAqB,SAAS,mIAChC;CAEF,IAAI,OAAO,aAAa,YAAY,OAAO,aAAa,YAAY,OAAO,aAAa,YACtF,MAAM,IAAI,UACR,2DAA2D,OAAO,SAAS,EAC7E;CAGF,MAAM,iBAAiB,MAAM,cAAc,MAAM;;;;;;;CAQjD,MAAM,SAAS,kBAAkB;CACjC,MAAM,WAAW,UAAU,MAAM,eAAe;EAC9C,UAAU,eAAe;EACzB,MAAM,eAAe;EACrB,eAAe,0BAA0B,eAAe,YAAY,QAAQ,QAAQ,IAAI,CAAC;EACzF,MAAM,eAAe;CACvB,CAAC;CAED,IAAI;EACF,MAAM,SAAS,OAAO,aAAa,YAC9B,CAAC,QAAQ,KAAK,EAAE,SAAS,QAAQ,QAAQ,CAAC,KAC1C,CAAC,SAAS,SAAS,IAAI;EAE5B,MAAM,WAAW,MAAM,SAAS,OAAO,SAAS,QAAQ,QAAQ,IAAI,UAAU,cAAc;EAC5F,IAAI,OAAO,SAAS;EAEpB,MAAM,UAAU,SAAS,WAAW,SAAS,eAAe,WAAW;EAEvE,IAAI,SAAS,eAAe,oBAAoB,OAC9C,OAAO,MAAM,gBAAgB,MAAM,SAAS,gBAAgB,SAAS,QAAQ,QAAQ,IAAI,KAAA,GAAW,SAAS,SAAS,cAAc;EAEtI,IAAI,SAAS,OAAO,GAAG,QAAQ,IAAI;EAEnC,MAAM,kBAAkB,SAAS,eAAe;EAChD,MAAM,eAAe,SAAS;EAE9B,IAAI;EAEJ,IAAI,mBAAmB,cAAc;GACnC,MAAM,YAAY,OAAO,oBAAoB,WAAW,kBAAkB,CAAC;GAC3E,MAAM,eAAe,KAAK,cAAc,SAAS,UAAU,OAAO;GAClE,kBAAkB,gBAAgB,kBAAkB,IAAI,GAAG,YAAY;EACzE;EAEA,OAAO;GAAE,MAAM,aAAa,IAAI;GAAG,QAAQ,SAAS;GAAgB,WAAW;EAAgB;CACjG,UAAU;EACR,IAAI,CAAC,QAAQ,MAAM,SAAS,MAAM;CACpC;AACF"}
1
+ {"version":3,"file":"index.js","names":[],"sources":["../../src/render/index.ts"],"sourcesContent":["import { resolve, extname } from 'node:path'\nimport { resolveConfig } from '../config/index.ts'\nimport { runTransformers } from '../transformers/index.ts'\nimport { createPlaintext } from '../plaintext.ts'\nimport { stripForHtml, stripForPlaintext } from '../utils/output-markers.ts'\nimport defu from 'defu'\nimport type { Component } from 'vue'\nimport type { MaizzleConfig } from '../types/index.ts'\nimport { createRenderer } from './createRenderer.ts'\nimport { getActiveRenderer } from './active.ts'\nimport { normalizeComponentSources } from '../utils/componentSources.ts'\n\nexport type { Renderer, RenderedTemplate, CreateRendererOptions } from './createRenderer.ts'\nexport { createRenderer } from './createRenderer.ts'\n\nexport interface RenderResult {\n html: string\n config: MaizzleConfig\n plaintext?: string\n}\n\n/**\n * Render a Vue SFC email template to a fully-transformed HTML string.\n * Accepts a file path or a raw SFC source string.\n */\nexport async function render(\n template: string | Component,\n config?: Partial<MaizzleConfig>,\n): Promise<RenderResult> {\n if (template == null) {\n throw new Error(\n `render() received ${template}. If you used \\`import X from './x.vue'\\`, Node cannot load .vue files natively — pass the path string instead: render('./x.vue').`,\n )\n }\n if (typeof template !== 'string' && typeof template !== 'object' && typeof template !== 'function') {\n throw new TypeError(\n `render() expected a file path or SFC source string, got ${typeof template}.`,\n )\n }\n\n const resolvedConfig = await resolveConfig(config)\n\n /**\n * Reuse a renderer started by the Vite plugin when one is active.\n * Spinning up a fresh Vite SSR server inside a host Vite dev process\n * (e.g. TanStack Start) collides on env wiring and throws\n * \"outsideEmitter undefined\".\n */\n const active = getActiveRenderer()\n const renderer = active ?? await createRenderer({\n markdown: resolvedConfig.markdown,\n root: resolvedConfig.root,\n componentDirs: normalizeComponentSources(resolvedConfig.components?.source, process.cwd()),\n vite: resolvedConfig.vite,\n })\n\n try {\n const isFile = typeof template === 'string'\n && ['.vue', '.md'].includes(extname(template))\n && !template.includes('\\n')\n\n const rendered = await renderer.render(isFile ? resolve(template) : template, resolvedConfig)\n let html = rendered.html\n\n const doctype = rendered.doctype ?? rendered.templateConfig.doctype ?? '<!DOCTYPE html>'\n\n if (rendered.templateConfig.useTransformers !== false) {\n html = await runTransformers(html, rendered.templateConfig, isFile ? resolve(template) : undefined, doctype, rendered.tailwindBlocks)\n }\n if (doctype) html = `${doctype}\\n${html}`\n\n const globalPlaintext = rendered.templateConfig.plaintext\n const sfcPlaintext = rendered.plaintext\n\n let plaintextResult: string | undefined\n\n if (globalPlaintext || sfcPlaintext) {\n const globalCfg = typeof globalPlaintext === 'object' ? globalPlaintext : {}\n const stripOptions = defu(sfcPlaintext?.options, globalCfg.options)\n plaintextResult = createPlaintext(stripForPlaintext(html), stripOptions)\n }\n\n return { html: stripForHtml(html), config: rendered.templateConfig, plaintext: plaintextResult }\n } finally {\n if (!active) await renderer.close()\n }\n}\n"],"mappings":";;;;;;;;;;;;;;AAyBA,eAAsB,OACpB,UACA,QACuB;CACvB,IAAI,YAAY,MACd,MAAM,IAAI,MACR,qBAAqB,SAAS,mIAChC;CAEF,IAAI,OAAO,aAAa,YAAY,OAAO,aAAa,YAAY,OAAO,aAAa,YACtF,MAAM,IAAI,UACR,2DAA2D,OAAO,SAAS,EAC7E;CAGF,MAAM,iBAAiB,MAAM,cAAc,MAAM;;;;;;;CAQjD,MAAM,SAAS,kBAAkB;CACjC,MAAM,WAAW,UAAU,MAAM,eAAe;EAC9C,UAAU,eAAe;EACzB,MAAM,eAAe;EACrB,eAAe,0BAA0B,eAAe,YAAY,QAAQ,QAAQ,IAAI,CAAC;EACzF,MAAM,eAAe;CACvB,CAAC;CAED,IAAI;EACF,MAAM,SAAS,OAAO,aAAa,YAC9B,CAAC,QAAQ,KAAK,CAAC,CAAC,SAAS,QAAQ,QAAQ,CAAC,KAC1C,CAAC,SAAS,SAAS,IAAI;EAE5B,MAAM,WAAW,MAAM,SAAS,OAAO,SAAS,QAAQ,QAAQ,IAAI,UAAU,cAAc;EAC5F,IAAI,OAAO,SAAS;EAEpB,MAAM,UAAU,SAAS,WAAW,SAAS,eAAe,WAAW;EAEvE,IAAI,SAAS,eAAe,oBAAoB,OAC9C,OAAO,MAAM,gBAAgB,MAAM,SAAS,gBAAgB,SAAS,QAAQ,QAAQ,IAAI,KAAA,GAAW,SAAS,SAAS,cAAc;EAEtI,IAAI,SAAS,OAAO,GAAG,QAAQ,IAAI;EAEnC,MAAM,kBAAkB,SAAS,eAAe;EAChD,MAAM,eAAe,SAAS;EAE9B,IAAI;EAEJ,IAAI,mBAAmB,cAAc;GACnC,MAAM,YAAY,OAAO,oBAAoB,WAAW,kBAAkB,CAAC;GAC3E,MAAM,eAAe,KAAK,cAAc,SAAS,UAAU,OAAO;GAClE,kBAAkB,gBAAgB,kBAAkB,IAAI,GAAG,YAAY;EACzE;EAEA,OAAO;GAAE,MAAM,aAAa,IAAI;GAAG,QAAQ,SAAS;GAAgB,WAAW;EAAgB;CACjG,UAAU;EACR,IAAI,CAAC,QAAQ,MAAM,SAAS,MAAM;CACpC;AACF"}
@@ -1 +1 @@
1
- {"version":3,"file":"injectFonts.js","names":[],"sources":["../../src/render/injectFonts.ts"],"sourcesContent":["import type { ChildNode, Element } from 'domhandler'\nimport type { FontRegistration } from '../composables/renderContext.ts'\nimport { decodeStyleEntities } from '../utils/decodeStyleEntities.ts'\n\nconst TAILWIND_IMPORT_RE = /((@import|@reference)\\s+[\"'](tailwindcss|@maizzle\\/tailwindcss)|@tailwind\\s)/\n\nfunction getText(el: Element): string {\n return (el.children || [])\n .filter((c: any) => c.type === 'text')\n .map((c: any) => c.data)\n .join('')\n}\n\n/**\n * Inject font `<link>` tags into `<head>` and merge `@theme` declarations\n * into the template's existing Tailwind `<style>` block (so the\n * `font-{slug}` utilities are generated in the same compilation unit\n * as the Tailwind import). Without a Tailwind import, emits plain\n * `.font-{slug}` class rules so the utility still works.\n */\nexport function injectFonts(\n dom: ChildNode[],\n fonts: FontRegistration[],\n parseDom: (html: string) => ChildNode[],\n walk: (ast: ChildNode[], cb: (n: ChildNode) => void) => void,\n): void {\n if (!fonts.length) return\n\n let head: Element | undefined\n let tailwindStyle: Element | undefined\n\n walk(dom, (node) => {\n const el = node as Element\n if (!el.name) return\n if (el.name === 'head' && !head) head = el\n if (el.name === 'style' && !tailwindStyle && TAILWIND_IMPORT_RE.test(decodeStyleEntities(getText(el)))) {\n tailwindStyle = el\n }\n })\n\n if (!head) return\n\n const linkHtml = fonts\n .map(f => `<link href=\"${f.url}\" rel=\"stylesheet\" media=\"screen\">`)\n .join('')\n const linkNodes = parseDom(linkHtml)\n for (const child of linkNodes) {\n (child as any).parent = head\n }\n head.children = [...(head.children || []), ...linkNodes] as any\n\n if (tailwindStyle) {\n const themeDecls = fonts\n .map(f => ` --font-${f.slug}: ${f.declaration};`)\n .join('\\n')\n const existing = getText(tailwindStyle)\n tailwindStyle.children = [{\n type: 'text',\n data: `${existing}\\n@theme {\\n${themeDecls}\\n}\\n`,\n parent: tailwindStyle,\n } as any]\n } else {\n const classRules = fonts\n .map(f => `.font-${f.slug} { font-family: ${f.declaration}; }`)\n .join('\\n')\n const styleNodes = parseDom(`<style>\\n${classRules}\\n</style>`)\n for (const child of styleNodes) {\n (child as any).parent = head\n }\n head.children = [...(head.children || []), ...styleNodes] as any\n }\n}\n"],"mappings":";;AAIA,MAAM,qBAAqB;AAE3B,SAAS,QAAQ,IAAqB;CACpC,QAAQ,GAAG,YAAY,CAAC,GACrB,QAAQ,MAAW,EAAE,SAAS,MAAM,EACpC,KAAK,MAAW,EAAE,IAAI,EACtB,KAAK,EAAE;AACZ;;;;;;;;AASA,SAAgB,YACd,KACA,OACA,UACA,MACM;CACN,IAAI,CAAC,MAAM,QAAQ;CAEnB,IAAI;CACJ,IAAI;CAEJ,KAAK,MAAM,SAAS;EAClB,MAAM,KAAK;EACX,IAAI,CAAC,GAAG,MAAM;EACd,IAAI,GAAG,SAAS,UAAU,CAAC,MAAM,OAAO;EACxC,IAAI,GAAG,SAAS,WAAW,CAAC,iBAAiB,mBAAmB,KAAK,oBAAoB,QAAQ,EAAE,CAAC,CAAC,GACnG,gBAAgB;CAEpB,CAAC;CAED,IAAI,CAAC,MAAM;CAKX,MAAM,YAAY,SAHD,MACd,KAAI,MAAK,eAAe,EAAE,IAAI,mCAAmC,EACjE,KAAK,EAC0B,CAAC;CACnC,KAAK,MAAM,SAAS,WAClB,MAAe,SAAS;CAE1B,KAAK,WAAW,CAAC,GAAI,KAAK,YAAY,CAAC,GAAI,GAAG,SAAS;CAEvD,IAAI,eAAe;EACjB,MAAM,aAAa,MAChB,KAAI,MAAK,YAAY,EAAE,KAAK,IAAI,EAAE,YAAY,EAAE,EAChD,KAAK,IAAI;EACZ,MAAM,WAAW,QAAQ,aAAa;EACtC,cAAc,WAAW,CAAC;GACxB,MAAM;GACN,MAAM,GAAG,SAAS,cAAc,WAAW;GAC3C,QAAQ;EACV,CAAQ;CACV,OAAO;EAIL,MAAM,aAAa,SAAS,YAHT,MAChB,KAAI,MAAK,SAAS,EAAE,KAAK,kBAAkB,EAAE,YAAY,IAAI,EAC7D,KAAK,IACyC,EAAE,WAAW;EAC9D,KAAK,MAAM,SAAS,YAClB,MAAe,SAAS;EAE1B,KAAK,WAAW,CAAC,GAAI,KAAK,YAAY,CAAC,GAAI,GAAG,UAAU;CAC1D;AACF"}
1
+ {"version":3,"file":"injectFonts.js","names":[],"sources":["../../src/render/injectFonts.ts"],"sourcesContent":["import type { ChildNode, Element } from 'domhandler'\nimport type { FontRegistration } from '../composables/renderContext.ts'\nimport { decodeStyleEntities } from '../utils/decodeStyleEntities.ts'\n\nconst TAILWIND_IMPORT_RE = /((@import|@reference)\\s+[\"'](tailwindcss|@maizzle\\/tailwindcss)|@tailwind\\s)/\n\nfunction getText(el: Element): string {\n return (el.children || [])\n .filter((c: any) => c.type === 'text')\n .map((c: any) => c.data)\n .join('')\n}\n\n/**\n * Inject font `<link>` tags into `<head>` and merge `@theme` declarations\n * into the template's existing Tailwind `<style>` block (so the\n * `font-{slug}` utilities are generated in the same compilation unit\n * as the Tailwind import). Without a Tailwind import, emits plain\n * `.font-{slug}` class rules so the utility still works.\n */\nexport function injectFonts(\n dom: ChildNode[],\n fonts: FontRegistration[],\n parseDom: (html: string) => ChildNode[],\n walk: (ast: ChildNode[], cb: (n: ChildNode) => void) => void,\n): void {\n if (!fonts.length) return\n\n let head: Element | undefined\n let tailwindStyle: Element | undefined\n\n walk(dom, (node) => {\n const el = node as Element\n if (!el.name) return\n if (el.name === 'head' && !head) head = el\n if (el.name === 'style' && !tailwindStyle && TAILWIND_IMPORT_RE.test(decodeStyleEntities(getText(el)))) {\n tailwindStyle = el\n }\n })\n\n if (!head) return\n\n const linkHtml = fonts\n .map(f => `<link href=\"${f.url}\" rel=\"stylesheet\" media=\"screen\">`)\n .join('')\n const linkNodes = parseDom(linkHtml)\n for (const child of linkNodes) {\n (child as any).parent = head\n }\n head.children = [...(head.children || []), ...linkNodes] as any\n\n if (tailwindStyle) {\n const themeDecls = fonts\n .map(f => ` --font-${f.slug}: ${f.declaration};`)\n .join('\\n')\n const existing = getText(tailwindStyle)\n tailwindStyle.children = [{\n type: 'text',\n data: `${existing}\\n@theme {\\n${themeDecls}\\n}\\n`,\n parent: tailwindStyle,\n } as any]\n } else {\n const classRules = fonts\n .map(f => `.font-${f.slug} { font-family: ${f.declaration}; }`)\n .join('\\n')\n const styleNodes = parseDom(`<style>\\n${classRules}\\n</style>`)\n for (const child of styleNodes) {\n (child as any).parent = head\n }\n head.children = [...(head.children || []), ...styleNodes] as any\n }\n}\n"],"mappings":";;AAIA,MAAM,qBAAqB;AAE3B,SAAS,QAAQ,IAAqB;CACpC,QAAQ,GAAG,YAAY,CAAC,EAAA,CACrB,QAAQ,MAAW,EAAE,SAAS,MAAM,CAAC,CACrC,KAAK,MAAW,EAAE,IAAI,CAAC,CACvB,KAAK,EAAE;AACZ;;;;;;;;AASA,SAAgB,YACd,KACA,OACA,UACA,MACM;CACN,IAAI,CAAC,MAAM,QAAQ;CAEnB,IAAI;CACJ,IAAI;CAEJ,KAAK,MAAM,SAAS;EAClB,MAAM,KAAK;EACX,IAAI,CAAC,GAAG,MAAM;EACd,IAAI,GAAG,SAAS,UAAU,CAAC,MAAM,OAAO;EACxC,IAAI,GAAG,SAAS,WAAW,CAAC,iBAAiB,mBAAmB,KAAK,oBAAoB,QAAQ,EAAE,CAAC,CAAC,GACnG,gBAAgB;CAEpB,CAAC;CAED,IAAI,CAAC,MAAM;CAKX,MAAM,YAAY,SAHD,MACd,KAAI,MAAK,eAAe,EAAE,IAAI,mCAAmC,CAAC,CAClE,KAAK,EAC0B,CAAC;CACnC,KAAK,MAAM,SAAS,WAClB,MAAe,SAAS;CAE1B,KAAK,WAAW,CAAC,GAAI,KAAK,YAAY,CAAC,GAAI,GAAG,SAAS;CAEvD,IAAI,eAAe;EACjB,MAAM,aAAa,MAChB,KAAI,MAAK,YAAY,EAAE,KAAK,IAAI,EAAE,YAAY,EAAE,CAAC,CACjD,KAAK,IAAI;EACZ,MAAM,WAAW,QAAQ,aAAa;EACtC,cAAc,WAAW,CAAC;GACxB,MAAM;GACN,MAAM,GAAG,SAAS,cAAc,WAAW;GAC3C,QAAQ;EACV,CAAQ;CACV,OAAO;EAIL,MAAM,aAAa,SAAS,YAHT,MAChB,KAAI,MAAK,SAAS,EAAE,KAAK,kBAAkB,EAAE,YAAY,IAAI,CAAC,CAC9D,KAAK,IACyC,EAAE,WAAW;EAC9D,KAAK,MAAM,SAAS,YAClB,MAAe,SAAS;EAE1B,KAAK,WAAW,CAAC,GAAI,KAAK,YAAY,CAAC,GAAI,GAAG,UAAU;CAC1D;AACF"}
@@ -0,0 +1,31 @@
1
+ import { MaizzleConfig } from "../../types/config.js";
2
+ //#region src/render/parallel/buildWorker.d.ts
3
+ interface BuildWorkerData {
4
+ /** Template paths (glob-relative, as produced on the main thread) for this batch. */
5
+ templatePaths: string[];
6
+ /** Config file path to reload (undefined → load maizzle.config from cwd). */
7
+ configPath?: string;
8
+ /** Serialized, post-beforeCreate config data from the main thread. */
9
+ configData: Partial<MaizzleConfig>;
10
+ outputPath: string;
11
+ outputExtension: string;
12
+ contentBase: string;
13
+ }
14
+ interface BuildWorkerResult {
15
+ files: string[];
16
+ sfcAfterBuildCount: number;
17
+ }
18
+ /**
19
+ * Build one batch of templates in a worker thread.
20
+ *
21
+ * Config function hooks (beforeRender/afterRender/afterTransform) can't cross
22
+ * the thread boundary, so the worker reloads the config module to recover them,
23
+ * then overlays the main thread's serialized config DATA on top — so
24
+ * beforeCreate mutations and resolved values win while the reloaded config only
25
+ * backfills the lost functions. beforeCreate/afterBuild are owned by the main
26
+ * thread and never run here.
27
+ */
28
+ declare function run(data: BuildWorkerData): Promise<BuildWorkerResult>;
29
+ //#endregion
30
+ export { BuildWorkerData, BuildWorkerResult, run };
31
+ //# sourceMappingURL=buildWorker.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"buildWorker.d.ts","names":[],"sources":["../../../src/render/parallel/buildWorker.ts"],"mappings":";;UAQiB,eAAA;;EAEf,aAAA;EAFe;EAIf,UAAA;;EAEA,UAAA,EAAY,OAAO,CAAC,aAAA;EACpB,UAAA;EACA,eAAA;EACA,WAAA;AAAA;AAAA,UAGe,iBAAA;EACf,KAAA;EACA,kBAAkB;AAAA;;AALP;AAGb;;;;AAEoB;AAyBpB;;;iBAAsB,GAAA,CAAI,IAAA,EAAM,eAAA,GAAkB,OAAA,CAAQ,iBAAA"}
@@ -0,0 +1,66 @@
1
+ import { resolveConfig } from "../../config/index.js";
2
+ import { EventManager } from "../../events/index.js";
3
+ import { normalizeComponentSources } from "../../utils/componentSources.js";
4
+ import { createRenderer } from "../createRenderer.js";
5
+ import { buildTemplate } from "../buildTemplate.js";
6
+ import { createDefu } from "defu";
7
+ //#region src/render/parallel/buildWorker.ts
8
+ /**
9
+ * Overlay config data with array-replace (not defu's default concat) so the
10
+ * main thread's arrays — content globs, plugin/source lists — fully replace the
11
+ * reloaded config's instead of duplicating every entry.
12
+ */
13
+ const mergeConfig = createDefu((obj, key, value) => {
14
+ if (Array.isArray(value)) {
15
+ if (!(key in obj)) obj[key] = value;
16
+ return true;
17
+ }
18
+ });
19
+ /**
20
+ * Build one batch of templates in a worker thread.
21
+ *
22
+ * Config function hooks (beforeRender/afterRender/afterTransform) can't cross
23
+ * the thread boundary, so the worker reloads the config module to recover them,
24
+ * then overlays the main thread's serialized config DATA on top — so
25
+ * beforeCreate mutations and resolved values win while the reloaded config only
26
+ * backfills the lost functions. beforeCreate/afterBuild are owned by the main
27
+ * thread and never run here.
28
+ */
29
+ async function run(data) {
30
+ const { templatePaths, configPath, configData, outputPath, outputExtension, contentBase } = data;
31
+ const config = mergeConfig(configData, await resolveConfig(configPath));
32
+ const events = new EventManager();
33
+ events.registerConfig(config);
34
+ const renderer = await createRenderer({
35
+ markdown: config.markdown,
36
+ root: config.root,
37
+ componentDirs: normalizeComponentSources(config.components?.source, process.cwd()),
38
+ vite: config.vite
39
+ });
40
+ const files = [];
41
+ let sfcAfterBuildCount = 0;
42
+ try {
43
+ for (const templatePath of templatePaths) {
44
+ const result = await buildTemplate(templatePath, {
45
+ config,
46
+ renderer,
47
+ events,
48
+ outputPath,
49
+ outputExtension,
50
+ contentBase
51
+ });
52
+ files.push(...result.files);
53
+ sfcAfterBuildCount += result.sfcAfterBuildCount;
54
+ }
55
+ } finally {
56
+ await renderer.close();
57
+ }
58
+ return {
59
+ files,
60
+ sfcAfterBuildCount
61
+ };
62
+ }
63
+ //#endregion
64
+ export { run };
65
+
66
+ //# sourceMappingURL=buildWorker.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"buildWorker.js","names":[],"sources":["../../../src/render/parallel/buildWorker.ts"],"sourcesContent":["import { createDefu } from 'defu'\nimport { resolveConfig } from '../../config/index.ts'\nimport { createRenderer } from '../createRenderer.ts'\nimport { EventManager } from '../../events/index.ts'\nimport { normalizeComponentSources } from '../../utils/componentSources.ts'\nimport { buildTemplate } from '../buildTemplate.ts'\nimport type { MaizzleConfig } from '../../types/index.ts'\n\nexport interface BuildWorkerData {\n /** Template paths (glob-relative, as produced on the main thread) for this batch. */\n templatePaths: string[]\n /** Config file path to reload (undefined → load maizzle.config from cwd). */\n configPath?: string\n /** Serialized, post-beforeCreate config data from the main thread. */\n configData: Partial<MaizzleConfig>\n outputPath: string\n outputExtension: string\n contentBase: string\n}\n\nexport interface BuildWorkerResult {\n files: string[]\n sfcAfterBuildCount: number\n}\n\n/**\n * Overlay config data with array-replace (not defu's default concat) so the\n * main thread's arrays — content globs, plugin/source lists — fully replace the\n * reloaded config's instead of duplicating every entry.\n */\nconst mergeConfig = createDefu((obj, key, value) => {\n if (Array.isArray(value)) {\n if (!(key in obj)) obj[key as keyof typeof obj] = value\n return true\n }\n})\n\n/**\n * Build one batch of templates in a worker thread.\n *\n * Config function hooks (beforeRender/afterRender/afterTransform) can't cross\n * the thread boundary, so the worker reloads the config module to recover them,\n * then overlays the main thread's serialized config DATA on top — so\n * beforeCreate mutations and resolved values win while the reloaded config only\n * backfills the lost functions. beforeCreate/afterBuild are owned by the main\n * thread and never run here.\n */\nexport async function run(data: BuildWorkerData): Promise<BuildWorkerResult> {\n const { templatePaths, configPath, configData, outputPath, outputExtension, contentBase } = data\n\n const reloaded = await resolveConfig(configPath)\n const config = mergeConfig(configData, reloaded) as MaizzleConfig\n\n const events = new EventManager()\n events.registerConfig(config)\n\n const renderer = await createRenderer({\n markdown: config.markdown,\n root: config.root,\n componentDirs: normalizeComponentSources(config.components?.source, process.cwd()),\n vite: config.vite,\n })\n\n const files: string[] = []\n let sfcAfterBuildCount = 0\n\n try {\n for (const templatePath of templatePaths) {\n const result = await buildTemplate(templatePath, {\n config,\n renderer,\n events,\n outputPath,\n outputExtension,\n contentBase,\n })\n files.push(...result.files)\n sfcAfterBuildCount += result.sfcAfterBuildCount\n }\n } finally {\n await renderer.close()\n }\n\n return { files, sfcAfterBuildCount }\n}\n"],"mappings":";;;;;;;;;;;;AA8BA,MAAM,cAAc,YAAY,KAAK,KAAK,UAAU;CAClD,IAAI,MAAM,QAAQ,KAAK,GAAG;EACxB,IAAI,EAAE,OAAO,MAAM,IAAI,OAA2B;EAClD,OAAO;CACT;AACF,CAAC;;;;;;;;;;;AAYD,eAAsB,IAAI,MAAmD;CAC3E,MAAM,EAAE,eAAe,YAAY,YAAY,YAAY,iBAAiB,gBAAgB;CAG5F,MAAM,SAAS,YAAY,YAAY,MADhB,cAAc,UAAU,CACA;CAE/C,MAAM,SAAS,IAAI,aAAa;CAChC,OAAO,eAAe,MAAM;CAE5B,MAAM,WAAW,MAAM,eAAe;EACpC,UAAU,OAAO;EACjB,MAAM,OAAO;EACb,eAAe,0BAA0B,OAAO,YAAY,QAAQ,QAAQ,IAAI,CAAC;EACjF,MAAM,OAAO;CACf,CAAC;CAED,MAAM,QAAkB,CAAC;CACzB,IAAI,qBAAqB;CAEzB,IAAI;EACF,KAAK,MAAM,gBAAgB,eAAe;GACxC,MAAM,SAAS,MAAM,cAAc,cAAc;IAC/C;IACA;IACA;IACA;IACA;IACA;GACF,CAAC;GACD,MAAM,KAAK,GAAG,OAAO,KAAK;GAC1B,sBAAsB,OAAO;EAC/B;CACF,UAAU;EACR,MAAM,SAAS,MAAM;CACvB;CAEA,OAAO;EAAE;EAAO;CAAmB;AACrC"}
@@ -0,0 +1,28 @@
1
+ // Tinypool worker entry. Plain JS so it loads in a raw worker thread without a
2
+ // TS toolchain. In the published dist the implementation is already compiled to
3
+ // `.js` and is imported natively (fast); during dev/tests only the `.ts` source
4
+ // exists and is loaded through jiti.
5
+ import { fileURLToPath, pathToFileURL } from 'node:url'
6
+ import { existsSync } from 'node:fs'
7
+
8
+ let implPromise
9
+
10
+ function loadImpl() {
11
+ if (!implPromise) {
12
+ const jsPath = fileURLToPath(new URL('./buildWorker.js', import.meta.url))
13
+ if (existsSync(jsPath)) {
14
+ implPromise = import(pathToFileURL(jsPath).href)
15
+ } else {
16
+ const tsPath = fileURLToPath(new URL('./buildWorker.ts', import.meta.url))
17
+ implPromise = import('jiti').then(({ createJiti }) =>
18
+ createJiti(fileURLToPath(import.meta.url)).import(tsPath),
19
+ )
20
+ }
21
+ }
22
+ return implPromise
23
+ }
24
+
25
+ export default async function buildWorker(data) {
26
+ const impl = await loadImpl()
27
+ return impl.run(data)
28
+ }
@@ -1 +1 @@
1
- {"version":3,"file":"codeBlockExtract.d.ts","names":[],"sources":["../../../src/render/plugins/codeBlockExtract.ts"],"mappings":";;;;;AASA;;;;AAA0C;iBAA1B,gBAAA,CAAA,GAAoB,MAAM"}
1
+ {"version":3,"file":"codeBlockExtract.d.ts","names":[],"sources":["../../../src/render/plugins/codeBlockExtract.ts"],"mappings":";;;;;AASA;;;;AAA0C;iBAA1B,gBAAA,IAAoB,MAAM"}
@@ -1 +1 @@
1
- {"version":3,"file":"codeBlockExtract.js","names":[],"sources":["../../../src/render/plugins/codeBlockExtract.ts"],"sourcesContent":["import type { Plugin } from 'vite'\n\n/**\n * Vite plugin that extracts raw slot content from <CodeBlock> tags\n * and passes it as a :code prop before Vue compiles the template.\n *\n * This lets users write HTML naturally inside CodeBlock slots without\n * Vue attempting to compile it as template syntax.\n */\nexport function codeBlockExtract(): Plugin {\n // Matches <CodeBlock ...>content</CodeBlock> (and kebab-case <code-block>)\n const re = /<(CodeBlock|code-block)((?:\\s[^>]*?)?)>([\\s\\S]*?)<\\/\\1>/g\n\n return {\n name: 'maizzle:code-block-extract',\n enforce: 'pre',\n transform(code: string, id: string) {\n if (!id.endsWith('.vue') && !id.endsWith('.md')) return\n if (!code.includes('CodeBlock') && !code.includes('code-block')) return\n\n const transformed = code.replace(re, (_match, tag, attrs, content) => {\n // Skip if already has a :code or v-bind:code prop\n if (/(?:^|\\s):code\\b/.test(attrs) || /v-bind:code\\b/.test(attrs)) return _match\n\n /**\n * Strip leading/trailing blank lines, then dedent based on\n * the minimum indent of non-empty lines (à la min-indent).\n */\n const stripped = content.replace(/^\\n+/, '').replace(/\\s+$/, '')\n if (!stripped) return _match\n\n const minIndent = stripped.match(/^[ \\t]*(?=\\S)/gm)\n ?.reduce((min: number, ws: string) => Math.min(min, ws.length), Infinity) ?? 0\n\n const dedented = minIndent > 0\n ? stripped.replace(new RegExp(`^[ \\\\t]{${minIndent}}`, 'gm'), '')\n : stripped\n\n // HTML-escape for safe embedding in a static attribute value.\n const escaped = dedented\n .replace(/&/g, '&amp;')\n .replace(/\"/g, '&quot;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n\n return `<${tag}${attrs} code=\"${escaped}\" />`\n })\n\n if (transformed !== code) {\n return { code: transformed, map: null }\n }\n },\n }\n}\n"],"mappings":";;;;;;;;AASA,SAAgB,mBAA2B;CAEzC,MAAM,KAAK;CAEX,OAAO;EACL,MAAM;EACN,SAAS;EACT,UAAU,MAAc,IAAY;GAClC,IAAI,CAAC,GAAG,SAAS,MAAM,KAAK,CAAC,GAAG,SAAS,KAAK,GAAG;GACjD,IAAI,CAAC,KAAK,SAAS,WAAW,KAAK,CAAC,KAAK,SAAS,YAAY,GAAG;GAEjE,MAAM,cAAc,KAAK,QAAQ,KAAK,QAAQ,KAAK,OAAO,YAAY;IAEpE,IAAI,kBAAkB,KAAK,KAAK,KAAK,gBAAgB,KAAK,KAAK,GAAG,OAAO;;;;;IAMzE,MAAM,WAAW,QAAQ,QAAQ,QAAQ,EAAE,EAAE,QAAQ,QAAQ,EAAE;IAC/D,IAAI,CAAC,UAAU,OAAO;IAEtB,MAAM,YAAY,SAAS,MAAM,iBAAiB,GAC9C,QAAQ,KAAa,OAAe,KAAK,IAAI,KAAK,GAAG,MAAM,GAAG,QAAQ,KAAK;IAa/E,OAAO,IAAI,MAAM,MAAM,UAXN,YAAY,IACzB,SAAS,QAAQ,IAAI,OAAO,WAAW,UAAU,IAAI,IAAI,GAAG,EAAE,IAC9D,UAID,QAAQ,MAAM,OAAO,EACrB,QAAQ,MAAM,QAAQ,EACtB,QAAQ,MAAM,MAAM,EACpB,QAAQ,MAAM,MAEqB,EAAE;GAC1C,CAAC;GAED,IAAI,gBAAgB,MAClB,OAAO;IAAE,MAAM;IAAa,KAAK;GAAK;EAE1C;CACF;AACF"}
1
+ {"version":3,"file":"codeBlockExtract.js","names":[],"sources":["../../../src/render/plugins/codeBlockExtract.ts"],"sourcesContent":["import type { Plugin } from 'vite'\n\n/**\n * Vite plugin that extracts raw slot content from <CodeBlock> tags\n * and passes it as a :code prop before Vue compiles the template.\n *\n * This lets users write HTML naturally inside CodeBlock slots without\n * Vue attempting to compile it as template syntax.\n */\nexport function codeBlockExtract(): Plugin {\n // Matches <CodeBlock ...>content</CodeBlock> (and kebab-case <code-block>)\n const re = /<(CodeBlock|code-block)((?:\\s[^>]*?)?)>([\\s\\S]*?)<\\/\\1>/g\n\n return {\n name: 'maizzle:code-block-extract',\n enforce: 'pre',\n transform(code: string, id: string) {\n if (!id.endsWith('.vue') && !id.endsWith('.md')) return\n if (!code.includes('CodeBlock') && !code.includes('code-block')) return\n\n const transformed = code.replace(re, (_match, tag, attrs, content) => {\n // Skip if already has a :code or v-bind:code prop\n if (/(?:^|\\s):code\\b/.test(attrs) || /v-bind:code\\b/.test(attrs)) return _match\n\n /**\n * Strip leading/trailing blank lines, then dedent based on\n * the minimum indent of non-empty lines (à la min-indent).\n */\n const stripped = content.replace(/^\\n+/, '').replace(/\\s+$/, '')\n if (!stripped) return _match\n\n const minIndent = stripped.match(/^[ \\t]*(?=\\S)/gm)\n ?.reduce((min: number, ws: string) => Math.min(min, ws.length), Infinity) ?? 0\n\n const dedented = minIndent > 0\n ? stripped.replace(new RegExp(`^[ \\\\t]{${minIndent}}`, 'gm'), '')\n : stripped\n\n // HTML-escape for safe embedding in a static attribute value.\n const escaped = dedented\n .replace(/&/g, '&amp;')\n .replace(/\"/g, '&quot;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n\n return `<${tag}${attrs} code=\"${escaped}\" />`\n })\n\n if (transformed !== code) {\n return { code: transformed, map: null }\n }\n },\n }\n}\n"],"mappings":";;;;;;;;AASA,SAAgB,mBAA2B;CAEzC,MAAM,KAAK;CAEX,OAAO;EACL,MAAM;EACN,SAAS;EACT,UAAU,MAAc,IAAY;GAClC,IAAI,CAAC,GAAG,SAAS,MAAM,KAAK,CAAC,GAAG,SAAS,KAAK,GAAG;GACjD,IAAI,CAAC,KAAK,SAAS,WAAW,KAAK,CAAC,KAAK,SAAS,YAAY,GAAG;GAEjE,MAAM,cAAc,KAAK,QAAQ,KAAK,QAAQ,KAAK,OAAO,YAAY;IAEpE,IAAI,kBAAkB,KAAK,KAAK,KAAK,gBAAgB,KAAK,KAAK,GAAG,OAAO;;;;;IAMzE,MAAM,WAAW,QAAQ,QAAQ,QAAQ,EAAE,CAAC,CAAC,QAAQ,QAAQ,EAAE;IAC/D,IAAI,CAAC,UAAU,OAAO;IAEtB,MAAM,YAAY,SAAS,MAAM,iBAAiB,CAAC,EAC/C,QAAQ,KAAa,OAAe,KAAK,IAAI,KAAK,GAAG,MAAM,GAAG,QAAQ,KAAK;IAa/E,OAAO,IAAI,MAAM,MAAM,UAXN,YAAY,IACzB,SAAS,QAAQ,IAAI,OAAO,WAAW,UAAU,IAAI,IAAI,GAAG,EAAE,IAC9D,SAAA,CAID,QAAQ,MAAM,OAAO,CAAC,CACtB,QAAQ,MAAM,QAAQ,CAAC,CACvB,QAAQ,MAAM,MAAM,CAAC,CACrB,QAAQ,MAAM,MAEqB,EAAE;GAC1C,CAAC;GAED,IAAI,gBAAgB,MAClB,OAAO;IAAE,MAAM;IAAa,KAAK;GAAK;EAE1C;CACF;AACF"}
@@ -1 +1 @@
1
- {"version":3,"file":"markdownExtract.d.ts","names":[],"sources":["../../../src/render/plugins/markdownExtract.ts"],"mappings":";;;;;AASA;;;iBAAgB,eAAA,CAAA,GAAmB,MAAM"}
1
+ {"version":3,"file":"markdownExtract.d.ts","names":[],"sources":["../../../src/render/plugins/markdownExtract.ts"],"mappings":";;;;;AASA;;;iBAAgB,eAAA,IAAmB,MAAM"}
@@ -1 +1 @@
1
- {"version":3,"file":"markdownExtract.js","names":[],"sources":["../../../src/render/plugins/markdownExtract.ts"],"sourcesContent":["import { dirname, resolve } from 'node:path'\nimport { readFileSync } from 'node:fs'\nimport type { Plugin } from 'vite'\n\n/**\n * Vite plugin that pre-processes <Markdown> tags:\n * - Extracts slot content, dedents it, and passes as :content prop\n * - Resolves `src` prop to read file contents at build time\n */\nexport function markdownExtract(): Plugin {\n const re = /<(Markdown|markdown)((?:\\s[^>]*?)?)>([\\s\\S]*?)<\\/\\1>/g\n const selfClosingRe = /<(Markdown|markdown)((?:\\s[^>]*?\\bsrc\\s*=\\s*\"[^\"]*\"[^>]*?))\\/>/g\n\n return {\n name: 'maizzle:markdown-extract',\n enforce: 'pre',\n transform(code: string, id: string) {\n if (!id.endsWith('.vue') && !id.endsWith('.md')) return\n if (!code.includes('Markdown') && !code.includes('markdown')) return\n\n let transformed = code\n\n // Handle <Markdown>content</Markdown>\n transformed = transformed.replace(re, (_match, tag, attrs, content) => {\n if (/(?:^|\\s):content\\b/.test(attrs) || /v-bind:content\\b/.test(attrs)) return _match\n\n const stripped = content.replace(/^\\n+/, '').replace(/\\s+$/, '')\n if (!stripped) return _match\n\n const minIndent = stripped.match(/^[ \\t]*(?=\\S)/gm)\n ?.reduce((min: number, ws: string) => Math.min(min, ws.length), Infinity) ?? 0\n\n const dedented = minIndent > 0\n ? stripped.replace(new RegExp(`^[ \\\\t]{${minIndent}}`, 'gm'), '')\n : stripped\n\n const escaped = dedented\n .replace(/&/g, '&amp;')\n .replace(/\"/g, '&quot;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n\n return `<${tag}${attrs} content=\"${escaped}\" />`\n })\n\n // Handle <Markdown src=\"./file.md\" /> — resolve and inline file content\n transformed = transformed.replace(selfClosingRe, (_match, tag, attrs) => {\n const srcMatch = attrs.match(/\\bsrc\\s*=\\s*\"([^\"]*)\"/)\n if (!srcMatch) return _match\n\n const srcPath = srcMatch[1]\n const resolvedPath = resolve(dirname(id), srcPath)\n\n let fileContent: string\n try {\n fileContent = readFileSync(resolvedPath, 'utf-8').trim()\n } catch {\n return _match\n }\n\n // Remove src prop, add content prop\n const cleanAttrs = attrs.replace(/\\s*\\bsrc\\s*=\\s*\"[^\"]*\"/, '')\n const escaped = fileContent\n .replace(/&/g, '&amp;')\n .replace(/\"/g, '&quot;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n\n return `<${tag}${cleanAttrs} content=\"${escaped}\" />`\n })\n\n if (transformed !== code) {\n return { code: transformed, map: null }\n }\n },\n }\n}\n"],"mappings":";;;;;;;;AASA,SAAgB,kBAA0B;CACxC,MAAM,KAAK;CACX,MAAM,gBAAgB;CAEtB,OAAO;EACL,MAAM;EACN,SAAS;EACT,UAAU,MAAc,IAAY;GAClC,IAAI,CAAC,GAAG,SAAS,MAAM,KAAK,CAAC,GAAG,SAAS,KAAK,GAAG;GACjD,IAAI,CAAC,KAAK,SAAS,UAAU,KAAK,CAAC,KAAK,SAAS,UAAU,GAAG;GAE9D,IAAI,cAAc;GAGlB,cAAc,YAAY,QAAQ,KAAK,QAAQ,KAAK,OAAO,YAAY;IACrE,IAAI,qBAAqB,KAAK,KAAK,KAAK,mBAAmB,KAAK,KAAK,GAAG,OAAO;IAE/E,MAAM,WAAW,QAAQ,QAAQ,QAAQ,EAAE,EAAE,QAAQ,QAAQ,EAAE;IAC/D,IAAI,CAAC,UAAU,OAAO;IAEtB,MAAM,YAAY,SAAS,MAAM,iBAAiB,GAC9C,QAAQ,KAAa,OAAe,KAAK,IAAI,KAAK,GAAG,MAAM,GAAG,QAAQ,KAAK;IAY/E,OAAO,IAAI,MAAM,MAAM,aAVN,YAAY,IACzB,SAAS,QAAQ,IAAI,OAAO,WAAW,UAAU,IAAI,IAAI,GAAG,EAAE,IAC9D,UAGD,QAAQ,MAAM,OAAO,EACrB,QAAQ,MAAM,QAAQ,EACtB,QAAQ,MAAM,MAAM,EACpB,QAAQ,MAAM,MAEwB,EAAE;GAC7C,CAAC;GAGD,cAAc,YAAY,QAAQ,gBAAgB,QAAQ,KAAK,UAAU;IACvE,MAAM,WAAW,MAAM,MAAM,uBAAuB;IACpD,IAAI,CAAC,UAAU,OAAO;IAEtB,MAAM,UAAU,SAAS;IACzB,MAAM,eAAe,QAAQ,QAAQ,EAAE,GAAG,OAAO;IAEjD,IAAI;IACJ,IAAI;KACF,cAAc,aAAa,cAAc,OAAO,EAAE,KAAK;IACzD,QAAQ;KACN,OAAO;IACT;IAUA,OAAO,IAAI,MAPQ,MAAM,QAAQ,0BAA0B,EAOjC,EAAE,YANZ,YACb,QAAQ,MAAM,OAAO,EACrB,QAAQ,MAAM,QAAQ,EACtB,QAAQ,MAAM,MAAM,EACpB,QAAQ,MAAM,MAE6B,EAAE;GAClD,CAAC;GAED,IAAI,gBAAgB,MAClB,OAAO;IAAE,MAAM;IAAa,KAAK;GAAK;EAE1C;CACF;AACF"}
1
+ {"version":3,"file":"markdownExtract.js","names":[],"sources":["../../../src/render/plugins/markdownExtract.ts"],"sourcesContent":["import { dirname, resolve } from 'node:path'\nimport { readFileSync } from 'node:fs'\nimport type { Plugin } from 'vite'\n\n/**\n * Vite plugin that pre-processes <Markdown> tags:\n * - Extracts slot content, dedents it, and passes as :content prop\n * - Resolves `src` prop to read file contents at build time\n */\nexport function markdownExtract(): Plugin {\n const re = /<(Markdown|markdown)((?:\\s[^>]*?)?)>([\\s\\S]*?)<\\/\\1>/g\n const selfClosingRe = /<(Markdown|markdown)((?:\\s[^>]*?\\bsrc\\s*=\\s*\"[^\"]*\"[^>]*?))\\/>/g\n\n return {\n name: 'maizzle:markdown-extract',\n enforce: 'pre',\n transform(code: string, id: string) {\n if (!id.endsWith('.vue') && !id.endsWith('.md')) return\n if (!code.includes('Markdown') && !code.includes('markdown')) return\n\n let transformed = code\n\n // Handle <Markdown>content</Markdown>\n transformed = transformed.replace(re, (_match, tag, attrs, content) => {\n if (/(?:^|\\s):content\\b/.test(attrs) || /v-bind:content\\b/.test(attrs)) return _match\n\n const stripped = content.replace(/^\\n+/, '').replace(/\\s+$/, '')\n if (!stripped) return _match\n\n const minIndent = stripped.match(/^[ \\t]*(?=\\S)/gm)\n ?.reduce((min: number, ws: string) => Math.min(min, ws.length), Infinity) ?? 0\n\n const dedented = minIndent > 0\n ? stripped.replace(new RegExp(`^[ \\\\t]{${minIndent}}`, 'gm'), '')\n : stripped\n\n const escaped = dedented\n .replace(/&/g, '&amp;')\n .replace(/\"/g, '&quot;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n\n return `<${tag}${attrs} content=\"${escaped}\" />`\n })\n\n // Handle <Markdown src=\"./file.md\" /> — resolve and inline file content\n transformed = transformed.replace(selfClosingRe, (_match, tag, attrs) => {\n const srcMatch = attrs.match(/\\bsrc\\s*=\\s*\"([^\"]*)\"/)\n if (!srcMatch) return _match\n\n const srcPath = srcMatch[1]\n const resolvedPath = resolve(dirname(id), srcPath)\n\n let fileContent: string\n try {\n fileContent = readFileSync(resolvedPath, 'utf-8').trim()\n } catch {\n return _match\n }\n\n // Remove src prop, add content prop\n const cleanAttrs = attrs.replace(/\\s*\\bsrc\\s*=\\s*\"[^\"]*\"/, '')\n const escaped = fileContent\n .replace(/&/g, '&amp;')\n .replace(/\"/g, '&quot;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n\n return `<${tag}${cleanAttrs} content=\"${escaped}\" />`\n })\n\n if (transformed !== code) {\n return { code: transformed, map: null }\n }\n },\n }\n}\n"],"mappings":";;;;;;;;AASA,SAAgB,kBAA0B;CACxC,MAAM,KAAK;CACX,MAAM,gBAAgB;CAEtB,OAAO;EACL,MAAM;EACN,SAAS;EACT,UAAU,MAAc,IAAY;GAClC,IAAI,CAAC,GAAG,SAAS,MAAM,KAAK,CAAC,GAAG,SAAS,KAAK,GAAG;GACjD,IAAI,CAAC,KAAK,SAAS,UAAU,KAAK,CAAC,KAAK,SAAS,UAAU,GAAG;GAE9D,IAAI,cAAc;GAGlB,cAAc,YAAY,QAAQ,KAAK,QAAQ,KAAK,OAAO,YAAY;IACrE,IAAI,qBAAqB,KAAK,KAAK,KAAK,mBAAmB,KAAK,KAAK,GAAG,OAAO;IAE/E,MAAM,WAAW,QAAQ,QAAQ,QAAQ,EAAE,CAAC,CAAC,QAAQ,QAAQ,EAAE;IAC/D,IAAI,CAAC,UAAU,OAAO;IAEtB,MAAM,YAAY,SAAS,MAAM,iBAAiB,CAAC,EAC/C,QAAQ,KAAa,OAAe,KAAK,IAAI,KAAK,GAAG,MAAM,GAAG,QAAQ,KAAK;IAY/E,OAAO,IAAI,MAAM,MAAM,aAVN,YAAY,IACzB,SAAS,QAAQ,IAAI,OAAO,WAAW,UAAU,IAAI,IAAI,GAAG,EAAE,IAC9D,SAAA,CAGD,QAAQ,MAAM,OAAO,CAAC,CACtB,QAAQ,MAAM,QAAQ,CAAC,CACvB,QAAQ,MAAM,MAAM,CAAC,CACrB,QAAQ,MAAM,MAEwB,EAAE;GAC7C,CAAC;GAGD,cAAc,YAAY,QAAQ,gBAAgB,QAAQ,KAAK,UAAU;IACvE,MAAM,WAAW,MAAM,MAAM,uBAAuB;IACpD,IAAI,CAAC,UAAU,OAAO;IAEtB,MAAM,UAAU,SAAS;IACzB,MAAM,eAAe,QAAQ,QAAQ,EAAE,GAAG,OAAO;IAEjD,IAAI;IACJ,IAAI;KACF,cAAc,aAAa,cAAc,OAAO,CAAC,CAAC,KAAK;IACzD,QAAQ;KACN,OAAO;IACT;IAUA,OAAO,IAAI,MAPQ,MAAM,QAAQ,0BAA0B,EAOjC,EAAE,YANZ,YACb,QAAQ,MAAM,OAAO,CAAC,CACtB,QAAQ,MAAM,QAAQ,CAAC,CACvB,QAAQ,MAAM,MAAM,CAAC,CACrB,QAAQ,MAAM,MAE6B,EAAE;GAClD,CAAC;GAED,IAAI,gBAAgB,MAClB,OAAO;IAAE,MAAM;IAAa,KAAK;GAAK;EAE1C;CACF;AACF"}
@@ -1 +1 @@
1
- {"version":3,"file":"rawExtract.d.ts","names":[],"sources":["../../../src/render/plugins/rawExtract.ts"],"mappings":";;;;;AASA;;;;AAAoC;iBAApB,UAAA,CAAA,GAAc,MAAM"}
1
+ {"version":3,"file":"rawExtract.d.ts","names":[],"sources":["../../../src/render/plugins/rawExtract.ts"],"mappings":";;;;;AASA;;;;AAAoC;iBAApB,UAAA,IAAc,MAAM"}
@@ -1 +1 @@
1
- {"version":3,"file":"rawExtract.js","names":[],"sources":["../../../src/render/plugins/rawExtract.ts"],"sourcesContent":["import type { Plugin } from 'vite'\n\n/**\n * Vite plugin that extracts raw slot content from <Raw> tags\n * and passes it as a :content prop before Vue compiles the template.\n *\n * Lets users write content (including `{{ }}` interpolation syntax used\n * by ESPs / Handlebars / Liquid) inside <Raw> without Vue parsing it.\n */\nexport function rawExtract(): Plugin {\n const re = /<(Raw)((?:\\s[^>]*?)?)>([\\s\\S]*?)<\\/\\1>/g\n\n return {\n name: 'maizzle:raw-extract',\n enforce: 'pre',\n transform(code: string, id: string) {\n if (!id.endsWith('.vue') && !id.endsWith('.md')) return\n if (!code.includes('Raw')) return\n\n const transformed = code.replace(re, (_match, tag, attrs, content) => {\n if (/(?:^|\\s):content\\b/.test(attrs) || /v-bind:content\\b/.test(attrs)) return _match\n\n const stripped = content.replace(/^\\n+/, '').replace(/\\s+$/, '')\n if (!stripped) return _match\n\n const minIndent = stripped.match(/^[ \\t]*(?=\\S)/gm)\n ?.reduce((min: number, ws: string) => Math.min(min, ws.length), Infinity) ?? 0\n\n const dedented = minIndent > 0\n ? stripped.replace(new RegExp(`^[ \\\\t]{${minIndent}}`, 'gm'), '')\n : stripped\n\n const escaped = dedented\n .replace(/&/g, '&amp;')\n .replace(/\"/g, '&quot;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n\n return `<${tag}${attrs} content=\"${escaped}\" />`\n })\n\n if (transformed !== code) {\n return { code: transformed, map: null }\n }\n },\n }\n}\n"],"mappings":";;;;;;;;AASA,SAAgB,aAAqB;CACnC,MAAM,KAAK;CAEX,OAAO;EACL,MAAM;EACN,SAAS;EACT,UAAU,MAAc,IAAY;GAClC,IAAI,CAAC,GAAG,SAAS,MAAM,KAAK,CAAC,GAAG,SAAS,KAAK,GAAG;GACjD,IAAI,CAAC,KAAK,SAAS,KAAK,GAAG;GAE3B,MAAM,cAAc,KAAK,QAAQ,KAAK,QAAQ,KAAK,OAAO,YAAY;IACpE,IAAI,qBAAqB,KAAK,KAAK,KAAK,mBAAmB,KAAK,KAAK,GAAG,OAAO;IAE/E,MAAM,WAAW,QAAQ,QAAQ,QAAQ,EAAE,EAAE,QAAQ,QAAQ,EAAE;IAC/D,IAAI,CAAC,UAAU,OAAO;IAEtB,MAAM,YAAY,SAAS,MAAM,iBAAiB,GAC9C,QAAQ,KAAa,OAAe,KAAK,IAAI,KAAK,GAAG,MAAM,GAAG,QAAQ,KAAK;IAY/E,OAAO,IAAI,MAAM,MAAM,aAVN,YAAY,IACzB,SAAS,QAAQ,IAAI,OAAO,WAAW,UAAU,IAAI,IAAI,GAAG,EAAE,IAC9D,UAGD,QAAQ,MAAM,OAAO,EACrB,QAAQ,MAAM,QAAQ,EACtB,QAAQ,MAAM,MAAM,EACpB,QAAQ,MAAM,MAEwB,EAAE;GAC7C,CAAC;GAED,IAAI,gBAAgB,MAClB,OAAO;IAAE,MAAM;IAAa,KAAK;GAAK;EAE1C;CACF;AACF"}
1
+ {"version":3,"file":"rawExtract.js","names":[],"sources":["../../../src/render/plugins/rawExtract.ts"],"sourcesContent":["import type { Plugin } from 'vite'\n\n/**\n * Vite plugin that extracts raw slot content from <Raw> tags\n * and passes it as a :content prop before Vue compiles the template.\n *\n * Lets users write content (including `{{ }}` interpolation syntax used\n * by ESPs / Handlebars / Liquid) inside <Raw> without Vue parsing it.\n */\nexport function rawExtract(): Plugin {\n const re = /<(Raw)((?:\\s[^>]*?)?)>([\\s\\S]*?)<\\/\\1>/g\n\n return {\n name: 'maizzle:raw-extract',\n enforce: 'pre',\n transform(code: string, id: string) {\n if (!id.endsWith('.vue') && !id.endsWith('.md')) return\n if (!code.includes('Raw')) return\n\n const transformed = code.replace(re, (_match, tag, attrs, content) => {\n if (/(?:^|\\s):content\\b/.test(attrs) || /v-bind:content\\b/.test(attrs)) return _match\n\n const stripped = content.replace(/^\\n+/, '').replace(/\\s+$/, '')\n if (!stripped) return _match\n\n const minIndent = stripped.match(/^[ \\t]*(?=\\S)/gm)\n ?.reduce((min: number, ws: string) => Math.min(min, ws.length), Infinity) ?? 0\n\n const dedented = minIndent > 0\n ? stripped.replace(new RegExp(`^[ \\\\t]{${minIndent}}`, 'gm'), '')\n : stripped\n\n const escaped = dedented\n .replace(/&/g, '&amp;')\n .replace(/\"/g, '&quot;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n\n return `<${tag}${attrs} content=\"${escaped}\" />`\n })\n\n if (transformed !== code) {\n return { code: transformed, map: null }\n }\n },\n }\n}\n"],"mappings":";;;;;;;;AASA,SAAgB,aAAqB;CACnC,MAAM,KAAK;CAEX,OAAO;EACL,MAAM;EACN,SAAS;EACT,UAAU,MAAc,IAAY;GAClC,IAAI,CAAC,GAAG,SAAS,MAAM,KAAK,CAAC,GAAG,SAAS,KAAK,GAAG;GACjD,IAAI,CAAC,KAAK,SAAS,KAAK,GAAG;GAE3B,MAAM,cAAc,KAAK,QAAQ,KAAK,QAAQ,KAAK,OAAO,YAAY;IACpE,IAAI,qBAAqB,KAAK,KAAK,KAAK,mBAAmB,KAAK,KAAK,GAAG,OAAO;IAE/E,MAAM,WAAW,QAAQ,QAAQ,QAAQ,EAAE,CAAC,CAAC,QAAQ,QAAQ,EAAE;IAC/D,IAAI,CAAC,UAAU,OAAO;IAEtB,MAAM,YAAY,SAAS,MAAM,iBAAiB,CAAC,EAC/C,QAAQ,KAAa,OAAe,KAAK,IAAI,KAAK,GAAG,MAAM,GAAG,QAAQ,KAAK;IAY/E,OAAO,IAAI,MAAM,MAAM,aAVN,YAAY,IACzB,SAAS,QAAQ,IAAI,OAAO,WAAW,UAAU,IAAI,IAAI,GAAG,EAAE,IAC9D,SAAA,CAGD,QAAQ,MAAM,OAAO,CAAC,CACtB,QAAQ,MAAM,QAAQ,CAAC,CACvB,QAAQ,MAAM,MAAM,CAAC,CACrB,QAAQ,MAAM,MAEwB,EAAE;GAC7C,CAAC;GAED,IAAI,gBAAgB,MAClB,OAAO;IAAE,MAAM;IAAa,KAAK;GAAK;EAE1C;CACF;AACF"}
@@ -1 +1 @@
1
- {"version":3,"file":"rowSourceLocation.d.ts","names":[],"sources":["../../../src/render/plugins/rowSourceLocation.ts"],"mappings":";;;;;AAaA;;;;AAA2C;;;;;iBAA3B,iBAAA,CAAA,GAAqB,MAAM"}
1
+ {"version":3,"file":"rowSourceLocation.d.ts","names":[],"sources":["../../../src/render/plugins/rowSourceLocation.ts"],"mappings":";;;;;AAaA;;;;AAA2C;;;;;iBAA3B,iBAAA,IAAqB,MAAM"}
@@ -1 +1 @@
1
- {"version":3,"file":"rowSourceLocation.js","names":[],"sources":["../../../src/render/plugins/rowSourceLocation.ts"],"sourcesContent":["import type { Plugin } from 'vite'\n\n/**\n * Vite plugin that injects `data-maizzle-loc=\"<file>:<line>\"` into every\n * `<Row>`/`<row>` opening tag in user templates.\n *\n * Used by Row.vue's runtime to point the user at the exact line in their\n * template when they misuse Row (e.g. without a Column child).\n *\n * Only transforms inside `<template>` blocks of SFCs (or the entire file\n * for `.md` templates) so `<Row>` mentions in `<script>` blocks (e.g. in\n * string literals or comments) are left untouched.\n */\nexport function rowSourceLocation(): Plugin {\n const tagRe = /(<(?:Row|row))(\\b[^>]*?)(\\/?>)/g\n\n function injectLoc(html: string, htmlOffset: number, fullCode: string, id: string): string {\n return html.replace(tagRe, (match, tag, attrs, end, localOffset: number) => {\n if (/\\bdata-maizzle-loc\\s*=/.test(attrs)) return match\n const absoluteOffset = htmlOffset + localOffset\n const line = fullCode.slice(0, absoluteOffset).split('\\n').length\n return `${tag}${attrs} data-maizzle-loc=\"${id}:${line}\"${end}`\n })\n }\n\n return {\n name: 'maizzle:row-loc',\n enforce: 'pre',\n transform(code, id) {\n const isVue = id.endsWith('.vue')\n const isMd = id.endsWith('.md')\n if (!isVue && !isMd) return\n if (!code.includes('<Row') && !code.includes('<row')) return\n\n let transformed: string\n\n if (isVue) {\n /**\n * Replace inside every <template>...</template> block, leaving\n * <script> and <style> blocks alone.\n */\n const templateBlock = /(<template\\b[^>]*>)([\\s\\S]*?)(<\\/template>)/g\n transformed = code.replace(templateBlock, (_match, open, inner, close, offset: number) => {\n const innerOffset = offset + open.length\n return open + injectLoc(inner, innerOffset, code, id) + close\n })\n } else {\n transformed = injectLoc(code, 0, code, id)\n }\n\n if (transformed !== code) {\n return { code: transformed, map: null }\n }\n },\n }\n}\n"],"mappings":";;;;;;;;;;;;AAaA,SAAgB,oBAA4B;CAC1C,MAAM,QAAQ;CAEd,SAAS,UAAU,MAAc,YAAoB,UAAkB,IAAoB;EACzF,OAAO,KAAK,QAAQ,QAAQ,OAAO,KAAK,OAAO,KAAK,gBAAwB;GAC1E,IAAI,yBAAyB,KAAK,KAAK,GAAG,OAAO;GACjD,MAAM,iBAAiB,aAAa;GAEpC,OAAO,GAAG,MAAM,MAAM,qBAAqB,GAAG,GADjC,SAAS,MAAM,GAAG,cAAc,EAAE,MAAM,IAAI,EAAE,OACL,GAAG;EAC3D,CAAC;CACH;CAEA,OAAO;EACL,MAAM;EACN,SAAS;EACT,UAAU,MAAM,IAAI;GAClB,MAAM,QAAQ,GAAG,SAAS,MAAM;GAChC,MAAM,OAAO,GAAG,SAAS,KAAK;GAC9B,IAAI,CAAC,SAAS,CAAC,MAAM;GACrB,IAAI,CAAC,KAAK,SAAS,MAAM,KAAK,CAAC,KAAK,SAAS,MAAM,GAAG;GAEtD,IAAI;GAEJ,IAAI,OAMF,cAAc,KAAK,QAAQ,iDAAgB,QAAQ,MAAM,OAAO,OAAO,WAAmB;IAExF,OAAO,OAAO,UAAU,OADJ,SAAS,KAAK,QACU,MAAM,EAAE,IAAI;GAC1D,CAAC;QAED,cAAc,UAAU,MAAM,GAAG,MAAM,EAAE;GAG3C,IAAI,gBAAgB,MAClB,OAAO;IAAE,MAAM;IAAa,KAAK;GAAK;EAE1C;CACF;AACF"}
1
+ {"version":3,"file":"rowSourceLocation.js","names":[],"sources":["../../../src/render/plugins/rowSourceLocation.ts"],"sourcesContent":["import type { Plugin } from 'vite'\n\n/**\n * Vite plugin that injects `data-maizzle-loc=\"<file>:<line>\"` into every\n * `<Row>`/`<row>` opening tag in user templates.\n *\n * Used by Row.vue's runtime to point the user at the exact line in their\n * template when they misuse Row (e.g. without a Column child).\n *\n * Only transforms inside `<template>` blocks of SFCs (or the entire file\n * for `.md` templates) so `<Row>` mentions in `<script>` blocks (e.g. in\n * string literals or comments) are left untouched.\n */\nexport function rowSourceLocation(): Plugin {\n const tagRe = /(<(?:Row|row))(\\b[^>]*?)(\\/?>)/g\n\n function injectLoc(html: string, htmlOffset: number, fullCode: string, id: string): string {\n return html.replace(tagRe, (match, tag, attrs, end, localOffset: number) => {\n if (/\\bdata-maizzle-loc\\s*=/.test(attrs)) return match\n const absoluteOffset = htmlOffset + localOffset\n const line = fullCode.slice(0, absoluteOffset).split('\\n').length\n return `${tag}${attrs} data-maizzle-loc=\"${id}:${line}\"${end}`\n })\n }\n\n return {\n name: 'maizzle:row-loc',\n enforce: 'pre',\n transform(code, id) {\n const isVue = id.endsWith('.vue')\n const isMd = id.endsWith('.md')\n if (!isVue && !isMd) return\n if (!code.includes('<Row') && !code.includes('<row')) return\n\n let transformed: string\n\n if (isVue) {\n /**\n * Replace inside every <template>...</template> block, leaving\n * <script> and <style> blocks alone.\n */\n const templateBlock = /(<template\\b[^>]*>)([\\s\\S]*?)(<\\/template>)/g\n transformed = code.replace(templateBlock, (_match, open, inner, close, offset: number) => {\n const innerOffset = offset + open.length\n return open + injectLoc(inner, innerOffset, code, id) + close\n })\n } else {\n transformed = injectLoc(code, 0, code, id)\n }\n\n if (transformed !== code) {\n return { code: transformed, map: null }\n }\n },\n }\n}\n"],"mappings":";;;;;;;;;;;;AAaA,SAAgB,oBAA4B;CAC1C,MAAM,QAAQ;CAEd,SAAS,UAAU,MAAc,YAAoB,UAAkB,IAAoB;EACzF,OAAO,KAAK,QAAQ,QAAQ,OAAO,KAAK,OAAO,KAAK,gBAAwB;GAC1E,IAAI,yBAAyB,KAAK,KAAK,GAAG,OAAO;GACjD,MAAM,iBAAiB,aAAa;GAEpC,OAAO,GAAG,MAAM,MAAM,qBAAqB,GAAG,GADjC,SAAS,MAAM,GAAG,cAAc,CAAC,CAAC,MAAM,IAAI,CAAC,CAAC,OACL,GAAG;EAC3D,CAAC;CACH;CAEA,OAAO;EACL,MAAM;EACN,SAAS;EACT,UAAU,MAAM,IAAI;GAClB,MAAM,QAAQ,GAAG,SAAS,MAAM;GAChC,MAAM,OAAO,GAAG,SAAS,KAAK;GAC9B,IAAI,CAAC,SAAS,CAAC,MAAM;GACrB,IAAI,CAAC,KAAK,SAAS,MAAM,KAAK,CAAC,KAAK,SAAS,MAAM,GAAG;GAEtD,IAAI;GAEJ,IAAI,OAMF,cAAc,KAAK,QAAQ,iDAAgB,QAAQ,MAAM,OAAO,OAAO,WAAmB;IAExF,OAAO,OAAO,UAAU,OADJ,SAAS,KAAK,QACU,MAAM,EAAE,IAAI;GAC1D,CAAC;QAED,cAAc,UAAU,MAAM,GAAG,MAAM,EAAE;GAG3C,IAAI,gBAAgB,MAClB,OAAO;IAAE,MAAM;IAAa,KAAK;GAAK;EAE1C;CACF;AACF"}
@@ -1 +1 @@
1
- {"version":3,"file":"serve.d.ts","names":[],"sources":["../src/serve.ts"],"mappings":";;;;UAoCiB,YAAA;EACf,MAAA,GAAS,OAAO,CAAC,aAAA;;EAEjB,IAAA;EAH2B;EAK3B,IAAA;EAJgB;EAMhB,MAAA;AAAA;;;;;;AAAM;AAYR;;;iBAAsB,KAAA,CAAM,OAAA,GAAS,YAAA,GAAiB,OAAA,CAAA,aAAA;AAAA,iBA2lBtC,WAAA,CAAY,MAAA,EAAQ,aAAa,EAAE,WAAA"}
1
+ {"version":3,"file":"serve.d.ts","names":[],"sources":["../src/serve.ts"],"mappings":";;;;UAmDiB,YAAA;EACf,MAAA,GAAS,OAAO,CAAC,aAAA;;EAEjB,IAAA;EAH2B;EAK3B,IAAA;EAJgB;EAMhB,MAAA;AAAA;;;;;;AAAM;AAYR;;;iBAAsB,KAAA,CAAM,OAAA,GAAS,YAAA,GAAiB,OAAA,CAAA,aAAA;AAAA,iBA6lBtC,WAAA,CAAY,MAAA,EAAQ,aAAa,EAAE,WAAA"}