@maizzle/framework 6.0.0-rc.16 → 6.0.0-rc.18

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 (460) hide show
  1. package/bin/maizzle.mjs +1 -1
  2. package/dist/build.d.ts +20 -0
  3. package/dist/build.d.ts.map +1 -0
  4. package/dist/{build.mjs → build.js} +14 -10
  5. package/dist/build.js.map +1 -0
  6. package/dist/components/Body.vue +10 -0
  7. package/dist/components/Button.vue +70 -11
  8. package/dist/components/Column.vue +10 -0
  9. package/dist/components/Container.vue +29 -1
  10. package/dist/components/Html.vue +10 -0
  11. package/dist/components/Layout.vue +27 -12
  12. package/dist/components/MarkdownLayout.vue +39 -0
  13. package/dist/components/Overlap.vue +10 -0
  14. package/dist/components/Preheader.vue +3 -8
  15. package/dist/components/Row.vue +13 -2
  16. package/dist/components/Section.vue +10 -0
  17. package/dist/components/Tailwind.vue +43 -0
  18. package/dist/components/{utils.d.mts → utils.d.ts} +1 -1
  19. package/dist/components/utils.d.ts.map +1 -0
  20. package/dist/components/{utils.mjs → utils.js} +1 -1
  21. package/dist/components/utils.js.map +1 -0
  22. package/dist/composables/{defineConfig.d.mts → defineConfig.d.ts} +2 -2
  23. package/dist/composables/defineConfig.d.ts.map +1 -0
  24. package/dist/composables/{defineConfig.mjs → defineConfig.js} +3 -3
  25. package/dist/composables/defineConfig.js.map +1 -0
  26. package/dist/composables/{renderContext.d.mts → renderContext.d.ts} +11 -5
  27. package/dist/composables/renderContext.d.ts.map +1 -0
  28. package/dist/composables/{renderContext.mjs → renderContext.js} +1 -1
  29. package/dist/composables/renderContext.js.map +1 -0
  30. package/dist/composables/useBaseUrl.d.ts +19 -0
  31. package/dist/composables/useBaseUrl.d.ts.map +1 -0
  32. package/dist/composables/useBaseUrl.js +27 -0
  33. package/dist/composables/useBaseUrl.js.map +1 -0
  34. package/dist/composables/{useConfig.d.mts → useConfig.d.ts} +2 -2
  35. package/dist/composables/useConfig.d.ts.map +1 -0
  36. package/dist/composables/{useConfig.mjs → useConfig.js} +1 -1
  37. package/dist/composables/useConfig.js.map +1 -0
  38. package/dist/composables/{useDoctype.d.mts → useDoctype.d.ts} +1 -1
  39. package/dist/composables/useDoctype.d.ts.map +1 -0
  40. package/dist/composables/{useDoctype.mjs → useDoctype.js} +2 -2
  41. package/dist/composables/useDoctype.js.map +1 -0
  42. package/dist/composables/{useEvent.d.mts → useEvent.d.ts} +2 -2
  43. package/dist/composables/useEvent.d.ts.map +1 -0
  44. package/dist/composables/{useEvent.mjs → useEvent.js} +2 -2
  45. package/dist/composables/useEvent.js.map +1 -0
  46. package/dist/composables/{useFont.d.mts → useFont.d.ts} +1 -1
  47. package/dist/composables/useFont.d.ts.map +1 -0
  48. package/dist/composables/{useFont.mjs → useFont.js} +2 -2
  49. package/dist/composables/useFont.js.map +1 -0
  50. package/dist/composables/{useOutlookFallback.d.mts → useOutlookFallback.d.ts} +1 -1
  51. package/dist/composables/useOutlookFallback.d.ts.map +1 -0
  52. package/dist/composables/{useOutlookFallback.mjs → useOutlookFallback.js} +1 -1
  53. package/dist/composables/useOutlookFallback.js.map +1 -0
  54. package/dist/composables/{usePlaintext.d.mts → usePlaintext.d.ts} +1 -1
  55. package/dist/composables/usePlaintext.d.ts.map +1 -0
  56. package/dist/composables/{usePlaintext.mjs → usePlaintext.js} +2 -2
  57. package/dist/composables/usePlaintext.js.map +1 -0
  58. package/dist/composables/{usePreheader.d.mts → usePreheader.d.ts} +1 -1
  59. package/dist/composables/usePreheader.d.ts.map +1 -0
  60. package/dist/composables/{usePreheader.mjs → usePreheader.js} +2 -2
  61. package/dist/composables/usePreheader.js.map +1 -0
  62. package/dist/composables/useTransformers.d.ts +34 -0
  63. package/dist/composables/useTransformers.d.ts.map +1 -0
  64. package/dist/composables/useTransformers.js +49 -0
  65. package/dist/composables/useTransformers.js.map +1 -0
  66. package/dist/composables/useUrlQuery.d.ts +19 -0
  67. package/dist/composables/useUrlQuery.d.ts.map +1 -0
  68. package/dist/composables/useUrlQuery.js +27 -0
  69. package/dist/composables/useUrlQuery.js.map +1 -0
  70. package/dist/config/{defaults.d.mts → defaults.d.ts} +2 -2
  71. package/dist/config/defaults.d.ts.map +1 -0
  72. package/dist/config/{defaults.mjs → defaults.js} +8 -2
  73. package/dist/config/defaults.js.map +1 -0
  74. package/dist/config/{index.d.mts → index.d.ts} +4 -4
  75. package/dist/config/index.d.ts.map +1 -0
  76. package/dist/config/{index.mjs → index.js} +3 -3
  77. package/dist/config/index.js.map +1 -0
  78. package/dist/events/{index.d.mts → index.d.ts} +10 -4
  79. package/dist/events/index.d.ts.map +1 -0
  80. package/dist/events/{index.mjs → index.js} +20 -4
  81. package/dist/events/index.js.map +1 -0
  82. package/dist/index.d.ts +37 -0
  83. package/dist/index.js +37 -0
  84. package/dist/{plaintext.d.mts → plaintext.d.ts} +1 -1
  85. package/dist/plaintext.d.ts.map +1 -0
  86. package/dist/{plaintext.mjs → plaintext.js} +1 -1
  87. package/dist/plaintext.js.map +1 -0
  88. package/dist/{plugin.d.mts → plugin.d.ts} +2 -2
  89. package/dist/plugin.d.ts.map +1 -0
  90. package/dist/{plugin.mjs → plugin.js} +5 -5
  91. package/dist/plugin.js.map +1 -0
  92. package/dist/plugins/postcss/{mergeMediaQueries.d.mts → mergeMediaQueries.d.ts} +2 -2
  93. package/dist/plugins/postcss/mergeMediaQueries.d.ts.map +1 -0
  94. package/dist/plugins/postcss/{mergeMediaQueries.mjs → mergeMediaQueries.js} +1 -1
  95. package/dist/plugins/postcss/mergeMediaQueries.js.map +1 -0
  96. package/dist/plugins/postcss/{pruneVars.d.mts → pruneVars.d.ts} +1 -1
  97. package/dist/plugins/postcss/pruneVars.d.ts.map +1 -0
  98. package/dist/plugins/postcss/{pruneVars.mjs → pruneVars.js} +1 -1
  99. package/dist/plugins/postcss/pruneVars.js.map +1 -0
  100. package/dist/plugins/postcss/{quoteFontFamilies.d.mts → quoteFontFamilies.d.ts} +1 -1
  101. package/dist/plugins/postcss/quoteFontFamilies.d.ts.map +1 -0
  102. package/dist/plugins/postcss/{quoteFontFamilies.mjs → quoteFontFamilies.js} +1 -1
  103. package/dist/plugins/postcss/quoteFontFamilies.js.map +1 -0
  104. package/dist/plugins/postcss/{removeDeclarations.d.mts → removeDeclarations.d.ts} +1 -1
  105. package/dist/plugins/postcss/removeDeclarations.d.ts.map +1 -0
  106. package/dist/plugins/postcss/{removeDeclarations.mjs → removeDeclarations.js} +1 -1
  107. package/dist/plugins/postcss/removeDeclarations.js.map +1 -0
  108. package/dist/plugins/postcss/resolveMaizzleImports.d.ts +16 -0
  109. package/dist/plugins/postcss/resolveMaizzleImports.d.ts.map +1 -0
  110. package/dist/plugins/postcss/resolveMaizzleImports.js +40 -0
  111. package/dist/plugins/postcss/resolveMaizzleImports.js.map +1 -0
  112. package/dist/plugins/postcss/{resolveProps.d.mts → resolveProps.d.ts} +1 -1
  113. package/dist/plugins/postcss/resolveProps.d.ts.map +1 -0
  114. package/dist/plugins/postcss/{resolveProps.mjs → resolveProps.js} +1 -1
  115. package/dist/plugins/postcss/resolveProps.js.map +1 -0
  116. package/dist/plugins/postcss/{tailwindCleanup.d.mts → tailwindCleanup.d.ts} +2 -2
  117. package/dist/plugins/postcss/tailwindCleanup.d.ts.map +1 -0
  118. package/dist/plugins/postcss/{tailwindCleanup.mjs → tailwindCleanup.js} +1 -1
  119. package/dist/plugins/postcss/tailwindCleanup.js.map +1 -0
  120. package/dist/{prepare.d.mts → prepare.d.ts} +1 -1
  121. package/dist/prepare.d.ts.map +1 -0
  122. package/dist/{prepare.mjs → prepare.js} +4 -4
  123. package/dist/prepare.js.map +1 -0
  124. package/dist/render/{createRenderer.d.mts → createRenderer.d.ts} +4 -3
  125. package/dist/render/createRenderer.d.ts.map +1 -0
  126. package/dist/render/{createRenderer.mjs → createRenderer.js} +21 -12
  127. package/dist/render/createRenderer.js.map +1 -0
  128. package/dist/render/index.d.ts +18 -0
  129. package/dist/render/index.d.ts.map +1 -0
  130. package/dist/render/index.js +45 -0
  131. package/dist/render/index.js.map +1 -0
  132. package/dist/render/{injectFonts.d.mts → injectFonts.d.ts} +2 -2
  133. package/dist/render/injectFonts.d.ts.map +1 -0
  134. package/dist/render/{injectFonts.mjs → injectFonts.js} +2 -2
  135. package/dist/render/injectFonts.js.map +1 -0
  136. package/dist/render/plugins/{codeBlockExtract.d.mts → codeBlockExtract.d.ts} +1 -1
  137. package/dist/render/plugins/codeBlockExtract.d.ts.map +1 -0
  138. package/dist/render/plugins/{codeBlockExtract.mjs → codeBlockExtract.js} +1 -1
  139. package/dist/render/plugins/codeBlockExtract.js.map +1 -0
  140. package/dist/render/plugins/{markdownExtract.d.mts → markdownExtract.d.ts} +1 -1
  141. package/dist/render/plugins/markdownExtract.d.ts.map +1 -0
  142. package/dist/render/plugins/{markdownExtract.mjs → markdownExtract.js} +1 -1
  143. package/dist/render/plugins/markdownExtract.js.map +1 -0
  144. package/dist/render/plugins/{rawExtract.d.mts → rawExtract.d.ts} +1 -1
  145. package/dist/render/plugins/rawExtract.d.ts.map +1 -0
  146. package/dist/render/plugins/{rawExtract.mjs → rawExtract.js} +1 -1
  147. package/dist/render/plugins/rawExtract.js.map +1 -0
  148. package/dist/render/plugins/{rowSourceLocation.d.mts → rowSourceLocation.d.ts} +1 -1
  149. package/dist/render/plugins/rowSourceLocation.d.ts.map +1 -0
  150. package/dist/render/plugins/{rowSourceLocation.mjs → rowSourceLocation.js} +1 -1
  151. package/dist/render/plugins/rowSourceLocation.js.map +1 -0
  152. package/dist/{serve.d.mts → serve.d.ts} +2 -2
  153. package/dist/serve.d.ts.map +1 -0
  154. package/dist/{serve.mjs → serve.js} +23 -14
  155. package/dist/serve.js.map +1 -0
  156. package/dist/server/{compatibility.d.mts → compatibility.d.ts} +2 -2
  157. package/dist/server/compatibility.d.ts.map +1 -0
  158. package/dist/server/{compatibility.mjs → compatibility.js} +4 -4
  159. package/dist/server/compatibility.js.map +1 -0
  160. package/dist/server/{email.d.mts → email.d.ts} +2 -2
  161. package/dist/server/email.d.ts.map +1 -0
  162. package/dist/server/{email.mjs → email.js} +1 -1
  163. package/dist/server/email.js.map +1 -0
  164. package/dist/server/{linter.d.mts → linter.d.ts} +2 -2
  165. package/dist/server/linter.d.ts.map +1 -0
  166. package/dist/server/{linter.mjs → linter.js} +2 -2
  167. package/dist/server/linter.js.map +1 -0
  168. package/dist/server/{sfc-utils.d.mts → sfc-utils.d.ts} +1 -1
  169. package/dist/server/sfc-utils.d.ts.map +1 -0
  170. package/dist/server/{sfc-utils.mjs → sfc-utils.js} +1 -1
  171. package/dist/server/sfc-utils.js.map +1 -0
  172. package/dist/server/ui/App.vue +18 -0
  173. package/dist/server/ui/components/ui/command/Command.vue +4 -1
  174. package/dist/server/ui/pages/Preview.vue +78 -19
  175. package/dist/transformers/{addAttributes.d.mts → addAttributes.d.ts} +2 -2
  176. package/dist/transformers/addAttributes.d.ts.map +1 -0
  177. package/dist/transformers/{addAttributes.mjs → addAttributes.js} +3 -3
  178. package/dist/transformers/addAttributes.js.map +1 -0
  179. package/dist/transformers/{attributeToStyle.d.mts → attributeToStyle.d.ts} +2 -2
  180. package/dist/transformers/attributeToStyle.d.ts.map +1 -0
  181. package/dist/transformers/{attributeToStyle.mjs → attributeToStyle.js} +3 -3
  182. package/dist/transformers/attributeToStyle.js.map +1 -0
  183. package/dist/transformers/{base.d.mts → base.d.ts} +2 -2
  184. package/dist/transformers/base.d.ts.map +1 -0
  185. package/dist/transformers/{base.mjs → base.js} +6 -6
  186. package/dist/transformers/base.js.map +1 -0
  187. package/dist/transformers/{columnWidth.d.mts → columnWidth.d.ts} +1 -1
  188. package/dist/transformers/columnWidth.d.ts.map +1 -0
  189. package/dist/transformers/{columnWidth.mjs → columnWidth.js} +3 -3
  190. package/dist/transformers/columnWidth.js.map +1 -0
  191. package/dist/transformers/{entities.d.mts → entities.d.ts} +2 -2
  192. package/dist/transformers/entities.d.ts.map +1 -0
  193. package/dist/transformers/{entities.mjs → entities.js} +3 -3
  194. package/dist/transformers/entities.js.map +1 -0
  195. package/dist/transformers/filters/{defaults.d.mts → defaults.d.ts} +1 -1
  196. package/dist/transformers/filters/defaults.d.ts.map +1 -0
  197. package/dist/transformers/filters/{defaults.mjs → defaults.js} +1 -1
  198. package/dist/transformers/filters/defaults.js.map +1 -0
  199. package/dist/transformers/filters/{index.d.mts → index.d.ts} +2 -2
  200. package/dist/transformers/filters/index.d.ts.map +1 -0
  201. package/dist/transformers/filters/{index.mjs → index.js} +5 -5
  202. package/dist/transformers/filters/index.js.map +1 -0
  203. package/dist/transformers/{format.d.mts → format.d.ts} +2 -2
  204. package/dist/transformers/format.d.ts.map +1 -0
  205. package/dist/transformers/{format.mjs → format.js} +1 -1
  206. package/dist/transformers/format.js.map +1 -0
  207. package/dist/transformers/{index.d.mts → index.d.ts} +4 -3
  208. package/dist/transformers/index.d.ts.map +1 -0
  209. package/dist/transformers/index.js +113 -0
  210. package/dist/transformers/index.js.map +1 -0
  211. package/dist/transformers/{inlineCSS.d.mts → inlineCSS.d.ts} +2 -2
  212. package/dist/transformers/inlineCSS.d.ts.map +1 -0
  213. package/dist/transformers/{inlineCSS.mjs → inlineCSS.js} +12 -6
  214. package/dist/transformers/inlineCSS.js.map +1 -0
  215. package/dist/transformers/{inlineLink.d.mts → inlineLink.d.ts} +1 -1
  216. package/dist/transformers/inlineLink.d.ts.map +1 -0
  217. package/dist/transformers/{inlineLink.mjs → inlineLink.js} +3 -3
  218. package/dist/transformers/inlineLink.js.map +1 -0
  219. package/dist/transformers/{minify.d.mts → minify.d.ts} +2 -2
  220. package/dist/transformers/minify.d.ts.map +1 -0
  221. package/dist/transformers/{minify.mjs → minify.js} +1 -1
  222. package/dist/transformers/minify.js.map +1 -0
  223. package/dist/transformers/msoPlaceholders.d.ts +28 -0
  224. package/dist/transformers/msoPlaceholders.d.ts.map +1 -0
  225. package/dist/transformers/msoPlaceholders.js +89 -0
  226. package/dist/transformers/msoPlaceholders.js.map +1 -0
  227. package/dist/transformers/{purgeCSS.d.mts → purgeCSS.d.ts} +2 -2
  228. package/dist/transformers/purgeCSS.d.ts.map +1 -0
  229. package/dist/transformers/{purgeCSS.mjs → purgeCSS.js} +49 -7
  230. package/dist/transformers/purgeCSS.js.map +1 -0
  231. package/dist/transformers/{removeAttributes.d.mts → removeAttributes.d.ts} +2 -2
  232. package/dist/transformers/removeAttributes.d.ts.map +1 -0
  233. package/dist/transformers/{removeAttributes.mjs → removeAttributes.js} +3 -3
  234. package/dist/transformers/removeAttributes.js.map +1 -0
  235. package/dist/transformers/{replaceStrings.d.mts → replaceStrings.d.ts} +2 -2
  236. package/dist/transformers/replaceStrings.d.ts.map +1 -0
  237. package/dist/transformers/{replaceStrings.mjs → replaceStrings.js} +1 -1
  238. package/dist/transformers/replaceStrings.js.map +1 -0
  239. package/dist/transformers/{safeClassNames.d.mts → safeClassNames.d.ts} +2 -2
  240. package/dist/transformers/safeClassNames.d.ts.map +1 -0
  241. package/dist/transformers/{safeClassNames.mjs → safeClassNames.js} +3 -3
  242. package/dist/transformers/safeClassNames.js.map +1 -0
  243. package/dist/transformers/{shorthandCSS.d.mts → shorthandCSS.d.ts} +2 -2
  244. package/dist/transformers/shorthandCSS.d.ts.map +1 -0
  245. package/dist/transformers/{shorthandCSS.mjs → shorthandCSS.js} +3 -3
  246. package/dist/transformers/shorthandCSS.js.map +1 -0
  247. package/dist/transformers/{sixHex.d.mts → sixHex.d.ts} +2 -2
  248. package/dist/transformers/sixHex.d.ts.map +1 -0
  249. package/dist/transformers/{sixHex.mjs → sixHex.js} +3 -3
  250. package/dist/transformers/sixHex.js.map +1 -0
  251. package/dist/transformers/tailwindComponent.d.ts +16 -0
  252. package/dist/transformers/tailwindComponent.d.ts.map +1 -0
  253. package/dist/transformers/tailwindComponent.js +93 -0
  254. package/dist/transformers/tailwindComponent.js.map +1 -0
  255. package/dist/transformers/{tailwindcss.d.mts → tailwindcss.d.ts} +2 -2
  256. package/dist/transformers/tailwindcss.d.ts.map +1 -0
  257. package/dist/transformers/{tailwindcss.mjs → tailwindcss.js} +6 -58
  258. package/dist/transformers/tailwindcss.js.map +1 -0
  259. package/dist/transformers/{urlQuery.d.mts → urlQuery.d.ts} +2 -2
  260. package/dist/transformers/urlQuery.d.ts.map +1 -0
  261. package/dist/transformers/{urlQuery.mjs → urlQuery.js} +2 -2
  262. package/dist/transformers/urlQuery.js.map +1 -0
  263. package/dist/types/{config.d.mts → config.d.ts} +45 -4
  264. package/dist/types/config.d.ts.map +1 -0
  265. package/dist/types/{index.d.mts → index.d.ts} +2 -2
  266. package/dist/utils/ast/index.d.ts +4 -0
  267. package/dist/utils/ast/index.js +5 -0
  268. package/dist/utils/ast/{parser.d.mts → parser.d.ts} +1 -1
  269. package/dist/utils/ast/parser.d.ts.map +1 -0
  270. package/dist/utils/ast/{parser.mjs → parser.js} +1 -1
  271. package/dist/utils/ast/parser.js.map +1 -0
  272. package/dist/utils/ast/{serializer.d.mts → serializer.d.ts} +1 -1
  273. package/dist/utils/ast/serializer.d.ts.map +1 -0
  274. package/dist/utils/ast/{serializer.mjs → serializer.js} +2 -2
  275. package/dist/utils/ast/serializer.js.map +1 -0
  276. package/dist/utils/ast/{walker.d.mts → walker.d.ts} +1 -1
  277. package/dist/utils/ast/walker.d.ts.map +1 -0
  278. package/dist/utils/ast/{walker.mjs → walker.js} +1 -1
  279. package/dist/utils/ast/walker.js.map +1 -0
  280. package/dist/utils/compileTailwindCss.d.ts +16 -0
  281. package/dist/utils/compileTailwindCss.d.ts.map +1 -0
  282. package/dist/utils/compileTailwindCss.js +55 -0
  283. package/dist/utils/compileTailwindCss.js.map +1 -0
  284. package/dist/utils/{decodeStyleEntities.d.mts → decodeStyleEntities.d.ts} +1 -1
  285. package/dist/utils/decodeStyleEntities.d.ts.map +1 -0
  286. package/dist/utils/{decodeStyleEntities.mjs → decodeStyleEntities.js} +1 -1
  287. package/dist/utils/decodeStyleEntities.js.map +1 -0
  288. package/dist/utils/{detect.d.mts → detect.d.ts} +1 -1
  289. package/dist/utils/detect.d.ts.map +1 -0
  290. package/dist/utils/{detect.mjs → detect.js} +1 -1
  291. package/dist/utils/detect.js.map +1 -0
  292. package/dist/utils/{url.d.mts → url.d.ts} +1 -1
  293. package/dist/utils/url.d.ts.map +1 -0
  294. package/dist/utils/{url.mjs → url.js} +1 -1
  295. package/dist/utils/url.js.map +1 -0
  296. package/node_modules/@clack/core/CHANGELOG.md +8 -0
  297. package/node_modules/@clack/core/dist/index.d.mts +18 -4
  298. package/node_modules/@clack/core/dist/index.mjs +16 -10
  299. package/node_modules/@clack/core/dist/index.mjs.map +1 -1
  300. package/node_modules/@clack/core/package.json +5 -2
  301. package/node_modules/@clack/prompts/CHANGELOG.md +15 -0
  302. package/node_modules/@clack/prompts/README.md +107 -2
  303. package/node_modules/@clack/prompts/dist/index.d.mts +16 -11
  304. package/node_modules/@clack/prompts/dist/index.mjs +114 -107
  305. package/node_modules/@clack/prompts/dist/index.mjs.map +1 -1
  306. package/node_modules/@clack/prompts/package.json +7 -4
  307. package/node_modules/fast-string-truncated-width/dist/index.js +36 -96
  308. package/node_modules/fast-string-truncated-width/dist/types.d.ts +0 -3
  309. package/node_modules/fast-string-truncated-width/dist/utils.d.ts +3 -3
  310. package/node_modules/fast-string-truncated-width/dist/utils.js +14 -9
  311. package/node_modules/fast-string-truncated-width/package.json +1 -1
  312. package/node_modules/fast-string-truncated-width/readme.md +2 -3
  313. package/node_modules/fast-string-width/package.json +2 -2
  314. package/node_modules/fast-string-width/readme.md +0 -3
  315. package/node_modules/fast-wrap-ansi/lib/main.js +4 -1
  316. package/node_modules/fast-wrap-ansi/lib/main.js.map +1 -1
  317. package/node_modules/fast-wrap-ansi/package.json +2 -2
  318. package/node_modules/tinyexec/README.md +1 -1
  319. package/node_modules/tinyexec/dist/main.d.mts +6 -6
  320. package/node_modules/tinyexec/dist/main.mjs +126 -134
  321. package/node_modules/tinyexec/package.json +9 -9
  322. package/package.json +15 -8
  323. package/dist/build.d.mts +0 -19
  324. package/dist/build.d.mts.map +0 -1
  325. package/dist/build.mjs.map +0 -1
  326. package/dist/components/utils.d.mts.map +0 -1
  327. package/dist/components/utils.mjs.map +0 -1
  328. package/dist/composables/defineConfig.d.mts.map +0 -1
  329. package/dist/composables/defineConfig.mjs.map +0 -1
  330. package/dist/composables/renderContext.d.mts.map +0 -1
  331. package/dist/composables/renderContext.mjs.map +0 -1
  332. package/dist/composables/useConfig.d.mts.map +0 -1
  333. package/dist/composables/useConfig.mjs.map +0 -1
  334. package/dist/composables/useDoctype.d.mts.map +0 -1
  335. package/dist/composables/useDoctype.mjs.map +0 -1
  336. package/dist/composables/useEvent.d.mts.map +0 -1
  337. package/dist/composables/useEvent.mjs.map +0 -1
  338. package/dist/composables/useFont.d.mts.map +0 -1
  339. package/dist/composables/useFont.mjs.map +0 -1
  340. package/dist/composables/useOutlookFallback.d.mts.map +0 -1
  341. package/dist/composables/useOutlookFallback.mjs.map +0 -1
  342. package/dist/composables/usePlaintext.d.mts.map +0 -1
  343. package/dist/composables/usePlaintext.mjs.map +0 -1
  344. package/dist/composables/usePreheader.d.mts.map +0 -1
  345. package/dist/composables/usePreheader.mjs.map +0 -1
  346. package/dist/config/defaults.d.mts.map +0 -1
  347. package/dist/config/defaults.mjs.map +0 -1
  348. package/dist/config/index.d.mts.map +0 -1
  349. package/dist/config/index.mjs.map +0 -1
  350. package/dist/events/index.d.mts.map +0 -1
  351. package/dist/events/index.mjs.map +0 -1
  352. package/dist/index.d.mts +0 -34
  353. package/dist/index.mjs +0 -34
  354. package/dist/plaintext.d.mts.map +0 -1
  355. package/dist/plaintext.mjs.map +0 -1
  356. package/dist/plugin.d.mts.map +0 -1
  357. package/dist/plugin.mjs.map +0 -1
  358. package/dist/plugins/postcss/mergeMediaQueries.d.mts.map +0 -1
  359. package/dist/plugins/postcss/mergeMediaQueries.mjs.map +0 -1
  360. package/dist/plugins/postcss/pruneVars.d.mts.map +0 -1
  361. package/dist/plugins/postcss/pruneVars.mjs.map +0 -1
  362. package/dist/plugins/postcss/quoteFontFamilies.d.mts.map +0 -1
  363. package/dist/plugins/postcss/quoteFontFamilies.mjs.map +0 -1
  364. package/dist/plugins/postcss/removeDeclarations.d.mts.map +0 -1
  365. package/dist/plugins/postcss/removeDeclarations.mjs.map +0 -1
  366. package/dist/plugins/postcss/resolveProps.d.mts.map +0 -1
  367. package/dist/plugins/postcss/resolveProps.mjs.map +0 -1
  368. package/dist/plugins/postcss/tailwindCleanup.d.mts.map +0 -1
  369. package/dist/plugins/postcss/tailwindCleanup.mjs.map +0 -1
  370. package/dist/prepare.d.mts.map +0 -1
  371. package/dist/prepare.mjs.map +0 -1
  372. package/dist/render/createRenderer.d.mts.map +0 -1
  373. package/dist/render/createRenderer.mjs.map +0 -1
  374. package/dist/render/index.d.mts +0 -26
  375. package/dist/render/index.d.mts.map +0 -1
  376. package/dist/render/index.mjs +0 -46
  377. package/dist/render/index.mjs.map +0 -1
  378. package/dist/render/injectFonts.d.mts.map +0 -1
  379. package/dist/render/injectFonts.mjs.map +0 -1
  380. package/dist/render/plugins/codeBlockExtract.d.mts.map +0 -1
  381. package/dist/render/plugins/codeBlockExtract.mjs.map +0 -1
  382. package/dist/render/plugins/markdownExtract.d.mts.map +0 -1
  383. package/dist/render/plugins/markdownExtract.mjs.map +0 -1
  384. package/dist/render/plugins/rawExtract.d.mts.map +0 -1
  385. package/dist/render/plugins/rawExtract.mjs.map +0 -1
  386. package/dist/render/plugins/rowSourceLocation.d.mts.map +0 -1
  387. package/dist/render/plugins/rowSourceLocation.mjs.map +0 -1
  388. package/dist/serve.d.mts.map +0 -1
  389. package/dist/serve.mjs.map +0 -1
  390. package/dist/server/compatibility.d.mts.map +0 -1
  391. package/dist/server/compatibility.mjs.map +0 -1
  392. package/dist/server/email.d.mts.map +0 -1
  393. package/dist/server/email.mjs.map +0 -1
  394. package/dist/server/linter.d.mts.map +0 -1
  395. package/dist/server/linter.mjs.map +0 -1
  396. package/dist/server/sfc-utils.d.mts.map +0 -1
  397. package/dist/server/sfc-utils.mjs.map +0 -1
  398. package/dist/transformers/addAttributes.d.mts.map +0 -1
  399. package/dist/transformers/addAttributes.mjs.map +0 -1
  400. package/dist/transformers/attributeToStyle.d.mts.map +0 -1
  401. package/dist/transformers/attributeToStyle.mjs.map +0 -1
  402. package/dist/transformers/base.d.mts.map +0 -1
  403. package/dist/transformers/base.mjs.map +0 -1
  404. package/dist/transformers/columnWidth.d.mts.map +0 -1
  405. package/dist/transformers/columnWidth.mjs.map +0 -1
  406. package/dist/transformers/entities.d.mts.map +0 -1
  407. package/dist/transformers/entities.mjs.map +0 -1
  408. package/dist/transformers/filters/defaults.d.mts.map +0 -1
  409. package/dist/transformers/filters/defaults.mjs.map +0 -1
  410. package/dist/transformers/filters/index.d.mts.map +0 -1
  411. package/dist/transformers/filters/index.mjs.map +0 -1
  412. package/dist/transformers/format.d.mts.map +0 -1
  413. package/dist/transformers/format.mjs.map +0 -1
  414. package/dist/transformers/index.d.mts.map +0 -1
  415. package/dist/transformers/index.mjs +0 -85
  416. package/dist/transformers/index.mjs.map +0 -1
  417. package/dist/transformers/inlineCSS.d.mts.map +0 -1
  418. package/dist/transformers/inlineCSS.mjs.map +0 -1
  419. package/dist/transformers/inlineLink.d.mts.map +0 -1
  420. package/dist/transformers/inlineLink.mjs.map +0 -1
  421. package/dist/transformers/minify.d.mts.map +0 -1
  422. package/dist/transformers/minify.mjs.map +0 -1
  423. package/dist/transformers/msoWidthFromClass.d.mts +0 -19
  424. package/dist/transformers/msoWidthFromClass.d.mts.map +0 -1
  425. package/dist/transformers/msoWidthFromClass.mjs +0 -61
  426. package/dist/transformers/msoWidthFromClass.mjs.map +0 -1
  427. package/dist/transformers/purgeCSS.d.mts.map +0 -1
  428. package/dist/transformers/purgeCSS.mjs.map +0 -1
  429. package/dist/transformers/removeAttributes.d.mts.map +0 -1
  430. package/dist/transformers/removeAttributes.mjs.map +0 -1
  431. package/dist/transformers/replaceStrings.d.mts.map +0 -1
  432. package/dist/transformers/replaceStrings.mjs.map +0 -1
  433. package/dist/transformers/safeClassNames.d.mts.map +0 -1
  434. package/dist/transformers/safeClassNames.mjs.map +0 -1
  435. package/dist/transformers/shorthandCSS.d.mts.map +0 -1
  436. package/dist/transformers/shorthandCSS.mjs.map +0 -1
  437. package/dist/transformers/sixHex.d.mts.map +0 -1
  438. package/dist/transformers/sixHex.mjs.map +0 -1
  439. package/dist/transformers/tailwindcss.d.mts.map +0 -1
  440. package/dist/transformers/tailwindcss.mjs.map +0 -1
  441. package/dist/transformers/urlQuery.d.mts.map +0 -1
  442. package/dist/transformers/urlQuery.mjs.map +0 -1
  443. package/dist/types/config.d.mts.map +0 -1
  444. package/dist/utils/ast/index.d.mts +0 -4
  445. package/dist/utils/ast/index.mjs +0 -5
  446. package/dist/utils/ast/parser.d.mts.map +0 -1
  447. package/dist/utils/ast/parser.mjs.map +0 -1
  448. package/dist/utils/ast/serializer.d.mts.map +0 -1
  449. package/dist/utils/ast/serializer.mjs.map +0 -1
  450. package/dist/utils/ast/walker.d.mts.map +0 -1
  451. package/dist/utils/ast/walker.mjs.map +0 -1
  452. package/dist/utils/decodeStyleEntities.d.mts.map +0 -1
  453. package/dist/utils/decodeStyleEntities.mjs.map +0 -1
  454. package/dist/utils/detect.d.mts.map +0 -1
  455. package/dist/utils/detect.mjs.map +0 -1
  456. package/dist/utils/url.d.mts.map +0 -1
  457. package/dist/utils/url.mjs.map +0 -1
  458. /package/dist/components/{Divider.vue → Hr.vue} +0 -0
  459. /package/dist/types/{config.mjs → config.js} +0 -0
  460. /package/dist/types/{index.mjs → index.js} +0 -0
@@ -1,10 +1,10 @@
1
- import { MaizzleConfigKey } from "../composables/useConfig.mjs";
2
- import { RenderContextKey } from "../composables/renderContext.mjs";
3
- import { isLaravel } from "../utils/detect.mjs";
4
- import { rowSourceLocation } from "./plugins/rowSourceLocation.mjs";
5
- import { rawExtract } from "./plugins/rawExtract.mjs";
6
- import { codeBlockExtract } from "./plugins/codeBlockExtract.mjs";
7
- import { markdownExtract } from "./plugins/markdownExtract.mjs";
1
+ import { MaizzleConfigKey } from "../composables/useConfig.js";
2
+ import { RenderContextKey } from "../composables/renderContext.js";
3
+ import { isLaravel } from "../utils/detect.js";
4
+ import { rowSourceLocation } from "./plugins/rowSourceLocation.js";
5
+ import { rawExtract } from "./plugins/rawExtract.js";
6
+ import { codeBlockExtract } from "./plugins/codeBlockExtract.js";
7
+ import { markdownExtract } from "./plugins/markdownExtract.js";
8
8
  import { existsSync } from "node:fs";
9
9
  import { dirname, resolve } from "node:path";
10
10
  import { fileURLToPath } from "node:url";
@@ -34,6 +34,7 @@ const vueRouterPkgDir = dirname(fileURLToPath(import.meta.resolve("vue-router/pa
34
34
  async function createRenderer(options = {}) {
35
35
  const { dts = false, markdown: markdownOptionsRaw, root = process.cwd(), componentDirs = [], vite: userViteConfig } = options;
36
36
  const { shikiTheme = "github-light", ...markdownOptions } = markdownOptionsRaw ?? {};
37
+ const componentDirsAbs = [resolve(root, "components"), ...componentDirs.map((d) => resolve(root, d))];
37
38
  const dtsDir = isLaravel() ? resolve(process.cwd(), "resources/js/types/maizzle") : resolve(root, ".maizzle");
38
39
  const VIRTUAL_SFC_ID = "virtual:maizzle-sfc.vue";
39
40
  let virtualSfcSource = "";
@@ -62,6 +63,12 @@ async function createRenderer(options = {}) {
62
63
  headEnabled: true,
63
64
  wrapperDiv: false,
64
65
  wrapperClasses: "prose",
66
+ wrapperComponent: (id, raw) => {
67
+ const layout = (raw.match(/^---\r?\n([\s\S]*?)\r?\n---/)?.[1])?.match(/^[ \t]*layout[ \t]*:[ \t]*['"]?([A-Za-z][\w-]*|false|none)['"]?[ \t]*$/m)?.[1];
68
+ if (layout === "false" || layout === "none") return null;
69
+ if (layout) return layout;
70
+ return componentDirsAbs.some((d) => id === d || id.startsWith(`${d}/`)) ? null : "MarkdownLayout";
71
+ },
65
72
  markdownOptions: { async highlight(code, lang) {
66
73
  const { codeToHtml } = await import("shiki");
67
74
  return codeToHtml(code, {
@@ -173,7 +180,7 @@ async function createRenderer(options = {}) {
173
180
  const hasTeleports = ssrContext.teleports && Object.keys(ssrContext.teleports).length > 0;
174
181
  const hasFonts = (renderContext.fonts?.length ?? 0) > 0;
175
182
  if (hasTeleports || hasFonts) {
176
- const { parse: parseDom, serialize: serializeDom, walk } = await import("../utils/ast/index.mjs");
183
+ const { parse: parseDom, serialize: serializeDom, walk } = await import("../utils/ast/index.js");
177
184
  let dom = parseDom(html);
178
185
  if (hasTeleports) for (const [rawTarget, content] of Object.entries(ssrContext.teleports)) {
179
186
  if (!content) continue;
@@ -190,7 +197,7 @@ async function createRenderer(options = {}) {
190
197
  });
191
198
  }
192
199
  if (hasFonts) {
193
- const { injectFonts } = await import("./injectFonts.mjs");
200
+ const { injectFonts } = await import("./injectFonts.js");
194
201
  injectFonts(dom, renderContext.fonts, parseDom, walk);
195
202
  }
196
203
  html = serializeDom(dom);
@@ -200,12 +207,14 @@ async function createRenderer(options = {}) {
200
207
  const previewHtml = `<div style="display:none">${text}${" ͏ ".repeat(fillerCount)}${"­ ".repeat(shyCount)}\u00A0</div>`;
201
208
  html = html.replace(/<body([^>]*)>/, `<body$1>${previewHtml}`);
202
209
  }
210
+ html = html.replaceAll("<!--[-->", "").replaceAll("<!--]-->", "").replaceAll("<!--teleport start anchor-->", "").replaceAll("<!--teleport anchor-->", "").replaceAll("<!--teleport start-->", "").replaceAll("<!--teleport end-->", "");
203
211
  return {
204
212
  html,
205
213
  doctype: renderContext.doctype,
206
- templateConfig: renderContext.sfcConfig ?? config,
214
+ templateConfig: renderContext.sfcConfig ? defu(renderContext.sfcConfig, config) : config,
207
215
  sfcEventHandlers: renderContext.sfcEventHandlers,
208
- plaintext: renderContext.plaintext
216
+ plaintext: renderContext.plaintext,
217
+ tailwindBlocks: renderContext.tailwindBlocks
209
218
  };
210
219
  },
211
220
  async invalidate(filePath) {
@@ -223,4 +232,4 @@ async function createRenderer(options = {}) {
223
232
 
224
233
  //#endregion
225
234
  export { createRenderer };
226
- //# sourceMappingURL=createRenderer.mjs.map
235
+ //# sourceMappingURL=createRenderer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"createRenderer.js","names":["merge"],"sources":["../../src/render/createRenderer.ts"],"sourcesContent":["import { dirname, resolve } from 'node:path'\nimport { existsSync } 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 } 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 { 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 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 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 /** Additional component directories to register for auto-import */\n componentDirs?: string[]\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 // Absolute component dirs — used to skip auto-wrapping `.md` files that are\n // imported as reusable components (vs. entry-point email templates).\n const componentDirsAbs = [resolve(root, 'components'), ...componentDirs.map(d => resolve(root, d))]\n\n const dtsDir = isLaravel()\n ? resolve(process.cwd(), 'resources/js/types/maizzle')\n : resolve(root, '.maizzle')\n\n const VIRTUAL_SFC_ID = 'virtual:maizzle-sfc.vue'\n let virtualSfcSource = ''\n\n // Check for a user vite.config file in the project root\n const viteConfigFile = ['vite.config.ts', 'vite.config.js']\n .map(f => resolve(root, f))\n .find(f => existsSync(f))\n\n const maizzleConfig: InlineConfig = {\n configFile: viteConfigFile ?? 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 },\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 // No `layout:` set — default to the built-in `MarkdownLayout` for\n // entry-template `.md` files. Skip for `.md` files inside component\n // dirs, which are reusable fragments imported into other templates.\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 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 resolve(__dirname, '../components'),\n resolve(root, 'components'),\n ...componentDirs,\n ],\n dts: dts ? resolve(dtsDir, 'components.d.ts') : false,\n }),\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 // 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 the plugins.)\n fs: {\n allow: [process.cwd(), root, ...componentDirs, vuePkgDir, vueServerRendererPkgDir, unheadVuePkgDir, vueRouterPkgDir],\n },\n },\n appType: 'custom',\n logLevel: 'silent',\n optimizeDeps: {\n noDiscovery: true,\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 are concatenated.\n // This ensures Maizzle's critical settings (middlewareMode, appType, etc.) always win,\n // while user plugins and other options are included.\n const finalConfig = userViteConfig && !viteConfigFile\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 // String input goes through Vite — must use ssrLoadModule for injection keys\n // so they share the same module instance as the SFC\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 for (const plugin of config.vue.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, shyCount } = renderContext.preheader\n const filler = '\\u2007\\u034F '.repeat(fillerCount)\n const shys = '\\u00AD '.repeat(shyCount)\n const previewHtml = `<div style=\"display:none\">${text}${filler}${shys}\\u00A0</div>`\n html = html.replace(/<body([^>]*)>/, `<body$1>${previewHtml}`)\n }\n\n // Strip Vue SSR fragment markers + teleport anchor comments. These are\n // rendering hygiene, not transformer concerns — must run regardless of\n // `useTransformers` state. Fragment markers contain `-->`, which would\n // prematurely terminate MSO conditional comments downstream.\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 // Layer sfcConfig over config — sfcConfig is a partial override\n // emitted by composables (defineConfig, useTransformers, etc.).\n // A naive replacement (`sfcConfig ?? config`) drops defaults from\n // the resolved config when the SFC only sets a single key, since\n // the composables' inject() of globalConfig can return `{}` in\n // dev when ssrLoadModule and the SFC's auto-imported module\n // resolve to different module instances (different Symbols).\n templateConfig: renderContext.sfcConfig ? merge(renderContext.sfcConfig, config) : config,\n sfcEventHandlers: renderContext.sfcEventHandlers,\n plaintext: renderContext.plaintext,\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 }\n}\n"],"mappings":"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"}
@@ -0,0 +1,18 @@
1
+ import { MaizzleConfig } from "../types/config.js";
2
+ import { CreateRendererOptions, RenderedTemplate, Renderer, createRenderer } from "./createRenderer.js";
3
+ import { Component } from "vue";
4
+
5
+ //#region src/render/index.d.ts
6
+ interface RenderResult {
7
+ html: string;
8
+ config: MaizzleConfig;
9
+ plaintext?: string;
10
+ }
11
+ /**
12
+ * Render a Vue SFC email template to a fully-transformed HTML string.
13
+ * Accepts a file path or a raw SFC source string.
14
+ */
15
+ declare function render(template: string | Component, config?: Partial<MaizzleConfig>): Promise<RenderResult>;
16
+ //#endregion
17
+ export { type CreateRendererOptions, RenderResult, type RenderedTemplate, type Renderer, createRenderer, render };
18
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","names":[],"sources":["../../src/render/index.ts"],"mappings":";;;;;UAWiB,YAAA;EACf,IAAA;EACA,MAAA,EAAQ,aAAA;EACR,SAAA;AAAA;;;;;iBAOoB,MAAA,CACpB,QAAA,WAAmB,SAAA,EACnB,MAAA,GAAS,OAAA,CAAQ,aAAA,IAChB,OAAA,CAAQ,YAAA"}
@@ -0,0 +1,45 @@
1
+ import { resolveConfig } from "../config/index.js";
2
+ import { runTransformers } from "../transformers/index.js";
3
+ import { createRenderer } from "./createRenderer.js";
4
+ import { createPlaintext } from "../plaintext.js";
5
+ import { extname, resolve } from "node:path";
6
+
7
+ //#region src/render/index.ts
8
+ /**
9
+ * Render a Vue SFC email template to a fully-transformed HTML string.
10
+ * Accepts a file path or a raw SFC source string.
11
+ */
12
+ async function render(template, config) {
13
+ if (template == null) throw new Error(`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').`);
14
+ if (typeof template !== "string" && typeof template !== "object" && typeof template !== "function") throw new TypeError(`render() expected a file path or SFC source string, got ${typeof template}.`);
15
+ const resolvedConfig = await resolveConfig(config);
16
+ const renderer = await createRenderer({
17
+ markdown: resolvedConfig.markdown,
18
+ root: resolvedConfig.root,
19
+ componentDirs: [resolvedConfig.components?.source ?? []].flat(),
20
+ vite: resolvedConfig.vite
21
+ });
22
+ try {
23
+ const isFile = typeof template === "string" && [".vue", ".md"].includes(extname(template)) && !template.includes("\n");
24
+ const rendered = await renderer.render(isFile ? resolve(template) : template, resolvedConfig);
25
+ let html = rendered.html;
26
+ const doctype = rendered.doctype ?? rendered.templateConfig.doctype ?? "<!DOCTYPE html>";
27
+ if (rendered.templateConfig.useTransformers !== false) html = await runTransformers(html, rendered.templateConfig, isFile ? resolve(template) : void 0, doctype, rendered.tailwindBlocks);
28
+ html = `${doctype}\n${html}`;
29
+ const globalPlaintext = rendered.templateConfig.plaintext;
30
+ const sfcPlaintext = rendered.plaintext;
31
+ let plaintextResult;
32
+ if (globalPlaintext || sfcPlaintext) plaintextResult = createPlaintext(html, typeof globalPlaintext === "object" ? globalPlaintext : {});
33
+ return {
34
+ html,
35
+ config: rendered.templateConfig,
36
+ plaintext: plaintextResult
37
+ };
38
+ } finally {
39
+ await renderer.close();
40
+ }
41
+ }
42
+
43
+ //#endregion
44
+ export { createRenderer, render };
45
+ //# sourceMappingURL=index.js.map
@@ -0,0 +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 type { Component } from 'vue'\nimport type { MaizzleConfig } from '../types/index.ts'\nimport { createRenderer } from './createRenderer.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 const renderer = await createRenderer({\n markdown: resolvedConfig.markdown,\n root: resolvedConfig.root,\n componentDirs: [resolvedConfig.components?.source ?? []].flat(),\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 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 stripOptions = typeof globalPlaintext === 'object' ? globalPlaintext : {}\n plaintextResult = createPlaintext(html, stripOptions)\n }\n\n return { html, config: rendered.templateConfig, plaintext: plaintextResult }\n } finally {\n await renderer.close()\n }\n}\n"],"mappings":";;;;;;;;;;;AAqBA,eAAsB,OACpB,UACA,QACuB;AACvB,KAAI,YAAY,KACd,OAAM,IAAI,MACR,qBAAqB,SAAS,oIAC/B;AAEH,KAAI,OAAO,aAAa,YAAY,OAAO,aAAa,YAAY,OAAO,aAAa,WACtF,OAAM,IAAI,UACR,2DAA2D,OAAO,SAAS,GAC5E;CAGH,MAAM,iBAAiB,MAAM,cAAc,OAAO;CAClD,MAAM,WAAW,MAAM,eAAe;EACpC,UAAU,eAAe;EACzB,MAAM,eAAe;EACrB,eAAe,CAAC,eAAe,YAAY,UAAU,EAAE,CAAC,CAAC,MAAM;EAC/D,MAAM,eAAe;EACtB,CAAC;AAEF,KAAI;EACF,MAAM,SAAS,OAAO,aAAa,YAC9B,CAAC,QAAQ,MAAM,CAAC,SAAS,QAAQ,SAAS,CAAC,IAC3C,CAAC,SAAS,SAAS,KAAK;EAE7B,MAAM,WAAW,MAAM,SAAS,OAAO,SAAS,QAAQ,SAAS,GAAG,UAAU,eAAe;EAC7F,IAAI,OAAO,SAAS;EAEpB,MAAM,UAAU,SAAS,WAAW,SAAS,eAAe,WAAW;AAEvE,MAAI,SAAS,eAAe,oBAAoB,MAC9C,QAAO,MAAM,gBAAgB,MAAM,SAAS,gBAAgB,SAAS,QAAQ,SAAS,GAAG,QAAW,SAAS,SAAS,eAAe;AAEvI,SAAO,GAAG,QAAQ,IAAI;EAEtB,MAAM,kBAAkB,SAAS,eAAe;EAChD,MAAM,eAAe,SAAS;EAE9B,IAAI;AAEJ,MAAI,mBAAmB,aAErB,mBAAkB,gBAAgB,MADb,OAAO,oBAAoB,WAAW,kBAAkB,EAAE,CAC1B;AAGvD,SAAO;GAAE;GAAM,QAAQ,SAAS;GAAgB,WAAW;GAAiB;WACpE;AACR,QAAM,SAAS,OAAO"}
@@ -1,4 +1,4 @@
1
- import { FontRegistration } from "../composables/renderContext.mjs";
1
+ import { FontRegistration } from "../composables/renderContext.js";
2
2
  import { ChildNode } from "domhandler";
3
3
 
4
4
  //#region src/render/injectFonts.d.ts
@@ -12,4 +12,4 @@ import { ChildNode } from "domhandler";
12
12
  declare function injectFonts(dom: ChildNode[], fonts: FontRegistration[], parseDom: (html: string) => ChildNode[], walk: (ast: ChildNode[], cb: (n: ChildNode) => void) => void): void;
13
13
  //#endregion
14
14
  export { injectFonts };
15
- //# sourceMappingURL=injectFonts.d.mts.map
15
+ //# sourceMappingURL=injectFonts.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"injectFonts.d.ts","names":[],"sources":["../../src/render/injectFonts.ts"],"mappings":";;;;;;AAoBA;;;;;iBAAgB,WAAA,CACd,GAAA,EAAK,SAAA,IACL,KAAA,EAAO,gBAAA,IACP,QAAA,GAAW,IAAA,aAAiB,SAAA,IAC5B,IAAA,GAAO,GAAA,EAAK,SAAA,IAAa,EAAA,GAAK,CAAA,EAAG,SAAA"}
@@ -1,4 +1,4 @@
1
- import { decodeStyleEntities } from "../utils/decodeStyleEntities.mjs";
1
+ import { decodeStyleEntities } from "../utils/decodeStyleEntities.js";
2
2
 
3
3
  //#region src/render/injectFonts.ts
4
4
  const TAILWIND_IMPORT_RE = /((@import|@reference)\s+["'](tailwindcss|@maizzle\/tailwindcss)|@tailwind\s)/;
@@ -43,4 +43,4 @@ function injectFonts(dom, fonts, parseDom, walk) {
43
43
 
44
44
  //#endregion
45
45
  export { injectFonts };
46
- //# sourceMappingURL=injectFonts.mjs.map
46
+ //# sourceMappingURL=injectFonts.js.map
@@ -0,0 +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;AACpC,SAAQ,GAAG,YAAY,EAAE,EACtB,QAAQ,MAAW,EAAE,SAAS,OAAO,CACrC,KAAK,MAAW,EAAE,KAAK,CACvB,KAAK,GAAG;;;;;;;;;AAUb,SAAgB,YACd,KACA,OACA,UACA,MACM;AACN,KAAI,CAAC,MAAM,OAAQ;CAEnB,IAAI;CACJ,IAAI;AAEJ,MAAK,MAAM,SAAS;EAClB,MAAM,KAAK;AACX,MAAI,CAAC,GAAG,KAAM;AACd,MAAI,GAAG,SAAS,UAAU,CAAC,KAAM,QAAO;AACxC,MAAI,GAAG,SAAS,WAAW,CAAC,iBAAiB,mBAAmB,KAAK,oBAAoB,QAAQ,GAAG,CAAC,CAAC,CACpG,iBAAgB;GAElB;AAEF,KAAI,CAAC,KAAM;CAKX,MAAM,YAAY,SAHD,MACd,KAAI,MAAK,eAAe,EAAE,IAAI,oCAAoC,CAClE,KAAK,GAAG,CACyB;AACpC,MAAK,MAAM,SAAS,UAClB,CAAC,MAAc,SAAS;AAE1B,MAAK,WAAW,CAAC,GAAI,KAAK,YAAY,EAAE,EAAG,GAAG,UAAU;AAExD,KAAI,eAAe;EACjB,MAAM,aAAa,MAChB,KAAI,MAAK,YAAY,EAAE,KAAK,IAAI,EAAE,YAAY,GAAG,CACjD,KAAK,KAAK;EACb,MAAM,WAAW,QAAQ,cAAc;AACvC,gBAAc,WAAW,CAAC;GACxB,MAAM;GACN,MAAM,GAAG,SAAS,cAAc,WAAW;GAC3C,QAAQ;GACT,CAAQ;QACJ;EAIL,MAAM,aAAa,SAAS,YAHT,MAChB,KAAI,MAAK,SAAS,EAAE,KAAK,kBAAkB,EAAE,YAAY,KAAK,CAC9D,KAAK,KAAK,CACsC,YAAY;AAC/D,OAAK,MAAM,SAAS,WAClB,CAAC,MAAc,SAAS;AAE1B,OAAK,WAAW,CAAC,GAAI,KAAK,YAAY,EAAE,EAAG,GAAG,WAAW"}
@@ -11,4 +11,4 @@ import { Plugin } from "vite";
11
11
  declare function codeBlockExtract(): Plugin;
12
12
  //#endregion
13
13
  export { codeBlockExtract };
14
- //# sourceMappingURL=codeBlockExtract.d.mts.map
14
+ //# sourceMappingURL=codeBlockExtract.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"codeBlockExtract.d.ts","names":[],"sources":["../../../src/render/plugins/codeBlockExtract.ts"],"mappings":";;;;;AASA;;;;;iBAAgB,gBAAA,CAAA,GAAoB,MAAA"}
@@ -31,4 +31,4 @@ function codeBlockExtract() {
31
31
 
32
32
  //#endregion
33
33
  export { codeBlockExtract };
34
- //# sourceMappingURL=codeBlockExtract.mjs.map
34
+ //# sourceMappingURL=codeBlockExtract.js.map
@@ -0,0 +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 // Strip leading/trailing blank lines, then dedent based on\n // the minimum indent of non-empty lines (à la min-indent)\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;AAEX,QAAO;EACL,MAAM;EACN,SAAS;EACT,UAAU,MAAc,IAAY;AAClC,OAAI,CAAC,GAAG,SAAS,OAAO,IAAI,CAAC,GAAG,SAAS,MAAM,CAAE;AACjD,OAAI,CAAC,KAAK,SAAS,YAAY,IAAI,CAAC,KAAK,SAAS,aAAa,CAAE;GAEjE,MAAM,cAAc,KAAK,QAAQ,KAAK,QAAQ,KAAK,OAAO,YAAY;AAEpE,QAAI,kBAAkB,KAAK,MAAM,IAAI,gBAAgB,KAAK,MAAM,CAAE,QAAO;IAIzE,MAAM,WAAW,QAAQ,QAAQ,QAAQ,GAAG,CAAC,QAAQ,QAAQ,GAAG;AAChE,QAAI,CAAC,SAAU,QAAO;IAEtB,MAAM,YAAY,SAAS,MAAM,kBAAkB,EAC/C,QAAQ,KAAa,OAAe,KAAK,IAAI,KAAK,GAAG,OAAO,EAAE,SAAS,IAAI;AAa/E,WAAO,IAAI,MAAM,MAAM,UAXN,YAAY,IACzB,SAAS,QAAQ,IAAI,OAAO,WAAW,UAAU,IAAI,KAAK,EAAE,GAAG,GAC/D,UAID,QAAQ,MAAM,QAAQ,CACtB,QAAQ,MAAM,SAAS,CACvB,QAAQ,MAAM,OAAO,CACrB,QAAQ,MAAM,OAAO,CAEgB;KACxC;AAEF,OAAI,gBAAgB,KAClB,QAAO;IAAE,MAAM;IAAa,KAAK;IAAM;;EAG5C"}
@@ -9,4 +9,4 @@ import { Plugin } from "vite";
9
9
  declare function markdownExtract(): Plugin;
10
10
  //#endregion
11
11
  export { markdownExtract };
12
- //# sourceMappingURL=markdownExtract.d.mts.map
12
+ //# sourceMappingURL=markdownExtract.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"markdownExtract.d.ts","names":[],"sources":["../../../src/render/plugins/markdownExtract.ts"],"mappings":";;;;;AASA;;;iBAAgB,eAAA,CAAA,GAAmB,MAAA"}
@@ -47,4 +47,4 @@ function markdownExtract() {
47
47
 
48
48
  //#endregion
49
49
  export { markdownExtract };
50
- //# sourceMappingURL=markdownExtract.mjs.map
50
+ //# sourceMappingURL=markdownExtract.js.map
@@ -0,0 +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;AAEtB,QAAO;EACL,MAAM;EACN,SAAS;EACT,UAAU,MAAc,IAAY;AAClC,OAAI,CAAC,GAAG,SAAS,OAAO,IAAI,CAAC,GAAG,SAAS,MAAM,CAAE;AACjD,OAAI,CAAC,KAAK,SAAS,WAAW,IAAI,CAAC,KAAK,SAAS,WAAW,CAAE;GAE9D,IAAI,cAAc;AAGlB,iBAAc,YAAY,QAAQ,KAAK,QAAQ,KAAK,OAAO,YAAY;AACrE,QAAI,qBAAqB,KAAK,MAAM,IAAI,mBAAmB,KAAK,MAAM,CAAE,QAAO;IAE/E,MAAM,WAAW,QAAQ,QAAQ,QAAQ,GAAG,CAAC,QAAQ,QAAQ,GAAG;AAChE,QAAI,CAAC,SAAU,QAAO;IAEtB,MAAM,YAAY,SAAS,MAAM,kBAAkB,EAC/C,QAAQ,KAAa,OAAe,KAAK,IAAI,KAAK,GAAG,OAAO,EAAE,SAAS,IAAI;AAY/E,WAAO,IAAI,MAAM,MAAM,aAVN,YAAY,IACzB,SAAS,QAAQ,IAAI,OAAO,WAAW,UAAU,IAAI,KAAK,EAAE,GAAG,GAC/D,UAGD,QAAQ,MAAM,QAAQ,CACtB,QAAQ,MAAM,SAAS,CACvB,QAAQ,MAAM,OAAO,CACrB,QAAQ,MAAM,OAAO,CAEmB;KAC3C;AAGF,iBAAc,YAAY,QAAQ,gBAAgB,QAAQ,KAAK,UAAU;IACvE,MAAM,WAAW,MAAM,MAAM,wBAAwB;AACrD,QAAI,CAAC,SAAU,QAAO;IAEtB,MAAM,UAAU,SAAS;IACzB,MAAM,eAAe,QAAQ,QAAQ,GAAG,EAAE,QAAQ;IAElD,IAAI;AACJ,QAAI;AACF,mBAAc,aAAa,cAAc,QAAQ,CAAC,MAAM;YAClD;AACN,YAAO;;AAWT,WAAO,IAAI,MAPQ,MAAM,QAAQ,0BAA0B,GAAG,CAOlC,YANZ,YACb,QAAQ,MAAM,QAAQ,CACtB,QAAQ,MAAM,SAAS,CACvB,QAAQ,MAAM,OAAO,CACrB,QAAQ,MAAM,OAAO,CAEwB;KAChD;AAEF,OAAI,gBAAgB,KAClB,QAAO;IAAE,MAAM;IAAa,KAAK;IAAM;;EAG5C"}
@@ -11,4 +11,4 @@ import { Plugin } from "vite";
11
11
  declare function rawExtract(): Plugin;
12
12
  //#endregion
13
13
  export { rawExtract };
14
- //# sourceMappingURL=rawExtract.d.mts.map
14
+ //# sourceMappingURL=rawExtract.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rawExtract.d.ts","names":[],"sources":["../../../src/render/plugins/rawExtract.ts"],"mappings":";;;;;AASA;;;;;iBAAgB,UAAA,CAAA,GAAc,MAAA"}
@@ -31,4 +31,4 @@ function rawExtract() {
31
31
 
32
32
  //#endregion
33
33
  export { rawExtract };
34
- //# sourceMappingURL=rawExtract.mjs.map
34
+ //# sourceMappingURL=rawExtract.js.map
@@ -0,0 +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;AAEX,QAAO;EACL,MAAM;EACN,SAAS;EACT,UAAU,MAAc,IAAY;AAClC,OAAI,CAAC,GAAG,SAAS,OAAO,IAAI,CAAC,GAAG,SAAS,MAAM,CAAE;AACjD,OAAI,CAAC,KAAK,SAAS,MAAM,CAAE;GAE3B,MAAM,cAAc,KAAK,QAAQ,KAAK,QAAQ,KAAK,OAAO,YAAY;AACpE,QAAI,qBAAqB,KAAK,MAAM,IAAI,mBAAmB,KAAK,MAAM,CAAE,QAAO;IAE/E,MAAM,WAAW,QAAQ,QAAQ,QAAQ,GAAG,CAAC,QAAQ,QAAQ,GAAG;AAChE,QAAI,CAAC,SAAU,QAAO;IAEtB,MAAM,YAAY,SAAS,MAAM,kBAAkB,EAC/C,QAAQ,KAAa,OAAe,KAAK,IAAI,KAAK,GAAG,OAAO,EAAE,SAAS,IAAI;AAY/E,WAAO,IAAI,MAAM,MAAM,aAVN,YAAY,IACzB,SAAS,QAAQ,IAAI,OAAO,WAAW,UAAU,IAAI,KAAK,EAAE,GAAG,GAC/D,UAGD,QAAQ,MAAM,QAAQ,CACtB,QAAQ,MAAM,SAAS,CACvB,QAAQ,MAAM,OAAO,CACrB,QAAQ,MAAM,OAAO,CAEmB;KAC3C;AAEF,OAAI,gBAAgB,KAClB,QAAO;IAAE,MAAM;IAAa,KAAK;IAAM;;EAG5C"}
@@ -15,4 +15,4 @@ import { Plugin } from "vite";
15
15
  declare function rowSourceLocation(): Plugin;
16
16
  //#endregion
17
17
  export { rowSourceLocation };
18
- //# sourceMappingURL=rowSourceLocation.d.mts.map
18
+ //# sourceMappingURL=rowSourceLocation.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rowSourceLocation.d.ts","names":[],"sources":["../../../src/render/plugins/rowSourceLocation.ts"],"mappings":";;;;;AAaA;;;;;;;;;iBAAgB,iBAAA,CAAA,GAAqB,MAAA"}
@@ -42,4 +42,4 @@ function rowSourceLocation() {
42
42
 
43
43
  //#endregion
44
44
  export { rowSourceLocation };
45
- //# sourceMappingURL=rowSourceLocation.mjs.map
45
+ //# sourceMappingURL=rowSourceLocation.js.map
@@ -0,0 +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 // Replace inside every <template>...</template> block, leaving\n // <script> and <style> blocks alone.\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;AACzF,SAAO,KAAK,QAAQ,QAAQ,OAAO,KAAK,OAAO,KAAK,gBAAwB;AAC1E,OAAI,yBAAyB,KAAK,MAAM,CAAE,QAAO;GACjD,MAAM,iBAAiB,aAAa;AAEpC,UAAO,GAAG,MAAM,MAAM,qBAAqB,GAAG,GADjC,SAAS,MAAM,GAAG,eAAe,CAAC,MAAM,KAAK,CAAC,OACL,GAAG;IACzD;;AAGJ,QAAO;EACL,MAAM;EACN,SAAS;EACT,UAAU,MAAM,IAAI;GAClB,MAAM,QAAQ,GAAG,SAAS,OAAO;GACjC,MAAM,OAAO,GAAG,SAAS,MAAM;AAC/B,OAAI,CAAC,SAAS,CAAC,KAAM;AACrB,OAAI,CAAC,KAAK,SAAS,OAAO,IAAI,CAAC,KAAK,SAAS,OAAO,CAAE;GAEtD,IAAI;AAEJ,OAAI,MAIF,eAAc,KAAK,QADG,iDACqB,QAAQ,MAAM,OAAO,OAAO,WAAmB;AAExF,WAAO,OAAO,UAAU,OADJ,SAAS,KAAK,QACU,MAAM,GAAG,GAAG;KACxD;OAEF,eAAc,UAAU,MAAM,GAAG,MAAM,GAAG;AAG5C,OAAI,gBAAgB,KAClB,QAAO;IAAE,MAAM;IAAa,KAAK;IAAM;;EAG5C"}
@@ -1,4 +1,4 @@
1
- import { MaizzleConfig } from "./types/config.mjs";
1
+ import { MaizzleConfig } from "./types/config.js";
2
2
  import { ViteDevServer } from "vite";
3
3
 
4
4
  //#region src/serve.d.ts
@@ -22,4 +22,4 @@ declare function serve(options?: ServeOptions): Promise<ViteDevServer>;
22
22
  declare function printBanner(server: ViteDevServer, startupTime?: number): void;
23
23
  //#endregion
24
24
  export { ServeOptions, printBanner, serve };
25
- //# sourceMappingURL=serve.d.mts.map
25
+ //# sourceMappingURL=serve.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"serve.d.ts","names":[],"sources":["../src/serve.ts"],"mappings":";;;;UA+BiB,YAAA;EACf,MAAA,GAAS,OAAA,CAAQ,aAAA;;EAEjB,IAAA;EAH2B;EAK3B,MAAA;AAAA;;;;;;;;AAYF;;iBAAsB,KAAA,CAAM,OAAA,GAAS,YAAA,GAAiB,OAAA,CAAA,aAAA;AAAA,iBAujBtC,WAAA,CAAY,MAAA,EAAQ,aAAA,EAAe,WAAA"}
@@ -1,10 +1,10 @@
1
- import { resolveConfig } from "./config/index.mjs";
2
- import { runTransformers } from "./transformers/index.mjs";
3
- import { createRenderer } from "./render/createRenderer.mjs";
4
- import { createPlaintext } from "./plaintext.mjs";
5
- import { serveLint } from "./server/linter.mjs";
6
- import { serveCompatibility } from "./server/compatibility.mjs";
7
- import { sendEmail } from "./server/email.mjs";
1
+ import { resolveConfig } from "./config/index.js";
2
+ import { runTransformers } from "./transformers/index.js";
3
+ import { createRenderer } from "./render/createRenderer.js";
4
+ import { createPlaintext } from "./plaintext.js";
5
+ import { serveLint } from "./server/linter.js";
6
+ import { serveCompatibility } from "./server/compatibility.js";
7
+ import { sendEmail } from "./server/email.js";
8
8
  import { createRequire } from "node:module";
9
9
  import { readFileSync } from "node:fs";
10
10
  import { basename, dirname, matchesGlob, resolve } from "node:path";
@@ -12,6 +12,7 @@ import { glob } from "tinyglobby";
12
12
  import { fileURLToPath } from "node:url";
13
13
  import { createLogger, createServer } from "vite";
14
14
  import vue from "@vitejs/plugin-vue";
15
+ import { renderUnicodeCompact } from "uqr";
15
16
  import tailwindcss from "@tailwindcss/vite";
16
17
  import { createHighlighter } from "shiki";
17
18
 
@@ -70,7 +71,9 @@ async function serve(options = {}) {
70
71
  "lucide-vue-next",
71
72
  "class-variance-authority",
72
73
  "clsx",
73
- "tailwind-merge"
74
+ "tailwind-merge",
75
+ "culori",
76
+ "postcss-safe-parser"
74
77
  ].map((name) => ({
75
78
  find: name,
76
79
  replacement: pkg(name)
@@ -271,7 +274,7 @@ async function serveRenderedTemplate(url, config, renderer, res) {
271
274
  let html = rendered.html;
272
275
  const templateConfig = rendered.templateConfig;
273
276
  const doctype = rendered.doctype ?? templateConfig.doctype ?? "<!DOCTYPE html>";
274
- html = await runTransformers(html, templateConfig, absolutePath, doctype);
277
+ html = await runTransformers(html, templateConfig, absolutePath, doctype, rendered.tailwindBlocks);
275
278
  html = `${doctype}\n${html}`;
276
279
  res.setHeader("Content-Type", "text/html");
277
280
  res.end(html);
@@ -303,7 +306,7 @@ async function serveHighlightedSource(url, config, renderer, res) {
303
306
  let html = rendered.html;
304
307
  const templateConfig = rendered.templateConfig;
305
308
  const doctype = rendered.doctype ?? templateConfig.doctype ?? "<!DOCTYPE html>";
306
- html = await runTransformers(html, templateConfig, absolutePath, doctype);
309
+ html = await runTransformers(html, templateConfig, absolutePath, doctype, rendered.tailwindBlocks);
307
310
  html = `${doctype}\n${html}`;
308
311
  const highlighted = (await getHighlighter()).codeToHtml(html, {
309
312
  lang: "html",
@@ -359,7 +362,7 @@ async function servePlaintext(url, config, renderer, res) {
359
362
  let html = rendered.html;
360
363
  const templateConfig = rendered.templateConfig;
361
364
  const doctype = rendered.doctype ?? templateConfig.doctype ?? "<!DOCTYPE html>";
362
- html = await runTransformers(html, templateConfig, absolutePath, doctype);
365
+ html = await runTransformers(html, templateConfig, absolutePath, doctype, rendered.tailwindBlocks);
363
366
  const plaintext = createPlaintext(html);
364
367
  res.setHeader("Content-Type", "text/plain");
365
368
  res.end(plaintext);
@@ -404,7 +407,7 @@ async function serveStats(url, config, renderer, res) {
404
407
  let html = rendered.html;
405
408
  const templateConfig = rendered.templateConfig;
406
409
  const doctype = rendered.doctype ?? templateConfig.doctype ?? "<!DOCTYPE html>";
407
- html = await runTransformers(html, templateConfig, absolutePath, doctype);
410
+ html = await runTransformers(html, templateConfig, absolutePath, doctype, rendered.tailwindBlocks);
408
411
  const sizeBytes = Buffer.byteLength(html, "utf-8");
409
412
  const totalImages = (html.match(/<img\b[^>]*>/gi) || []).length + (html.match(/url\s*\([^)]+\)/gi) || []).length;
410
413
  const links = (html.match(/<a\b[^>]*href\s*=/gi) || []).length;
@@ -461,7 +464,7 @@ async function serveEmailEndpoint(url, req, res, config, renderer) {
461
464
  let html = rendered.html;
462
465
  const templateConfig = rendered.templateConfig;
463
466
  const doctype = rendered.doctype ?? templateConfig.doctype ?? "<!DOCTYPE html>";
464
- html = await runTransformers(html, templateConfig, absolutePath, doctype);
467
+ html = await runTransformers(html, templateConfig, absolutePath, doctype, rendered.tailwindBlocks);
465
468
  html = `${doctype}\n${html}`;
466
469
  const text = createPlaintext(html);
467
470
  const result = await sendEmail({
@@ -493,6 +496,12 @@ function serveEmailConfig(config, res) {
493
496
  function printBanner(server, startupTime) {
494
497
  const info = server.config.logger.info;
495
498
  const time = startupTime ?? server._maizzleStartupTime;
499
+ const networkUrl = server.resolvedUrls?.network[0];
500
+ if (networkUrl) {
501
+ const qr = renderUnicodeCompact(networkUrl, { border: 1 });
502
+ info("");
503
+ info(qr.split("\n").map((line) => ` ${line}`).join("\n"));
504
+ }
496
505
  info("");
497
506
  info(` \x1b[32m\x1b[1mMAIZZLE\x1b[0m\x1b[32m v6.0.0\x1b[0m \x1b[2mready in\x1b[0m \x1b[1m${time}\x1b[0m ms`);
498
507
  info("");
@@ -511,4 +520,4 @@ function customLogger() {
511
520
 
512
521
  //#endregion
513
522
  export { printBanner, serve };
514
- //# sourceMappingURL=serve.mjs.map
523
+ //# sourceMappingURL=serve.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"serve.js","names":[],"sources":["../src/serve.ts"],"sourcesContent":["import { readFileSync } from 'node:fs'\nimport { dirname, resolve, basename, matchesGlob } from 'node:path'\nimport { fileURLToPath } from 'node:url'\nimport { createRequire } from 'node:module'\nimport { createServer, createLogger, type ViteDevServer } from 'vite'\nimport { renderUnicodeCompact } from 'uqr'\nimport vue from '@vitejs/plugin-vue'\nimport tailwindcss from '@tailwindcss/vite'\nimport { glob } from 'tinyglobby'\nimport { createHighlighter, type Highlighter } from 'shiki'\nimport { createPlaintext } from './plaintext.ts'\nimport { resolveConfig } from './config/index.ts'\nimport { runTransformers } from './transformers/index.ts'\nimport { createRenderer, type Renderer } from './render/createRenderer.ts'\nimport { serveCompatibility } from './server/compatibility.ts'\nimport { serveLint } from './server/linter.ts'\nimport { sendEmail } from './server/email.ts'\nimport type { MaizzleConfig } from './types/index.ts'\n\nconst __dirname = dirname(fileURLToPath(import.meta.url))\nconst devUIDir = resolve(__dirname, 'server/ui')\n\nconst require = createRequire(import.meta.url)\nconst pkg = (name: string) => {\n const resolved = require.resolve(name).replace(/\\\\/g, '/')\n const marker = `node_modules/${name}`\n const idx = resolved.lastIndexOf(marker)\n\n return resolved.slice(0, idx + marker.length)\n}\n\nexport interface ServeOptions {\n config?: Partial<MaizzleConfig> | string\n /** Expose the server on the network (e.g. --host) */\n host?: boolean | string\n /** When true, suppresses the banner/URL output (used by the Vite plugin, which prints its own) */\n silent?: boolean\n}\n\n/**\n * Start the Maizzle dev server.\n *\n * Creates two things:\n * 1. A Vite dev server for the dev UI (sidebar + preview, with Vue + Tailwind for the UI itself)\n * 2. A Renderer instance for SSR rendering email templates\n *\n * Template rendering goes through the Renderer, not the Vite dev server.\n */\nexport async function serve(options: ServeOptions = {}) {\n const start = performance.now()\n\n let config = await resolveConfig(options.config)\n const port = config.server?.port ?? 3000\n\n // Create a renderer for SSR rendering email templates (with dts for dev)\n let renderer = await createRenderer({ dts: true, markdown: config.markdown, root: config.root, componentDirs: [config.components?.source ?? []].flat(), vite: config.vite })\n\n const server = await createServer({\n configFile: false,\n plugins: [\n // Vue and Tailwind are only for the dev UI SPA, not for email templates\n vue(),\n tailwindcss(),\n maizzleDevPlugin(config, renderer, options.config),\n ],\n resolve: {\n dedupe: ['vue'],\n alias: [\n { find: '@', replacement: devUIDir },\n { find: 'vue', replacement: resolve(pkg('vue'), 'dist/vue.runtime.esm-bundler.js') },\n ...['vue-router', 'reka-ui', '@vueuse/core', '@vueuse/shared', 'lucide-vue-next', 'class-variance-authority', 'clsx', 'tailwind-merge', 'culori', 'postcss-safe-parser']\n .map(name => ({ find: name, replacement: pkg(name) })),\n ],\n },\n cacheDir: resolve(devUIDir, '.vite'),\n optimizeDeps: {\n noDiscovery: true,\n include: [\n 'vue',\n 'vue-router',\n 'lucide-vue-next',\n '@vueuse/core',\n '@vueuse/shared',\n 'reka-ui',\n 'class-variance-authority',\n 'clsx',\n 'tailwind-merge',\n 'culori',\n 'postcss-safe-parser',\n ],\n },\n server: {\n port,\n host: options.host,\n fs: {\n allow: [process.cwd(), config.root ?? process.cwd(), devUIDir, ...['vue', 'vue-router', 'reka-ui', '@vueuse/core', '@vueuse/shared', 'lucide-vue-next', 'class-variance-authority', 'clsx', 'tailwind-merge', 'culori', 'postcss-safe-parser'].map(pkg)],\n },\n },\n customLogger: customLogger(),\n })\n\n // Store renderer ref on server for cleanup\n const originalClose = server.close.bind(server)\n server.close = async () => {\n await renderer.close()\n return originalClose()\n }\n\n await server.listen()\n\n const startupTime = Math.round(performance.now() - start)\n\n if (!options.silent) {\n printBanner(server, startupTime)\n }\n\n // Expose startup time so the plugin can print it later\n ; (server as any)._maizzleStartupTime = startupTime\n\n return server\n}\n\n/**\n * Internal Vite plugin that adds Maizzle middleware and file watching to the dev UI server.\n */\nfunction maizzleDevPlugin(\n config: MaizzleConfig,\n renderer: Renderer,\n configInput: Partial<MaizzleConfig> | string | undefined,\n) {\n return {\n name: 'maizzle:dev',\n enforce: 'pre' as const,\n\n hotUpdate: {\n order: 'pre' as const,\n handler({ file }: { file: string }) {\n // Prevent Tailwind/Vue from triggering a full reload for email template files.\n // Maizzle handles these via custom HMR events in the watcher below.\n if (isTemplateFile(file)) {\n return []\n }\n },\n },\n\n configureServer(server: ViteDevServer) {\n // File watching\n const defaultWatchPaths = [\n 'maizzle.config.js',\n 'maizzle.config.ts',\n 'tailwind.config.js',\n 'tailwind.config.ts',\n 'locales/**',\n ]\n\n const userWatchPaths = config.server?.watch ?? []\n const watchPaths = [...defaultWatchPaths, ...userWatchPaths]\n const isWatchedFile = (file: string) => watchPaths.some(p => matchesGlob(file, p))\n\n for (const watchPath of watchPaths) {\n server.watcher.add(watchPath)\n }\n\n server.watcher.on('add', async (file) => {\n if (isTemplateFile(file)) {\n await renderer.invalidateAll()\n server.ws.send({ type: 'custom', event: 'maizzle:templates-changed' })\n }\n })\n\n server.watcher.on('unlink', async (file) => {\n if (isTemplateFile(file)) {\n await renderer.invalidateAll()\n server.ws.send({ type: 'custom', event: 'maizzle:templates-changed' })\n }\n })\n\n server.watcher.on('change', async (file) => {\n if (isWatchedFile(file)) {\n config = await resolveConfig(configInput)\n\n // Recreate the renderer so config changes (e.g. markdown.shikiTheme) take effect\n await renderer.close()\n renderer = await createRenderer({ dts: true, markdown: config.markdown, root: config.root, componentDirs: [config.components?.source ?? []].flat(), vite: config.vite })\n\n // Push UI-relevant config bits so the dev UI reacts to live edits\n // without a page reload. Uses the same shape as the initial inject.\n server.ws.send({ type: 'custom', event: 'maizzle:config-updated', data: buildUiConfig(config) })\n }\n\n // Invalidate all renderer modules so component and config changes\n // are picked up on the next render (Tailwind recompiles with fresh content)\n await renderer.invalidateAll()\n\n if (\n isTemplateFile(file)\n || isWatchedFile(file)\n ) {\n server.ws.send({ type: 'custom', event: 'maizzle:template-updated', data: { file } })\n }\n })\n\n // API middleware (before Vite's middleware)\n server.middlewares.use(async (req: any, res: any, next: any) => {\n const url = req.url || '/'\n\n if (url === '/__maizzle/templates') {\n return serveTemplateList(config, res)\n }\n\n if (url.startsWith('/__maizzle/render/')) {\n return await serveRenderedTemplate(url, config, renderer, res)\n }\n\n if (url.startsWith('/__maizzle/source/')) {\n return await serveHighlightedSource(url, config, renderer, res)\n }\n\n if (url.startsWith('/__maizzle/compatibility/')) {\n return await serveCompatibility(url, res, config, [config.components?.source ?? []].flat())\n }\n\n if (url.startsWith('/__maizzle/lint/')) {\n return await serveLint(url, res, config, [config.components?.source ?? []].flat())\n }\n\n if (url.startsWith('/__maizzle/vue-source/')) {\n return await serveVueSource(url, config, res)\n }\n\n if (url.startsWith('/__maizzle/plaintext/')) {\n return await servePlaintext(url, config, renderer, res)\n }\n\n if (url.startsWith('/__maizzle/stats/')) {\n return await serveStats(url, config, renderer, res)\n }\n\n if (url.startsWith('/__maizzle/email/') && req.method === 'POST') {\n return await serveEmailEndpoint(url, req, res, config, renderer)\n }\n\n if (url === '/__maizzle/email-config') {\n return serveEmailConfig(config, res)\n }\n\n next()\n })\n\n // Dev UI fallback (after Vite's middleware)\n return () => {\n server.middlewares.use(async (req: any, res: any, next: any) => {\n if (isNavigationRequest(req)) {\n return await serveDevUI(server, res, req.url || '/', config)\n }\n\n next()\n })\n }\n },\n }\n}\n\nfunction isTemplateFile(file: string): boolean {\n return (file.endsWith('.vue') || file.endsWith('.md')) && !file.includes('server/ui')\n}\n\nfunction isNavigationRequest(req: any): boolean {\n const accept = req.headers?.accept || ''\n return req.method === 'GET' && accept.includes('text/html')\n}\n\n/**\n * Shape exposed to the dev UI both at initial HTML load (as\n * `window.__MAIZZLE_CONFIG__`) and on the `maizzle:config-updated` HMR event.\n * Add UI-visible config bits here; consumers on both ends pick up automatically.\n */\nfunction buildUiConfig(config: MaizzleConfig) {\n return {\n checks: config.server?.checks ?? true,\n }\n}\n\nasync function serveDevUI(server: ViteDevServer, res: any, url: string, config: MaizzleConfig) {\n let indexHtml = readFileSync(resolve(devUIDir, 'index.html'), 'utf-8')\n\n indexHtml = indexHtml.replace('./main.ts', `/@fs/${resolve(devUIDir, 'main.ts')}`)\n indexHtml = indexHtml.replace('./favicon.svg', `/@fs/${resolve(devUIDir, 'favicon.svg')}`)\n\n const configScript = `<script>window.__MAIZZLE_CONFIG__ = ${JSON.stringify(buildUiConfig(config))};</script>`\n indexHtml = indexHtml.replace('</head>', `${configScript}</head>`)\n\n const transformed = await server.transformIndexHtml(url, indexHtml)\n\n res.setHeader('Content-Type', 'text/html')\n res.end(transformed)\n}\n\nasync function serveTemplateList(config: MaizzleConfig, res: any) {\n const contentPatterns = config.content ?? ['emails/**/*.vue']\n const templates = await glob(contentPatterns)\n\n const data = templates.map(t => ({\n name: basename(t).replace(/\\.(vue|md)$/, ''),\n path: t,\n href: '/' + t.replace(/\\.(vue|md)$/, ''),\n }))\n\n res.setHeader('Content-Type', 'application/json')\n res.end(JSON.stringify(data))\n}\n\n/**\n * SSR render a .vue template using the Renderer (not the dev UI server).\n */\nasync function serveRenderedTemplate(url: string, config: MaizzleConfig, renderer: Renderer, res: any) {\n const templateSlug = url.replace('/__maizzle/render/', '').replace(/\\?.*$/, '')\n\n const contentPatterns = config.content ?? ['emails/**/*.vue']\n const templates = await glob(contentPatterns)\n const match = templates.find(t => t.replace(/\\.(vue|md)$/, '') === templateSlug)\n\n if (!match) {\n res.statusCode = 404\n res.end('Template not found')\n return\n }\n\n try {\n const absolutePath = resolve(match)\n\n // Invalidate all modules so template + component changes are picked up\n await renderer.invalidateAll()\n\n const rendered = await renderer.render(absolutePath, config)\n let html = rendered.html\n\n const templateConfig = rendered.templateConfig\n const doctype = rendered.doctype ?? templateConfig.doctype ?? '<!DOCTYPE html>'\n\n html = await runTransformers(html, templateConfig, absolutePath, doctype, rendered.tailwindBlocks)\n html = `${doctype}\\n${html}`\n\n res.setHeader('Content-Type', 'text/html')\n res.end(html)\n } catch (error: any) {\n res.statusCode = 500\n res.end(`<pre>${error.stack || error.message}</pre>`)\n }\n}\n\nlet highlighter: Highlighter | null = null\n\nasync function getHighlighter() {\n if (!highlighter) {\n highlighter = await createHighlighter({\n themes: ['laserwave'],\n langs: ['html', 'vue'],\n })\n }\n return highlighter\n}\n\nasync function serveHighlightedSource(url: string, config: MaizzleConfig, renderer: Renderer, res: any) {\n const templateSlug = url.replace('/__maizzle/source/', '').replace(/\\?.*$/, '')\n\n const contentPatterns = config.content ?? ['emails/**/*.vue']\n const templates = await glob(contentPatterns)\n const match = templates.find(t => t.replace(/\\.(vue|md)$/, '') === templateSlug)\n\n if (!match) {\n res.statusCode = 404\n res.end('Template not found')\n return\n }\n\n try {\n const absolutePath = resolve(match)\n\n await renderer.invalidateAll()\n\n const rendered = await renderer.render(absolutePath, config)\n let html = rendered.html\n\n const templateConfig = rendered.templateConfig\n const doctype = rendered.doctype ?? templateConfig.doctype ?? '<!DOCTYPE html>'\n html = await runTransformers(html, templateConfig, absolutePath, doctype, rendered.tailwindBlocks)\n\n html = `${doctype}\\n${html}`\n\n const hl = await getHighlighter()\n const highlighted = hl.codeToHtml(html, {\n lang: 'html',\n theme: 'laserwave',\n transformers: [{\n line(node, line) {\n node.properties['data-line'] = line\n },\n }],\n })\n\n res.setHeader('Content-Type', 'text/html')\n res.end(highlighted)\n } catch (error: any) {\n res.statusCode = 500\n res.end(`<pre>${error.stack || error.message}</pre>`)\n }\n}\n\nasync function serveVueSource(url: string, config: MaizzleConfig, res: any) {\n const templateSlug = url.replace('/__maizzle/vue-source/', '').replace(/\\?.*$/, '')\n\n const contentPatterns = config.content ?? ['emails/**/*.vue']\n const templates = await glob(contentPatterns)\n const match = templates.find(t => t.replace(/\\.(vue|md)$/, '') === templateSlug)\n\n if (!match) {\n res.statusCode = 404\n res.end('Template not found')\n return\n }\n\n try {\n const source = readFileSync(resolve(match), 'utf-8')\n const lang = match.endsWith('.md') ? 'html' : 'vue'\n\n const hl = await getHighlighter()\n const highlighted = hl.codeToHtml(source, {\n lang,\n theme: 'laserwave',\n transformers: [{\n line(node, line) {\n node.properties['data-line'] = line\n },\n }],\n })\n\n res.setHeader('Content-Type', 'text/html')\n res.end(highlighted)\n } catch (error: any) {\n res.statusCode = 500\n res.end(`<pre>${error.stack || error.message}</pre>`)\n }\n}\n\nasync function servePlaintext(url: string, config: MaizzleConfig, renderer: Renderer, res: any) {\n const templateSlug = url.replace('/__maizzle/plaintext/', '').replace(/\\?.*$/, '')\n\n const contentPatterns = config.content ?? ['emails/**/*.vue']\n const templates = await glob(contentPatterns)\n const match = templates.find(t => t.replace(/\\.(vue|md)$/, '') === templateSlug)\n\n if (!match) {\n res.statusCode = 404\n res.end('Template not found')\n return\n }\n\n try {\n const absolutePath = resolve(match)\n await renderer.invalidateAll()\n\n const rendered = await renderer.render(absolutePath, config)\n let html = rendered.html\n const templateConfig = rendered.templateConfig\n const doctype = rendered.doctype ?? templateConfig.doctype ?? '<!DOCTYPE html>'\n html = await runTransformers(html, templateConfig, absolutePath, doctype, rendered.tailwindBlocks)\n\n const plaintext = createPlaintext(html)\n\n res.setHeader('Content-Type', 'text/plain')\n res.end(plaintext)\n } catch (error: any) {\n res.statusCode = 500\n res.end(error.message)\n }\n}\n\nfunction humanFileSize(bytes: number, si = false, dp = 2) {\n const threshold = si ? 1000 : 1024\n\n if (Math.abs(bytes) < threshold) {\n return bytes + ' B'\n }\n\n const units = ['KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB']\n let u = -1\n const r = 10 ** dp\n\n do {\n bytes /= threshold\n ++u\n } while (Math.round(Math.abs(bytes) * r) / r >= threshold && u < units.length - 1)\n\n return bytes.toFixed(dp) + ' ' + units[u]\n}\n\nasync function serveStats(url: string, config: MaizzleConfig, renderer: Renderer, res: any) {\n const templateSlug = url.replace('/__maizzle/stats/', '').replace(/\\?.*$/, '')\n\n const contentPatterns = config.content ?? ['emails/**/*.vue']\n const templates = await glob(contentPatterns)\n const match = templates.find(t => t.replace(/\\.(vue|md)$/, '') === templateSlug)\n\n if (!match) {\n res.statusCode = 404\n res.end(JSON.stringify({ error: 'Template not found' }))\n return\n }\n\n try {\n const absolutePath = resolve(match)\n await renderer.invalidateAll()\n\n const rendered = await renderer.render(absolutePath, config)\n let html = rendered.html\n const templateConfig = rendered.templateConfig\n const doctype = rendered.doctype ?? templateConfig.doctype ?? '<!DOCTYPE html>'\n html = await runTransformers(html, templateConfig, absolutePath, doctype, rendered.tailwindBlocks)\n\n const sizeBytes = Buffer.byteLength(html, 'utf-8')\n\n // Count images: <img> tags and CSS background images\n const imgTags = (html.match(/<img\\b[^>]*>/gi) || []).length\n const bgImages = (html.match(/url\\s*\\([^)]+\\)/gi) || []).length\n const totalImages = imgTags + bgImages\n\n // Count links\n const links = (html.match(/<a\\b[^>]*href\\s*=/gi) || []).length\n\n res.setHeader('Content-Type', 'application/json')\n res.end(JSON.stringify({\n size: {\n bytes: sizeBytes,\n formatted: humanFileSize(sizeBytes),\n },\n images: totalImages,\n links,\n }))\n } catch (error: any) {\n res.statusCode = 500\n res.end(JSON.stringify({ error: error.message }))\n }\n}\n\nasync function serveEmailEndpoint(url: string, req: any, res: any, config: MaizzleConfig, renderer: Renderer) {\n const templateSlug = url.replace('/__maizzle/email/', '').replace(/\\?.*$/, '')\n\n const contentPatterns = config.content ?? ['emails/**/*.vue']\n const templates = await glob(contentPatterns)\n const match = templates.find(t => t.replace(/\\.(vue|md)$/, '') === templateSlug)\n\n if (!match) {\n res.statusCode = 404\n res.end(JSON.stringify({ success: false, message: 'Template not found' }))\n return\n }\n\n let body = ''\n for await (const chunk of req) body += chunk\n\n let payload: { to: string[]; subject: string }\n\n try {\n payload = JSON.parse(body)\n } catch {\n res.statusCode = 400\n res.end(JSON.stringify({ success: false, message: 'Invalid JSON' }))\n return\n }\n\n if (!payload.to?.length) {\n res.statusCode = 400\n res.end(JSON.stringify({ success: false, message: 'Missing recipients' }))\n return\n }\n\n try {\n const absolutePath = resolve(match)\n await renderer.invalidateAll()\n\n const rendered = await renderer.render(absolutePath, config)\n let html = rendered.html\n const templateConfig = rendered.templateConfig\n const doctype = rendered.doctype ?? templateConfig.doctype ?? '<!DOCTYPE html>'\n html = await runTransformers(html, templateConfig, absolutePath, doctype, rendered.tailwindBlocks)\n html = `${doctype}\\n${html}`\n\n const text = createPlaintext(html)\n\n const result = await sendEmail(\n { to: payload.to, subject: payload.subject, html, text },\n config,\n templateConfig,\n )\n\n res.setHeader('Content-Type', 'application/json')\n res.end(JSON.stringify(result))\n } catch (error: any) {\n res.statusCode = 500\n res.end(JSON.stringify({ success: false, message: error.message }))\n }\n}\n\nfunction serveEmailConfig(config: MaizzleConfig, res: any) {\n const emailConfig = config.server?.email\n res.setHeader('Content-Type', 'application/json')\n res.end(JSON.stringify({\n to: emailConfig?.to ? (Array.isArray(emailConfig.to) ? emailConfig.to : [emailConfig.to]) : [],\n from: emailConfig?.from ?? '',\n subject: emailConfig?.subject ?? '',\n hasTransport: !!emailConfig?.transport,\n }))\n}\n\nexport function printBanner(server: ViteDevServer, startupTime?: number) {\n const info = server.config.logger.info\n const time = startupTime ?? (server as any)._maizzleStartupTime\n\n const networkUrl = server.resolvedUrls?.network[0]\n if (networkUrl) {\n const qr = renderUnicodeCompact(networkUrl, { border: 1 })\n info('')\n info(qr.split('\\n').map(line => ` ${line}`).join('\\n'))\n }\n\n info('')\n info(` \\x1b[32m\\x1b[1mMAIZZLE\\x1b[0m\\x1b[32m v6.0.0\\x1b[0m \\x1b[2mready in\\x1b[0m \\x1b[1m${time}\\x1b[0m ms`)\n info('')\n server.printUrls()\n info('')\n}\n\nfunction customLogger() {\n const logger = createLogger('info')\n const warn = logger.warn\n\n logger.warn = (message, options) => {\n if (typeof message === 'string' && message.includes('<tr> cannot be child of <table>')) {\n return\n }\n\n warn(message, options)\n }\n\n return logger\n}\n"],"mappings":"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"}
@@ -1,4 +1,4 @@
1
- import { MaizzleConfig } from "../types/config.mjs";
1
+ import { MaizzleConfig } from "../types/config.js";
2
2
  //#region src/server/compatibility.d.ts
3
3
  interface Feature {
4
4
  slug: string;
@@ -54,4 +54,4 @@ declare function initCompatibility(): Promise<Indexes | null>;
54
54
  declare function serveCompatibility(url: string, res: any, config: MaizzleConfig, componentDirs: string[]): Promise<void>;
55
55
  //#endregion
56
56
  export { initCompatibility, serveCompatibility };
57
- //# sourceMappingURL=compatibility.d.mts.map
57
+ //# sourceMappingURL=compatibility.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"compatibility.d.ts","names":[],"sources":["../../src/server/compatibility.ts"],"mappings":";;UAiBU,OAAA;EACR,IAAA;EACA,KAAA;EACA,GAAA;EACA,QAAA;EACA,KAAA;AAAA;AAAA,UAoBQ,OAAA;EACR,SAAA;IAAa,SAAA;IAAmB,SAAA;IAAmB,WAAA;IAAqB,OAAA;IAAiB,KAAA;EAAA;EACzF,eAAA,EAAiB,MAAA;EACjB,OAAA,EAAS,GAAA,SAAY,OAAA;EACrB,YAAA,EAAc,GAAA,SAAY,KAAA;IAAQ,KAAA;IAAe,OAAA,EAAS,OAAA;EAAA;EAC1D,SAAA,EAAW,GAAA,SAAY,OAAA;EACvB,eAAA,EAAiB,GAAA,SAAY,OAAA;EAC7B,cAAA,EAAgB,GAAA,SAAY,OAAA;EAC5B,gBAAA,EAAkB,GAAA,SAAY,OAAA;EAC9B,WAAA,EAAa,GAAA,SAAY,OAAA;EACzB,OAAA,EAAS,GAAA,SAAY,OAAA;EACrB,YAAA,GAAe,OAAA;EACf,YAAA,GAAe,OAAA;EACf,UAAA,GAAa,OAAA;EACb,WAAA,GAAc,OAAA;EACd,cAAA,GAAiB,OAAA;EACjB,OAAA,EAAS,GAAA,SAAY,OAAA;EACrB,QAAA,EAAU,GAAA,SAAY,OAAA;EACtB,aAAA,EAAe,GAAA,SAAY,OAAA;EAC3B,cAAA,EAAgB,GAAA,SAAY,OAAA;EAC5B,WAAA,GAAc,OAAA;EACd,YAAA,GAAe,OAAA;EACf,eAAA,GAAkB,OAAA;EAClB,eAAA,GAAkB,OAAA;EAClB,mBAAA,GAAsB,OAAA;EACtB,aAAA,GAAgB,OAAA;EAChB,eAAA,GAAkB,OAAA;EAClB,QAAA,EAAU,GAAA,SAAY,OAAA;EATK;EAW3B,MAAA,EAAQ,GAAA;IAAc,KAAA;IAAe,GAAA;EAAA;AAAA;AAAA,iBAsOjB,iBAAA,CAAA,GAAqB,OAAA,CAAQ,OAAA;AAAA,iBA+jB7B,kBAAA,CACpB,GAAA,UACA,GAAA,OACA,MAAA,EAAQ,aAAA,EACR,aAAA,aAAuB,OAAA"}
@@ -1,6 +1,6 @@
1
- import { tailwindcss } from "../transformers/tailwindcss.mjs";
2
- import { buildComponentMap, findComponentTags, parseSfcBlocks } from "./sfc-utils.mjs";
3
- import { scanLint } from "./linter.mjs";
1
+ import { tailwindcss } from "../transformers/tailwindcss.js";
2
+ import { buildComponentMap, findComponentTags, parseSfcBlocks } from "./sfc-utils.js";
3
+ import { scanLint } from "./linter.js";
4
4
  import { readFileSync } from "node:fs";
5
5
  import { resolve } from "node:path";
6
6
  import { Parser } from "htmlparser2";
@@ -908,4 +908,4 @@ async function serveCompatibility(url, res, config, componentDirs) {
908
908
 
909
909
  //#endregion
910
910
  export { initCompatibility, serveCompatibility };
911
- //# sourceMappingURL=compatibility.mjs.map
911
+ //# sourceMappingURL=compatibility.js.map