@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
@@ -0,0 +1 @@
1
+ {"version":3,"file":"msoPlaceholders.d.ts","names":[],"sources":["../../src/transformers/msoPlaceholders.ts"],"mappings":";;;;;AA4CA;;;;;;;;;;;;;;;;;;;iBAAgB,eAAA,CAAgB,GAAA,EAAK,SAAA,KAAc,SAAA"}
@@ -0,0 +1,89 @@
1
+ import { walk } from "../utils/ast/walker.js";
2
+ import "../utils/ast/index.js";
3
+
4
+ //#region src/transformers/msoPlaceholders.ts
5
+ const RE_MAX_WIDTH = /(?:^|;\s*)max-width:\s*([^;]+)/i;
6
+ const RE_WIDTH = /(?:^|;\s*)width:\s*([^;]+)/i;
7
+ const RE_PERCENT = /^[\d.]+%$/;
8
+ const PADDING_DECL_RE = /(?:^|;)\s*(padding(?:-[a-z-]+)?\s*:\s*[^;]+)/gi;
9
+ function resolveWidth(value) {
10
+ const trimmed = value.trim();
11
+ if (RE_PERCENT.test(trimmed)) return trimmed;
12
+ const m = trimmed.match(/^([\d.]+)(px|rem|em|pt)?$/i);
13
+ if (!m) return null;
14
+ const n = parseFloat(m[1]);
15
+ switch ((m[2] || "px").toLowerCase()) {
16
+ case "px": return `${Math.round(n)}px`;
17
+ case "rem":
18
+ case "em": return `${Math.round(n * 16)}px`;
19
+ case "pt": return `${Math.round(n * 1.333)}px`;
20
+ default: return null;
21
+ }
22
+ }
23
+ /**
24
+ * Resolve all `__MAIZZLE_MSO*__` placeholders inside MSO conditional comments
25
+ * by reading inlined style + `data-*` markers on the paired elements.
26
+ *
27
+ * Two placeholder families:
28
+ *
29
+ * MSOW (`__MAIZZLE_MSOW_{id}__`) — emitted by `<Container>` and `<Section>`.
30
+ * Source element is marked with `data-maizzle-msow-id`. Reads inlined
31
+ * `max-width:` (falls back to `width:`) and normalizes to px. Falls
32
+ * back to `data-maizzle-msow-fallback` (default `600px`) when the
33
+ * value can't be parsed.
34
+ *
35
+ * MSOTDSTYLE (`__MAIZZLE_MSOTDSTYLE_{id}__`) — emitted by `<Container>`'s
36
+ * MSO `<td>`. Source element is marked with `data-maizzle-mso-td-id`.
37
+ * Copies every `padding*` declaration from inlined style and appends
38
+ * the `data-maizzle-mso-style` value (the user's `msoStyle` prop).
39
+ * Empty input resolves to '' so the placeholder collapses cleanly.
40
+ *
41
+ * Single collect-walk + single substitute-walk: the same Container div
42
+ * carries both marker kinds, so one element visit fills both maps.
43
+ */
44
+ function msoPlaceholders(dom) {
45
+ const widths = /* @__PURE__ */ new Map();
46
+ const tdStyles = /* @__PURE__ */ new Map();
47
+ walk(dom, (node) => {
48
+ const a = node.attribs;
49
+ if (!a) return;
50
+ const msowId = a["data-maizzle-msow-id"];
51
+ const tdId = a["data-maizzle-mso-td-id"];
52
+ if (!msowId && !tdId) return;
53
+ const style = a.style ?? "";
54
+ if (msowId) {
55
+ delete a["data-maizzle-msow-id"];
56
+ const fallback = a["data-maizzle-msow-fallback"] ?? "600px";
57
+ delete a["data-maizzle-msow-fallback"];
58
+ const raw = style.match(RE_MAX_WIDTH)?.[1] ?? style.match(RE_WIDTH)?.[1];
59
+ const resolved = raw ? resolveWidth(raw) : null;
60
+ widths.set(msowId, resolved ?? fallback);
61
+ }
62
+ if (tdId) {
63
+ delete a["data-maizzle-mso-td-id"];
64
+ const msoStyle = (a["data-maizzle-mso-style"] ?? "").trim().replace(/;\s*$/, "");
65
+ delete a["data-maizzle-mso-style"];
66
+ const parts = [];
67
+ if (style) for (const m of style.matchAll(PADDING_DECL_RE)) parts.push(m[1].trim());
68
+ if (msoStyle) parts.push(msoStyle);
69
+ tdStyles.set(tdId, parts.length ? ` style="${parts.join("; ")}"` : "");
70
+ }
71
+ });
72
+ if (widths.size === 0 && tdStyles.size === 0) return dom;
73
+ walk(dom, (node) => {
74
+ if (node.type !== "comment") return;
75
+ let data = node.data;
76
+ if (!data) return;
77
+ const hasMsow = widths.size > 0 && data.includes("__MAIZZLE_MSOW_");
78
+ const hasTd = tdStyles.size > 0 && data.includes("__MAIZZLE_MSOTDSTYLE_");
79
+ if (!hasMsow && !hasTd) return;
80
+ if (hasMsow) for (const [id, val] of widths) data = data.replaceAll(`__MAIZZLE_MSOW_${id}__`, val);
81
+ if (hasTd) for (const [id, val] of tdStyles) data = data.replaceAll(`__MAIZZLE_MSOTDSTYLE_${id}__`, val);
82
+ node.data = data;
83
+ });
84
+ return dom;
85
+ }
86
+
87
+ //#endregion
88
+ export { msoPlaceholders };
89
+ //# sourceMappingURL=msoPlaceholders.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"msoPlaceholders.js","names":[],"sources":["../../src/transformers/msoPlaceholders.ts"],"sourcesContent":["import { walk } from '../utils/ast/index.ts'\nimport type { ChildNode, Element } from 'domhandler'\n\nconst RE_MAX_WIDTH = /(?:^|;\\s*)max-width:\\s*([^;]+)/i\nconst RE_WIDTH = /(?:^|;\\s*)width:\\s*([^;]+)/i\nconst RE_PERCENT = /^[\\d.]+%$/\nconst PADDING_DECL_RE = /(?:^|;)\\s*(padding(?:-[a-z-]+)?\\s*:\\s*[^;]+)/gi\n\nfunction resolveWidth(value: string): string | null {\n const trimmed = value.trim()\n if (RE_PERCENT.test(trimmed)) return trimmed\n const m = trimmed.match(/^([\\d.]+)(px|rem|em|pt)?$/i)\n if (!m) return null\n const n = parseFloat(m[1])\n switch ((m[2] || 'px').toLowerCase()) {\n case 'px': return `${Math.round(n)}px`\n case 'rem':\n case 'em': return `${Math.round(n * 16)}px`\n case 'pt': return `${Math.round(n * 1.333)}px`\n default: return null\n }\n}\n\n/**\n * Resolve all `__MAIZZLE_MSO*__` placeholders inside MSO conditional comments\n * by reading inlined style + `data-*` markers on the paired elements.\n *\n * Two placeholder families:\n *\n * MSOW (`__MAIZZLE_MSOW_{id}__`) — emitted by `<Container>` and `<Section>`.\n * Source element is marked with `data-maizzle-msow-id`. Reads inlined\n * `max-width:` (falls back to `width:`) and normalizes to px. Falls\n * back to `data-maizzle-msow-fallback` (default `600px`) when the\n * value can't be parsed.\n *\n * MSOTDSTYLE (`__MAIZZLE_MSOTDSTYLE_{id}__`) — emitted by `<Container>`'s\n * MSO `<td>`. Source element is marked with `data-maizzle-mso-td-id`.\n * Copies every `padding*` declaration from inlined style and appends\n * the `data-maizzle-mso-style` value (the user's `msoStyle` prop).\n * Empty input resolves to '' so the placeholder collapses cleanly.\n *\n * Single collect-walk + single substitute-walk: the same Container div\n * carries both marker kinds, so one element visit fills both maps.\n */\nexport function msoPlaceholders(dom: ChildNode[]): ChildNode[] {\n const widths = new Map<string, string>()\n const tdStyles = new Map<string, string>()\n\n walk(dom, (node) => {\n const el = node as Element\n const a = el.attribs\n if (!a) return\n\n const msowId = a['data-maizzle-msow-id']\n const tdId = a['data-maizzle-mso-td-id']\n if (!msowId && !tdId) return\n\n const style = a.style ?? ''\n\n if (msowId) {\n delete a['data-maizzle-msow-id']\n const fallback = a['data-maizzle-msow-fallback'] ?? '600px'\n delete a['data-maizzle-msow-fallback']\n const raw = style.match(RE_MAX_WIDTH)?.[1] ?? style.match(RE_WIDTH)?.[1]\n const resolved = raw ? resolveWidth(raw) : null\n widths.set(msowId, resolved ?? fallback)\n }\n\n if (tdId) {\n delete a['data-maizzle-mso-td-id']\n const msoStyle = (a['data-maizzle-mso-style'] ?? '').trim().replace(/;\\s*$/, '')\n delete a['data-maizzle-mso-style']\n\n const parts: string[] = []\n if (style) {\n for (const m of style.matchAll(PADDING_DECL_RE)) {\n parts.push(m[1].trim())\n }\n }\n if (msoStyle) parts.push(msoStyle)\n\n tdStyles.set(tdId, parts.length ? ` style=\"${parts.join('; ')}\"` : '')\n }\n })\n\n if (widths.size === 0 && tdStyles.size === 0) return dom\n\n walk(dom, (node) => {\n if (node.type !== 'comment') return\n let data = (node as any).data as string\n if (!data) return\n const hasMsow = widths.size > 0 && data.includes('__MAIZZLE_MSOW_')\n const hasTd = tdStyles.size > 0 && data.includes('__MAIZZLE_MSOTDSTYLE_')\n if (!hasMsow && !hasTd) return\n\n if (hasMsow) {\n for (const [id, val] of widths) {\n data = data.replaceAll(`__MAIZZLE_MSOW_${id}__`, val)\n }\n }\n if (hasTd) {\n for (const [id, val] of tdStyles) {\n data = data.replaceAll(`__MAIZZLE_MSOTDSTYLE_${id}__`, val)\n }\n }\n ;(node as any).data = data\n })\n\n return dom\n}\n"],"mappings":";;;;AAGA,MAAM,eAAe;AACrB,MAAM,WAAW;AACjB,MAAM,aAAa;AACnB,MAAM,kBAAkB;AAExB,SAAS,aAAa,OAA8B;CAClD,MAAM,UAAU,MAAM,MAAM;AAC5B,KAAI,WAAW,KAAK,QAAQ,CAAE,QAAO;CACrC,MAAM,IAAI,QAAQ,MAAM,6BAA6B;AACrD,KAAI,CAAC,EAAG,QAAO;CACf,MAAM,IAAI,WAAW,EAAE,GAAG;AAC1B,UAAS,EAAE,MAAM,MAAM,aAAa,EAApC;EACE,KAAK,KAAM,QAAO,GAAG,KAAK,MAAM,EAAE,CAAC;EACnC,KAAK;EACL,KAAK,KAAM,QAAO,GAAG,KAAK,MAAM,IAAI,GAAG,CAAC;EACxC,KAAK,KAAM,QAAO,GAAG,KAAK,MAAM,IAAI,MAAM,CAAC;EAC3C,QAAS,QAAO;;;;;;;;;;;;;;;;;;;;;;;;AAyBpB,SAAgB,gBAAgB,KAA+B;CAC7D,MAAM,yBAAS,IAAI,KAAqB;CACxC,MAAM,2BAAW,IAAI,KAAqB;AAE1C,MAAK,MAAM,SAAS;EAElB,MAAM,IADK,KACE;AACb,MAAI,CAAC,EAAG;EAER,MAAM,SAAS,EAAE;EACjB,MAAM,OAAO,EAAE;AACf,MAAI,CAAC,UAAU,CAAC,KAAM;EAEtB,MAAM,QAAQ,EAAE,SAAS;AAEzB,MAAI,QAAQ;AACV,UAAO,EAAE;GACT,MAAM,WAAW,EAAE,iCAAiC;AACpD,UAAO,EAAE;GACT,MAAM,MAAM,MAAM,MAAM,aAAa,GAAG,MAAM,MAAM,MAAM,SAAS,GAAG;GACtE,MAAM,WAAW,MAAM,aAAa,IAAI,GAAG;AAC3C,UAAO,IAAI,QAAQ,YAAY,SAAS;;AAG1C,MAAI,MAAM;AACR,UAAO,EAAE;GACT,MAAM,YAAY,EAAE,6BAA6B,IAAI,MAAM,CAAC,QAAQ,SAAS,GAAG;AAChF,UAAO,EAAE;GAET,MAAM,QAAkB,EAAE;AAC1B,OAAI,MACF,MAAK,MAAM,KAAK,MAAM,SAAS,gBAAgB,CAC7C,OAAM,KAAK,EAAE,GAAG,MAAM,CAAC;AAG3B,OAAI,SAAU,OAAM,KAAK,SAAS;AAElC,YAAS,IAAI,MAAM,MAAM,SAAS,WAAW,MAAM,KAAK,KAAK,CAAC,KAAK,GAAG;;GAExE;AAEF,KAAI,OAAO,SAAS,KAAK,SAAS,SAAS,EAAG,QAAO;AAErD,MAAK,MAAM,SAAS;AAClB,MAAI,KAAK,SAAS,UAAW;EAC7B,IAAI,OAAQ,KAAa;AACzB,MAAI,CAAC,KAAM;EACX,MAAM,UAAU,OAAO,OAAO,KAAK,KAAK,SAAS,kBAAkB;EACnE,MAAM,QAAQ,SAAS,OAAO,KAAK,KAAK,SAAS,wBAAwB;AACzE,MAAI,CAAC,WAAW,CAAC,MAAO;AAExB,MAAI,QACF,MAAK,MAAM,CAAC,IAAI,QAAQ,OACtB,QAAO,KAAK,WAAW,kBAAkB,GAAG,KAAK,IAAI;AAGzD,MAAI,MACF,MAAK,MAAM,CAAC,IAAI,QAAQ,SACtB,QAAO,KAAK,WAAW,wBAAwB,GAAG,KAAK,IAAI;AAG9D,EAAC,KAAa,OAAO;GACtB;AAEF,QAAO"}
@@ -1,4 +1,4 @@
1
- import { CssConfig } from "../types/config.mjs";
1
+ import { CssConfig } from "../types/config.js";
2
2
  import { ChildNode } from "domhandler";
3
3
 
4
4
  //#region src/transformers/purgeCSS.d.ts
@@ -20,4 +20,4 @@ import { ChildNode } from "domhandler";
20
20
  declare function purgeCSS(dom: ChildNode[], config?: CssConfig): ChildNode[];
21
21
  //#endregion
22
22
  export { purgeCSS };
23
- //# sourceMappingURL=purgeCSS.d.mts.map
23
+ //# sourceMappingURL=purgeCSS.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"purgeCSS.d.ts","names":[],"sources":["../../src/transformers/purgeCSS.ts"],"mappings":";;;;;;AAiDA;;;;;;;;;;;;;iBAAgB,QAAA,CAAS,GAAA,EAAK,SAAA,IAAa,MAAA,GAAQ,SAAA,GAAiB,SAAA"}
@@ -1,9 +1,8 @@
1
- import { parse } from "../utils/ast/parser.mjs";
2
- import { walk } from "../utils/ast/walker.mjs";
3
- import { serialize } from "../utils/ast/serializer.mjs";
4
- import "../utils/ast/index.mjs";
1
+ import { parse } from "../utils/ast/parser.js";
2
+ import { walk } from "../utils/ast/walker.js";
3
+ import { serialize } from "../utils/ast/serializer.js";
4
+ import "../utils/ast/index.js";
5
5
  import { defu } from "defu";
6
- import postcss from "postcss";
7
6
  import safeParser from "postcss-safe-parser";
8
7
  import { selectAll } from "css-select";
9
8
  import { comb } from "email-comb";
@@ -63,8 +62,51 @@ function purgeCSS(dom, config = {}) {
63
62
  }, DEFAULT_OPTIONS);
64
63
  const safelist = [...DEFAULT_SAFELIST, ...userSafelist];
65
64
  dom = deepPurge(dom, safelist);
65
+ /**
66
+ * Shield embed style tags from email-comb. Comb has no skip option,
67
+ * so it strips CSS comments and drops class refs it can't match
68
+ * against visible CSS. Swap each embed tag's body for a unique
69
+ * stub rule (`.maizzle-keep-N{}`) so comb keeps the tag, then
70
+ * whitelist that stub plus every selector from the original
71
+ * CSS so comb leaves matching refs alone elsewhere — and
72
+ * finally restore the original CSS once comb has run.
73
+ */
74
+ const stash = [];
75
+ const extraWhitelist = [];
76
+ walk(dom, (node) => {
77
+ const el = node;
78
+ if (el.name !== "style" || !el.attribs) return;
79
+ if (!("embed" in el.attribs) && !("data-embed" in el.attribs)) return;
80
+ const textNode = el.children?.find((c) => c.type === "text");
81
+ if (!textNode?.data) return;
82
+ const token = `.maizzle-keep-${stash.length}`;
83
+ extraWhitelist.push(token);
84
+ for (const m of textNode.data.matchAll(/(?<![\w-])[.#][a-zA-Z_][\w-]*/g)) extraWhitelist.push(m[0]);
85
+ stash.push({
86
+ token,
87
+ original: textNode.data,
88
+ textNode
89
+ });
90
+ textNode.data = `${token}{}`;
91
+ });
92
+ if (extraWhitelist.length) options.whitelist = [...options.whitelist ?? [], ...extraWhitelist];
66
93
  const { result } = comb(serialize(dom), options);
94
+ /**
95
+ * Comb returns a fresh string, so we work off the post-parse tree:
96
+ * find each embed style tag whose body still starts with the stub
97
+ * token we planted earlier and swap the original CSS back in.
98
+ */
67
99
  let purgedDom = parse(result);
100
+ if (stash.length) walk(purgedDom, (node) => {
101
+ const el = node;
102
+ if (el.name !== "style" || !el.attribs) return;
103
+ if (!("embed" in el.attribs) && !("data-embed" in el.attribs)) return;
104
+ const textNode = el.children?.find((c) => c.type === "text");
105
+ if (!textNode?.data) return;
106
+ const trimmed = textNode.data.trim();
107
+ const match = stash.find((s) => trimmed === `${s.token}{}` || trimmed.startsWith(`${s.token}{`));
108
+ if (match) textNode.data = match.original;
109
+ });
68
110
  walk(purgedDom, (node) => {
69
111
  const el = node;
70
112
  if (el.name === "style" && el.attribs) {
@@ -94,7 +136,7 @@ function deepPurge(dom, safelist) {
94
136
  if ("data-embed" in el.attribs || "embed" in el.attribs) return;
95
137
  const textNode = el.children?.find((c) => c.type === "text");
96
138
  if (!textNode?.data?.trim()) return;
97
- const root = postcss.parse(textNode.data, { parser: safeParser });
139
+ const root = safeParser(textNode.data);
98
140
  root.walkRules((rule) => {
99
141
  if (rule.parent?.type === "atrule") return;
100
142
  const selectors = rule.selectors ?? [rule.selector];
@@ -129,4 +171,4 @@ function deepPurge(dom, safelist) {
129
171
 
130
172
  //#endregion
131
173
  export { purgeCSS };
132
- //# sourceMappingURL=purgeCSS.mjs.map
174
+ //# sourceMappingURL=purgeCSS.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"purgeCSS.js","names":["merge"],"sources":["../../src/transformers/purgeCSS.ts"],"sourcesContent":["import { comb } from 'email-comb'\nimport { defu as merge } from 'defu'\nimport safeParser from 'postcss-safe-parser'\nimport { selectAll } from 'css-select'\nimport type { ChildNode, Element } from 'domhandler'\nimport { parse, serialize, walk } from '../utils/ast/index.ts'\nimport type { CssConfig } from '../types/config.ts'\n\nconst DEFAULT_SAFELIST: string[] = [\n '*body*', // Gmail\n '.gmail*', // Gmail\n '.apple*', // Apple Mail\n '.ios*', // Mail on iOS\n '.ox-*', // Open-Xchange\n '.outlook*', // Outlook.com\n '[data-ogs*', // Outlook.com\n '.bloop_container', // Airmail\n '.Singleton', // Apple Mail 10\n '.unused', // Notes 8\n '.moz-text-html', // Thunderbird\n '.mail-detail-content', // Comcast, Libero webmail\n '*edo*', // Edison (all)\n '#*', // Freenet uses #msgBody\n '.lang*', // Fenced code blocks\n]\n\nconst DEFAULT_OPTIONS = {\n backend: [\n { heads: '{{', tails: '}}' },\n { heads: '{%', tails: '%}' },\n ],\n whitelist: [...DEFAULT_SAFELIST],\n}\n\n/**\n * Remove unused CSS transformer.\n *\n * Uses `email-comb` to strip CSS selectors and corresponding class/id\n * references that are not matched anywhere in the HTML body.\n *\n * Enable by setting `css.purge: true` (or passing options).\n * The user-supplied options are merged on top of the defaults, so\n * `safelist` values are **appended** to the built-in safelist rather\n * than replacing it.\n *\n * Accepts `ChildNode[]` as input, serializes internally before passing\n * to email-comb (which requires a raw HTML string), then parses the\n * result back to `ChildNode[]` so it fits in the DOM pipeline.\n */\nexport function purgeCSS(dom: ChildNode[], config: CssConfig = {}): ChildNode[] {\n const option = config.purge\n\n if (!option) return dom\n\n const userOptions = typeof option === 'object' ? option : {}\n\n // Merge user options on top of defaults.\n // defu merges objects deeply; for arrays it appends user values.\n // We want the user safelist appended to the default safelist,\n // so we build whitelist manually.\n const userSafelist = Array.isArray((userOptions as any).safelist)\n ? (userOptions as any).safelist as string[]\n : []\n\n const { safelist: _discard, ...restUserOptions } = userOptions as any\n\n const options = merge(\n { ...restUserOptions, whitelist: [...DEFAULT_SAFELIST, ...userSafelist] },\n DEFAULT_OPTIONS,\n )\n\n // Deep purge first: DOM-aware selector removal using PostCSS + css-select.\n // Runs before email-comb so that email-comb can clean up orphaned classes\n // in HTML attributes left behind by removed CSS rules.\n const safelist = [...DEFAULT_SAFELIST, ...userSafelist]\n dom = deepPurge(dom, safelist)\n\n /**\n * Shield embed style tags from email-comb. Comb has no skip option,\n * so it strips CSS comments and drops class refs it can't match\n * against visible CSS. Swap each embed tag's body for a unique\n * stub rule (`.maizzle-keep-N{}`) so comb keeps the tag, then\n * whitelist that stub plus every selector from the original\n * CSS so comb leaves matching refs alone elsewhere — and\n * finally restore the original CSS once comb has run.\n */\n const stash: { token: string; original: string; textNode: any }[] = []\n const extraWhitelist: string[] = []\n walk(dom, (node) => {\n const el = node as Element\n if (el.name !== 'style' || !el.attribs) return\n if (!('embed' in el.attribs) && !('data-embed' in el.attribs)) return\n const textNode = el.children?.find((c: any) => c.type === 'text') as any\n if (!textNode?.data) return\n const idx = stash.length\n const token = `.maizzle-keep-${idx}`\n extraWhitelist.push(token)\n for (const m of textNode.data.matchAll(/(?<![\\w-])[.#][a-zA-Z_][\\w-]*/g)) {\n extraWhitelist.push(m[0])\n }\n stash.push({ token, original: textNode.data, textNode })\n textNode.data = `${token}{}`\n })\n\n if (extraWhitelist.length) {\n options.whitelist = [...(options.whitelist as string[] ?? []), ...extraWhitelist]\n }\n\n const { result } = comb(serialize(dom), options)\n\n /**\n * Comb returns a fresh string, so we work off the post-parse tree:\n * find each embed style tag whose body still starts with the stub\n * token we planted earlier and swap the original CSS back in.\n */\n let purgedDom = parse(result)\n\n if (stash.length) {\n walk(purgedDom, (node) => {\n const el = node as Element\n if (el.name !== 'style' || !el.attribs) return\n if (!('embed' in el.attribs) && !('data-embed' in el.attribs)) return\n const textNode = el.children?.find((c: any) => c.type === 'text') as any\n if (!textNode?.data) return\n const trimmed = textNode.data.trim()\n const match = stash.find(s => trimmed === `${s.token}{}` || trimmed.startsWith(`${s.token}{`))\n if (match) textNode.data = match.original\n })\n }\n\n // Clean up data-embed/embed attributes — no longer needed after purging\n walk(purgedDom, (node) => {\n const el = node as Element\n if (el.name === 'style' && el.attribs) {\n delete el.attribs['data-embed']\n delete el.attribs.embed\n }\n })\n\n return purgedDom\n}\n\n/**\n * Deep purge: uses PostCSS to parse CSS in non-embedded style tags,\n * then checks each selector against the DOM with css-select.\n * Removes rules where no selector matches any element.\n */\nfunction isSafelisted(selector: string, safelist: string[]): boolean {\n return safelist.some((pattern) => {\n if (pattern.startsWith('*') && pattern.endsWith('*')) {\n return selector.includes(pattern.slice(1, -1))\n }\n if (pattern.endsWith('*')) {\n return selector.startsWith(pattern.slice(0, -1))\n }\n if (pattern.startsWith('*')) {\n return selector.endsWith(pattern.slice(1))\n }\n return selector === pattern\n })\n}\n\nfunction deepPurge(dom: ChildNode[], safelist: string[]): ChildNode[] {\n walk(dom, (node) => {\n const el = node as Element\n\n if (el.name !== 'style' || !el.attribs) return\n if ('data-embed' in el.attribs || 'embed' in el.attribs) return\n\n const textNode = el.children?.find((c: any) => c.type === 'text') as any\n if (!textNode?.data?.trim()) return\n\n const root = safeParser(textNode.data)\n\n root.walkRules((rule) => {\n // Skip rules inside @media or other at-rules — those may target\n // states we can't match statically (hover, responsive, etc.)\n if (rule.parent?.type === 'atrule') return\n\n const selectors = rule.selectors ?? [rule.selector]\n const matched = selectors.filter((sel) => {\n // Keep safelisted selectors\n if (isSafelisted(sel, safelist)) return true\n\n // Skip pseudo-classes/elements that can't be matched statically.\n // Functional pseudos like :not(), :is(), :where(), :has() are\n // matchable by css-select, so we only skip dynamic/state ones.\n if (/::[\\w-]/.test(sel)) return true\n if (/(?<!:):(?!not\\b|is\\b|where\\b|has\\b)[\\w-]/.test(sel.replace(/\\\\./g, ''))) return true\n\n try {\n return selectAll(sel, dom).length > 0\n } catch {\n // If css-select can't parse the selector, keep it\n return true\n }\n })\n\n if (matched.length === 0) {\n rule.remove()\n } else if (matched.length < selectors.length) {\n rule.selectors = matched\n }\n })\n\n // Remove empty at-rules\n root.walkAtRules((atRule) => {\n if (atRule.nodes?.length === 0) {\n atRule.remove()\n }\n })\n\n const purgedCss = root.toString()\n\n if (purgedCss.trim()) {\n textNode.data = purgedCss\n } else {\n // Remove the style tag entirely if empty\n const parent = el.parent\n if (parent && 'children' in parent) {\n const idx = parent.children.indexOf(el as any)\n if (idx !== -1) parent.children.splice(idx, 1)\n }\n }\n })\n\n return dom\n}\n"],"mappings":";;;;;;;;;;AAQA,MAAM,mBAA6B;CACjC;CACA;CACA;CACA;CACA;CACA;CACA;CACA;CACA;CACA;CACA;CACA;CACA;CACA;CACA;CACD;AAED,MAAM,kBAAkB;CACtB,SAAS,CACP;EAAE,OAAO;EAAM,OAAO;EAAM,EAC5B;EAAE,OAAO;EAAM,OAAO;EAAM,CAC7B;CACD,WAAW,CAAC,GAAG,iBAAiB;CACjC;;;;;;;;;;;;;;;;AAiBD,SAAgB,SAAS,KAAkB,SAAoB,EAAE,EAAe;CAC9E,MAAM,SAAS,OAAO;AAEtB,KAAI,CAAC,OAAQ,QAAO;CAEpB,MAAM,cAAc,OAAO,WAAW,WAAW,SAAS,EAAE;CAM5D,MAAM,eAAe,MAAM,QAAS,YAAoB,SAAS,GAC5D,YAAoB,WACrB,EAAE;CAEN,MAAM,EAAE,UAAU,UAAU,GAAG,oBAAoB;CAEnD,MAAM,UAAUA,KACd;EAAE,GAAG;EAAiB,WAAW,CAAC,GAAG,kBAAkB,GAAG,aAAa;EAAE,EACzE,gBACD;CAKD,MAAM,WAAW,CAAC,GAAG,kBAAkB,GAAG,aAAa;AACvD,OAAM,UAAU,KAAK,SAAS;;;;;;;;;;CAW9B,MAAM,QAA8D,EAAE;CACtE,MAAM,iBAA2B,EAAE;AACnC,MAAK,MAAM,SAAS;EAClB,MAAM,KAAK;AACX,MAAI,GAAG,SAAS,WAAW,CAAC,GAAG,QAAS;AACxC,MAAI,EAAE,WAAW,GAAG,YAAY,EAAE,gBAAgB,GAAG,SAAU;EAC/D,MAAM,WAAW,GAAG,UAAU,MAAM,MAAW,EAAE,SAAS,OAAO;AACjE,MAAI,CAAC,UAAU,KAAM;EAErB,MAAM,QAAQ,iBADF,MAAM;AAElB,iBAAe,KAAK,MAAM;AAC1B,OAAK,MAAM,KAAK,SAAS,KAAK,SAAS,iCAAiC,CACtE,gBAAe,KAAK,EAAE,GAAG;AAE3B,QAAM,KAAK;GAAE;GAAO,UAAU,SAAS;GAAM;GAAU,CAAC;AACxD,WAAS,OAAO,GAAG,MAAM;GACzB;AAEF,KAAI,eAAe,OACjB,SAAQ,YAAY,CAAC,GAAI,QAAQ,aAAyB,EAAE,EAAG,GAAG,eAAe;CAGnF,MAAM,EAAE,WAAW,KAAK,UAAU,IAAI,EAAE,QAAQ;;;;;;CAOhD,IAAI,YAAY,MAAM,OAAO;AAE7B,KAAI,MAAM,OACR,MAAK,YAAY,SAAS;EACxB,MAAM,KAAK;AACX,MAAI,GAAG,SAAS,WAAW,CAAC,GAAG,QAAS;AACxC,MAAI,EAAE,WAAW,GAAG,YAAY,EAAE,gBAAgB,GAAG,SAAU;EAC/D,MAAM,WAAW,GAAG,UAAU,MAAM,MAAW,EAAE,SAAS,OAAO;AACjE,MAAI,CAAC,UAAU,KAAM;EACrB,MAAM,UAAU,SAAS,KAAK,MAAM;EACpC,MAAM,QAAQ,MAAM,MAAK,MAAK,YAAY,GAAG,EAAE,MAAM,OAAO,QAAQ,WAAW,GAAG,EAAE,MAAM,GAAG,CAAC;AAC9F,MAAI,MAAO,UAAS,OAAO,MAAM;GACjC;AAIJ,MAAK,YAAY,SAAS;EACxB,MAAM,KAAK;AACX,MAAI,GAAG,SAAS,WAAW,GAAG,SAAS;AACrC,UAAO,GAAG,QAAQ;AAClB,UAAO,GAAG,QAAQ;;GAEpB;AAEF,QAAO;;;;;;;AAQT,SAAS,aAAa,UAAkB,UAA6B;AACnE,QAAO,SAAS,MAAM,YAAY;AAChC,MAAI,QAAQ,WAAW,IAAI,IAAI,QAAQ,SAAS,IAAI,CAClD,QAAO,SAAS,SAAS,QAAQ,MAAM,GAAG,GAAG,CAAC;AAEhD,MAAI,QAAQ,SAAS,IAAI,CACvB,QAAO,SAAS,WAAW,QAAQ,MAAM,GAAG,GAAG,CAAC;AAElD,MAAI,QAAQ,WAAW,IAAI,CACzB,QAAO,SAAS,SAAS,QAAQ,MAAM,EAAE,CAAC;AAE5C,SAAO,aAAa;GACpB;;AAGJ,SAAS,UAAU,KAAkB,UAAiC;AACpE,MAAK,MAAM,SAAS;EAClB,MAAM,KAAK;AAEX,MAAI,GAAG,SAAS,WAAW,CAAC,GAAG,QAAS;AACxC,MAAI,gBAAgB,GAAG,WAAW,WAAW,GAAG,QAAS;EAEzD,MAAM,WAAW,GAAG,UAAU,MAAM,MAAW,EAAE,SAAS,OAAO;AACjE,MAAI,CAAC,UAAU,MAAM,MAAM,CAAE;EAE7B,MAAM,OAAO,WAAW,SAAS,KAAK;AAEtC,OAAK,WAAW,SAAS;AAGvB,OAAI,KAAK,QAAQ,SAAS,SAAU;GAEpC,MAAM,YAAY,KAAK,aAAa,CAAC,KAAK,SAAS;GACnD,MAAM,UAAU,UAAU,QAAQ,QAAQ;AAExC,QAAI,aAAa,KAAK,SAAS,CAAE,QAAO;AAKxC,QAAI,UAAU,KAAK,IAAI,CAAE,QAAO;AAChC,QAAI,2CAA2C,KAAK,IAAI,QAAQ,QAAQ,GAAG,CAAC,CAAE,QAAO;AAErF,QAAI;AACF,YAAO,UAAU,KAAK,IAAI,CAAC,SAAS;YAC9B;AAEN,YAAO;;KAET;AAEF,OAAI,QAAQ,WAAW,EACrB,MAAK,QAAQ;YACJ,QAAQ,SAAS,UAAU,OACpC,MAAK,YAAY;IAEnB;AAGF,OAAK,aAAa,WAAW;AAC3B,OAAI,OAAO,OAAO,WAAW,EAC3B,QAAO,QAAQ;IAEjB;EAEF,MAAM,YAAY,KAAK,UAAU;AAEjC,MAAI,UAAU,MAAM,CAClB,UAAS,OAAO;OACX;GAEL,MAAM,SAAS,GAAG;AAClB,OAAI,UAAU,cAAc,QAAQ;IAClC,MAAM,MAAM,OAAO,SAAS,QAAQ,GAAU;AAC9C,QAAI,QAAQ,GAAI,QAAO,SAAS,OAAO,KAAK,EAAE;;;GAGlD;AAEF,QAAO"}
@@ -1,4 +1,4 @@
1
- import { AttributesConfig } from "../types/config.mjs";
1
+ import { AttributesConfig } from "../types/config.js";
2
2
  import { ChildNode } from "domhandler";
3
3
 
4
4
  //#region src/transformers/removeAttributes.d.ts
@@ -28,4 +28,4 @@ import { ChildNode } from "domhandler";
28
28
  declare function removeAttributes(dom: ChildNode[], config?: AttributesConfig): ChildNode[];
29
29
  //#endregion
30
30
  export { removeAttributes };
31
- //# sourceMappingURL=removeAttributes.d.mts.map
31
+ //# sourceMappingURL=removeAttributes.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"removeAttributes.d.ts","names":[],"sources":["../../src/transformers/removeAttributes.ts"],"mappings":";;;;;;AAkCA;;;;;;;;;;;;;;;;;;;;;iBAAgB,gBAAA,CAAiB,GAAA,EAAK,SAAA,IAAa,MAAA,GAAQ,gBAAA,GAAwB,SAAA"}
@@ -1,5 +1,5 @@
1
- import { walk } from "../utils/ast/walker.mjs";
2
- import "../utils/ast/index.mjs";
1
+ import { walk } from "../utils/ast/walker.js";
2
+ import "../utils/ast/index.js";
3
3
 
4
4
  //#region src/transformers/removeAttributes.ts
5
5
  /**
@@ -60,4 +60,4 @@ function removeAttributes(dom, config = {}) {
60
60
 
61
61
  //#endregion
62
62
  export { removeAttributes };
63
- //# sourceMappingURL=removeAttributes.mjs.map
63
+ //# sourceMappingURL=removeAttributes.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"removeAttributes.js","names":[],"sources":["../../src/transformers/removeAttributes.ts"],"sourcesContent":["import type { ChildNode, Element } from 'domhandler'\nimport { walk } from '../utils/ast/index.ts'\nimport type { AttributesConfig } from '../types/config.ts'\n\ninterface RemoveAttributeConfig {\n name: string\n value?: string | RegExp | boolean\n}\n\ntype RemoveAttributeOption = string | RemoveAttributeConfig\n\n/**\n * Remove attributes transformer.\n *\n * Removes specified HTML attributes from elements.\n *\n * By default, removes empty `style` and `class` attributes.\n *\n * Supports:\n * - String: removes attribute when empty (boolean or empty string)\n * - Object with name and value: removes when attribute matches exactly\n * - Object with name and RegExp value: removes when attribute value matches regex\n *\n * Configured via `remove` array:\n * ```js\n * {\n * remove: [\n * 'data-src', // Remove empty data-src attributes\n * { name: 'id', value: 'test' }, // Remove id=\"test\" exactly\n * { name: 'data-id', value: /\\d/ } // Remove data-id when value contains digits\n * ]\n * }\n * ```\n */\nexport function removeAttributes(dom: ChildNode[], config: AttributesConfig = {}): ChildNode[] {\n const removeOptions = config.remove\n\n // Always remove empty style and class attributes by default\n const alwaysRemove: RemoveAttributeOption[] = ['style', 'class']\n\n // Parse user options\n let userOptions: RemoveAttributeOption[] = []\n if (Array.isArray(removeOptions)) {\n userOptions = removeOptions as RemoveAttributeOption[]\n }\n\n // Combine default and user options\n const attributesToRemove: RemoveAttributeOption[] = [...alwaysRemove, ...userOptions]\n\n if (attributesToRemove.length === 0) {\n return dom\n }\n\n walk(dom, (node) => {\n const el = node as Element\n if (!el.attribs) return\n\n for (const attr of attributesToRemove) {\n let attrName: string\n let attrValue: string | RegExp | boolean | undefined\n\n if (typeof attr === 'string') {\n attrName = attr\n attrValue = true // Remove when value is empty (boolean true or empty string)\n } else {\n attrName = attr.name\n attrValue = attr.value\n }\n\n const currentValue = el.attribs[attrName]\n\n // Skip if attribute doesn't exist\n if (currentValue === undefined) continue\n\n let shouldRemove = false\n\n if (typeof attrValue === 'boolean') {\n // Remove if value is empty (boolean true is treated as no-value attribute)\n shouldRemove = currentValue === '' || (currentValue as unknown) === true\n } else if (typeof attrValue === 'string') {\n // Remove if value matches exactly\n shouldRemove = currentValue === attrValue\n } else if (attrValue instanceof RegExp) {\n // Remove if value matches regex\n shouldRemove = attrValue.test(currentValue)\n } else {\n // Default: remove if empty\n shouldRemove = currentValue === ''\n }\n\n if (shouldRemove) {\n delete el.attribs[attrName]\n }\n }\n })\n\n return dom\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;AAkCA,SAAgB,iBAAiB,KAAkB,SAA2B,EAAE,EAAe;CAC7F,MAAM,gBAAgB,OAAO;CAG7B,MAAM,eAAwC,CAAC,SAAS,QAAQ;CAGhE,IAAI,cAAuC,EAAE;AAC7C,KAAI,MAAM,QAAQ,cAAc,CAC9B,eAAc;CAIhB,MAAM,qBAA8C,CAAC,GAAG,cAAc,GAAG,YAAY;AAErF,KAAI,mBAAmB,WAAW,EAChC,QAAO;AAGT,MAAK,MAAM,SAAS;EAClB,MAAM,KAAK;AACX,MAAI,CAAC,GAAG,QAAS;AAEjB,OAAK,MAAM,QAAQ,oBAAoB;GACrC,IAAI;GACJ,IAAI;AAEJ,OAAI,OAAO,SAAS,UAAU;AAC5B,eAAW;AACX,gBAAY;UACP;AACL,eAAW,KAAK;AAChB,gBAAY,KAAK;;GAGnB,MAAM,eAAe,GAAG,QAAQ;AAGhC,OAAI,iBAAiB,OAAW;GAEhC,IAAI,eAAe;AAEnB,OAAI,OAAO,cAAc,UAEvB,gBAAe,iBAAiB,MAAO,iBAA6B;YAC3D,OAAO,cAAc,SAE9B,gBAAe,iBAAiB;YACvB,qBAAqB,OAE9B,gBAAe,UAAU,KAAK,aAAa;OAG3C,gBAAe,iBAAiB;AAGlC,OAAI,aACF,QAAO,GAAG,QAAQ;;GAGtB;AAEF,QAAO"}
@@ -1,4 +1,4 @@
1
- import { MaizzleConfig } from "../types/config.mjs";
1
+ import { MaizzleConfig } from "../types/config.js";
2
2
 
3
3
  //#region src/transformers/replaceStrings.d.ts
4
4
  /**
@@ -13,4 +13,4 @@ import { MaizzleConfig } from "../types/config.mjs";
13
13
  declare function replaceStrings(html: string, config?: MaizzleConfig): string;
14
14
  //#endregion
15
15
  export { replaceStrings };
16
- //# sourceMappingURL=replaceStrings.d.mts.map
16
+ //# sourceMappingURL=replaceStrings.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"replaceStrings.d.ts","names":[],"sources":["../../src/transformers/replaceStrings.ts"],"mappings":";;;;;AAWA;;;;;;;iBAAgB,cAAA,CAAe,IAAA,UAAc,MAAA,GAAQ,aAAA"}
@@ -16,4 +16,4 @@ function replaceStrings(html, config = {}) {
16
16
 
17
17
  //#endregion
18
18
  export { replaceStrings };
19
- //# sourceMappingURL=replaceStrings.mjs.map
19
+ //# sourceMappingURL=replaceStrings.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"replaceStrings.js","names":[],"sources":["../../src/transformers/replaceStrings.ts"],"sourcesContent":["import type { MaizzleConfig } from '../types/config.ts'\n\n/**\n * Replace strings transformer.\n *\n * Replaces strings in the HTML using the key-value pairs defined in\n * `config.replaceStrings`. Each key is treated as a regular expression\n * pattern (case-insensitive, global), and the value is the replacement.\n *\n * Character classes must be escaped in keys, e.g. `\\\\s` for `\\s`.\n */\nexport function replaceStrings(html: string, config: MaizzleConfig = {}): string {\n const replacements = config.replaceStrings\n\n if (!replacements || Object.keys(replacements).length === 0) return html\n\n return Object.entries(replacements).reduce(\n (result, [pattern, replacement]) => result.replace(new RegExp(pattern, 'gi'), replacement),\n html,\n )\n}\n"],"mappings":";;;;;;;;;;AAWA,SAAgB,eAAe,MAAc,SAAwB,EAAE,EAAU;CAC/E,MAAM,eAAe,OAAO;AAE5B,KAAI,CAAC,gBAAgB,OAAO,KAAK,aAAa,CAAC,WAAW,EAAG,QAAO;AAEpE,QAAO,OAAO,QAAQ,aAAa,CAAC,QACjC,QAAQ,CAAC,SAAS,iBAAiB,OAAO,QAAQ,IAAI,OAAO,SAAS,KAAK,EAAE,YAAY,EAC1F,KACD"}
@@ -1,4 +1,4 @@
1
- import { CssConfig } from "../types/config.mjs";
1
+ import { CssConfig } from "../types/config.js";
2
2
  import { ChildNode } from "domhandler";
3
3
 
4
4
  //#region src/transformers/safeClassNames.d.ts
@@ -19,4 +19,4 @@ import { ChildNode } from "domhandler";
19
19
  declare function safeClassNames(dom: ChildNode[], config?: CssConfig): ChildNode[];
20
20
  //#endregion
21
21
  export { safeClassNames };
22
- //# sourceMappingURL=safeClassNames.d.mts.map
22
+ //# sourceMappingURL=safeClassNames.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"safeClassNames.d.ts","names":[],"sources":["../../src/transformers/safeClassNames.ts"],"mappings":";;;;;;AAsGA;;;;;;;;;;;;iBAAgB,cAAA,CAAe,GAAA,EAAK,SAAA,IAAa,MAAA,GAAQ,SAAA,GAAiB,SAAA"}
@@ -1,5 +1,5 @@
1
- import { walk } from "../utils/ast/walker.mjs";
2
- import "../utils/ast/index.mjs";
1
+ import { walk } from "../utils/ast/walker.js";
2
+ import "../utils/ast/index.js";
3
3
  import postcss from "postcss";
4
4
  import safeParser from "postcss-safe-parser";
5
5
 
@@ -100,4 +100,4 @@ function safeClassNames(dom, config = {}) {
100
100
 
101
101
  //#endregion
102
102
  export { safeClassNames };
103
- //# sourceMappingURL=safeClassNames.mjs.map
103
+ //# sourceMappingURL=safeClassNames.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"safeClassNames.js","names":[],"sources":["../../src/transformers/safeClassNames.ts"],"sourcesContent":["import postcss from 'postcss'\nimport safeParser from 'postcss-safe-parser'\nimport type { ChildNode, Element } from 'domhandler'\nimport { walk } from '../utils/ast/index.ts'\nimport type { CssConfig } from '../types/config.ts'\n\nconst DEFAULT_REPLACEMENTS: Record<string, string> = {\n ':': '-',\n '/': '-',\n '%': 'pc',\n '.': '_',\n ',': '_',\n '#': '_',\n '[': '',\n ']': '',\n '(': '',\n ')': '',\n '{': '',\n '}': '',\n '!': '-i',\n '&': 'and-',\n '<': 'lt-',\n '=': 'eq-',\n '>': 'gt-',\n '|': 'or-',\n '@': 'at-',\n '?': 'q-',\n '\\\\': '-',\n '\"': '-',\n \"'\": '-',\n '*': '-',\n '+': '-',\n ';': '-',\n '^': '-',\n '`': '-',\n '~': '-',\n '$': '-',\n}\n\nfunction escapeForRegex(s: string): string {\n return s.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&')\n}\n\n/**\n * Replace escaped special characters in CSS selectors.\n *\n * Tailwind generates selectors like `.sm\\:text-base`. This function\n * replaces the `\\:` with `-` (or whatever the configured replacement is)\n * so the selector becomes `.sm-text-base`, which is safe for email clients.\n */\nfunction processCssSelectors(css: string, replacements: Record<string, string>): string {\n // Matches \\<char> in CSS selectors — e.g. \\: \\/ \\. \\[ etc.\n const selectorRegex = new RegExp(\n `\\\\\\\\(${Object.keys(replacements).map(escapeForRegex).join('|')})`,\n 'g',\n )\n\n return postcss([\n (root: postcss.Root) => {\n root.walkRules((rule: postcss.Rule) => {\n rule.selector = rule.selector\n .replace(selectorRegex, (_matched, char) => replacements[char] ?? _matched)\n // Handle CSS unicode escape for comma (\\2c → _)\n .replaceAll('\\\\2c ', '_')\n })\n },\n ]).process(css, { parser: safeParser }).css\n}\n\n/**\n * Replace unsafe special characters in a class attribute value.\n *\n * Splits on whitespace and replaces each char from the replacements map\n * in each class token individually.\n */\nfunction processClassAttr(classStr: string, replacements: Record<string, string>): string {\n return classStr\n .split(/\\s+/)\n .filter(Boolean)\n .map((cls) => {\n for (const [from, to] of Object.entries(replacements)) {\n cls = cls.split(from).join(to)\n }\n return cls\n })\n .join(' ')\n}\n\n/**\n * Safe class names transformer.\n *\n * Replaces unsafe characters (`:`, `/`, `[`, `]`, etc.) in:\n * - CSS selectors inside `<style>` tags\n * - HTML `class` attributes\n *\n * This makes Tailwind utility classes like `sm:text-base` safe for\n * email clients that cannot handle escaped characters in class names.\n *\n * Enabled by default. Disable by setting `css.safe` to `false`.\n * Customize replacements by passing a `Record<string, string>` — user\n * values are merged on top of the defaults.\n */\nexport function safeClassNames(dom: ChildNode[], config: CssConfig = {}): ChildNode[] {\n const option = config.safe ?? true\n\n if (!option) return dom\n\n const replacements: Record<string, string> =\n option && typeof option === 'object'\n ? { ...DEFAULT_REPLACEMENTS, ...option }\n : DEFAULT_REPLACEMENTS\n\n walk(dom, (node) => {\n const el = node as Element\n\n // Process CSS selectors inside <style> tags\n if (el.name === 'style' && el.children?.length) {\n const text = el.children.find((c) => c.type === 'text') as any\n if (text?.data?.trim()) {\n text.data = processCssSelectors(text.data, replacements)\n }\n }\n\n // Replace special chars in class attributes\n if ('attribs' in el && el.attribs?.class) {\n el.attribs.class = processClassAttr(el.attribs.class, replacements)\n }\n })\n\n return dom\n}\n"],"mappings":";;;;;;AAMA,MAAM,uBAA+C;CACnD,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,MAAM;CACN,MAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACL,KAAK;CACN;AAED,SAAS,eAAe,GAAmB;AACzC,QAAO,EAAE,QAAQ,uBAAuB,OAAO;;;;;;;;;AAUjD,SAAS,oBAAoB,KAAa,cAA8C;CAEtF,MAAM,gBAAgB,IAAI,OACxB,QAAQ,OAAO,KAAK,aAAa,CAAC,IAAI,eAAe,CAAC,KAAK,IAAI,CAAC,IAChE,IACD;AAED,QAAO,QAAQ,EACZ,SAAuB;AACtB,OAAK,WAAW,SAAuB;AACrC,QAAK,WAAW,KAAK,SAClB,QAAQ,gBAAgB,UAAU,SAAS,aAAa,SAAS,SAAS,CAE1E,WAAW,SAAS,IAAI;IAC3B;GAEL,CAAC,CAAC,QAAQ,KAAK,EAAE,QAAQ,YAAY,CAAC,CAAC;;;;;;;;AAS1C,SAAS,iBAAiB,UAAkB,cAA8C;AACxF,QAAO,SACJ,MAAM,MAAM,CACZ,OAAO,QAAQ,CACf,KAAK,QAAQ;AACZ,OAAK,MAAM,CAAC,MAAM,OAAO,OAAO,QAAQ,aAAa,CACnD,OAAM,IAAI,MAAM,KAAK,CAAC,KAAK,GAAG;AAEhC,SAAO;GACP,CACD,KAAK,IAAI;;;;;;;;;;;;;;;;AAiBd,SAAgB,eAAe,KAAkB,SAAoB,EAAE,EAAe;CACpF,MAAM,SAAS,OAAO,QAAQ;AAE9B,KAAI,CAAC,OAAQ,QAAO;CAEpB,MAAM,eACJ,UAAU,OAAO,WAAW,WACxB;EAAE,GAAG;EAAsB,GAAG;EAAQ,GACtC;AAEN,MAAK,MAAM,SAAS;EAClB,MAAM,KAAK;AAGX,MAAI,GAAG,SAAS,WAAW,GAAG,UAAU,QAAQ;GAC9C,MAAM,OAAO,GAAG,SAAS,MAAM,MAAM,EAAE,SAAS,OAAO;AACvD,OAAI,MAAM,MAAM,MAAM,CACpB,MAAK,OAAO,oBAAoB,KAAK,MAAM,aAAa;;AAK5D,MAAI,aAAa,MAAM,GAAG,SAAS,MACjC,IAAG,QAAQ,QAAQ,iBAAiB,GAAG,QAAQ,OAAO,aAAa;GAErE;AAEF,QAAO"}
@@ -1,4 +1,4 @@
1
- import { CssConfig } from "../types/config.mjs";
1
+ import { CssConfig } from "../types/config.js";
2
2
  import { ChildNode } from "domhandler";
3
3
 
4
4
  //#region src/transformers/shorthandCSS.d.ts
@@ -21,4 +21,4 @@ import { ChildNode } from "domhandler";
21
21
  declare function shorthandCSS(dom: ChildNode[], config?: CssConfig): ChildNode[];
22
22
  //#endregion
23
23
  export { shorthandCSS };
24
- //# sourceMappingURL=shorthandCSS.d.mts.map
24
+ //# sourceMappingURL=shorthandCSS.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shorthandCSS.d.ts","names":[],"sources":["../../src/transformers/shorthandCSS.ts"],"mappings":";;;;;;AA2BA;;;;;;;;;;;;;;iBAAgB,YAAA,CAAa,GAAA,EAAK,SAAA,IAAa,MAAA,GAAQ,SAAA,GAAiB,SAAA"}
@@ -1,5 +1,5 @@
1
- import { walk } from "../utils/ast/walker.mjs";
2
- import "../utils/ast/index.mjs";
1
+ import { walk } from "../utils/ast/walker.js";
2
+ import "../utils/ast/index.js";
3
3
  import postcss from "postcss";
4
4
  import safeParser from "postcss-safe-parser";
5
5
  import mergeLonghand from "postcss-merge-longhand";
@@ -45,4 +45,4 @@ function shorthandCSS(dom, config = {}) {
45
45
 
46
46
  //#endregion
47
47
  export { shorthandCSS };
48
- //# sourceMappingURL=shorthandCSS.mjs.map
48
+ //# sourceMappingURL=shorthandCSS.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shorthandCSS.js","names":[],"sources":["../../src/transformers/shorthandCSS.ts"],"sourcesContent":["import postcss from 'postcss'\nimport safeParser from 'postcss-safe-parser'\nimport mergeLonghand from 'postcss-merge-longhand'\nimport type { ChildNode, Element } from 'domhandler'\nimport { walk } from '../utils/ast/index.ts'\nimport type { CssConfig } from '../types/config.ts'\n\ninterface ShorthandCssOptions {\n tags?: string[]\n}\n\n/**\n * Shorthand CSS transformer.\n *\n * Rewrites longhand CSS inside `style` attributes with shorthand syntax.\n * Works with margin, padding, and border when all sides are specified.\n *\n * For example:\n * `margin-left: 2px; margin-right: 2px; margin-top: 4px; margin-bottom: 4px`\n * becomes:\n * `margin: 4px 2px`\n *\n * Enabled via `css.shorthand`:\n * - `true`: enable for all tags\n * - `{ tags: ['td', 'div'] }`: enable only for specified tags\n * - `false` or omitted: disabled\n */\nexport function shorthandCSS(dom: ChildNode[], config: CssConfig = {}): ChildNode[] {\n const option = config.shorthand\n\n // Disabled by default\n if (!option) {\n return dom\n }\n\n // Parse options\n const options: ShorthandCssOptions = typeof option === 'object' ? option : {}\n const allowedTags = options.tags ?? []\n const hasTagFilter = allowedTags.length > 0\n\n walk(dom, (node) => {\n const el = node as Element\n\n // Skip if no attribs or no style\n if (!el.attribs?.style) {\n return\n }\n\n // Skip if tag filter is active and this tag is not allowed\n if (hasTagFilter && !allowedTags.includes(el.name)) {\n return\n }\n\n const styleValue = el.attribs.style\n\n try {\n // Process the style with postcss-merge-longhand\n // Wrap in a dummy selector since postcss needs a rule\n const { css } = postcss()\n .use(mergeLonghand)\n .process(`div { ${styleValue} }`, { parser: safeParser })\n\n // Extract the content between the braces\n const match = css.match(/div\\s*\\{\\s*([^}]+)\\s*\\}/)\n if (match && match[1]) {\n const newStyle = match[1].trim()\n if (newStyle !== styleValue) {\n el.attribs.style = newStyle\n }\n }\n } catch {\n // If processing fails, keep the original style\n }\n })\n\n return dom\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;AA2BA,SAAgB,aAAa,KAAkB,SAAoB,EAAE,EAAe;CAClF,MAAM,SAAS,OAAO;AAGtB,KAAI,CAAC,OACH,QAAO;CAKT,MAAM,eAD+B,OAAO,WAAW,WAAW,SAAS,EAAE,EACjD,QAAQ,EAAE;CACtC,MAAM,eAAe,YAAY,SAAS;AAE1C,MAAK,MAAM,SAAS;EAClB,MAAM,KAAK;AAGX,MAAI,CAAC,GAAG,SAAS,MACf;AAIF,MAAI,gBAAgB,CAAC,YAAY,SAAS,GAAG,KAAK,CAChD;EAGF,MAAM,aAAa,GAAG,QAAQ;AAE9B,MAAI;GAGF,MAAM,EAAE,QAAQ,SAAS,CACtB,IAAI,cAAc,CAClB,QAAQ,SAAS,WAAW,KAAK,EAAE,QAAQ,YAAY,CAAC;GAG3D,MAAM,QAAQ,IAAI,MAAM,0BAA0B;AAClD,OAAI,SAAS,MAAM,IAAI;IACrB,MAAM,WAAW,MAAM,GAAG,MAAM;AAChC,QAAI,aAAa,WACf,IAAG,QAAQ,QAAQ;;UAGjB;GAGR;AAEF,QAAO"}
@@ -1,4 +1,4 @@
1
- import { CssConfig } from "../types/config.mjs";
1
+ import { CssConfig } from "../types/config.js";
2
2
  import { ChildNode } from "domhandler";
3
3
 
4
4
  //#region src/transformers/sixHex.d.ts
@@ -13,4 +13,4 @@ import { ChildNode } from "domhandler";
13
13
  declare function sixHex(dom: ChildNode[], config?: CssConfig): ChildNode[];
14
14
  //#endregion
15
15
  export { sixHex };
16
- //# sourceMappingURL=sixHex.d.mts.map
16
+ //# sourceMappingURL=sixHex.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sixHex.d.ts","names":[],"sources":["../../src/transformers/sixHex.ts"],"mappings":";;;;;;AAeA;;;;;;iBAAgB,MAAA,CAAO,GAAA,EAAK,SAAA,IAAa,MAAA,GAAQ,SAAA,GAAiB,SAAA"}
@@ -1,5 +1,5 @@
1
- import { walk } from "../utils/ast/walker.mjs";
2
- import "../utils/ast/index.mjs";
1
+ import { walk } from "../utils/ast/walker.js";
2
+ import "../utils/ast/index.js";
3
3
  import { conv } from "color-shorthand-hex-to-six-digit";
4
4
 
5
5
  //#region src/transformers/sixHex.ts
@@ -27,4 +27,4 @@ function sixHex(dom, config = {}) {
27
27
 
28
28
  //#endregion
29
29
  export { sixHex };
30
- //# sourceMappingURL=sixHex.mjs.map
30
+ //# sourceMappingURL=sixHex.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sixHex.js","names":[],"sources":["../../src/transformers/sixHex.ts"],"sourcesContent":["import { conv } from 'color-shorthand-hex-to-six-digit'\nimport type { ChildNode, Element } from 'domhandler'\nimport { walk } from '../utils/ast/index.ts'\nimport type { CssConfig } from '../types/config.ts'\n\nconst targets = new Set(['bgcolor', 'color'])\n\n/**\n * Six-digit HEX transformer.\n *\n * Converts 3-digit HEX color codes to 6-digit in `bgcolor` and `color`\n * attributes, for better email client compatibility.\n *\n * Enabled by default via `css.sixHex`.\n */\nexport function sixHex(dom: ChildNode[], config: CssConfig = {}): ChildNode[] {\n if (config.sixHex === false) {\n return dom\n }\n\n walk(dom, (node) => {\n const el = node as Element\n\n if (!el.attribs) {\n return\n }\n\n for (const attr of targets) {\n const value = el.attribs[attr]\n\n if (value) {\n el.attribs[attr] = conv(value)\n }\n }\n })\n\n return dom\n}\n"],"mappings":";;;;;AAKA,MAAM,UAAU,IAAI,IAAI,CAAC,WAAW,QAAQ,CAAC;;;;;;;;;AAU7C,SAAgB,OAAO,KAAkB,SAAoB,EAAE,EAAe;AAC5E,KAAI,OAAO,WAAW,MACpB,QAAO;AAGT,MAAK,MAAM,SAAS;EAClB,MAAM,KAAK;AAEX,MAAI,CAAC,GAAG,QACN;AAGF,OAAK,MAAM,QAAQ,SAAS;GAC1B,MAAM,QAAQ,GAAG,QAAQ;AAEzB,OAAI,MACF,IAAG,QAAQ,QAAQ,KAAK,MAAM;;GAGlC;AAEF,QAAO"}
@@ -0,0 +1,16 @@
1
+ import { MaizzleConfig } from "../types/config.js";
2
+ import { TailwindBlock } from "../composables/renderContext.js";
3
+ import { ChildNode } from "domhandler";
4
+
5
+ //#region src/transformers/tailwindComponent.d.ts
6
+ /**
7
+ * Compile Tailwind CSS for each top-level <Tailwind> block in the render
8
+ * context. Nested <Tailwind> instances are flattened: their classes flow
9
+ * up to the outermost block, their `#config` slot (if any) is ignored.
10
+ * One <style> per outermost block is appended to <head>; marker comments
11
+ * are stripped after.
12
+ */
13
+ declare function tailwindComponent(dom: ChildNode[], blocks: TailwindBlock[], config: MaizzleConfig, filePath?: string): Promise<ChildNode[]>;
14
+ //#endregion
15
+ export { tailwindComponent };
16
+ //# sourceMappingURL=tailwindComponent.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tailwindComponent.d.ts","names":[],"sources":["../../src/transformers/tailwindComponent.ts"],"mappings":";;;;;;;AA0BA;;;;;iBAAsB,iBAAA,CACpB,GAAA,EAAK,SAAA,IACL,MAAA,EAAQ,aAAA,IACR,MAAA,EAAQ,aAAA,EACR,QAAA,YACC,OAAA,CAAQ,SAAA"}
@@ -0,0 +1,93 @@
1
+ import { walk } from "../utils/ast/walker.js";
2
+ import "../utils/ast/index.js";
3
+ import { compileTailwindCss } from "../utils/compileTailwindCss.js";
4
+ import { resolve } from "node:path";
5
+
6
+ //#region src/transformers/tailwindComponent.ts
7
+ const DEFAULT_SEED = "@import \"@maizzle/tailwindcss\";";
8
+ const OPEN_RE = /^mz-tw:(\S+)$/;
9
+ const CLOSE_RE = /^\/mz-tw:(\S+)$/;
10
+ /**
11
+ * Compile Tailwind CSS for each top-level <Tailwind> block in the render
12
+ * context. Nested <Tailwind> instances are flattened: their classes flow
13
+ * up to the outermost block, their `#config` slot (if any) is ignored.
14
+ * One <style> per outermost block is appended to <head>; marker comments
15
+ * are stripped after.
16
+ */
17
+ async function tailwindComponent(dom, blocks, config, filePath) {
18
+ if (!blocks.length) return dom;
19
+ const map = /* @__PURE__ */ new Map();
20
+ for (const b of blocks) map.set(b.id, {
21
+ id: b.id,
22
+ configCss: b.css,
23
+ nested: false,
24
+ classes: /* @__PURE__ */ new Set()
25
+ });
26
+ const stack = [];
27
+ const markers = [];
28
+ walk(dom, (node) => {
29
+ if (node.type === "comment") {
30
+ const data = node.data;
31
+ const open = data.match(OPEN_RE);
32
+ const close = data.match(CLOSE_RE);
33
+ if (open) {
34
+ const id = open[1];
35
+ const meta = map.get(id);
36
+ if (meta && stack.length > 0) meta.nested = true;
37
+ if (meta) stack.push(id);
38
+ markers.push(node);
39
+ } else if (close) {
40
+ const id = close[1];
41
+ if (stack[stack.length - 1] === id) stack.pop();
42
+ markers.push(node);
43
+ }
44
+ return;
45
+ }
46
+ const el = node;
47
+ if (el.attribs?.class && stack.length > 0) map.get(stack[0]).classes.add(el.attribs.class);
48
+ });
49
+ const fromPath = filePath ?? resolve(process.cwd(), "template.vue");
50
+ let head;
51
+ walk(dom, (n) => {
52
+ if (!head && n.name === "head") head = n;
53
+ });
54
+ if (!head) throw new Error("`Tailwind` component requires `Head` component to be present in the template.");
55
+ for (const meta of map.values()) {
56
+ if (meta.nested) continue;
57
+ const css = (await compileTailwindCss(buildCssInput(meta.configCss, meta.classes), config, `${fromPath}?tw=${meta.id}`)).trim();
58
+ if (!css) continue;
59
+ const styleNode = {
60
+ type: "tag",
61
+ name: "style",
62
+ attribs: { raw: "" },
63
+ children: [],
64
+ parent: head,
65
+ prev: null,
66
+ next: null
67
+ };
68
+ styleNode.children = [{
69
+ type: "text",
70
+ data: css,
71
+ parent: styleNode,
72
+ prev: null,
73
+ next: null
74
+ }];
75
+ head.children.push(styleNode);
76
+ }
77
+ for (const c of markers) {
78
+ const parent = c.parent;
79
+ if (!parent?.children) continue;
80
+ const i = parent.children.indexOf(c);
81
+ if (i >= 0) parent.children.splice(i, 1);
82
+ }
83
+ return dom;
84
+ }
85
+ function buildCssInput(configCss, classes) {
86
+ const seed = configCss ?? DEFAULT_SEED;
87
+ if (!classes.size) return seed;
88
+ return `${seed}\n@source inline("${[...classes].join(" ").replace(/"/g, "\\\"")}");`;
89
+ }
90
+
91
+ //#endregion
92
+ export { tailwindComponent };
93
+ //# sourceMappingURL=tailwindComponent.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tailwindComponent.js","names":[],"sources":["../../src/transformers/tailwindComponent.ts"],"sourcesContent":["import { resolve } from 'node:path'\nimport type { ChildNode, Element, Comment } from 'domhandler'\nimport { walk } from '../utils/ast/index.ts'\nimport { compileTailwindCss } from '../utils/compileTailwindCss.ts'\nimport type { TailwindBlock } from '../composables/renderContext.ts'\nimport type { MaizzleConfig } from '../types/config.ts'\n\nconst DEFAULT_SEED = '@import \"@maizzle/tailwindcss\";'\n\ninterface BlockMeta {\n id: string\n configCss?: string\n nested: boolean\n classes: Set<string>\n}\n\nconst OPEN_RE = /^mz-tw:(\\S+)$/\nconst CLOSE_RE = /^\\/mz-tw:(\\S+)$/\n\n/**\n * Compile Tailwind CSS for each top-level <Tailwind> block in the render\n * context. Nested <Tailwind> instances are flattened: their classes flow\n * up to the outermost block, their `#config` slot (if any) is ignored.\n * One <style> per outermost block is appended to <head>; marker comments\n * are stripped after.\n */\nexport async function tailwindComponent(\n dom: ChildNode[],\n blocks: TailwindBlock[],\n config: MaizzleConfig,\n filePath?: string,\n): Promise<ChildNode[]> {\n if (!blocks.length) return dom\n\n const map = new Map<string, BlockMeta>()\n for (const b of blocks) {\n map.set(b.id, { id: b.id, configCss: b.css, nested: false, classes: new Set() })\n }\n\n const stack: string[] = []\n const markers: Comment[] = []\n\n walk(dom, (node) => {\n if (node.type === 'comment') {\n const data = (node as Comment).data\n const open = data.match(OPEN_RE)\n const close = data.match(CLOSE_RE)\n if (open) {\n const id = open[1]\n const meta = map.get(id)\n if (meta && stack.length > 0) meta.nested = true\n if (meta) stack.push(id)\n markers.push(node as Comment)\n } else if (close) {\n const id = close[1]\n if (stack[stack.length - 1] === id) stack.pop()\n markers.push(node as Comment)\n }\n return\n }\n\n const el = node as Element\n // Always assign to the OUTERMOST active marker (stack[0]) so nested\n // <Tailwind> blocks merge their classes into the parent's scope.\n if (el.attribs?.class && stack.length > 0) {\n map.get(stack[0])!.classes.add(el.attribs.class)\n }\n })\n\n const fromPath = filePath ?? resolve(process.cwd(), 'template.vue')\n\n let head: Element | undefined\n walk(dom, (n) => {\n if (!head && (n as Element).name === 'head') head = n as Element\n })\n\n if (!head) {\n throw new Error('`Tailwind` component requires `Head` component to be present in the template.')\n }\n\n // Compile + inject one <style raw> per outermost block. `raw` opts the\n // existing tailwindcss transformer out of recompiling already-compiled CSS.\n for (const meta of map.values()) {\n if (meta.nested) continue\n\n const cssInput = buildCssInput(meta.configCss, meta.classes)\n const css = (await compileTailwindCss(cssInput, config, `${fromPath}?tw=${meta.id}`)).trim()\n if (!css) continue\n\n const styleNode: Element = {\n type: 'tag',\n name: 'style',\n attribs: { raw: '' },\n children: [],\n parent: head,\n prev: null,\n next: null,\n } as any\n\n const textNode = {\n type: 'text',\n data: css,\n parent: styleNode,\n prev: null,\n next: null,\n } as any\n\n styleNode.children = [textNode]\n head.children.push(styleNode)\n }\n\n // Strip marker comments from their parents\n for (const c of markers) {\n const parent = c.parent as Element | null\n if (!parent?.children) continue\n const i = parent.children.indexOf(c)\n if (i >= 0) parent.children.splice(i, 1)\n }\n\n return dom\n}\n\nfunction buildCssInput(configCss: string | undefined, classes: Set<string>): string {\n const seed = configCss ?? DEFAULT_SEED\n\n if (!classes.size) return seed\n\n const inline = [...classes].join(' ').replace(/\"/g, '\\\\\"')\n return `${seed}\\n@source inline(\"${inline}\");`\n}\n"],"mappings":";;;;;;AAOA,MAAM,eAAe;AASrB,MAAM,UAAU;AAChB,MAAM,WAAW;;;;;;;;AASjB,eAAsB,kBACpB,KACA,QACA,QACA,UACsB;AACtB,KAAI,CAAC,OAAO,OAAQ,QAAO;CAE3B,MAAM,sBAAM,IAAI,KAAwB;AACxC,MAAK,MAAM,KAAK,OACd,KAAI,IAAI,EAAE,IAAI;EAAE,IAAI,EAAE;EAAI,WAAW,EAAE;EAAK,QAAQ;EAAO,yBAAS,IAAI,KAAK;EAAE,CAAC;CAGlF,MAAM,QAAkB,EAAE;CAC1B,MAAM,UAAqB,EAAE;AAE7B,MAAK,MAAM,SAAS;AAClB,MAAI,KAAK,SAAS,WAAW;GAC3B,MAAM,OAAQ,KAAiB;GAC/B,MAAM,OAAO,KAAK,MAAM,QAAQ;GAChC,MAAM,QAAQ,KAAK,MAAM,SAAS;AAClC,OAAI,MAAM;IACR,MAAM,KAAK,KAAK;IAChB,MAAM,OAAO,IAAI,IAAI,GAAG;AACxB,QAAI,QAAQ,MAAM,SAAS,EAAG,MAAK,SAAS;AAC5C,QAAI,KAAM,OAAM,KAAK,GAAG;AACxB,YAAQ,KAAK,KAAgB;cACpB,OAAO;IAChB,MAAM,KAAK,MAAM;AACjB,QAAI,MAAM,MAAM,SAAS,OAAO,GAAI,OAAM,KAAK;AAC/C,YAAQ,KAAK,KAAgB;;AAE/B;;EAGF,MAAM,KAAK;AAGX,MAAI,GAAG,SAAS,SAAS,MAAM,SAAS,EACtC,KAAI,IAAI,MAAM,GAAG,CAAE,QAAQ,IAAI,GAAG,QAAQ,MAAM;GAElD;CAEF,MAAM,WAAW,YAAY,QAAQ,QAAQ,KAAK,EAAE,eAAe;CAEnE,IAAI;AACJ,MAAK,MAAM,MAAM;AACf,MAAI,CAAC,QAAS,EAAc,SAAS,OAAQ,QAAO;GACpD;AAEF,KAAI,CAAC,KACH,OAAM,IAAI,MAAM,gFAAgF;AAKlG,MAAK,MAAM,QAAQ,IAAI,QAAQ,EAAE;AAC/B,MAAI,KAAK,OAAQ;EAGjB,MAAM,OAAO,MAAM,mBADF,cAAc,KAAK,WAAW,KAAK,QAAQ,EACZ,QAAQ,GAAG,SAAS,MAAM,KAAK,KAAK,EAAE,MAAM;AAC5F,MAAI,CAAC,IAAK;EAEV,MAAM,YAAqB;GACzB,MAAM;GACN,MAAM;GACN,SAAS,EAAE,KAAK,IAAI;GACpB,UAAU,EAAE;GACZ,QAAQ;GACR,MAAM;GACN,MAAM;GACP;AAUD,YAAU,WAAW,CARJ;GACf,MAAM;GACN,MAAM;GACN,QAAQ;GACR,MAAM;GACN,MAAM;GACP,CAE8B;AAC/B,OAAK,SAAS,KAAK,UAAU;;AAI/B,MAAK,MAAM,KAAK,SAAS;EACvB,MAAM,SAAS,EAAE;AACjB,MAAI,CAAC,QAAQ,SAAU;EACvB,MAAM,IAAI,OAAO,SAAS,QAAQ,EAAE;AACpC,MAAI,KAAK,EAAG,QAAO,SAAS,OAAO,GAAG,EAAE;;AAG1C,QAAO;;AAGT,SAAS,cAAc,WAA+B,SAA8B;CAClF,MAAM,OAAO,aAAa;AAE1B,KAAI,CAAC,QAAQ,KAAM,QAAO;AAG1B,QAAO,GAAG,KAAK,oBADA,CAAC,GAAG,QAAQ,CAAC,KAAK,IAAI,CAAC,QAAQ,MAAM,OAAM,CAChB"}
@@ -1,4 +1,4 @@
1
- import { MaizzleConfig } from "../types/config.mjs";
1
+ import { MaizzleConfig } from "../types/config.js";
2
2
  import { ChildNode } from "domhandler";
3
3
 
4
4
  //#region src/transformers/tailwindcss.d.ts
@@ -21,4 +21,4 @@ import { ChildNode } from "domhandler";
21
21
  declare function tailwindcss(dom: ChildNode[], config: MaizzleConfig, filePath?: string): Promise<ChildNode[]>;
22
22
  //#endregion
23
23
  export { tailwindcss };
24
- //# sourceMappingURL=tailwindcss.d.mts.map
24
+ //# sourceMappingURL=tailwindcss.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tailwindcss.d.ts","names":[],"sources":["../../src/transformers/tailwindcss.ts"],"mappings":";;;;;;AA0EA;;;;;;;;;;;;;;iBAAsB,WAAA,CAAY,GAAA,EAAK,SAAA,IAAa,MAAA,EAAQ,aAAA,EAAe,QAAA,YAAoB,OAAA,CAAQ,SAAA"}