@ahmedrowaihi/pdf-forge-preview 1.0.0-canary.0

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 (351) hide show
  1. package/.next/BUILD_ID +1 -0
  2. package/.next/app-path-routes-manifest.json +7 -0
  3. package/.next/build/chunks/[root-of-the-server]__12fb5caf._.js +233 -0
  4. package/.next/build/chunks/[root-of-the-server]__12fb5caf._.js.map +8 -0
  5. package/.next/build/chunks/[root-of-the-server]__242deb00._.js +500 -0
  6. package/.next/build/chunks/[root-of-the-server]__242deb00._.js.map +11 -0
  7. package/.next/build/chunks/[turbopack-node]_transforms_postcss_ts_ad9a1eec._.js +13 -0
  8. package/.next/build/chunks/[turbopack-node]_transforms_postcss_ts_ad9a1eec._.js.map +5 -0
  9. package/.next/build/chunks/[turbopack]_runtime.js +795 -0
  10. package/.next/build/chunks/[turbopack]_runtime.js.map +10 -0
  11. package/.next/build/chunks/node_modules__pnpm_806d01c0._.js +6758 -0
  12. package/.next/build/chunks/node_modules__pnpm_806d01c0._.js.map +47 -0
  13. package/.next/build/package.json +1 -0
  14. package/.next/build/postcss.js +6 -0
  15. package/.next/build/postcss.js.map +5 -0
  16. package/.next/build-manifest.json +20 -0
  17. package/.next/diagnostics/build-diagnostics.json +6 -0
  18. package/.next/diagnostics/framework.json +1 -0
  19. package/.next/export-marker.json +6 -0
  20. package/.next/fallback-build-manifest.json +12 -0
  21. package/.next/images-manifest.json +66 -0
  22. package/.next/next-minimal-server.js.nft.json +1 -0
  23. package/.next/next-server.js.nft.json +1 -0
  24. package/.next/package.json +1 -0
  25. package/.next/prerender-manifest.json +65 -0
  26. package/.next/required-server-files.js +163 -0
  27. package/.next/required-server-files.json +163 -0
  28. package/.next/routes-manifest.json +77 -0
  29. package/.next/server/app/_global-error/page/app-paths-manifest.json +3 -0
  30. package/.next/server/app/_global-error/page/build-manifest.json +17 -0
  31. package/.next/server/app/_global-error/page/next-font-manifest.json +6 -0
  32. package/.next/server/app/_global-error/page/react-loadable-manifest.json +1 -0
  33. package/.next/server/app/_global-error/page/server-reference-manifest.json +4 -0
  34. package/.next/server/app/_global-error/page.js +11 -0
  35. package/.next/server/app/_global-error/page.js.map +5 -0
  36. package/.next/server/app/_global-error/page.js.nft.json +1 -0
  37. package/.next/server/app/_global-error/page_client-reference-manifest.js +2 -0
  38. package/.next/server/app/_global-error.html +2 -0
  39. package/.next/server/app/_global-error.meta +15 -0
  40. package/.next/server/app/_global-error.rsc +13 -0
  41. package/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +5 -0
  42. package/.next/server/app/_global-error.segments/_full.segment.rsc +13 -0
  43. package/.next/server/app/_global-error.segments/_head.segment.rsc +6 -0
  44. package/.next/server/app/_global-error.segments/_index.segment.rsc +4 -0
  45. package/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -0
  46. package/.next/server/app/_not-found/page/app-paths-manifest.json +3 -0
  47. package/.next/server/app/_not-found/page/build-manifest.json +17 -0
  48. package/.next/server/app/_not-found/page/next-font-manifest.json +16 -0
  49. package/.next/server/app/_not-found/page/react-loadable-manifest.json +1 -0
  50. package/.next/server/app/_not-found/page/server-reference-manifest.json +20 -0
  51. package/.next/server/app/_not-found/page.js +14 -0
  52. package/.next/server/app/_not-found/page.js.map +5 -0
  53. package/.next/server/app/_not-found/page.js.nft.json +1 -0
  54. package/.next/server/app/_not-found/page_client-reference-manifest.js +2 -0
  55. package/.next/server/app/favicon.ico/route/app-paths-manifest.json +3 -0
  56. package/.next/server/app/favicon.ico/route/build-manifest.json +11 -0
  57. package/.next/server/app/favicon.ico/route.js +6 -0
  58. package/.next/server/app/favicon.ico/route.js.map +5 -0
  59. package/.next/server/app/favicon.ico/route.js.nft.json +1 -0
  60. package/.next/server/app/favicon.ico.body +0 -0
  61. package/.next/server/app/favicon.ico.meta +1 -0
  62. package/.next/server/app/page/app-paths-manifest.json +3 -0
  63. package/.next/server/app/page/build-manifest.json +17 -0
  64. package/.next/server/app/page/next-font-manifest.json +16 -0
  65. package/.next/server/app/page/react-loadable-manifest.json +1 -0
  66. package/.next/server/app/page/server-reference-manifest.json +20 -0
  67. package/.next/server/app/page.js +17 -0
  68. package/.next/server/app/page.js.map +5 -0
  69. package/.next/server/app/page.js.nft.json +1 -0
  70. package/.next/server/app/page_client-reference-manifest.js +2 -0
  71. package/.next/server/app/preview/[...slug]/page/app-paths-manifest.json +3 -0
  72. package/.next/server/app/preview/[...slug]/page/build-manifest.json +17 -0
  73. package/.next/server/app/preview/[...slug]/page/next-font-manifest.json +16 -0
  74. package/.next/server/app/preview/[...slug]/page/react-loadable-manifest.json +1 -0
  75. package/.next/server/app/preview/[...slug]/page/server-reference-manifest.json +65 -0
  76. package/.next/server/app/preview/[...slug]/page.js +19 -0
  77. package/.next/server/app/preview/[...slug]/page.js.map +5 -0
  78. package/.next/server/app/preview/[...slug]/page.js.nft.json +1 -0
  79. package/.next/server/app/preview/[...slug]/page_client-reference-manifest.js +2 -0
  80. package/.next/server/app-paths-manifest.json +7 -0
  81. package/.next/server/chunks/730ea_preview-server__next-internal_server_app_favicon_ico_route_actions_a71a8ae7.js +3 -0
  82. package/.next/server/chunks/730ea_preview-server__next-internal_server_app_favicon_ico_route_actions_a71a8ae7.js.map +1 -0
  83. package/.next/server/chunks/[externals]_next_dist_a6d89067._.js +3 -0
  84. package/.next/server/chunks/[externals]_next_dist_a6d89067._.js.map +1 -0
  85. package/.next/server/chunks/[root-of-the-server]__a62cd78d._.js +21 -0
  86. package/.next/server/chunks/[root-of-the-server]__a62cd78d._.js.map +1 -0
  87. package/.next/server/chunks/[turbopack]_runtime.js +795 -0
  88. package/.next/server/chunks/[turbopack]_runtime.js.map +10 -0
  89. package/.next/server/chunks/ssr/730ea_preview-server__next-internal_server_app__global-error_page_actions_986e2de5.js +3 -0
  90. package/.next/server/chunks/ssr/730ea_preview-server__next-internal_server_app__global-error_page_actions_986e2de5.js.map +1 -0
  91. package/.next/server/chunks/ssr/[root-of-the-server]__025eaae9._.js +3 -0
  92. package/.next/server/chunks/ssr/[root-of-the-server]__025eaae9._.js.map +1 -0
  93. package/.next/server/chunks/ssr/[root-of-the-server]__1536282c._.js +4 -0
  94. package/.next/server/chunks/ssr/[root-of-the-server]__1536282c._.js.map +1 -0
  95. package/.next/server/chunks/ssr/[root-of-the-server]__15cf9d36._.js +3 -0
  96. package/.next/server/chunks/ssr/[root-of-the-server]__15cf9d36._.js.map +1 -0
  97. package/.next/server/chunks/ssr/[root-of-the-server]__450f653e._.js +3 -0
  98. package/.next/server/chunks/ssr/[root-of-the-server]__450f653e._.js.map +1 -0
  99. package/.next/server/chunks/ssr/[root-of-the-server]__4874d0d7._.js +3 -0
  100. package/.next/server/chunks/ssr/[root-of-the-server]__4874d0d7._.js.map +1 -0
  101. package/.next/server/chunks/ssr/[root-of-the-server]__49771cdc._.js +3 -0
  102. package/.next/server/chunks/ssr/[root-of-the-server]__49771cdc._.js.map +1 -0
  103. package/.next/server/chunks/ssr/[root-of-the-server]__9ff74047._.js +3 -0
  104. package/.next/server/chunks/ssr/[root-of-the-server]__9ff74047._.js.map +1 -0
  105. package/.next/server/chunks/ssr/[root-of-the-server]__a9be37b1._.js +3 -0
  106. package/.next/server/chunks/ssr/[root-of-the-server]__a9be37b1._.js.map +1 -0
  107. package/.next/server/chunks/ssr/[root-of-the-server]__aeefe74e._.js +3 -0
  108. package/.next/server/chunks/ssr/[root-of-the-server]__aeefe74e._.js.map +1 -0
  109. package/.next/server/chunks/ssr/[root-of-the-server]__b13738d2._.js +3 -0
  110. package/.next/server/chunks/ssr/[root-of-the-server]__b13738d2._.js.map +1 -0
  111. package/.next/server/chunks/ssr/[root-of-the-server]__bb428c83._.js +4 -0
  112. package/.next/server/chunks/ssr/[root-of-the-server]__bb428c83._.js.map +1 -0
  113. package/.next/server/chunks/ssr/[root-of-the-server]__c61333e3._.js +10 -0
  114. package/.next/server/chunks/ssr/[root-of-the-server]__c61333e3._.js.map +1 -0
  115. package/.next/server/chunks/ssr/[root-of-the-server]__cb0c7b1a._.js +4 -0
  116. package/.next/server/chunks/ssr/[root-of-the-server]__cb0c7b1a._.js.map +1 -0
  117. package/.next/server/chunks/ssr/[root-of-the-server]__dafcae4c._.js +3 -0
  118. package/.next/server/chunks/ssr/[root-of-the-server]__dafcae4c._.js.map +1 -0
  119. package/.next/server/chunks/ssr/[root-of-the-server]__e47d0c07._.js +3 -0
  120. package/.next/server/chunks/ssr/[root-of-the-server]__e47d0c07._.js.map +1 -0
  121. package/.next/server/chunks/ssr/[root-of-the-server]__ecac8617._.js +31 -0
  122. package/.next/server/chunks/ssr/[root-of-the-server]__ecac8617._.js.map +1 -0
  123. package/.next/server/chunks/ssr/[root-of-the-server]__fd4bda25._.js +3 -0
  124. package/.next/server/chunks/ssr/[root-of-the-server]__fd4bda25._.js.map +1 -0
  125. package/.next/server/chunks/ssr/[turbopack]_runtime.js +795 -0
  126. package/.next/server/chunks/ssr/[turbopack]_runtime.js.map +10 -0
  127. package/.next/server/chunks/ssr/_19494208._.js +19 -0
  128. package/.next/server/chunks/ssr/_19494208._.js.map +1 -0
  129. package/.next/server/chunks/ssr/_810f54e9._.js +4 -0
  130. package/.next/server/chunks/ssr/_810f54e9._.js.map +1 -0
  131. package/.next/server/chunks/ssr/_aa01e67e._.js +4 -0
  132. package/.next/server/chunks/ssr/_aa01e67e._.js.map +1 -0
  133. package/.next/server/chunks/ssr/_bf10718b._.js +4 -0
  134. package/.next/server/chunks/ssr/_bf10718b._.js.map +1 -0
  135. package/.next/server/chunks/ssr/_c57c12df._.js +4 -0
  136. package/.next/server/chunks/ssr/_c57c12df._.js.map +1 -0
  137. package/.next/server/chunks/ssr/_d9b57cb8._.js +5 -0
  138. package/.next/server/chunks/ssr/_d9b57cb8._.js.map +1 -0
  139. package/.next/server/chunks/ssr/_e2a766a8._.js +3 -0
  140. package/.next/server/chunks/ssr/_e2a766a8._.js.map +1 -0
  141. package/.next/server/chunks/ssr/_e2b86512._.js +8 -0
  142. package/.next/server/chunks/ssr/_e2b86512._.js.map +1 -0
  143. package/.next/server/chunks/ssr/_ee8ea3aa._.js +19 -0
  144. package/.next/server/chunks/ssr/_ee8ea3aa._.js.map +1 -0
  145. package/.next/server/chunks/ssr/e8449_next_dist_5870db32._.js +3 -0
  146. package/.next/server/chunks/ssr/e8449_next_dist_5870db32._.js.map +1 -0
  147. package/.next/server/chunks/ssr/e8449_next_dist_client_components_builtin_forbidden_5542069c.js +3 -0
  148. package/.next/server/chunks/ssr/e8449_next_dist_client_components_builtin_forbidden_5542069c.js.map +1 -0
  149. package/.next/server/chunks/ssr/e8449_next_dist_client_components_builtin_global-error_d0870f3a.js +3 -0
  150. package/.next/server/chunks/ssr/e8449_next_dist_client_components_builtin_global-error_d0870f3a.js.map +1 -0
  151. package/.next/server/chunks/ssr/e8449_next_dist_client_components_builtin_unauthorized_4c01c8d5.js +3 -0
  152. package/.next/server/chunks/ssr/e8449_next_dist_client_components_builtin_unauthorized_4c01c8d5.js.map +1 -0
  153. package/.next/server/chunks/ssr/e8449_next_dist_client_components_d90ace34._.js +3 -0
  154. package/.next/server/chunks/ssr/e8449_next_dist_client_components_d90ace34._.js.map +1 -0
  155. package/.next/server/chunks/ssr/e8449_next_dist_esm_af7aafb8._.js +6 -0
  156. package/.next/server/chunks/ssr/e8449_next_dist_esm_af7aafb8._.js.map +1 -0
  157. package/.next/server/chunks/ssr/e8449_next_dist_esm_build_templates_app-page_f16f0848.js +4 -0
  158. package/.next/server/chunks/ssr/e8449_next_dist_esm_build_templates_app-page_f16f0848.js.map +1 -0
  159. package/.next/server/chunks/ssr/e8449_next_dist_f0d8a2cc._.js +4 -0
  160. package/.next/server/chunks/ssr/e8449_next_dist_f0d8a2cc._.js.map +1 -0
  161. package/.next/server/chunks/ssr/node_modules__pnpm_37510d96._.js +5 -0
  162. package/.next/server/chunks/ssr/node_modules__pnpm_37510d96._.js.map +1 -0
  163. package/.next/server/chunks/ssr/packages_preview-server_src_app_897ecf1c._.js +3 -0
  164. package/.next/server/chunks/ssr/packages_preview-server_src_app_897ecf1c._.js.map +1 -0
  165. package/.next/server/chunks/ssr/packages_preview-server_src_e17a2a9b._.js +3 -0
  166. package/.next/server/chunks/ssr/packages_preview-server_src_e17a2a9b._.js.map +1 -0
  167. package/.next/server/functions-config-manifest.json +4 -0
  168. package/.next/server/interception-route-rewrite-manifest.js +1 -0
  169. package/.next/server/middleware-build-manifest.js +21 -0
  170. package/.next/server/middleware-manifest.json +6 -0
  171. package/.next/server/next-font-manifest.js +1 -0
  172. package/.next/server/next-font-manifest.json +34 -0
  173. package/.next/server/pages/500.html +2 -0
  174. package/.next/server/pages-manifest.json +3 -0
  175. package/.next/server/server-reference-manifest.js +1 -0
  176. package/.next/server/server-reference-manifest.json +80 -0
  177. package/.next/static/cdYIhKFtJ0GB-yJK5ywz_/_buildManifest.js +11 -0
  178. package/.next/static/cdYIhKFtJ0GB-yJK5ywz_/_clientMiddlewareManifest.json +1 -0
  179. package/.next/static/cdYIhKFtJ0GB-yJK5ywz_/_ssgManifest.js +1 -0
  180. package/.next/static/chunks/0464c7ff175ee6ff.js +1 -0
  181. package/.next/static/chunks/10468413db24762a.js +14 -0
  182. package/.next/static/chunks/2de338262e51ef94.js +3 -0
  183. package/.next/static/chunks/37adc260f85da877.js +1 -0
  184. package/.next/static/chunks/521eee9903bc4d1f.js +5 -0
  185. package/.next/static/chunks/6051bd38272cb442.js +14 -0
  186. package/.next/static/chunks/8d433d4b9d701456.js +1 -0
  187. package/.next/static/chunks/92ba72595aad9df6.js +1 -0
  188. package/.next/static/chunks/959ed978a6e89a66.js +1 -0
  189. package/.next/static/chunks/a6dad97d9634a72d.js +1 -0
  190. package/.next/static/chunks/a6dad97d9634a72d.js.map +1 -0
  191. package/.next/static/chunks/a6db6456c5b75734.js +1 -0
  192. package/.next/static/chunks/c3ece0a7e3e07076.js +1 -0
  193. package/.next/static/chunks/d1b29a74f6814a03.css +3 -0
  194. package/.next/static/chunks/f7ec22614fe1c1fe.js +1 -0
  195. package/.next/static/chunks/turbopack-e062a8e6d6034eb0.js +4 -0
  196. package/.next/static/media/1bffadaabf893a1e-s.7cd81963.woff2 +0 -0
  197. package/.next/static/media/2bbe8d2671613f1f-s.76dcb0b2.woff2 +0 -0
  198. package/.next/static/media/2c55a0e60120577a-s.2a48534a.woff2 +0 -0
  199. package/.next/static/media/5476f68d60460930-s.c995e352.woff2 +0 -0
  200. package/.next/static/media/83afe278b6a6bb3c-s.p.3a6ba036.woff2 +0 -0
  201. package/.next/static/media/9c72aa0f40e4eef8-s.18a48cbc.woff2 +0 -0
  202. package/.next/static/media/SFMonoBold-s.p.b90ec775.otf +0 -0
  203. package/.next/static/media/SFMonoHeavy-s.p.545fe93b.otf +0 -0
  204. package/.next/static/media/SFMonoLight-s.p.7c5363a6.otf +0 -0
  205. package/.next/static/media/SFMonoMedium-s.p.a4fc9904.otf +0 -0
  206. package/.next/static/media/SFMonoRegular-s.p.04ea7bf3.otf +0 -0
  207. package/.next/static/media/SFMonoSemibold-s.p.2d2ddb43.otf +0 -0
  208. package/.next/static/media/ad66f9afd8947f86-s.7a40eb73.woff2 +0 -0
  209. package/.next/static/media/favicon.678eb597.ico +0 -0
  210. package/.next/static/media/logo.22a370b0.png +0 -0
  211. package/.next/trace +1 -0
  212. package/.next/trace-build +1 -0
  213. package/.next/turbopack +0 -0
  214. package/.next/types/routes.d.ts +58 -0
  215. package/.next/types/validator.ts +70 -0
  216. package/CHANGELOG.md +12 -0
  217. package/LICENSE.md +8 -0
  218. package/index.mjs +17 -0
  219. package/jsx-runtime/jsx-dev-runtime.js +26 -0
  220. package/module-punycode.d.ts +3 -0
  221. package/next-env.d.ts +6 -0
  222. package/next.config.mjs +15 -0
  223. package/package.json +78 -0
  224. package/postcss.config.js +5 -0
  225. package/readme.md +33 -0
  226. package/scripts/build-preview-server.mts +25 -0
  227. package/scripts/dev.mts +57 -0
  228. package/scripts/seed.mts +36 -0
  229. package/src/actions/export-single-template.ts +74 -0
  230. package/src/actions/get-template-path-from-slug.ts +32 -0
  231. package/src/actions/get-templates-directory-metadata-action.ts +19 -0
  232. package/src/actions/render-template-by-path.tsx +313 -0
  233. package/src/actions/safe-action.ts +15 -0
  234. package/src/animated-icons-data/help.json +1082 -0
  235. package/src/animated-icons-data/link.json +1309 -0
  236. package/src/animated-icons-data/load.json +443 -0
  237. package/src/animated-icons-data/mail.json +1320 -0
  238. package/src/app/env.ts +14 -0
  239. package/src/app/favicon.ico +0 -0
  240. package/src/app/fonts/SFMono/SFMonoBold.otf +0 -0
  241. package/src/app/fonts/SFMono/SFMonoBoldItalic.otf +0 -0
  242. package/src/app/fonts/SFMono/SFMonoHeavy.otf +0 -0
  243. package/src/app/fonts/SFMono/SFMonoHeavyItalic.otf +0 -0
  244. package/src/app/fonts/SFMono/SFMonoLight.otf +0 -0
  245. package/src/app/fonts/SFMono/SFMonoLightItalic.otf +0 -0
  246. package/src/app/fonts/SFMono/SFMonoMedium.otf +0 -0
  247. package/src/app/fonts/SFMono/SFMonoMediumItalic.otf +0 -0
  248. package/src/app/fonts/SFMono/SFMonoRegular.otf +0 -0
  249. package/src/app/fonts/SFMono/SFMonoRegularItalic.otf +0 -0
  250. package/src/app/fonts/SFMono/SFMonoSemibold.otf +0 -0
  251. package/src/app/fonts/SFMono/SFMonoSemiboldItalic.otf +0 -0
  252. package/src/app/fonts.ts +39 -0
  253. package/src/app/globals.css +136 -0
  254. package/src/app/layout.tsx +46 -0
  255. package/src/app/logo.png +0 -0
  256. package/src/app/page.tsx +52 -0
  257. package/src/app/preview/[...slug]/download-button.tsx +138 -0
  258. package/src/app/preview/[...slug]/error-overlay.tsx +58 -0
  259. package/src/app/preview/[...slug]/page.tsx +90 -0
  260. package/src/app/preview/[...slug]/preview.tsx +249 -0
  261. package/src/app/preview/[...slug]/template-frame.tsx +68 -0
  262. package/src/components/button.tsx +101 -0
  263. package/src/components/code-container.tsx +169 -0
  264. package/src/components/code-snippet.tsx +9 -0
  265. package/src/components/code.tsx +185 -0
  266. package/src/components/heading.tsx +113 -0
  267. package/src/components/icons/icon-arrow-down.tsx +16 -0
  268. package/src/components/icons/icon-base.tsx +26 -0
  269. package/src/components/icons/icon-bug.tsx +19 -0
  270. package/src/components/icons/icon-button.tsx +23 -0
  271. package/src/components/icons/icon-check.tsx +19 -0
  272. package/src/components/icons/icon-clipboard.tsx +40 -0
  273. package/src/components/icons/icon-cloud-alert.tsx +18 -0
  274. package/src/components/icons/icon-cloud-check.tsx +17 -0
  275. package/src/components/icons/icon-download.tsx +19 -0
  276. package/src/components/icons/icon-file.tsx +19 -0
  277. package/src/components/icons/icon-folder-open.tsx +19 -0
  278. package/src/components/icons/icon-folder.tsx +18 -0
  279. package/src/components/icons/icon-hide-sidebar.tsx +23 -0
  280. package/src/components/icons/icon-image.tsx +19 -0
  281. package/src/components/icons/icon-info.tsx +18 -0
  282. package/src/components/icons/icon-link.tsx +14 -0
  283. package/src/components/icons/icon-loader.tsx +16 -0
  284. package/src/components/icons/icon-monitor.tsx +19 -0
  285. package/src/components/icons/icon-moon.tsx +16 -0
  286. package/src/components/icons/icon-phone.tsx +26 -0
  287. package/src/components/icons/icon-reload.tsx +18 -0
  288. package/src/components/icons/icon-source.tsx +19 -0
  289. package/src/components/icons/icon-stamp.tsx +14 -0
  290. package/src/components/icons/icon-sun.tsx +74 -0
  291. package/src/components/icons/icon-warning.tsx +31 -0
  292. package/src/components/index.ts +7 -0
  293. package/src/components/logo.tsx +41 -0
  294. package/src/components/resizable-wrapper.tsx +269 -0
  295. package/src/components/shell.tsx +95 -0
  296. package/src/components/sidebar/file-tree-directory-children.tsx +142 -0
  297. package/src/components/sidebar/file-tree-directory.tsx +92 -0
  298. package/src/components/sidebar/file-tree.tsx +31 -0
  299. package/src/components/sidebar/index.ts +1 -0
  300. package/src/components/sidebar/sidebar.tsx +46 -0
  301. package/src/components/text.tsx +99 -0
  302. package/src/components/toolbar/checking-results.tsx +150 -0
  303. package/src/components/toolbar/code-preview-line-link.tsx +39 -0
  304. package/src/components/toolbar/results-table.tsx +0 -0
  305. package/src/components/toolbar/results.tsx +52 -0
  306. package/src/components/toolbar/toolbar-button.tsx +52 -0
  307. package/src/components/toolbar/use-cached-state.ts +36 -0
  308. package/src/components/toolbar.tsx +182 -0
  309. package/src/components/tooltip-content.tsx +31 -0
  310. package/src/components/tooltip.tsx +19 -0
  311. package/src/components/topbar/active-view-toggle-group.tsx +86 -0
  312. package/src/components/topbar/emulated-dark-mode-toggle.tsx +58 -0
  313. package/src/components/topbar/view-size-controls.tsx +173 -0
  314. package/src/components/topbar.tsx +59 -0
  315. package/src/contexts/preview.tsx +91 -0
  316. package/src/contexts/templates.tsx +57 -0
  317. package/src/contexts/toolbar.tsx +22 -0
  318. package/src/hooks/use-clamped-state.ts +24 -0
  319. package/src/hooks/use-fragment-identifier.ts +14 -0
  320. package/src/hooks/use-hot-reload.ts +31 -0
  321. package/src/hooks/use-rendering-metadata.ts +37 -0
  322. package/src/hooks/use-template-rendering-result.ts +58 -0
  323. package/src/utils/cn.ts +6 -0
  324. package/src/utils/constants.ts +8 -0
  325. package/src/utils/contains-template.ts +52 -0
  326. package/src/utils/convert-stack-with-sourcemap.ts +79 -0
  327. package/src/utils/copy-text-to-clipboard.ts +7 -0
  328. package/src/utils/create-jsx-runtime.ts +47 -0
  329. package/src/utils/esbuild/escape-string-for-regex.ts +3 -0
  330. package/src/utils/esbuild/renderring-utilities-exporter.ts +64 -0
  331. package/src/utils/get-line-and-column-from-offset.ts +11 -0
  332. package/src/utils/get-template-component.ts +158 -0
  333. package/src/utils/get-templates-directory-metadata.ts +195 -0
  334. package/src/utils/index.ts +6 -0
  335. package/src/utils/language-map.ts +7 -0
  336. package/src/utils/load-stream.ts +15 -0
  337. package/src/utils/register-spinner-autostopping.ts +28 -0
  338. package/src/utils/result.ts +49 -0
  339. package/src/utils/run-bundled-code.ts +86 -0
  340. package/src/utils/sanitize.ts +6 -0
  341. package/src/utils/sleep.ts +3 -0
  342. package/src/utils/snake-to-camel.ts +5 -0
  343. package/src/utils/static-node-modules-for-vm.ts +93 -0
  344. package/src/utils/style-text.ts +11 -0
  345. package/src/utils/types/as.ts +26 -0
  346. package/src/utils/types/error-object.ts +11 -0
  347. package/src/utils/types/hot-reload-change.ts +13 -0
  348. package/src/utils/types/template.ts +8 -0
  349. package/src/utils/unreachable.ts +8 -0
  350. package/templates/.gitkeep +0 -0
  351. package/tsconfig.json +47 -0
@@ -0,0 +1,249 @@
1
+ 'use client';
2
+
3
+ import { usePathname, useRouter, useSearchParams } from 'next/navigation';
4
+ import { useState } from 'react';
5
+ import { flushSync } from 'react-dom';
6
+ import { Toaster } from 'sonner';
7
+ import { useDebouncedCallback } from 'use-debounce';
8
+ import { Topbar } from '../../../components';
9
+ import { CodeContainer } from '../../../components/code-container';
10
+ import {
11
+ makeIframeDocumentBubbleEvents,
12
+ ResizableWrapper,
13
+ } from '../../../components/resizable-wrapper';
14
+ import { useToolbarState } from '../../../components/toolbar';
15
+ import { Tooltip } from '../../../components/tooltip';
16
+ import { ActiveViewToggleGroup } from '../../../components/topbar/active-view-toggle-group';
17
+ import { EmulatedDarkModeToggle } from '../../../components/topbar/emulated-dark-mode-toggle';
18
+ import { ViewSizeControls } from '../../../components/topbar/view-size-controls';
19
+ import { usePreviewContext } from '../../../contexts/preview';
20
+ import { useClampedState } from '../../../hooks/use-clamped-state';
21
+ import { cn } from '../../../utils';
22
+ import { DownloadButton } from './download-button';
23
+ import { ErrorOverlay } from './error-overlay';
24
+ import { TemplateFrame } from './template-frame';
25
+
26
+ interface PreviewProps extends React.ComponentProps<'div'> {
27
+ templateTitle: string;
28
+ }
29
+
30
+ const Preview = ({ templateTitle, className, ...props }: PreviewProps) => {
31
+ const { renderingResult, renderedTemplateMetadata } = usePreviewContext();
32
+
33
+ const router = useRouter();
34
+ const pathname = usePathname();
35
+ const searchParams = useSearchParams();
36
+
37
+ const isDarkModeEnabled = searchParams.get('dark') !== null;
38
+ const activeView = searchParams.get('view') ?? 'preview';
39
+ const activeLang = searchParams.get('lang') ?? 'tsx';
40
+
41
+ const handleDarkModeChange = (enabled: boolean) => {
42
+ const params = new URLSearchParams(searchParams);
43
+ if (enabled) {
44
+ params.set('dark', '');
45
+ } else {
46
+ params.delete('dark');
47
+ }
48
+ router.push(`${pathname}?${params.toString()}${location.hash}`);
49
+ };
50
+
51
+ const handleViewChange = (view: string) => {
52
+ const params = new URLSearchParams(searchParams);
53
+ params.set('view', view);
54
+ router.push(`${pathname}?${params.toString()}${location.hash}`);
55
+ };
56
+
57
+ const handleLangChange = (lang: string) => {
58
+ const params = new URLSearchParams(searchParams);
59
+ params.set('view', 'source');
60
+ params.set('lang', lang);
61
+ const isSameLang = searchParams.get('lang') === lang;
62
+ router.push(
63
+ `${pathname}?${params.toString()}${isSameLang ? location.hash : ''}`,
64
+ );
65
+ };
66
+
67
+ const hasRenderingMetadata = typeof renderedTemplateMetadata !== 'undefined';
68
+ const hasErrors = 'error' in renderingResult;
69
+
70
+ const [maxWidth, setMaxWidth] = useState(Number.POSITIVE_INFINITY);
71
+ const [maxHeight, setMaxHeight] = useState(Number.POSITIVE_INFINITY);
72
+ const minWidth = 220;
73
+ const minHeight = minWidth * 1.6;
74
+ const storedWidth = searchParams.get('width');
75
+ const storedHeight = searchParams.get('height');
76
+ const A4_WIDTH = 794;
77
+ const A4_HEIGHT = 1123;
78
+
79
+ const [width, setWidth] = useClampedState(
80
+ storedWidth ? Number.parseInt(storedWidth, 10) : A4_WIDTH,
81
+ minWidth,
82
+ maxWidth,
83
+ );
84
+ const [height, setHeight] = useClampedState(
85
+ storedHeight ? Number.parseInt(storedHeight, 10) : A4_HEIGHT,
86
+ minHeight,
87
+ maxHeight,
88
+ );
89
+
90
+ const handleSaveViewSize = useDebouncedCallback(() => {
91
+ const params = new URLSearchParams(searchParams);
92
+ params.set('width', width.toString());
93
+ params.set('height', height.toString());
94
+ router.push(`${pathname}?${params.toString()}${location.hash}`);
95
+ }, 300);
96
+
97
+ const { toggled: toolbarToggled } = useToolbarState();
98
+
99
+ return (
100
+ <>
101
+ <Topbar templateTitle={templateTitle}>
102
+ {activeView === 'preview' ? (
103
+ <>
104
+ <EmulatedDarkModeToggle
105
+ enabled={isDarkModeEnabled}
106
+ onChange={(enabled) => handleDarkModeChange(enabled)}
107
+ />
108
+ <ViewSizeControls
109
+ setViewHeight={(height) => {
110
+ setHeight(height);
111
+ flushSync(() => {
112
+ handleSaveViewSize();
113
+ });
114
+ }}
115
+ setViewWidth={(width) => {
116
+ setWidth(width);
117
+ flushSync(() => {
118
+ handleSaveViewSize();
119
+ });
120
+ }}
121
+ viewHeight={height}
122
+ viewWidth={width}
123
+ minWidth={minWidth}
124
+ minHeight={minHeight}
125
+ />
126
+ </>
127
+ ) : null}
128
+ <ActiveViewToggleGroup
129
+ activeView={activeView}
130
+ setActiveView={handleViewChange}
131
+ />
132
+ {hasRenderingMetadata ? (
133
+ <div className="flex justify-end">
134
+ <DownloadButton
135
+ templateSlug={templateTitle}
136
+ htmlMarkup={renderedTemplateMetadata.markup}
137
+ darkMode={isDarkModeEnabled}
138
+ />
139
+ </div>
140
+ ) : null}
141
+ </Topbar>
142
+
143
+ <div
144
+ {...props}
145
+ className={cn(
146
+ 'h-[calc(100%-3.5rem-2.375rem)] will-change-[height] flex p-4 transition-[height] duration-300 relative',
147
+ activeView === 'preview' && 'bg-gray-200',
148
+ activeView === 'preview' && isDarkModeEnabled && 'bg-gray-400',
149
+ toolbarToggled && 'h-[calc(100%-3.5rem-13rem)]',
150
+ className,
151
+ )}
152
+ ref={(element) => {
153
+ const observer = new ResizeObserver((entry) => {
154
+ const [elementEntry] = entry;
155
+ if (elementEntry) {
156
+ setMaxWidth(elementEntry.contentRect.width);
157
+ setMaxHeight(elementEntry.contentRect.height);
158
+ }
159
+ });
160
+
161
+ if (element) {
162
+ observer.observe(element);
163
+ }
164
+
165
+ return () => {
166
+ observer.disconnect();
167
+ };
168
+ }}
169
+ >
170
+ {hasErrors ? <ErrorOverlay error={renderingResult.error} /> : null}
171
+
172
+ {hasRenderingMetadata ? (
173
+ <>
174
+ {activeView === 'preview' && (
175
+ <ResizableWrapper
176
+ minHeight={minHeight}
177
+ minWidth={minWidth}
178
+ maxHeight={maxHeight}
179
+ maxWidth={maxWidth}
180
+ height={height}
181
+ onResizeEnd={() => {
182
+ handleSaveViewSize();
183
+ }}
184
+ onResize={(value, direction) => {
185
+ const isHorizontal =
186
+ direction === 'east' || direction === 'west';
187
+ if (isHorizontal) {
188
+ setWidth(Math.round(value));
189
+ } else {
190
+ setHeight(Math.round(value));
191
+ }
192
+ }}
193
+ width={width}
194
+ >
195
+ <TemplateFrame
196
+ className="max-h-full rounded-lg bg-white [color-scheme:auto]"
197
+ darkMode={isDarkModeEnabled}
198
+ markup={renderedTemplateMetadata.markup}
199
+ width={width}
200
+ height={height}
201
+ title={templateTitle}
202
+ ref={(iframe) => {
203
+ if (!iframe) return;
204
+
205
+ return makeIframeDocumentBubbleEvents(iframe);
206
+ }}
207
+ />
208
+ </ResizableWrapper>
209
+ )}
210
+
211
+ {activeView === 'source' && (
212
+ <div className="h-full w-full">
213
+ <div className="m-auto h-full flex max-w-3xl p-6">
214
+ <Tooltip.Provider>
215
+ <CodeContainer
216
+ activeLang={activeLang}
217
+ basename={renderedTemplateMetadata.basename}
218
+ markups={[
219
+ {
220
+ language: 'tsx',
221
+ extension: renderedTemplateMetadata.extname,
222
+ content: renderedTemplateMetadata.reactMarkup,
223
+ },
224
+ {
225
+ language: 'html',
226
+ content: renderedTemplateMetadata.prettyMarkup,
227
+ },
228
+ {
229
+ language: 'markdown',
230
+ extension: 'md',
231
+ content: renderedTemplateMetadata.plainText,
232
+ },
233
+ ]}
234
+ setActiveLang={handleLangChange}
235
+ />
236
+ </Tooltip.Provider>
237
+ </div>
238
+ </div>
239
+ )}
240
+ </>
241
+ ) : null}
242
+
243
+ <Toaster />
244
+ </div>
245
+ </>
246
+ );
247
+ };
248
+
249
+ export default Preview;
@@ -0,0 +1,68 @@
1
+ import type { ComponentProps } from 'react';
2
+ import { forwardRef, useEffect, useRef } from 'react';
3
+
4
+ interface TemplateFrameProps extends ComponentProps<'iframe'> {
5
+ markup: string;
6
+ width: number;
7
+ height: number;
8
+ darkMode?: boolean;
9
+ }
10
+
11
+ const applyDarkMode = (contentDocument: Document, darkMode: boolean) => {
12
+ // Add or remove 'dark' class on the body/html element
13
+ // Templates can use Theme component with variant="dark" to define CSS variables
14
+ // Example: <Theme variant="dark" css={`--color-bg: #1a1a1a;`} />
15
+ if (darkMode) {
16
+ contentDocument.documentElement.classList.add('dark');
17
+ contentDocument.body.classList.add('dark');
18
+ } else {
19
+ contentDocument.documentElement.classList.remove('dark');
20
+ contentDocument.body.classList.remove('dark');
21
+ }
22
+ };
23
+
24
+ export const TemplateFrame = forwardRef<HTMLIFrameElement, TemplateFrameProps>(
25
+ function TemplateFrame(
26
+ { markup, width, height, darkMode = false, ...rest },
27
+ ref,
28
+ ) {
29
+ const iframeRef = useRef<HTMLIFrameElement>(null);
30
+
31
+ // Merge refs: both internal ref and forwarded ref
32
+ const setRefs = (element: HTMLIFrameElement | null) => {
33
+ iframeRef.current = element;
34
+ if (typeof ref === 'function') {
35
+ ref(element);
36
+ } else if (ref) {
37
+ ref.current = element;
38
+ }
39
+ };
40
+
41
+ useEffect(() => {
42
+ const iframe = iframeRef.current;
43
+ if (!iframe) return;
44
+
45
+ const { contentDocument } = iframe;
46
+ if (!contentDocument) return;
47
+
48
+ applyDarkMode(contentDocument, darkMode);
49
+ }, [darkMode, markup]);
50
+
51
+ return (
52
+ <iframe
53
+ ref={setRefs}
54
+ srcDoc={markup}
55
+ width={width}
56
+ height={height}
57
+ onLoad={(event) => {
58
+ const iframe = event.currentTarget;
59
+ const { contentDocument } = iframe;
60
+ if (!contentDocument) return;
61
+
62
+ applyDarkMode(contentDocument, darkMode);
63
+ }}
64
+ {...rest}
65
+ />
66
+ );
67
+ },
68
+ );
@@ -0,0 +1,101 @@
1
+ 'use client';
2
+ import { DotLottieReact } from '@lottiefiles/dotlottie-react';
3
+ import * as SlotPrimitive from '@radix-ui/react-slot';
4
+ import type * as React from 'react';
5
+ import animatedLoadIcon from '../animated-icons-data/load.json';
6
+ import { cn } from '../utils/cn';
7
+ import { unreachable } from '../utils/unreachable';
8
+
9
+ type RootProps = React.ComponentProps<'button'>;
10
+
11
+ type Appearance = 'white' | 'gradient';
12
+ type Size = '1' | '2' | '3' | '4';
13
+
14
+ interface ButtonProps extends RootProps {
15
+ asChild?: boolean;
16
+ appearance?: Appearance;
17
+ size?: Size;
18
+ loading?: boolean;
19
+ }
20
+
21
+ export const Button = ({
22
+ asChild,
23
+ appearance = 'white',
24
+ className,
25
+ children,
26
+ size = '2',
27
+ loading,
28
+ ref,
29
+ ...props
30
+ }: ButtonProps) => {
31
+ const Root = asChild ? SlotPrimitive.Slot : 'button';
32
+
33
+ return (
34
+ <Root
35
+ ref={ref}
36
+ type="button"
37
+ {...props}
38
+ className={cn(
39
+ getSize(size),
40
+ getAppearance(appearance),
41
+ 'inline-flex items-center justify-center gap-2 border font-medium',
42
+ className,
43
+ )}
44
+ aria-disabled={loading}
45
+ >
46
+ <span
47
+ className={cn(
48
+ '-ml-7 opacity-0 transition-opacity duration-200',
49
+ loading && 'opacity-100',
50
+ )}
51
+ >
52
+ <DotLottieReact
53
+ data={animatedLoadIcon}
54
+ autoplay={false}
55
+ className="h-5 w-5"
56
+ loop={true}
57
+ />
58
+ </span>
59
+ <SlotPrimitive.Slottable>{children}</SlotPrimitive.Slottable>
60
+ </Root>
61
+ );
62
+ };
63
+
64
+ Button.displayName = 'Button';
65
+
66
+ const getAppearance = (appearance: Appearance | undefined) => {
67
+ switch (appearance) {
68
+ case undefined:
69
+ case 'white':
70
+ return [
71
+ 'border-white bg-white text-black transition-colors duration-200 ease-in-out',
72
+ 'hover:bg-white/90',
73
+ 'focus:bg-white/90 focus:outline-hidden focus:ring-2 focus:ring-white/20',
74
+ 'mt-2 mb-2 aria-disabled:border-transparent aria-disabled:bg-slate-11',
75
+ ];
76
+ case 'gradient':
77
+ return [
78
+ 'bg-gradient border-[#34343A] backdrop-blur-[1.25rem]',
79
+ 'hover:bg-gradient-hover',
80
+ 'focus:bg-gradient-hover focus:outline-hidden focus:ring-2 focus:ring-white/20',
81
+ ];
82
+ default:
83
+ unreachable(appearance);
84
+ }
85
+ };
86
+
87
+ const getSize = (size: Size | undefined) => {
88
+ switch (size) {
89
+ case '1':
90
+ return '';
91
+ case undefined:
92
+ case '2':
93
+ return 'text-[.875rem] h-8 px-3 rounded-md gap-2';
94
+ case '3':
95
+ return 'text-[.875rem] h-10 px-4 rounded-md gap-2';
96
+ case '4':
97
+ return 'text-base h-11 px-4 rounded-md gap-2';
98
+ default:
99
+ unreachable(size);
100
+ }
101
+ };
@@ -0,0 +1,169 @@
1
+ import { LayoutGroup, motion } from 'framer-motion';
2
+ import type { Language } from 'prism-react-renderer';
3
+ import * as React from 'react';
4
+ import { copyTextToClipboard } from '../utils';
5
+ import { tabTransition } from '../utils/constants';
6
+ import languageMap from '../utils/language-map';
7
+ import { Code } from './code';
8
+ import { IconButton } from './icons/icon-button';
9
+ import { IconCheck } from './icons/icon-check';
10
+ import { IconClipboard } from './icons/icon-clipboard';
11
+ import { IconDownload } from './icons/icon-download';
12
+ import { Tooltip } from './tooltip';
13
+
14
+ interface CodeContainerProps {
15
+ markups: MarkupProps[];
16
+ basename: string;
17
+ activeLang: string;
18
+ setActiveLang: (lang: string) => void;
19
+ }
20
+
21
+ interface MarkupProps {
22
+ language: Language;
23
+ extension?: string;
24
+ content: string;
25
+ }
26
+
27
+ export const CodeContainer: React.FC<Readonly<CodeContainerProps>> = ({
28
+ markups,
29
+ basename: filename,
30
+ activeLang,
31
+ setActiveLang,
32
+ }) => {
33
+ const activeMarkup = markups.find(({ language }) => activeLang === language);
34
+ if (!activeMarkup) {
35
+ throw new Error('No markup found for the active language!', {
36
+ cause: {
37
+ activeLang,
38
+ markups,
39
+ },
40
+ });
41
+ }
42
+
43
+ const codeId = React.useId();
44
+
45
+ return (
46
+ <div
47
+ className="relative max-h-[650px] w-full h-full whitespace-pre rounded-md border border-slate-6 text-sm"
48
+ style={{
49
+ lineHeight: '130%',
50
+ background:
51
+ 'linear-gradient(145.37deg, rgba(255, 255, 255, 0.09) -8.75%, rgba(255, 255, 255, 0.027) 83.95%)',
52
+ boxShadow: 'rgb(0 0 0 / 10%) 0px 5px 30px -5px',
53
+ }}
54
+ >
55
+ <div className="h-9 border-b border-slate-6">
56
+ <div className="flex">
57
+ <LayoutGroup id={codeId}>
58
+ {markups.map(({ language }) => {
59
+ const isCurrentLang = activeLang === language;
60
+ return (
61
+ <motion.button
62
+ className={`relative px-4 py-[8px] font-sans text-sm font-medium transition duration-200 ease-in-out hover:text-slate-12 ${
63
+ activeLang !== language ? 'text-slate-11' : 'text-slate-12'
64
+ }`}
65
+ key={language}
66
+ onClick={() => {
67
+ setActiveLang(language);
68
+ }}
69
+ >
70
+ {isCurrentLang ? (
71
+ <motion.span
72
+ animate={{ opacity: 1 }}
73
+ className="absolute bottom-0 left-0 right-0 top-0 bg-slate-4"
74
+ exit={{ opacity: 0 }}
75
+ initial={{ opacity: 0 }}
76
+ layoutId="code"
77
+ transition={tabTransition}
78
+ />
79
+ ) : null}
80
+ {languageMap[language]}
81
+ </motion.button>
82
+ );
83
+ })}
84
+ </LayoutGroup>
85
+ </div>
86
+ <CopyToClipboardButton content={activeMarkup.content} />
87
+ <DownloadButton
88
+ content={activeMarkup.content}
89
+ filename={`${filename}.${activeMarkup.extension || activeMarkup.language}`}
90
+ />
91
+ </div>
92
+ <div className="h-[calc(100%-2.25rem)]">
93
+ <Code language={activeLang}>{activeMarkup.content}</Code>
94
+ </div>
95
+ </div>
96
+ );
97
+ };
98
+
99
+ interface CopyToClipboardButtonProps {
100
+ content: string;
101
+ }
102
+
103
+ const CopyToClipboardButton = ({ content }: CopyToClipboardButtonProps) => {
104
+ const [isCopied, setIsCopied] = React.useState(false);
105
+
106
+ const unsetIsCopiedTimeout = React.useRef<NodeJS.Timeout>(undefined);
107
+ React.useEffect(() => {
108
+ setIsCopied(false);
109
+ clearTimeout(unsetIsCopiedTimeout.current);
110
+ unsetIsCopiedTimeout.current = undefined;
111
+ }, [content]);
112
+
113
+ return (
114
+ <Tooltip>
115
+ <Tooltip.Trigger
116
+ asChild
117
+ className="absolute right-2 top-2 hidden md:block"
118
+ >
119
+ <IconButton
120
+ onClick={async () => {
121
+ setIsCopied(true);
122
+ await copyTextToClipboard(content);
123
+ unsetIsCopiedTimeout.current = setTimeout(() => {
124
+ setIsCopied(false);
125
+ }, 3000);
126
+ }}
127
+ >
128
+ {isCopied ? <IconCheck /> : <IconClipboard />}
129
+ </IconButton>
130
+ </Tooltip.Trigger>
131
+ <Tooltip.Content>Copy to Clipboard</Tooltip.Content>
132
+ </Tooltip>
133
+ );
134
+ };
135
+
136
+ interface DownloadButtonProps {
137
+ content: string;
138
+ filename: string;
139
+ }
140
+
141
+ const DownloadButton = ({ content, filename }: DownloadButtonProps) => {
142
+ const generatedUrl = React.useMemo(() => {
143
+ const file = new File([content], filename);
144
+ return URL.createObjectURL(file);
145
+ }, [content, filename]);
146
+ const url = React.useSyncExternalStore(
147
+ () => () => {},
148
+ () => generatedUrl,
149
+ () => undefined,
150
+ );
151
+
152
+ return (
153
+ <Tooltip>
154
+ <Tooltip.Trigger
155
+ asChild
156
+ className="text-gray-11 absolute right-8 top-2 hidden md:block"
157
+ >
158
+ <a
159
+ className="text-slate-11 transition duration-200 ease-in-out hover:text-slate-12"
160
+ download={filename}
161
+ href={url}
162
+ >
163
+ <IconDownload />
164
+ </a>
165
+ </Tooltip.Trigger>
166
+ <Tooltip.Content>Download</Tooltip.Content>
167
+ </Tooltip>
168
+ );
169
+ };
@@ -0,0 +1,9 @@
1
+ const CodeSnippet = ({ children }) => {
2
+ return (
3
+ <code className="m-0.5 inline-block rounded-md bg-white/10 p-1 font-mono leading-none text-slate-12">
4
+ {children}
5
+ </code>
6
+ );
7
+ };
8
+
9
+ export default CodeSnippet;