react-email 4.0.16 → 4.1.0-canary.10

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 (470) hide show
  1. package/CHANGELOG.md +65 -0
  2. package/dev/index.js +40 -0
  3. package/dev/package.json +13 -0
  4. package/dist/index.js +1 -1
  5. package/package.json +13 -46
  6. package/readme.md +16 -0
  7. package/src/commands/build.ts +306 -0
  8. package/src/commands/dev.ts +27 -0
  9. package/src/commands/export.ts +204 -0
  10. package/src/commands/start.ts +38 -0
  11. package/src/index.ts +55 -0
  12. package/src/utils/__snapshots__/tree.spec.ts.snap +27 -0
  13. package/src/utils/esbuild/renderring-utilities-exporter.ts +1 -1
  14. package/src/utils/get-emails-directory-metadata.spec.ts +1 -1
  15. package/src/utils/get-preview-server-location.ts +51 -0
  16. package/src/utils/index.ts +2 -6
  17. package/src/utils/packageJson.ts +4 -0
  18. package/src/utils/preview/get-env-variables-for-preview-app.ts +14 -0
  19. package/src/utils/preview/hot-reloading/create-dependency-graph.spec.ts +284 -0
  20. package/src/utils/preview/hot-reloading/create-dependency-graph.ts +321 -0
  21. package/src/utils/preview/hot-reloading/get-imported-modules.spec.ts +151 -0
  22. package/src/utils/preview/hot-reloading/get-imported-modules.ts +49 -0
  23. package/src/utils/preview/hot-reloading/resolve-path-aliases.spec.ts +11 -0
  24. package/src/utils/preview/hot-reloading/resolve-path-aliases.ts +32 -0
  25. package/src/utils/preview/hot-reloading/setup-hot-reloading.ts +121 -0
  26. package/src/utils/preview/hot-reloading/test/tsconfig.json +8 -0
  27. package/src/utils/preview/index.ts +2 -0
  28. package/src/utils/preview/serve-static-file.ts +51 -0
  29. package/src/utils/preview/start-dev-server.ts +234 -0
  30. package/src/utils/tree.spec.ts +5 -0
  31. package/src/utils/tree.ts +76 -0
  32. package/src/utils/types/hot-reload-change.ts +1 -1
  33. package/src/utils/types/hot-reload-event.ts +1 -1
  34. package/tsconfig.json +4 -10
  35. package/dist/cli/index.d.mts +0 -1
  36. package/dist/cli/index.d.ts +0 -1
  37. package/dist/cli/index.js +0 -2785
  38. package/dist/cli/index.mjs +0 -1361
  39. package/dist/index.d.mts +0 -20
  40. package/dist/index.d.ts +0 -20
  41. package/dist/index.mjs +0 -21
  42. package/dist/package/index.d.mts +0 -33
  43. package/dist/package/index.d.ts +0 -33
  44. package/dist/package/index.js +0 -62
  45. package/dist/package/index.mjs +0 -7
  46. package/dist/preview/.next/BUILD_ID +0 -1
  47. package/dist/preview/.next/app-build-manifest.json +0 -44
  48. package/dist/preview/.next/app-path-routes-manifest.json +0 -6
  49. package/dist/preview/.next/build-manifest.json +0 -33
  50. package/dist/preview/.next/diagnostics/build-diagnostics.json +0 -6
  51. package/dist/preview/.next/diagnostics/framework.json +0 -1
  52. package/dist/preview/.next/export-marker.json +0 -6
  53. package/dist/preview/.next/images-manifest.json +0 -57
  54. package/dist/preview/.next/next-minimal-server.js.nft.json +0 -1
  55. package/dist/preview/.next/next-server.js.nft.json +0 -1
  56. package/dist/preview/.next/package.json +0 -1
  57. package/dist/preview/.next/prerender-manifest.json +0 -41
  58. package/dist/preview/.next/react-loadable-manifest.json +0 -1
  59. package/dist/preview/.next/required-server-files.json +0 -311
  60. package/dist/preview/.next/routes-manifest.json +0 -64
  61. package/dist/preview/.next/server/app/_not-found/page.js +0 -1
  62. package/dist/preview/.next/server/app/_not-found/page.js.nft.json +0 -1
  63. package/dist/preview/.next/server/app/_not-found/page_client-reference-manifest.js +0 -1
  64. package/dist/preview/.next/server/app/favicon.ico/route.js +0 -1
  65. package/dist/preview/.next/server/app/favicon.ico/route.js.nft.json +0 -1
  66. package/dist/preview/.next/server/app/favicon.ico.body +0 -0
  67. package/dist/preview/.next/server/app/favicon.ico.meta +0 -1
  68. package/dist/preview/.next/server/app/page.js +0 -1
  69. package/dist/preview/.next/server/app/page.js.nft.json +0 -1
  70. package/dist/preview/.next/server/app/page_client-reference-manifest.js +0 -1
  71. package/dist/preview/.next/server/app/preview/[...slug]/page.js +0 -321
  72. package/dist/preview/.next/server/app/preview/[...slug]/page.js.nft.json +0 -1
  73. package/dist/preview/.next/server/app/preview/[...slug]/page_client-reference-manifest.js +0 -1
  74. package/dist/preview/.next/server/app-paths-manifest.json +0 -6
  75. package/dist/preview/.next/server/chunks/134.js +0 -6
  76. package/dist/preview/.next/server/chunks/235.js +0 -15
  77. package/dist/preview/.next/server/chunks/275.js +0 -1
  78. package/dist/preview/.next/server/chunks/343.js +0 -20
  79. package/dist/preview/.next/server/chunks/428.js +0 -14
  80. package/dist/preview/.next/server/chunks/963.js +0 -1
  81. package/dist/preview/.next/server/functions-config-manifest.json +0 -4
  82. package/dist/preview/.next/server/interception-route-rewrite-manifest.js +0 -1
  83. package/dist/preview/.next/server/middleware-build-manifest.js +0 -1
  84. package/dist/preview/.next/server/middleware-manifest.json +0 -6
  85. package/dist/preview/.next/server/middleware-react-loadable-manifest.js +0 -1
  86. package/dist/preview/.next/server/next-font-manifest.js +0 -1
  87. package/dist/preview/.next/server/next-font-manifest.json +0 -1
  88. package/dist/preview/.next/server/pages/500.html +0 -1
  89. package/dist/preview/.next/server/pages/_app.js +0 -1
  90. package/dist/preview/.next/server/pages/_app.js.nft.json +0 -1
  91. package/dist/preview/.next/server/pages/_document.js +0 -1
  92. package/dist/preview/.next/server/pages/_document.js.nft.json +0 -1
  93. package/dist/preview/.next/server/pages/_error.js +0 -1
  94. package/dist/preview/.next/server/pages/_error.js.nft.json +0 -1
  95. package/dist/preview/.next/server/pages-manifest.json +0 -5
  96. package/dist/preview/.next/server/server-reference-manifest.js +0 -1
  97. package/dist/preview/.next/server/server-reference-manifest.json +0 -1
  98. package/dist/preview/.next/server/webpack-runtime.js +0 -1
  99. package/dist/preview/.next/static/chunks/107-3043079e7cb8bcae.js +0 -1
  100. package/dist/preview/.next/static/chunks/293-297b1eb2241f9a70.js +0 -1
  101. package/dist/preview/.next/static/chunks/3bd82e28-cda2c00a924937c5.js +0 -1
  102. package/dist/preview/.next/static/chunks/45-1021fac82f766268.js +0 -1
  103. package/dist/preview/.next/static/chunks/484-e38a627386aae911.js +0 -1
  104. package/dist/preview/.next/static/chunks/589-817d8691661d370e.js +0 -1
  105. package/dist/preview/.next/static/chunks/902-c34acb56733e0ce1.js +0 -1
  106. package/dist/preview/.next/static/chunks/app/_not-found/page-4cbc7dce3ad33336.js +0 -1
  107. package/dist/preview/.next/static/chunks/app/layout-269b5cbd8f4cd2e3.js +0 -1
  108. package/dist/preview/.next/static/chunks/app/page-2dbfb5b2dc4b1191.js +0 -1
  109. package/dist/preview/.next/static/chunks/app/preview/[...slug]/page-79e0c9b528a62f8b.js +0 -1
  110. package/dist/preview/.next/static/chunks/f33a14d2-ec7c5f0b91818561.js +0 -6
  111. package/dist/preview/.next/static/chunks/framework-b887e9fc751a9906.js +0 -1
  112. package/dist/preview/.next/static/chunks/main-9a03e7ba8acb1900.js +0 -1
  113. package/dist/preview/.next/static/chunks/main-app-976577a424e11c75.js +0 -1
  114. package/dist/preview/.next/static/chunks/pages/_app-542a93a5a214e1c0.js +0 -1
  115. package/dist/preview/.next/static/chunks/pages/_error-d5fe1b1612642f76.js +0 -1
  116. package/dist/preview/.next/static/chunks/polyfills-42372ed130431b0a.js +0 -1
  117. package/dist/preview/.next/static/chunks/webpack-31c45daa2bd82a7b.js +0 -1
  118. package/dist/preview/.next/static/css/ac3decd5d6736fbe.css +0 -3
  119. package/dist/preview/.next/static/media/05613964ce6c782e-s.p.otf +0 -0
  120. package/dist/preview/.next/static/media/11c6126b9369e85e-s.p.otf +0 -0
  121. package/dist/preview/.next/static/media/26a46d62cd723877-s.woff2 +0 -0
  122. package/dist/preview/.next/static/media/26cb97734d8cb717-s.p.otf +0 -0
  123. package/dist/preview/.next/static/media/55c55f0601d81cf3-s.woff2 +0 -0
  124. package/dist/preview/.next/static/media/581909926a08bbc8-s.woff2 +0 -0
  125. package/dist/preview/.next/static/media/8e9860b6e62d6359-s.woff2 +0 -0
  126. package/dist/preview/.next/static/media/97e0cb1ae144a2a9-s.woff2 +0 -0
  127. package/dist/preview/.next/static/media/bb6462617151f6b7-s.p.otf +0 -0
  128. package/dist/preview/.next/static/media/cf6daef822ab0142-s.p.otf +0 -0
  129. package/dist/preview/.next/static/media/df0a9ae256c0569c-s.woff2 +0 -0
  130. package/dist/preview/.next/static/media/e4051546b3043204-s.p.otf +0 -0
  131. package/dist/preview/.next/static/media/e4af272ccee01ff0-s.p.woff2 +0 -0
  132. package/dist/preview/.next/static/media/logo.2ce2a759.png +0 -0
  133. package/dist/preview/.next/static/yxjkw7Y7HRLmRWkl3S43z/_buildManifest.js +0 -1
  134. package/dist/preview/.next/static/yxjkw7Y7HRLmRWkl3S43z/_ssgManifest.js +0 -1
  135. package/dist/preview/.next/trace +0 -27
  136. package/dist/preview/.next/types/app/layout.ts +0 -84
  137. package/dist/preview/.next/types/app/page.ts +0 -84
  138. package/dist/preview/.next/types/app/preview/[...slug]/page.ts +0 -84
  139. package/dist/preview/.next/types/cache-life.d.ts +0 -141
  140. package/dist/preview/.next/types/package.json +0 -1
  141. package/module-punycode.d.ts +0 -3
  142. package/next-env.d.ts +0 -5
  143. package/next.config.js +0 -22
  144. package/postcss.config.js +0 -8
  145. package/scripts/build-preview-server.mjs +0 -33
  146. package/scripts/fill-caniemail-data.mjs +0 -36
  147. package/src/actions/email-validation/caniemail-data.ts +0 -85993
  148. package/src/actions/email-validation/check-compatibility.ts +0 -333
  149. package/src/actions/email-validation/check-images.spec.tsx +0 -100
  150. package/src/actions/email-validation/check-images.ts +0 -160
  151. package/src/actions/email-validation/check-links.spec.tsx +0 -113
  152. package/src/actions/email-validation/check-links.ts +0 -113
  153. package/src/actions/email-validation/get-code-location-from-ast-element.ts +0 -18
  154. package/src/actions/email-validation/quick-fetch.ts +0 -14
  155. package/src/actions/get-email-path-from-slug.ts +0 -32
  156. package/src/actions/get-emails-directory-metadata-action.ts +0 -19
  157. package/src/actions/render-email-by-path.tsx +0 -121
  158. package/src/animated-icons-data/help.json +0 -1082
  159. package/src/animated-icons-data/link.json +0 -1309
  160. package/src/animated-icons-data/load.json +0 -443
  161. package/src/animated-icons-data/mail.json +0 -1320
  162. package/src/app/env.ts +0 -15
  163. package/src/app/favicon.ico +0 -0
  164. package/src/app/fonts/SFMono/SFMonoBold.otf +0 -0
  165. package/src/app/fonts/SFMono/SFMonoBoldItalic.otf +0 -0
  166. package/src/app/fonts/SFMono/SFMonoHeavy.otf +0 -0
  167. package/src/app/fonts/SFMono/SFMonoHeavyItalic.otf +0 -0
  168. package/src/app/fonts/SFMono/SFMonoLight.otf +0 -0
  169. package/src/app/fonts/SFMono/SFMonoLightItalic.otf +0 -0
  170. package/src/app/fonts/SFMono/SFMonoMedium.otf +0 -0
  171. package/src/app/fonts/SFMono/SFMonoMediumItalic.otf +0 -0
  172. package/src/app/fonts/SFMono/SFMonoRegular.otf +0 -0
  173. package/src/app/fonts/SFMono/SFMonoRegularItalic.otf +0 -0
  174. package/src/app/fonts/SFMono/SFMonoSemibold.otf +0 -0
  175. package/src/app/fonts/SFMono/SFMonoSemiboldItalic.otf +0 -0
  176. package/src/app/fonts.ts +0 -39
  177. package/src/app/globals.css +0 -15
  178. package/src/app/layout.tsx +0 -43
  179. package/src/app/logo.png +0 -0
  180. package/src/app/page.tsx +0 -46
  181. package/src/app/preview/[...slug]/page.tsx +0 -157
  182. package/src/app/preview/[...slug]/preview.tsx +0 -216
  183. package/src/app/preview/[...slug]/rendering-error.tsx +0 -40
  184. package/src/commands/testing/out/magic-links/aws-verify-email.html +0 -165
  185. package/src/commands/testing/out/magic-links/linear-login-code.html +0 -90
  186. package/src/commands/testing/out/magic-links/notion-magic-link.html +0 -76
  187. package/src/commands/testing/out/magic-links/plaid-verify-identity.html +0 -78
  188. package/src/commands/testing/out/magic-links/raycast-magic-link.html +0 -91
  189. package/src/commands/testing/out/magic-links/slack-confirm.html +0 -240
  190. package/src/commands/testing/out/newsletters/codepen-challengers.html +0 -548
  191. package/src/commands/testing/out/newsletters/google-play-policy-update.html +0 -339
  192. package/src/commands/testing/out/newsletters/stack-overflow-tips.html +0 -231
  193. package/src/commands/testing/out/notifications/github-access-token.html +0 -104
  194. package/src/commands/testing/out/notifications/papermark-year-in-review.html +0 -317
  195. package/src/commands/testing/out/notifications/vercel-invite-user.html +0 -171
  196. package/src/commands/testing/out/notifications/yelp-recent-login.html +0 -195
  197. package/src/commands/testing/out/receipts/apple-receipt.html +0 -677
  198. package/src/commands/testing/out/receipts/nike-receipt.html +0 -724
  199. package/src/commands/testing/out/reset-password/dropbox-reset-password.html +0 -98
  200. package/src/commands/testing/out/reset-password/twitch-reset-password.html +0 -220
  201. package/src/commands/testing/out/reviews/airbnb-review.html +0 -206
  202. package/src/commands/testing/out/reviews/amazon-review.html +0 -356
  203. package/src/commands/testing/out/static/airbnb-logo.png +0 -0
  204. package/src/commands/testing/out/static/airbnb-review-user.jpg +0 -0
  205. package/src/commands/testing/out/static/amazon-book.jpg +0 -0
  206. package/src/commands/testing/out/static/amazon-facebook.jpg +0 -0
  207. package/src/commands/testing/out/static/amazon-instagram.jpg +0 -0
  208. package/src/commands/testing/out/static/amazon-logo.png +0 -0
  209. package/src/commands/testing/out/static/amazon-prime-logo.png +0 -0
  210. package/src/commands/testing/out/static/amazon-rating.gif +0 -0
  211. package/src/commands/testing/out/static/amazon-twitter.jpg +0 -0
  212. package/src/commands/testing/out/static/apple-card-icon.png +0 -0
  213. package/src/commands/testing/out/static/apple-hbo-max-icon.jpeg +0 -0
  214. package/src/commands/testing/out/static/apple-logo.png +0 -0
  215. package/src/commands/testing/out/static/apple-wallet.png +0 -0
  216. package/src/commands/testing/out/static/aws-logo.png +0 -0
  217. package/src/commands/testing/out/static/codepen-challengers.png +0 -0
  218. package/src/commands/testing/out/static/codepen-cube.png +0 -0
  219. package/src/commands/testing/out/static/codepen-pro.png +0 -0
  220. package/src/commands/testing/out/static/dropbox-logo.png +0 -0
  221. package/src/commands/testing/out/static/github.png +0 -0
  222. package/src/commands/testing/out/static/google-play-academy.png +0 -0
  223. package/src/commands/testing/out/static/google-play-chat.png +0 -0
  224. package/src/commands/testing/out/static/google-play-footer.png +0 -0
  225. package/src/commands/testing/out/static/google-play-header.png +0 -0
  226. package/src/commands/testing/out/static/google-play-icon.png +0 -0
  227. package/src/commands/testing/out/static/google-play-logo.png +0 -0
  228. package/src/commands/testing/out/static/google-play-pl.png +0 -0
  229. package/src/commands/testing/out/static/google-play.png +0 -0
  230. package/src/commands/testing/out/static/koala-logo.png +0 -0
  231. package/src/commands/testing/out/static/linear-logo.png +0 -0
  232. package/src/commands/testing/out/static/netlify-logo.png +0 -0
  233. package/src/commands/testing/out/static/nike-logo.png +0 -0
  234. package/src/commands/testing/out/static/nike-phone.png +0 -0
  235. package/src/commands/testing/out/static/nike-product.png +0 -0
  236. package/src/commands/testing/out/static/nike-recomendation-1.png +0 -0
  237. package/src/commands/testing/out/static/nike-recomendation-2.png +0 -0
  238. package/src/commands/testing/out/static/nike-recomendation-3.png +0 -0
  239. package/src/commands/testing/out/static/nike-recomendation-4.png +0 -0
  240. package/src/commands/testing/out/static/notion-logo.png +0 -0
  241. package/src/commands/testing/out/static/plaid-logo.png +0 -0
  242. package/src/commands/testing/out/static/raycast-bg.png +0 -0
  243. package/src/commands/testing/out/static/raycast-logo.png +0 -0
  244. package/src/commands/testing/out/static/slack-facebook.png +0 -0
  245. package/src/commands/testing/out/static/slack-linkedin.png +0 -0
  246. package/src/commands/testing/out/static/slack-logo.png +0 -0
  247. package/src/commands/testing/out/static/slack-twitter.png +0 -0
  248. package/src/commands/testing/out/static/stack-overflow-header.png +0 -0
  249. package/src/commands/testing/out/static/stack-overflow-logo-sm.png +0 -0
  250. package/src/commands/testing/out/static/stack-overflow-logo.png +0 -0
  251. package/src/commands/testing/out/static/stripe-logo.png +0 -0
  252. package/src/commands/testing/out/static/twitch-icon-facebook.png +0 -0
  253. package/src/commands/testing/out/static/twitch-icon-twitter.png +0 -0
  254. package/src/commands/testing/out/static/twitch-logo.png +0 -0
  255. package/src/commands/testing/out/static/vercel-arrow.png +0 -0
  256. package/src/commands/testing/out/static/vercel-logo.png +0 -0
  257. package/src/commands/testing/out/static/vercel-team.png +0 -0
  258. package/src/commands/testing/out/static/vercel-user.png +0 -0
  259. package/src/commands/testing/out/static/yelp-footer.png +0 -0
  260. package/src/commands/testing/out/static/yelp-header.png +0 -0
  261. package/src/commands/testing/out/static/yelp-logo.png +0 -0
  262. package/src/commands/testing/out/welcome/koala-welcome.html +0 -90
  263. package/src/commands/testing/out/welcome/netlify-welcome.html +0 -199
  264. package/src/commands/testing/out/welcome/stripe-welcome.html +0 -153
  265. package/src/components/button.tsx +0 -101
  266. package/src/components/code-container.tsx +0 -164
  267. package/src/components/code-snippet.tsx +0 -9
  268. package/src/components/code.tsx +0 -184
  269. package/src/components/heading.tsx +0 -113
  270. package/src/components/icons/icon-arrow-down.tsx +0 -16
  271. package/src/components/icons/icon-base.tsx +0 -26
  272. package/src/components/icons/icon-bug.tsx +0 -19
  273. package/src/components/icons/icon-button.tsx +0 -23
  274. package/src/components/icons/icon-check.tsx +0 -19
  275. package/src/components/icons/icon-clipboard.tsx +0 -40
  276. package/src/components/icons/icon-download.tsx +0 -19
  277. package/src/components/icons/icon-email.tsx +0 -18
  278. package/src/components/icons/icon-file.tsx +0 -19
  279. package/src/components/icons/icon-folder-open.tsx +0 -19
  280. package/src/components/icons/icon-folder.tsx +0 -18
  281. package/src/components/icons/icon-hide-sidebar.tsx +0 -23
  282. package/src/components/icons/icon-image.tsx +0 -19
  283. package/src/components/icons/icon-info.tsx +0 -18
  284. package/src/components/icons/icon-link.tsx +0 -14
  285. package/src/components/icons/icon-monitor.tsx +0 -19
  286. package/src/components/icons/icon-phone.tsx +0 -26
  287. package/src/components/icons/icon-reload.tsx +0 -18
  288. package/src/components/icons/icon-source.tsx +0 -19
  289. package/src/components/icons/icon-stamp.tsx +0 -14
  290. package/src/components/icons/icon-warning.tsx +0 -31
  291. package/src/components/index.ts +0 -7
  292. package/src/components/logo.tsx +0 -63
  293. package/src/components/resizable-wrapper.tsx +0 -173
  294. package/src/components/send.tsx +0 -134
  295. package/src/components/shell.tsx +0 -95
  296. package/src/components/sidebar/file-tree-directory-children.tsx +0 -134
  297. package/src/components/sidebar/file-tree-directory.tsx +0 -92
  298. package/src/components/sidebar/file-tree.tsx +0 -31
  299. package/src/components/sidebar/index.ts +0 -1
  300. package/src/components/sidebar/sidebar.tsx +0 -43
  301. package/src/components/text.tsx +0 -99
  302. package/src/components/toolbar/checking-results.tsx +0 -150
  303. package/src/components/toolbar/code-preview-line-link.tsx +0 -40
  304. package/src/components/toolbar/compatibility.tsx +0 -113
  305. package/src/components/toolbar/linter.tsx +0 -278
  306. package/src/components/toolbar/results.tsx +0 -51
  307. package/src/components/toolbar/spam-assassin.tsx +0 -155
  308. package/src/components/toolbar/toolbar-button.tsx +0 -52
  309. package/src/components/toolbar/use-cached-state.ts +0 -33
  310. package/src/components/toolbar.tsx +0 -349
  311. package/src/components/tooltip-content.tsx +0 -31
  312. package/src/components/tooltip.tsx +0 -19
  313. package/src/components/topbar/active-view-toggle-group.tsx +0 -86
  314. package/src/components/topbar/view-size-controls.tsx +0 -247
  315. package/src/components/topbar.tsx +0 -59
  316. package/src/contexts/emails.tsx +0 -59
  317. package/src/contexts/fragment-identifier.tsx +0 -48
  318. package/src/contexts/preview.tsx +0 -79
  319. package/src/hooks/use-clamped-state.ts +0 -24
  320. package/src/hooks/use-email-rendering-result.ts +0 -58
  321. package/src/hooks/use-fragment-identifier.ts +0 -14
  322. package/src/hooks/use-hot-reload.ts +0 -31
  323. package/src/hooks/use-icon-animation.ts +0 -41
  324. package/src/hooks/use-rendering-metadata.ts +0 -36
  325. package/src/package/body/dist/index.d.mts +0 -6
  326. package/src/package/body/dist/index.d.ts +0 -6
  327. package/src/package/body/dist/index.js +0 -79
  328. package/src/package/body/dist/index.mjs +0 -45
  329. package/src/package/button/dist/index.d.mts +0 -6
  330. package/src/package/button/dist/index.d.ts +0 -6
  331. package/src/package/button/dist/index.js +0 -252
  332. package/src/package/button/dist/index.mjs +0 -218
  333. package/src/package/code-block/dist/index.d.mts +0 -4906
  334. package/src/package/code-block/dist/index.d.ts +0 -4906
  335. package/src/package/code-block/dist/index.js +0 -18205
  336. package/src/package/code-block/dist/index.mjs +0 -18133
  337. package/src/package/code-inline/dist/index.d.mts +0 -11
  338. package/src/package/code-inline/dist/index.d.ts +0 -11
  339. package/src/package/code-inline/dist/index.js +0 -106
  340. package/src/package/code-inline/dist/index.mjs +0 -72
  341. package/src/package/column/dist/index.d.mts +0 -6
  342. package/src/package/column/dist/index.d.ts +0 -6
  343. package/src/package/column/dist/index.js +0 -79
  344. package/src/package/column/dist/index.mjs +0 -45
  345. package/src/package/components/dist/index.d.mts +0 -20
  346. package/src/package/components/dist/index.d.ts +0 -20
  347. package/src/package/components/dist/index.js +0 -62
  348. package/src/package/components/dist/index.mjs +0 -21
  349. package/src/package/container/dist/index.d.mts +0 -6
  350. package/src/package/container/dist/index.d.ts +0 -6
  351. package/src/package/container/dist/index.js +0 -93
  352. package/src/package/container/dist/index.mjs +0 -59
  353. package/src/package/font/dist/index.d.mts +0 -25
  354. package/src/package/font/dist/index.d.ts +0 -25
  355. package/src/package/font/dist/index.js +0 -55
  356. package/src/package/font/dist/index.mjs +0 -28
  357. package/src/package/head/dist/index.d.mts +0 -6
  358. package/src/package/head/dist/index.d.ts +0 -6
  359. package/src/package/head/dist/index.js +0 -83
  360. package/src/package/head/dist/index.mjs +0 -49
  361. package/src/package/heading/dist/index.d.mts +0 -43
  362. package/src/package/heading/dist/index.d.ts +0 -43
  363. package/src/package/heading/dist/index.js +0 -113
  364. package/src/package/heading/dist/index.mjs +0 -79
  365. package/src/package/hr/dist/index.d.mts +0 -6
  366. package/src/package/hr/dist/index.d.ts +0 -6
  367. package/src/package/hr/dist/index.js +0 -89
  368. package/src/package/hr/dist/index.mjs +0 -55
  369. package/src/package/html/dist/index.d.mts +0 -6
  370. package/src/package/html/dist/index.d.ts +0 -6
  371. package/src/package/html/dist/index.js +0 -79
  372. package/src/package/html/dist/index.mjs +0 -45
  373. package/src/package/img/dist/index.d.mts +0 -6
  374. package/src/package/img/dist/index.d.ts +0 -6
  375. package/src/package/img/dist/index.js +0 -94
  376. package/src/package/img/dist/index.mjs +0 -60
  377. package/src/package/link/dist/index.d.mts +0 -6
  378. package/src/package/link/dist/index.d.ts +0 -6
  379. package/src/package/link/dist/index.js +0 -90
  380. package/src/package/link/dist/index.mjs +0 -56
  381. package/src/package/markdown/dist/index.d.mts +0 -15
  382. package/src/package/markdown/dist/index.d.ts +0 -15
  383. package/src/package/markdown/dist/index.js +0 -92
  384. package/src/package/markdown/dist/index.mjs +0 -58
  385. package/src/package/preview/dist/index.d.mts +0 -12
  386. package/src/package/preview/dist/index.d.ts +0 -12
  387. package/src/package/preview/dist/index.js +0 -108
  388. package/src/package/preview/dist/index.mjs +0 -73
  389. package/src/package/render/dist/browser/index.d.mts +0 -24
  390. package/src/package/render/dist/browser/index.d.ts +0 -24
  391. package/src/package/render/dist/browser/index.js +0 -250
  392. package/src/package/render/dist/browser/index.mjs +0 -214
  393. package/src/package/render/dist/index.d.mts +0 -23
  394. package/src/package/render/dist/index.d.ts +0 -23
  395. package/src/package/render/dist/index.js +0 -768
  396. package/src/package/render/dist/index.mjs +0 -733
  397. package/src/package/render/dist/node/index.d.mts +0 -27
  398. package/src/package/render/dist/node/index.d.ts +0 -27
  399. package/src/package/render/dist/node/index.js +0 -212
  400. package/src/package/render/dist/node/index.mjs +0 -176
  401. package/src/package/row/dist/index.d.mts +0 -10
  402. package/src/package/row/dist/index.d.ts +0 -10
  403. package/src/package/row/dist/index.js +0 -93
  404. package/src/package/row/dist/index.mjs +0 -59
  405. package/src/package/section/dist/index.d.mts +0 -6
  406. package/src/package/section/dist/index.d.ts +0 -6
  407. package/src/package/section/dist/index.js +0 -93
  408. package/src/package/section/dist/index.mjs +0 -59
  409. package/src/package/tailwind/dist/index.d.ts +0 -19
  410. package/src/package/tailwind/dist/index.js +0 -48
  411. package/src/package/tailwind/dist/index.mjs +0 -17167
  412. package/src/package/tailwind/dist/tailwindcss/config.d.ts +0 -376
  413. package/src/package/tailwind/dist/tailwindcss/generated/.gitkeep +0 -0
  414. package/src/package/tailwind/dist/tailwindcss/generated/colors.d.ts +0 -298
  415. package/src/package/tailwind/dist/tailwindcss/generated/corePluginList.d.ts +0 -1
  416. package/src/package/tailwind/dist/tailwindcss/generated/default-theme.d.ts +0 -397
  417. package/src/package/tailwind/dist/tailwindcss/index.d.ts +0 -11
  418. package/src/package/text/dist/index.d.mts +0 -6
  419. package/src/package/text/dist/index.d.ts +0 -6
  420. package/src/package/text/dist/index.js +0 -89
  421. package/src/package/text/dist/index.mjs +0 -55
  422. package/src/utils/__snapshots__/get-email-component.spec.ts.snap +0 -3
  423. package/src/utils/caniemail/all-css-properties.ts +0 -358
  424. package/src/utils/caniemail/ast/__snapshots__/get-object-variables.spec.ts.snap +0 -74
  425. package/src/utils/caniemail/ast/__snapshots__/get-used-style-properties.spec.ts.snap +0 -24
  426. package/src/utils/caniemail/ast/get-object-variables.spec.ts +0 -19
  427. package/src/utils/caniemail/ast/get-object-variables.ts +0 -61
  428. package/src/utils/caniemail/ast/get-used-style-properties.spec.ts +0 -23
  429. package/src/utils/caniemail/ast/get-used-style-properties.ts +0 -91
  430. package/src/utils/caniemail/get-compatibility-stats-for-entry.ts +0 -118
  431. package/src/utils/caniemail/get-css-functions.ts +0 -25
  432. package/src/utils/caniemail/get-css-property-names.ts +0 -32
  433. package/src/utils/caniemail/get-css-property-with-value.ts +0 -14
  434. package/src/utils/caniemail/get-css-unit.ts +0 -3
  435. package/src/utils/caniemail/get-element-attributes.ts +0 -7
  436. package/src/utils/caniemail/get-element-names.ts +0 -20
  437. package/src/utils/caniemail/tailwind/generate-tailwind-rules.ts +0 -30
  438. package/src/utils/caniemail/tailwind/get-tailwind-config.ts +0 -187
  439. package/src/utils/caniemail/tailwind/get-tailwind-metadata.spec.ts +0 -25
  440. package/src/utils/caniemail/tailwind/get-tailwind-metadata.ts +0 -45
  441. package/src/utils/caniemail/tailwind/setup-tailwind-context.ts +0 -15
  442. package/src/utils/cn.ts +0 -6
  443. package/src/utils/constants.ts +0 -6
  444. package/src/utils/contains-email-template.spec.ts +0 -124
  445. package/src/utils/contains-email-template.ts +0 -33
  446. package/src/utils/copy-text-to-clipboard.ts +0 -7
  447. package/src/utils/get-email-component.spec.ts +0 -41
  448. package/src/utils/get-email-component.ts +0 -134
  449. package/src/utils/get-line-and-column-from-offset.spec.ts +0 -11
  450. package/src/utils/get-line-and-column-from-offset.ts +0 -11
  451. package/src/utils/improve-error-with-sourcemap.ts +0 -86
  452. package/src/utils/js-email-detection.spec.ts +0 -24
  453. package/src/utils/language-map.ts +0 -7
  454. package/src/utils/linting.ts +0 -60
  455. package/src/utils/load-stream.ts +0 -15
  456. package/src/utils/result.ts +0 -49
  457. package/src/utils/run-bundled-code.ts +0 -64
  458. package/src/utils/sanitize.ts +0 -6
  459. package/src/utils/static-node-modules-for-vm.ts +0 -93
  460. package/src/utils/testing/js-email-export-default.js +0 -17
  461. package/src/utils/testing/js-email-test.js +0 -18
  462. package/src/utils/testing/mdx-email-test.js +0 -128
  463. package/src/utils/testing/request-response-email.tsx +0 -9
  464. package/src/utils/types/as.ts +0 -26
  465. package/src/utils/types/email-template.ts +0 -8
  466. package/src/utils/types/error-object.ts +0 -11
  467. package/src/utils/unreachable.ts +0 -8
  468. package/tailwind-internals.d.ts +0 -133
  469. package/tailwind.config.ts +0 -99
  470. /package/src/{components/toolbar/results-table.tsx → utils/preview/hot-reloading/test/some-file.ts} +0 -0
@@ -1,113 +0,0 @@
1
- import { useRef, useState } from 'react';
2
- import { toast } from 'sonner';
3
- import { nicenames } from '../../actions/email-validation/caniemail-data';
4
- import {
5
- type CompatibilityCheckingResult,
6
- checkCompatibility,
7
- } from '../../actions/email-validation/check-compatibility';
8
- import { sanitize } from '../../utils';
9
- import { loadStream } from '../../utils/load-stream';
10
- import { IconWarning } from '../icons/icon-warning';
11
- import { CodePreviewLineLink } from './code-preview-line-link';
12
- import { Results } from './results';
13
-
14
- export const useCompatibility = ({
15
- reactMarkup,
16
- emailPath,
17
-
18
- initialResults,
19
- }: {
20
- reactMarkup: string;
21
- emailPath: string;
22
-
23
- initialResults?: CompatibilityCheckingResult[];
24
- }) => {
25
- const [results, setResults] = useState(initialResults);
26
-
27
- const [loading, setLoading] = useState(false);
28
- const isLoadingRef = useRef(false);
29
-
30
- const load = async () => {
31
- if (isLoadingRef.current) return;
32
- isLoadingRef.current = true;
33
- setLoading(true);
34
-
35
- setResults([]);
36
- let rawResults: CompatibilityCheckingResult[] = [];
37
-
38
- try {
39
- const stream = await checkCompatibility(reactMarkup, emailPath);
40
- for await (const result of loadStream(stream)) {
41
- if (result.status !== 'error') continue;
42
- setResults((current) => {
43
- if (!current) {
44
- return [result];
45
- }
46
- rawResults = [...current, result];
47
- return rawResults;
48
- });
49
- }
50
- } catch (exception) {
51
- console.error(exception);
52
- toast.error(JSON.stringify(exception));
53
- } finally {
54
- setLoading(false);
55
- isLoadingRef.current = false;
56
- }
57
-
58
- return rawResults;
59
- };
60
-
61
- return [results, { loading, load }] as const;
62
- };
63
-
64
- interface CompatibilityProps {
65
- results: CompatibilityCheckingResult[] | undefined;
66
- }
67
-
68
- export const Compatibility = ({ results }: CompatibilityProps) => {
69
- return (
70
- <Results>
71
- {results?.map((result, i) => {
72
- const statsReportedNotWorking = Object.entries(
73
- result.statsPerEmailClient,
74
- ).filter(([, stats]) => stats.status === 'error');
75
- const unsupportedClientsString = statsReportedNotWorking
76
- .map(([emailClient]) => nicenames.family[emailClient])
77
- .join(', ');
78
-
79
- return (
80
- <Results.Row key={i}>
81
- <Results.Column>
82
- <span className="flex text-red-400 uppercase gap-2 items-center">
83
- <IconWarning />
84
- {sanitize(result.entry.title)}
85
- </span>
86
- </Results.Column>
87
- <Results.Column>
88
- {statsReportedNotWorking.length > 0
89
- ? `Not supported in ${unsupportedClientsString}`
90
- : null}
91
-
92
- <a
93
- href={result.entry.url}
94
- className="underline ml-2 decoration-slate-9 decoration-1 hover:decoration-slate-11 transition-colors hover:text-slate-12"
95
- rel="noreferrer"
96
- target="_blank"
97
- >
98
- More ↗
99
- </a>
100
- </Results.Column>
101
- <Results.Column className="font-mono text-slate-11 text-right">
102
- <CodePreviewLineLink
103
- line={result.location.start.line}
104
- column={result.location.start.column}
105
- type="react"
106
- />
107
- </Results.Column>
108
- </Results.Row>
109
- );
110
- })}
111
- </Results>
112
- );
113
- };
@@ -1,278 +0,0 @@
1
- import prettyBytes from 'pretty-bytes';
2
- import { Children, useRef, useState } from 'react';
3
- import type { ImageCheckingResult } from '../../actions/email-validation/check-images';
4
- import type { LinkCheckingResult } from '../../actions/email-validation/check-links';
5
- import { cn, sanitize } from '../../utils';
6
- import { getLintingSources, loadLintingRowsFrom } from '../../utils/linting';
7
- import { IconWarning } from '../icons/icon-warning';
8
- import { CodePreviewLineLink } from './code-preview-line-link';
9
- import { Results } from './results';
10
-
11
- export type LintingRow =
12
- | {
13
- source: 'image';
14
- result: ImageCheckingResult;
15
- }
16
- | {
17
- source: 'link';
18
- result: LinkCheckingResult;
19
- };
20
-
21
- interface LinterProps {
22
- rows: LintingRow[] | undefined;
23
- }
24
-
25
- export const useLinter = ({
26
- markup,
27
-
28
- initialRows,
29
- }: {
30
- markup: string;
31
-
32
- initialRows?: LintingRow[];
33
- }) => {
34
- const [rows, setRows] = useState<LintingRow[] | undefined>(initialRows);
35
-
36
- const sources = getLintingSources(
37
- markup,
38
- 'location' in global
39
- ? `${global.location.protocol}//${global.location.host}`
40
- : '',
41
- );
42
-
43
- const [loading, setLoading] = useState(false);
44
- const isStreaming = useRef(false);
45
-
46
- const load = async () => {
47
- if (isStreaming.current) return;
48
- isStreaming.current = true;
49
- setLoading(true);
50
-
51
- setRows([]);
52
- try {
53
- let rows: LintingRow[] = [];
54
- for await (const row of loadLintingRowsFrom(sources)) {
55
- setRows((current) => {
56
- if (!current) {
57
- return [row];
58
- }
59
- const newArray = [...current, row];
60
- newArray.sort((a, b) => {
61
- if (a.result.status === 'error' && b.result.status === 'warning') {
62
- return -1;
63
- }
64
-
65
- if (a.result.status === 'warning' && b.result.status === 'error') {
66
- return 1;
67
- }
68
-
69
- return 0;
70
- });
71
- rows = newArray;
72
- return newArray;
73
- });
74
- }
75
- return rows;
76
- } finally {
77
- setLoading(false);
78
- isStreaming.current = false;
79
- }
80
- };
81
-
82
- return [rows, { loading, load }] as const;
83
- };
84
-
85
- export const Linter = ({ rows }: LinterProps) => {
86
- if (rows === undefined) return null;
87
-
88
- return (
89
- <Results>
90
- {rows.map((row, i) => {
91
- if (row.source === 'link') {
92
- const failingCheck = row.result.checks.find(
93
- (check) => check.passed === false,
94
- )!;
95
- const metadata: React.ReactNode[] = [];
96
- for (const check of row.result.checks) {
97
- if (
98
- check.type === 'fetch_attempt' &&
99
- check.metadata.fetchStatusCode
100
- ) {
101
- metadata.push(<>HTTP {check.metadata.fetchStatusCode}</>);
102
- }
103
- }
104
- metadata.push(
105
- <CodePreviewLineLink
106
- line={row.result.codeLocation.line}
107
- column={row.result.codeLocation.column}
108
- type="html"
109
- />,
110
- );
111
- return (
112
- <Result status={row.result.status} key={i}>
113
- <Result.Name>{sanitize(failingCheck.type)}</Result.Name>
114
- <Result.Description>
115
- {failingCheck.type === 'security'
116
- ? 'Insecure URL, use HTTPS instead of HTTP'
117
- : null}
118
- {failingCheck.type === 'fetch_attempt' &&
119
- failingCheck.metadata.fetchStatusCode &&
120
- failingCheck.metadata.fetchStatusCode >= 300 &&
121
- failingCheck.metadata.fetchStatusCode < 400
122
- ? 'There was a redirect, the content may have been moved'
123
- : null}
124
- {failingCheck.type === 'fetch_attempt' &&
125
- failingCheck.metadata.fetchStatusCode &&
126
- failingCheck.metadata.fetchStatusCode >= 400
127
- ? 'The link is broken'
128
- : null}
129
- {failingCheck.type === 'fetch_attempt' &&
130
- failingCheck.metadata.fetchStatusCode === undefined
131
- ? 'The link could not be reached'
132
- : null}
133
- {failingCheck.type === 'syntax'
134
- ? 'The link is broken due to invalid syntax'
135
- : null}
136
-
137
- <span className="ml-2 text-ellipsis overflow-hidden text-nowrap max-w-[30ch]">
138
- {row.result.link}
139
- </span>
140
- </Result.Description>
141
- <Result.Metadata>{metadata}</Result.Metadata>
142
- </Result>
143
- );
144
- }
145
-
146
- if (row.source === 'image') {
147
- const failingCheck = row.result.checks.find(
148
- (check) => check.passed === false,
149
- )!;
150
- const metadata: React.ReactNode[] = [];
151
- for (const check of row.result.checks) {
152
- if (check.type === 'image_size' && check.metadata.byteCount) {
153
- metadata.push(prettyBytes(check.metadata.byteCount));
154
- }
155
- if (
156
- check.type === 'fetch_attempt' &&
157
- check.metadata.fetchStatusCode
158
- ) {
159
- metadata.push(<>HTTP {check.metadata.fetchStatusCode}</>);
160
- }
161
- }
162
- metadata.push(
163
- <CodePreviewLineLink
164
- line={row.result.codeLocation.line}
165
- column={row.result.codeLocation.column}
166
- type="html"
167
- />,
168
- );
169
- return (
170
- <Result status={row.result.status} key={i}>
171
- <Result.Name>{sanitize(failingCheck.type)}</Result.Name>
172
- <Result.Description>
173
- {failingCheck.type === 'security'
174
- ? 'Insecure URL, use HTTPS instead of HTTP'
175
- : null}
176
- {failingCheck.type === 'fetch_attempt' &&
177
- failingCheck.metadata.fetchStatusCode &&
178
- failingCheck.metadata.fetchStatusCode >= 300 &&
179
- failingCheck.metadata.fetchStatusCode < 400
180
- ? 'There was a redirect, the image may have been moved'
181
- : null}
182
- {failingCheck.type === 'fetch_attempt' &&
183
- failingCheck.metadata.fetchStatusCode &&
184
- failingCheck.metadata.fetchStatusCode >= 400
185
- ? 'The image is broken'
186
- : null}
187
- {failingCheck.type === 'fetch_attempt' &&
188
- failingCheck.metadata.fetchStatusCode === undefined
189
- ? 'The image could not be reached'
190
- : null}
191
- {failingCheck.type === 'syntax'
192
- ? 'The image is broken due to an invalid source'
193
- : null}
194
-
195
- {failingCheck.type === 'accessibility'
196
- ? 'Missing alt text'
197
- : null}
198
-
199
- {failingCheck.type === 'image_size' &&
200
- failingCheck.metadata.byteCount
201
- ? 'This image is too large, keep it under 1mb'
202
- : null}
203
-
204
- <span className="ml-2 text-ellipsis overflow-hidden text-nowrap max-w-[30ch]">
205
- {row.result.source}
206
- </span>
207
- </Result.Description>
208
- <Result.Metadata>{metadata}</Result.Metadata>
209
- </Result>
210
- );
211
- }
212
-
213
- return undefined;
214
- })}
215
- </Results>
216
- );
217
- };
218
-
219
- interface ResultProps extends React.ComponentProps<typeof Results.Row> {
220
- status: 'error' | 'warning' | 'success';
221
- }
222
-
223
- const Result = ({ children, className, status, ...props }: ResultProps) => {
224
- return (
225
- <Results.Row
226
- data-status={status}
227
- {...props}
228
- className={cn('group/result', className)}
229
- >
230
- {children}
231
- </Results.Row>
232
- );
233
- };
234
-
235
- Result.Name = ({
236
- children,
237
- ...props
238
- }: React.ComponentProps<typeof Results.Column>) => {
239
- return (
240
- <Results.Column {...props}>
241
- <span className="flex uppercase gap-2 items-center group-data-[status=error]/result:text-red-400 group-data-[status=warning]/result:text-orange-300">
242
- <IconWarning />
243
- {typeof children === 'string' ? sanitize(children) : children}
244
- </span>
245
- </Results.Column>
246
- );
247
- };
248
-
249
- Result.Description = ({
250
- children,
251
- className,
252
- ...props
253
- }: React.ComponentProps<typeof Results.Column>) => {
254
- return <Results.Column {...props}>{children}</Results.Column>;
255
- };
256
-
257
- interface MetadatProps extends React.ComponentProps<typeof Results.Column> {
258
- children: React.ReactNode[];
259
- }
260
-
261
- Result.Metadata = ({ children, className, ...props }: MetadatProps) => {
262
- return (
263
- <Results.Column
264
- align="right"
265
- {...props}
266
- className={cn('font-mono text-slate-11', className)}
267
- >
268
- {Children.map(children, (child, index) => {
269
- return (
270
- <>
271
- {index > 0 ? ' · ' : null}
272
- {child}
273
- </>
274
- );
275
- })}
276
- </Results.Column>
277
- );
278
- };
@@ -1,51 +0,0 @@
1
- import { cn } from '../../utils';
2
-
3
- export const Results = ({
4
- children,
5
- className,
6
- ...props
7
- }: React.ComponentProps<'table'>) => {
8
- return (
9
- <table
10
- className={cn(
11
- 'group relative w-full border-collapse text-left text-slate-10 text-sm',
12
- className,
13
- )}
14
- >
15
- <tbody>{children}</tbody>
16
- </table>
17
- );
18
- };
19
-
20
- Results.Row = ({
21
- children,
22
- className,
23
- ...props
24
- }: React.ComponentProps<'tr'>) => {
25
- return (
26
- <tr
27
- className={cn(
28
- 'border-collapse align-bottom border-slate-6 border-b last:border-b-0',
29
- className,
30
- )}
31
- {...props}
32
- >
33
- {children}
34
- </tr>
35
- );
36
- };
37
-
38
- Results.Column = ({
39
- children,
40
- className,
41
- ...props
42
- }: React.ComponentProps<'td'>) => {
43
- return (
44
- <td
45
- className={cn('py-1.5 align-bottom font-regular', className)}
46
- {...props}
47
- >
48
- {children}
49
- </td>
50
- );
51
- };
@@ -1,155 +0,0 @@
1
- import { useRef, useState } from 'react';
2
- import { toast } from 'sonner';
3
- import { cn, sanitize } from '../../utils';
4
- import { IconWarning } from '../icons/icon-warning';
5
- import { Results } from './results';
6
-
7
- interface SpamAssassinProps {
8
- result: SpamCheckingResult | undefined;
9
- }
10
-
11
- export interface SpamCheckingResult {
12
- checks: {
13
- name: string;
14
- description: string;
15
- points: number;
16
- }[];
17
- isSpam: boolean;
18
- points: number;
19
- }
20
-
21
- function toSorted<T>(array: T[], sorter: (a: T, b: T) => number): T[] {
22
- const cloned = [...array];
23
- cloned.sort(sorter);
24
- return cloned;
25
- }
26
-
27
- export const useSpamAssassin = ({
28
- markup,
29
- plainText,
30
-
31
- initialResult,
32
- }: {
33
- markup: string;
34
- plainText: string;
35
-
36
- initialResult?: SpamCheckingResult;
37
- }) => {
38
- const [result, setResult] = useState<SpamCheckingResult | undefined>(
39
- initialResult,
40
- );
41
-
42
- const [loading, setLoading] = useState(false);
43
- const isLoadingRef = useRef(false);
44
-
45
- const load = async () => {
46
- if (isLoadingRef.current) return;
47
- isLoadingRef.current = true;
48
- setLoading(true);
49
-
50
- try {
51
- const response = await fetch('https://react.email/api/check-spam', {
52
- method: 'POST',
53
- headers: { 'Content-Type': 'application/json' },
54
- body: JSON.stringify({
55
- html: markup,
56
- plainText: plainText,
57
- }),
58
- });
59
-
60
- const responseBody = (await response.json()) as
61
- | { error: string }
62
- | SpamCheckingResult;
63
- if ('error' in responseBody) {
64
- toast.error(responseBody.error);
65
- } else {
66
- setResult(responseBody);
67
- return responseBody;
68
- }
69
- } catch (exception) {
70
- console.error(exception);
71
- toast.error(JSON.stringify(exception));
72
- } finally {
73
- setLoading(false);
74
- isLoadingRef.current = false;
75
- }
76
- };
77
-
78
- return [result, { loading, load }] as const;
79
- };
80
-
81
- export const SpamAssassin = ({ result }: SpamAssassinProps) => {
82
- return (
83
- <>
84
- {result ? (
85
- <Results>
86
- <Results.Row className="sticky border-b top-0">
87
- <Results.Column className="uppercase">
88
- <span className="flex gap-2 items-center">
89
- <IconWarning
90
- className={cn(
91
- result.points === 0 ? 'text-green-400' : null,
92
- result.points > 0 && result.points <= 1.5 ? null : null,
93
- result.points > 1.5 ? 'text-yellow-100' : null,
94
- result.points > 3 ? 'text-orange-400' : null,
95
- result.points >= 5 ? 'text-red-400' : null,
96
- )}
97
- />
98
- Score
99
- </span>
100
- </Results.Column>
101
- <Results.Column>
102
- {result.points === 0
103
- ? 'Congratulations! Your email is clean of abuse indicators.'
104
- : 'Higher scores are better'}
105
- </Results.Column>
106
- <Results.Column className="text-right tracking-tighter font-bold">
107
- <span
108
- className={cn(
109
- 'text-3xl',
110
- result.points === 0 ? 'text-green-400' : null,
111
- result.points > 0 && result.points <= 1.5 ? null : null,
112
- result.points > 1.5 ? 'text-yellow-200' : null,
113
- result.points > 3 ? 'text-orange-400' : null,
114
- result.points >= 5 ? 'text-red-400' : null,
115
- )}
116
- >
117
- {(10 - result.points).toFixed(1)}
118
- </span>{' '}
119
- <span className="text-lg">/ 10</span>
120
- </Results.Column>
121
- </Results.Row>
122
- {toSorted(result.checks, (a, b) => b.points - a.points).map(
123
- (check) => (
124
- <Results.Row key={check.name}>
125
- <Results.Column className="uppercase">
126
- <span className="flex gap-2 items-center">
127
- <IconWarning
128
- className={cn(
129
- check.points > 1 ? 'text-yellow-200' : null,
130
- check.points > 2 ? 'text-orange-400' : null,
131
- check.points > 3 ? 'text-red-400' : null,
132
- )}
133
- />
134
- {sanitize(check.name)}
135
- </span>
136
- </Results.Column>
137
- <Results.Column>{check.description}</Results.Column>
138
- <Results.Column
139
- className={cn(
140
- 'text-right font-mono tracking-tighter',
141
- check.points > 1 ? 'text-yellow-200' : null,
142
- check.points > 2 ? 'text-orange-400' : null,
143
- check.points > 3 ? 'text-red-400' : null,
144
- )}
145
- >
146
- -{check.points.toFixed(1)}
147
- </Results.Column>
148
- </Results.Row>
149
- ),
150
- )}
151
- </Results>
152
- ) : null}
153
- </>
154
- );
155
- };
@@ -1,52 +0,0 @@
1
- import { motion } from 'framer-motion';
2
- import { cn } from '../../utils';
3
- import { Tooltip } from '../tooltip';
4
-
5
- interface ToolbarButtonProps extends React.ComponentProps<'button'> {
6
- children: React.ReactNode;
7
- active?: boolean;
8
- tooltip?: React.ReactNode;
9
- delayDuration?: number;
10
- }
11
-
12
- export const ToolbarButton = ({
13
- children,
14
- className,
15
- active,
16
- tooltip,
17
- delayDuration = 500,
18
- ...props
19
- }: ToolbarButtonProps) => {
20
- return (
21
- <Tooltip.Provider>
22
- <Tooltip delayDuration={delayDuration}>
23
- <Tooltip.Trigger asChild>
24
- <button
25
- type="button"
26
- {...props}
27
- className={cn(
28
- 'h-full w-fit font-regular flex text-sm text-slate-10 items-center align-middle justify-center px-1 gap-2 relative',
29
- 'hover:text-slate-12 transition-colors',
30
- active && 'data-[state=active]:text-cyan-11',
31
- className,
32
- )}
33
- >
34
- {children}
35
- {active ? (
36
- <motion.span
37
- className="-bottom-px absolute rounded-sm left-0 w-full bg-cyan-11 h-px"
38
- layoutId="active-toolbar-button"
39
- transition={{
40
- type: 'spring',
41
- bounce: 0.2,
42
- duration: 0.6,
43
- }}
44
- />
45
- ) : null}
46
- </button>
47
- </Tooltip.Trigger>
48
- {tooltip ? <Tooltip.Content>{tooltip}</Tooltip.Content> : null}
49
- </Tooltip>
50
- </Tooltip.Provider>
51
- );
52
- };
@@ -1,33 +0,0 @@
1
- import { useSyncExternalStore } from 'react';
2
-
3
- export const useCachedState = <T>(key: string) => {
4
- let value: T | undefined = undefined;
5
- if ('localStorage' in global) {
6
- const storedValue = global.localStorage.getItem(key);
7
- if (storedValue !== null && storedValue !== 'undefined') {
8
- try {
9
- value = JSON.parse(storedValue) as T;
10
- } catch (exception) {
11
- console.warn(
12
- 'Failed to load stored value for',
13
- key,
14
- 'with value',
15
- storedValue,
16
- );
17
- }
18
- }
19
- }
20
-
21
- return [
22
- useSyncExternalStore(
23
- () => () => {},
24
- () => value,
25
- () => undefined,
26
- ),
27
- function setValue(newValue: T | undefined) {
28
- if ('localStorage' in global) {
29
- global.localStorage.setItem(key, JSON.stringify(newValue));
30
- }
31
- },
32
- ] as const;
33
- };