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

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 (221) hide show
  1. package/dist/build.d.ts.map +1 -1
  2. package/dist/build.js +10 -6
  3. package/dist/build.js.map +1 -1
  4. package/dist/components/Button.vue +17 -39
  5. package/dist/components/Container.vue +6 -6
  6. package/dist/components/Hr.vue +20 -120
  7. package/dist/components/Html.vue +19 -1
  8. package/dist/components/NotPlaintext.vue +14 -0
  9. package/dist/components/{Vml.vue → OutlookBg.vue} +1 -1
  10. package/dist/components/Plaintext.vue +14 -0
  11. package/dist/components/QrCode.vue +157 -0
  12. package/dist/components/Spacer.vue +28 -27
  13. package/dist/components/utils.js +1 -1
  14. package/dist/components/utils.js.map +1 -1
  15. package/dist/composables/defineConfig.js +1 -2
  16. package/dist/composables/defineConfig.js.map +1 -1
  17. package/dist/composables/renderContext.js +1 -1
  18. package/dist/composables/useBaseUrl.js +3 -4
  19. package/dist/composables/useBaseUrl.js.map +1 -1
  20. package/dist/composables/useConfig.js +1 -2
  21. package/dist/composables/useConfig.js.map +1 -1
  22. package/dist/composables/useDoctype.js +1 -2
  23. package/dist/composables/useDoctype.js.map +1 -1
  24. package/dist/composables/useEvent.js +1 -2
  25. package/dist/composables/useEvent.js.map +1 -1
  26. package/dist/composables/useFont.js +1 -2
  27. package/dist/composables/useFont.js.map +1 -1
  28. package/dist/composables/useOutlookFallback.js +1 -2
  29. package/dist/composables/useOutlookFallback.js.map +1 -1
  30. package/dist/composables/usePlaintext.d.ts +2 -0
  31. package/dist/composables/usePlaintext.d.ts.map +1 -1
  32. package/dist/composables/usePlaintext.js +2 -2
  33. package/dist/composables/usePlaintext.js.map +1 -1
  34. package/dist/composables/usePreheader.js +1 -2
  35. package/dist/composables/usePreheader.js.map +1 -1
  36. package/dist/composables/useTransformers.d.ts +3 -3
  37. package/dist/composables/useTransformers.js +4 -5
  38. package/dist/composables/useTransformers.js.map +1 -1
  39. package/dist/composables/useUrlQuery.js +3 -4
  40. package/dist/composables/useUrlQuery.js.map +1 -1
  41. package/dist/config/defaults.js +1 -1
  42. package/dist/config/index.js +1 -2
  43. package/dist/config/index.js.map +1 -1
  44. package/dist/events/index.js +1 -1
  45. package/dist/events/index.js.map +1 -1
  46. package/dist/index.d.ts +9 -9
  47. package/dist/index.js +4 -5
  48. package/dist/plaintext.js +3 -4
  49. package/dist/plaintext.js.map +1 -1
  50. package/dist/plugin.js +1 -2
  51. package/dist/plugin.js.map +1 -1
  52. package/dist/plugins/postcss/mergeMediaQueries.js +1 -2
  53. package/dist/plugins/postcss/mergeMediaQueries.js.map +1 -1
  54. package/dist/plugins/postcss/pruneVars.js +1 -1
  55. package/dist/plugins/postcss/pruneVars.js.map +1 -1
  56. package/dist/plugins/postcss/quoteFontFamilies.js +1 -1
  57. package/dist/plugins/postcss/quoteFontFamilies.js.map +1 -1
  58. package/dist/plugins/postcss/removeDeclarations.js +1 -1
  59. package/dist/plugins/postcss/removeDeclarations.js.map +1 -1
  60. package/dist/plugins/postcss/resolveMaizzleImports.js +1 -2
  61. package/dist/plugins/postcss/resolveMaizzleImports.js.map +1 -1
  62. package/dist/plugins/postcss/resolveProps.js +1 -1
  63. package/dist/plugins/postcss/resolveProps.js.map +1 -1
  64. package/dist/plugins/postcss/tailwindCleanup.js +1 -1
  65. package/dist/plugins/postcss/tailwindCleanup.js.map +1 -1
  66. package/dist/prepare.js +1 -2
  67. package/dist/prepare.js.map +1 -1
  68. package/dist/render/active.d.ts +8 -0
  69. package/dist/render/active.d.ts.map +1 -0
  70. package/dist/render/active.js +12 -0
  71. package/dist/render/active.js.map +1 -0
  72. package/dist/render/createRenderer.d.ts.map +1 -1
  73. package/dist/render/createRenderer.js +6 -9
  74. package/dist/render/createRenderer.js.map +1 -1
  75. package/dist/render/index.d.ts.map +1 -1
  76. package/dist/render/index.js +13 -6
  77. package/dist/render/index.js.map +1 -1
  78. package/dist/render/injectFonts.js +1 -2
  79. package/dist/render/injectFonts.js.map +1 -1
  80. package/dist/render/plugins/codeBlockExtract.js +1 -1
  81. package/dist/render/plugins/codeBlockExtract.js.map +1 -1
  82. package/dist/render/plugins/markdownExtract.js +1 -2
  83. package/dist/render/plugins/markdownExtract.js.map +1 -1
  84. package/dist/render/plugins/rawExtract.js +1 -1
  85. package/dist/render/plugins/rawExtract.js.map +1 -1
  86. package/dist/render/plugins/rowSourceLocation.js +1 -1
  87. package/dist/render/plugins/rowSourceLocation.js.map +1 -1
  88. package/dist/serve.d.ts +2 -0
  89. package/dist/serve.d.ts.map +1 -1
  90. package/dist/serve.js +12 -7
  91. package/dist/serve.js.map +1 -1
  92. package/dist/server/compatibility.js +1 -2
  93. package/dist/server/compatibility.js.map +1 -1
  94. package/dist/server/email.js +1 -2
  95. package/dist/server/email.js.map +1 -1
  96. package/dist/server/linter.js +1 -2
  97. package/dist/server/linter.js.map +1 -1
  98. package/dist/server/sfc-utils.js +1 -2
  99. package/dist/server/sfc-utils.js.map +1 -1
  100. package/dist/tests/render/_helpers.d.ts +6 -0
  101. package/dist/tests/render/_helpers.d.ts.map +1 -0
  102. package/dist/tests/render/_helpers.js +16 -0
  103. package/dist/tests/render/_helpers.js.map +1 -0
  104. package/dist/transformers/addAttributes.js +3 -4
  105. package/dist/transformers/addAttributes.js.map +1 -1
  106. package/dist/transformers/attributeToStyle.d.ts +27 -14
  107. package/dist/transformers/attributeToStyle.d.ts.map +1 -1
  108. package/dist/transformers/attributeToStyle.js +34 -20
  109. package/dist/transformers/attributeToStyle.js.map +1 -1
  110. package/dist/transformers/base.d.ts +66 -3
  111. package/dist/transformers/base.d.ts.map +1 -1
  112. package/dist/transformers/base.js +50 -24
  113. package/dist/transformers/base.js.map +1 -1
  114. package/dist/transformers/columnWidth.js +1 -2
  115. package/dist/transformers/columnWidth.js.map +1 -1
  116. package/dist/transformers/entities.d.ts +31 -2
  117. package/dist/transformers/entities.d.ts.map +1 -1
  118. package/dist/transformers/entities.js +39 -7
  119. package/dist/transformers/entities.js.map +1 -1
  120. package/dist/transformers/filters/defaults.js +1 -1
  121. package/dist/transformers/filters/defaults.js.map +1 -1
  122. package/dist/transformers/filters/index.d.ts +31 -10
  123. package/dist/transformers/filters/index.d.ts.map +1 -1
  124. package/dist/transformers/filters/index.js +36 -14
  125. package/dist/transformers/filters/index.js.map +1 -1
  126. package/dist/transformers/format.d.ts +14 -7
  127. package/dist/transformers/format.d.ts.map +1 -1
  128. package/dist/transformers/format.js +15 -11
  129. package/dist/transformers/format.js.map +1 -1
  130. package/dist/transformers/index.js +49 -29
  131. package/dist/transformers/index.js.map +1 -1
  132. package/dist/transformers/inlineCss.d.ts +84 -0
  133. package/dist/transformers/inlineCss.d.ts.map +1 -0
  134. package/dist/transformers/{inlineCSS.js → inlineCss.js} +24 -14
  135. package/dist/transformers/inlineCss.js.map +1 -0
  136. package/dist/transformers/inlineLink.d.ts +26 -5
  137. package/dist/transformers/inlineLink.d.ts.map +1 -1
  138. package/dist/transformers/inlineLink.js +31 -7
  139. package/dist/transformers/inlineLink.js.map +1 -1
  140. package/dist/transformers/minify.d.ts +13 -9
  141. package/dist/transformers/minify.d.ts.map +1 -1
  142. package/dist/transformers/minify.js +14 -13
  143. package/dist/transformers/minify.js.map +1 -1
  144. package/dist/transformers/msoPlaceholders.js +1 -2
  145. package/dist/transformers/msoPlaceholders.js.map +1 -1
  146. package/dist/transformers/purgeCss.d.ts +43 -0
  147. package/dist/transformers/purgeCss.d.ts.map +1 -0
  148. package/dist/transformers/{purgeCSS.js → purgeCss.js} +32 -25
  149. package/dist/transformers/purgeCss.js.map +1 -0
  150. package/dist/transformers/removeAttributes.d.ts +43 -20
  151. package/dist/transformers/removeAttributes.d.ts.map +1 -1
  152. package/dist/transformers/removeAttributes.js +34 -27
  153. package/dist/transformers/removeAttributes.js.map +1 -1
  154. package/dist/transformers/replaceStrings.js +1 -1
  155. package/dist/transformers/replaceStrings.js.map +1 -1
  156. package/dist/transformers/safeClassNames.js +1 -2
  157. package/dist/transformers/safeClassNames.js.map +1 -1
  158. package/dist/transformers/shorthandCss.d.ts +47 -0
  159. package/dist/transformers/shorthandCss.d.ts.map +1 -0
  160. package/dist/transformers/shorthandCss.js +61 -0
  161. package/dist/transformers/shorthandCss.js.map +1 -0
  162. package/dist/transformers/sixHex.d.ts +16 -7
  163. package/dist/transformers/sixHex.d.ts.map +1 -1
  164. package/dist/transformers/sixHex.js +21 -9
  165. package/dist/transformers/sixHex.js.map +1 -1
  166. package/dist/transformers/tailwindComponent.js +1 -2
  167. package/dist/transformers/tailwindComponent.js.map +1 -1
  168. package/dist/transformers/tailwindcss.js +1 -2
  169. package/dist/transformers/tailwindcss.js.map +1 -1
  170. package/dist/transformers/urlQuery.d.ts +26 -14
  171. package/dist/transformers/urlQuery.d.ts.map +1 -1
  172. package/dist/transformers/urlQuery.js +32 -20
  173. package/dist/transformers/urlQuery.js.map +1 -1
  174. package/dist/types/config.d.ts +71 -19
  175. package/dist/types/config.d.ts.map +1 -1
  176. package/dist/types/config.js +1 -1
  177. package/dist/types/index.d.ts +2 -2
  178. package/dist/types/index.js +1 -1
  179. package/dist/utils/ast/index.js +1 -2
  180. package/dist/utils/ast/parser.js +1 -2
  181. package/dist/utils/ast/parser.js.map +1 -1
  182. package/dist/utils/ast/serializer.js +1 -2
  183. package/dist/utils/ast/serializer.js.map +1 -1
  184. package/dist/utils/ast/walker.js +1 -1
  185. package/dist/utils/ast/walker.js.map +1 -1
  186. package/dist/utils/compileTailwindCss.js +1 -2
  187. package/dist/utils/compileTailwindCss.js.map +1 -1
  188. package/dist/utils/decodeStyleEntities.js +1 -1
  189. package/dist/utils/decodeStyleEntities.js.map +1 -1
  190. package/dist/utils/detect.js +1 -2
  191. package/dist/utils/detect.js.map +1 -1
  192. package/dist/utils/output-markers.d.ts +29 -0
  193. package/dist/utils/output-markers.d.ts.map +1 -0
  194. package/dist/utils/output-markers.js +68 -0
  195. package/dist/utils/output-markers.js.map +1 -0
  196. package/dist/utils/url.js +1 -2
  197. package/dist/utils/url.js.map +1 -1
  198. package/node_modules/maizzle/README.md +24 -0
  199. package/node_modules/maizzle/dist/commands/make/component.mjs +1 -1
  200. package/node_modules/maizzle/dist/commands/make/config.mjs +1 -1
  201. package/node_modules/maizzle/dist/commands/make/layout.mjs +3 -3
  202. package/node_modules/maizzle/dist/commands/make/scaffold.mjs +1 -1
  203. package/node_modules/maizzle/dist/commands/make/stubs/Layout.vue +146 -0
  204. package/node_modules/maizzle/dist/commands/make/stubs/component.vue +2 -4
  205. package/node_modules/maizzle/dist/commands/make/stubs/config.ts +1 -5
  206. package/node_modules/maizzle/dist/commands/make/template.mjs +1 -1
  207. package/node_modules/maizzle/dist/commands/new.mjs +29 -24
  208. package/node_modules/maizzle/dist/index.mjs +28 -8
  209. package/node_modules/maizzle/package.json +1 -1
  210. package/package.json +2 -2
  211. package/dist/transformers/inlineCSS.d.ts +0 -17
  212. package/dist/transformers/inlineCSS.d.ts.map +0 -1
  213. package/dist/transformers/inlineCSS.js.map +0 -1
  214. package/dist/transformers/purgeCSS.d.ts +0 -23
  215. package/dist/transformers/purgeCSS.d.ts.map +0 -1
  216. package/dist/transformers/purgeCSS.js.map +0 -1
  217. package/dist/transformers/shorthandCSS.d.ts +0 -24
  218. package/dist/transformers/shorthandCSS.d.ts.map +0 -1
  219. package/dist/transformers/shorthandCSS.js +0 -48
  220. package/dist/transformers/shorthandCSS.js.map +0 -1
  221. package/node_modules/maizzle/dist/commands/make/stubs/layout.vue +0 -39
@@ -1,17 +0,0 @@
1
- import { CssConfig } from "../types/config.js";
2
- import { ChildNode } from "domhandler";
3
-
4
- //#region src/transformers/inlineCSS.d.ts
5
- /**
6
- * Inline CSS transformer.
7
- *
8
- * Inlines CSS from `<style>` tags into inline style attributes on HTML elements.
9
- * This is important for email client compatibility (especially Outlook on Windows).
10
- *
11
- * Enabled when `css.inline` is set to `true` or an object with options.
12
- * All Juice options are supported and passed through directly.
13
- */
14
- declare function inlineCSS(dom: ChildNode[], config?: CssConfig): ChildNode[];
15
- //#endregion
16
- export { inlineCSS };
17
- //# sourceMappingURL=inlineCSS.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"inlineCSS.d.ts","names":[],"sources":["../../src/transformers/inlineCSS.ts"],"mappings":";;;;;;AAeA;;;;;;;iBAAgB,SAAA,CAAU,GAAA,EAAK,SAAA,IAAa,MAAA,GAAQ,SAAA,GAAiB,SAAA"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"inlineCSS.js","names":[],"sources":["../../src/transformers/inlineCSS.ts"],"sourcesContent":["import juice from 'juice'\nimport { walk, parse, serialize } from '../utils/ast/index.ts'\nimport type { ChildNode, Element } from 'domhandler'\nimport type { Options as JuiceOptions } from 'juice'\nimport type { CssConfig } from '../types/config.ts'\n\n/**\n * Inline CSS transformer.\n *\n * Inlines CSS from `<style>` tags into inline style attributes on HTML elements.\n * This is important for email client compatibility (especially Outlook on Windows).\n *\n * Enabled when `css.inline` is set to `true` or an object with options.\n * All Juice options are supported and passed through directly.\n */\nexport function inlineCSS(dom: ChildNode[], config: CssConfig = {}): ChildNode[] {\n const inline = config.inline\n\n // Disabled when inline is falsy or not an object/truthy\n if (!inline) {\n return dom\n }\n\n // Build options from config\n const options = typeof inline === 'object' ? inline : {}\n\n // Separate Maizzle-specific options from Juice options\n const {\n preferUnitlessValues = true,\n safelist,\n customCSS = '',\n styleToAttribute,\n excludedProperties,\n widthElements,\n heightElements,\n codeBlocks,\n ...juicePassthrough\n } = options\n\n // Configure Juice static properties\n juice.styleToAttribute = styleToAttribute ?? {}\n juice.excludedProperties = ['--tw-shadow', ...(excludedProperties ?? [])]\n juice.widthElements = (widthElements ?? ['img', 'video']).map(i => i.toUpperCase()) as unknown as HTMLElement[]\n juice.heightElements = (heightElements ?? ['img', 'video']).map(i => i.toUpperCase()) as unknown as HTMLElement[]\n\n // Add custom code blocks\n if (codeBlocks && typeof codeBlocks === 'object') {\n Object.entries(codeBlocks).forEach(([key, value]) => {\n if (value.start && value.end) {\n juice.codeBlocks[key] = value\n }\n })\n }\n\n // Handle style tags with embed attributes.\n // We add a marker attribute that persists through the pipeline,\n // then restore data-embed from it after Juice runs.\n walk(dom, (node) => {\n const el = node as Element\n if (el.name === 'style' && el.attribs) {\n // Sync data-embed ↔ embed. Use `in` so presence-only attrs\n // (<style embed> → attribs.embed === '') still count.\n if ('embed' in el.attribs && !('data-embed' in el.attribs)) {\n el.attribs['data-embed'] = ''\n }\n if ('data-embed' in el.attribs && !('embed' in el.attribs)) {\n el.attribs.embed = ''\n }\n\n // Add marker that persists through the pipeline\n if ('data-embed' in el.attribs) {\n el.attribs['data-maizzle-embed'] = ''\n }\n }\n })\n\n // Serialize for juice (juice requires a string)\n const serialized = serialize(dom)\n\n let inlinedHtml: string\n\n try {\n const juiceOptions: JuiceOptions = {\n removeStyleTags: juicePassthrough.removeStyleTags ?? false,\n removeInlinedSelectors: juicePassthrough.removeInlinedSelectors ?? true,\n applyWidthAttributes: juicePassthrough.applyWidthAttributes ?? true,\n applyHeightAttributes: juicePassthrough.applyHeightAttributes ?? true,\n preservedSelectors: safelist ?? [],\n ...customCSS ? { extraCss: customCSS } : {},\n inlineDuplicateProperties: juicePassthrough.inlineDuplicateProperties ?? true,\n ...juicePassthrough,\n }\n\n inlinedHtml = juice(serialized, juiceOptions)\n } catch {\n // If Juice fails, return the dom unchanged\n return dom\n }\n\n // Post-process for preferUnitlessValues\n const result = parse(inlinedHtml)\n\n walk(result, (node) => {\n const el = node as Element\n if (el.attribs?.style) {\n // Normalize style formatting: ensure spaces after : and ;\n let style = el.attribs.style\n .replace(/:\\s*/g, ': ')\n .replace(/;\\s*/g, '; ')\n .trimEnd()\n\n // Ensure trailing semicolon\n if (!style.endsWith(';')) {\n style += ';'\n }\n\n if (preferUnitlessValues) {\n style = style.replace(\n /\\b0(px|rem|em|%|vh|vw|vmin|vmax|in|cm|mm|pt|pc|ex|ch)\\b/g,\n '0'\n )\n }\n\n el.attribs.style = style\n }\n })\n\n /**\n * Restore `embed` from our marker so the purge step can detect\n * these tags and skip them. Drop `data-embed` (juice's name)\n * since it's redundant once `embed` is back, and let purge\n * strip `embed` itself at the end of its run.\n */\n walk(result, (node) => {\n const el = node as Element\n if (el.name === 'style' && el.attribs && 'data-maizzle-embed' in el.attribs) {\n el.attribs.embed = ''\n delete el.attribs['data-embed']\n delete el.attribs['data-maizzle-embed']\n }\n })\n\n return result\n}\n"],"mappings":";;;;;;;;;;;;;;;;AAeA,SAAgB,UAAU,KAAkB,SAAoB,EAAE,EAAe;CAC/E,MAAM,SAAS,OAAO;AAGtB,KAAI,CAAC,OACH,QAAO;CAOT,MAAM,EACJ,uBAAuB,MACvB,UACA,YAAY,IACZ,kBACA,oBACA,eACA,gBACA,YACA,GAAG,qBAZW,OAAO,WAAW,WAAW,SAAS,EAAE;AAgBxD,OAAM,mBAAmB,oBAAoB,EAAE;AAC/C,OAAM,qBAAqB,CAAC,eAAe,GAAI,sBAAsB,EAAE,CAAE;AACzE,OAAM,iBAAiB,iBAAiB,CAAC,OAAO,QAAQ,EAAE,KAAI,MAAK,EAAE,aAAa,CAAC;AACnF,OAAM,kBAAkB,kBAAkB,CAAC,OAAO,QAAQ,EAAE,KAAI,MAAK,EAAE,aAAa,CAAC;AAGrF,KAAI,cAAc,OAAO,eAAe,SACtC,QAAO,QAAQ,WAAW,CAAC,SAAS,CAAC,KAAK,WAAW;AACnD,MAAI,MAAM,SAAS,MAAM,IACvB,OAAM,WAAW,OAAO;GAE1B;AAMJ,MAAK,MAAM,SAAS;EAClB,MAAM,KAAK;AACX,MAAI,GAAG,SAAS,WAAW,GAAG,SAAS;AAGrC,OAAI,WAAW,GAAG,WAAW,EAAE,gBAAgB,GAAG,SAChD,IAAG,QAAQ,gBAAgB;AAE7B,OAAI,gBAAgB,GAAG,WAAW,EAAE,WAAW,GAAG,SAChD,IAAG,QAAQ,QAAQ;AAIrB,OAAI,gBAAgB,GAAG,QACrB,IAAG,QAAQ,wBAAwB;;GAGvC;CAGF,MAAM,aAAa,UAAU,IAAI;CAEjC,IAAI;AAEJ,KAAI;AAYF,gBAAc,MAAM,YAXe;GACjC,iBAAiB,iBAAiB,mBAAmB;GACrD,wBAAwB,iBAAiB,0BAA0B;GACnE,sBAAsB,iBAAiB,wBAAwB;GAC/D,uBAAuB,iBAAiB,yBAAyB;GACjE,oBAAoB,YAAY,EAAE;GAClC,GAAG,YAAY,EAAE,UAAU,WAAW,GAAG,EAAE;GAC3C,2BAA2B,iBAAiB,6BAA6B;GACzE,GAAG;GACJ,CAE4C;SACvC;AAEN,SAAO;;CAIT,MAAM,SAAS,MAAM,YAAY;AAEjC,MAAK,SAAS,SAAS;EACrB,MAAM,KAAK;AACX,MAAI,GAAG,SAAS,OAAO;GAErB,IAAI,QAAQ,GAAG,QAAQ,MACpB,QAAQ,SAAS,KAAK,CACtB,QAAQ,SAAS,KAAK,CACtB,SAAS;AAGZ,OAAI,CAAC,MAAM,SAAS,IAAI,CACtB,UAAS;AAGX,OAAI,qBACF,SAAQ,MAAM,QACZ,4DACA,IACD;AAGH,MAAG,QAAQ,QAAQ;;GAErB;;;;;;;AAQF,MAAK,SAAS,SAAS;EACrB,MAAM,KAAK;AACX,MAAI,GAAG,SAAS,WAAW,GAAG,WAAW,wBAAwB,GAAG,SAAS;AAC3E,MAAG,QAAQ,QAAQ;AACnB,UAAO,GAAG,QAAQ;AAClB,UAAO,GAAG,QAAQ;;GAEpB;AAEF,QAAO"}
@@ -1,23 +0,0 @@
1
- import { CssConfig } from "../types/config.js";
2
- import { ChildNode } from "domhandler";
3
-
4
- //#region src/transformers/purgeCSS.d.ts
5
- /**
6
- * Remove unused CSS transformer.
7
- *
8
- * Uses `email-comb` to strip CSS selectors and corresponding class/id
9
- * references that are not matched anywhere in the HTML body.
10
- *
11
- * Enable by setting `css.purge: true` (or passing options).
12
- * The user-supplied options are merged on top of the defaults, so
13
- * `safelist` values are **appended** to the built-in safelist rather
14
- * than replacing it.
15
- *
16
- * Accepts `ChildNode[]` as input, serializes internally before passing
17
- * to email-comb (which requires a raw HTML string), then parses the
18
- * result back to `ChildNode[]` so it fits in the DOM pipeline.
19
- */
20
- declare function purgeCSS(dom: ChildNode[], config?: CssConfig): ChildNode[];
21
- //#endregion
22
- export { purgeCSS };
23
- //# sourceMappingURL=purgeCSS.d.ts.map
@@ -1 +0,0 @@
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 +0,0 @@
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":"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"}
@@ -1,24 +0,0 @@
1
- import { CssConfig } from "../types/config.js";
2
- import { ChildNode } from "domhandler";
3
-
4
- //#region src/transformers/shorthandCSS.d.ts
5
- /**
6
- * Shorthand CSS transformer.
7
- *
8
- * Rewrites longhand CSS inside `style` attributes with shorthand syntax.
9
- * Works with margin, padding, and border when all sides are specified.
10
- *
11
- * For example:
12
- * `margin-left: 2px; margin-right: 2px; margin-top: 4px; margin-bottom: 4px`
13
- * becomes:
14
- * `margin: 4px 2px`
15
- *
16
- * Enabled via `css.shorthand`:
17
- * - `true`: enable for all tags
18
- * - `{ tags: ['td', 'div'] }`: enable only for specified tags
19
- * - `false` or omitted: disabled
20
- */
21
- declare function shorthandCSS(dom: ChildNode[], config?: CssConfig): ChildNode[];
22
- //#endregion
23
- export { shorthandCSS };
24
- //# sourceMappingURL=shorthandCSS.d.ts.map
@@ -1 +0,0 @@
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,48 +0,0 @@
1
- import { walk } from "../utils/ast/walker.js";
2
- import "../utils/ast/index.js";
3
- import postcss from "postcss";
4
- import safeParser from "postcss-safe-parser";
5
- import mergeLonghand from "postcss-merge-longhand";
6
-
7
- //#region src/transformers/shorthandCSS.ts
8
- /**
9
- * Shorthand CSS transformer.
10
- *
11
- * Rewrites longhand CSS inside `style` attributes with shorthand syntax.
12
- * Works with margin, padding, and border when all sides are specified.
13
- *
14
- * For example:
15
- * `margin-left: 2px; margin-right: 2px; margin-top: 4px; margin-bottom: 4px`
16
- * becomes:
17
- * `margin: 4px 2px`
18
- *
19
- * Enabled via `css.shorthand`:
20
- * - `true`: enable for all tags
21
- * - `{ tags: ['td', 'div'] }`: enable only for specified tags
22
- * - `false` or omitted: disabled
23
- */
24
- function shorthandCSS(dom, config = {}) {
25
- const option = config.shorthand;
26
- if (!option) return dom;
27
- const allowedTags = (typeof option === "object" ? option : {}).tags ?? [];
28
- const hasTagFilter = allowedTags.length > 0;
29
- walk(dom, (node) => {
30
- const el = node;
31
- if (!el.attribs?.style) return;
32
- if (hasTagFilter && !allowedTags.includes(el.name)) return;
33
- const styleValue = el.attribs.style;
34
- try {
35
- const { css } = postcss().use(mergeLonghand).process(`div { ${styleValue} }`, { parser: safeParser });
36
- const match = css.match(/div\s*\{\s*([^}]+)\s*\}/);
37
- if (match && match[1]) {
38
- const newStyle = match[1].trim();
39
- if (newStyle !== styleValue) el.attribs.style = newStyle;
40
- }
41
- } catch {}
42
- });
43
- return dom;
44
- }
45
-
46
- //#endregion
47
- export { shorthandCSS };
48
- //# sourceMappingURL=shorthandCSS.js.map
@@ -1 +0,0 @@
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,39 +0,0 @@
1
- <script setup lang="ts">
2
- defineProps({
3
- bodyClass: {
4
- type: String,
5
- default: ''
6
- }
7
- })
8
- </script>
9
-
10
- <template>
11
- <html lang="en" dir="ltr" xmlns:v="urn:schemas-microsoft-com:vml" xmlns:o="urn:schemas-microsoft-com:office:office">
12
- <head>
13
- <meta charset="utf-8">
14
- <meta name="x-apple-disable-message-reformatting">
15
- <meta name="viewport" content="width=device-width, initial-scale=1">
16
- <meta name="format-detection" content="telephone=no, date=no, address=no, email=no, url=no">
17
- <meta name="color-scheme" content="light dark">
18
- <meta name="supported-color-schemes" content="light dark">
19
- <!--[if mso]>
20
- <noscript>
21
- <xml>
22
- <o:OfficeDocumentSettings xmlns:o="urn:schemas-microsoft-com:office:office">
23
- <o:PixelsPerInch>96</o:PixelsPerInch>
24
- </o:OfficeDocumentSettings>
25
- </xml>
26
- </noscript>
27
- <style>
28
- td,th,div,p,a,h1,h2,h3,h4,h5,h6 {font-family: "Segoe UI", sans-serif; mso-line-height-rule: exactly;}
29
- </style>
30
- <![endif]-->
31
- <style>
32
- @import "@maizzle/tailwindcss";
33
- </style>
34
- </head>
35
- <body :class="['m-0 p-0 w-full [word-break:break-word] [-webkit-font-smoothing:antialiased]', bodyClass]">
36
- <slot />
37
- </body>
38
- </html>
39
- </template>