@tscircuit/fake-snippets 0.0.1

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 (396) hide show
  1. package/.github/CODEOWNERS +1 -0
  2. package/.github/workflows/bun-formatcheck.yml +26 -0
  3. package/.github/workflows/bun-test.yml +28 -0
  4. package/.github/workflows/bun-typecheck.yml +26 -0
  5. package/.github/workflows/bundle-size-analysis.yml +79 -0
  6. package/.github/workflows/playwright-test.yml +37 -0
  7. package/.github/workflows/stale.yml +40 -0
  8. package/.github/workflows/update-snapshots.yml +49 -0
  9. package/CONTRIBUTING.md +59 -0
  10. package/LICENSE +21 -0
  11. package/README.md +113 -0
  12. package/biome.json +60 -0
  13. package/bun-tests/fake-snippets-api/fixtures/get-circuit-json.ts +148 -0
  14. package/bun-tests/fake-snippets-api/fixtures/get-test-server.ts +96 -0
  15. package/bun-tests/fake-snippets-api/fixtures/start-server.ts +44 -0
  16. package/bun-tests/fake-snippets-api/routes/accounts/get_account_balance.test.ts +18 -0
  17. package/bun-tests/fake-snippets-api/routes/health.test.ts +9 -0
  18. package/bun-tests/fake-snippets-api/routes/order_files/get.test.ts +48 -0
  19. package/bun-tests/fake-snippets-api/routes/order_files/upload.test.ts +77 -0
  20. package/bun-tests/fake-snippets-api/routes/orders/create.test.ts +19 -0
  21. package/bun-tests/fake-snippets-api/routes/orders/get.test.ts +38 -0
  22. package/bun-tests/fake-snippets-api/routes/orders/list.test.ts +30 -0
  23. package/bun-tests/fake-snippets-api/routes/orders/update.test.ts +46 -0
  24. package/bun-tests/fake-snippets-api/routes/snippets/add_star.test.ts +114 -0
  25. package/bun-tests/fake-snippets-api/routes/snippets/create.test.ts +28 -0
  26. package/bun-tests/fake-snippets-api/routes/snippets/delete.test.ts +106 -0
  27. package/bun-tests/fake-snippets-api/routes/snippets/download.test.ts +90 -0
  28. package/bun-tests/fake-snippets-api/routes/snippets/generate_from_jlcpcb.test.ts +25 -0
  29. package/bun-tests/fake-snippets-api/routes/snippets/get_image.test.ts +113 -0
  30. package/bun-tests/fake-snippets-api/routes/snippets/images.test.ts +112 -0
  31. package/bun-tests/fake-snippets-api/routes/snippets/list.test.ts +62 -0
  32. package/bun-tests/fake-snippets-api/routes/snippets/list_newest.test.ts +48 -0
  33. package/bun-tests/fake-snippets-api/routes/snippets/list_trending.test.ts +69 -0
  34. package/bun-tests/fake-snippets-api/routes/snippets/remove_star.test.ts +110 -0
  35. package/bun-tests/fake-snippets-api/routes/snippets/search.test.ts +75 -0
  36. package/bun-tests/fake-snippets-api/routes/snippets/star-count.test.ts +44 -0
  37. package/bun-tests/fake-snippets-api/routes/snippets/update.test.ts +116 -0
  38. package/bun-tests/parts-engine.test.ts +18 -0
  39. package/bun.lockb +0 -0
  40. package/bunfig.toml +2 -0
  41. package/components.json +20 -0
  42. package/dist/assets/editor_example_1-1000w.webp +0 -0
  43. package/dist/assets/editor_example_1-1200w.webp +0 -0
  44. package/dist/assets/editor_example_1-1600w.webp +0 -0
  45. package/dist/assets/editor_example_1-2000w.webp +0 -0
  46. package/dist/assets/editor_example_1-400w.webp +0 -0
  47. package/dist/assets/editor_example_1-600w.webp +0 -0
  48. package/dist/assets/editor_example_1-800w.webp +0 -0
  49. package/dist/assets/editor_example_1_more_square-1000w.webp +0 -0
  50. package/dist/assets/editor_example_1_more_square-1200w.webp +0 -0
  51. package/dist/assets/editor_example_1_more_square-1600w.webp +0 -0
  52. package/dist/assets/editor_example_1_more_square-2000w.webp +0 -0
  53. package/dist/assets/editor_example_1_more_square-400w.webp +0 -0
  54. package/dist/assets/editor_example_1_more_square-600w.webp +0 -0
  55. package/dist/assets/editor_example_1_more_square-800w.webp +0 -0
  56. package/dist/assets/editor_example_2-1000w.webp +0 -0
  57. package/dist/assets/editor_example_2-1200w.webp +0 -0
  58. package/dist/assets/editor_example_2-1600w.webp +0 -0
  59. package/dist/assets/editor_example_2-2000w.webp +0 -0
  60. package/dist/assets/editor_example_2-400w.webp +0 -0
  61. package/dist/assets/editor_example_2-600w.webp +0 -0
  62. package/dist/assets/editor_example_2-800w.webp +0 -0
  63. package/dist/assets/example_schematic-1000w.webp +0 -0
  64. package/dist/assets/example_schematic-1200w.webp +0 -0
  65. package/dist/assets/example_schematic-1600w.webp +0 -0
  66. package/dist/assets/example_schematic-2000w.webp +0 -0
  67. package/dist/assets/example_schematic-400w.webp +0 -0
  68. package/dist/assets/example_schematic-600w.webp +0 -0
  69. package/dist/assets/example_schematic-800w.webp +0 -0
  70. package/dist/bundle.js +3270 -0
  71. package/dist/robots.txt +9 -0
  72. package/dist/sitemap.xml +118 -0
  73. package/docs/CIRCUIT_JSON_SOURCE_COMPONENT_OVERVIEW.md +151 -0
  74. package/fake-snippets-api/README.md +6 -0
  75. package/fake-snippets-api/biome.json +47 -0
  76. package/fake-snippets-api/bun.lockb +0 -0
  77. package/fake-snippets-api/lib/db/autoload-dev-snippets.ts +84 -0
  78. package/fake-snippets-api/lib/db/autoload-snippets.json +24 -0
  79. package/fake-snippets-api/lib/db/db-client.ts +343 -0
  80. package/fake-snippets-api/lib/db/schema.ts +112 -0
  81. package/fake-snippets-api/lib/db/seed.ts +1608 -0
  82. package/fake-snippets-api/lib/middleware/with-ctx-error.ts +26 -0
  83. package/fake-snippets-api/lib/middleware/with-db.ts +15 -0
  84. package/fake-snippets-api/lib/middleware/with-error-handling.ts +24 -0
  85. package/fake-snippets-api/lib/middleware/with-optional-session-auth.ts +34 -0
  86. package/fake-snippets-api/lib/middleware/with-request-logging.ts +54 -0
  87. package/fake-snippets-api/lib/middleware/with-session-auth.ts +39 -0
  88. package/fake-snippets-api/lib/middleware/with-winter-spec.ts +24 -0
  89. package/fake-snippets-api/next-env.d.ts +5 -0
  90. package/fake-snippets-api/routes/api/accounts/get.ts +21 -0
  91. package/fake-snippets-api/routes/api/accounts/get_account_balance.ts +22 -0
  92. package/fake-snippets-api/routes/api/accounts/update.ts +32 -0
  93. package/fake-snippets-api/routes/api/ai/[...anyroute].ts +31 -0
  94. package/fake-snippets-api/routes/api/ai.ts +2 -0
  95. package/fake-snippets-api/routes/api/aistream/[...anyroute].ts +65 -0
  96. package/fake-snippets-api/routes/api/health.ts +9 -0
  97. package/fake-snippets-api/routes/api/internal/sessions/create_without_auth.ts +63 -0
  98. package/fake-snippets-api/routes/api/order_files/get.ts +28 -0
  99. package/fake-snippets-api/routes/api/order_files/upload.ts +43 -0
  100. package/fake-snippets-api/routes/api/orders/create.ts +41 -0
  101. package/fake-snippets-api/routes/api/orders/get.ts +28 -0
  102. package/fake-snippets-api/routes/api/orders/list.ts +15 -0
  103. package/fake-snippets-api/routes/api/orders/update.ts +50 -0
  104. package/fake-snippets-api/routes/api/snippets/add_star.ts +42 -0
  105. package/fake-snippets-api/routes/api/snippets/create.ts +55 -0
  106. package/fake-snippets-api/routes/api/snippets/delete.ts +41 -0
  107. package/fake-snippets-api/routes/api/snippets/download.ts +148 -0
  108. package/fake-snippets-api/routes/api/snippets/generate_from_jlcpcb.ts +55 -0
  109. package/fake-snippets-api/routes/api/snippets/get.ts +50 -0
  110. package/fake-snippets-api/routes/api/snippets/get_image.ts +65 -0
  111. package/fake-snippets-api/routes/api/snippets/images/[author]/[snippet_name]/[typeFormat].ts +74 -0
  112. package/fake-snippets-api/routes/api/snippets/list.ts +28 -0
  113. package/fake-snippets-api/routes/api/snippets/list_newest.ts +13 -0
  114. package/fake-snippets-api/routes/api/snippets/list_trending.ts +21 -0
  115. package/fake-snippets-api/routes/api/snippets/remove_star.ts +42 -0
  116. package/fake-snippets-api/routes/api/snippets/search.ts +18 -0
  117. package/fake-snippets-api/routes/api/snippets/update.ts +86 -0
  118. package/favicon.ico +0 -0
  119. package/index.html +23 -0
  120. package/landing.html +23 -0
  121. package/package.json +164 -0
  122. package/playwright-tests/ai-page.spec.ts +19 -0
  123. package/playwright-tests/cmd-click.spec.ts +43 -0
  124. package/playwright-tests/dashboard-page.spec.ts +10 -0
  125. package/playwright-tests/editor-page.spec.ts +15 -0
  126. package/playwright-tests/files-dialog.spec.ts +19 -0
  127. package/playwright-tests/footprint-dialog/footprint-dialog.spec.ts +27 -0
  128. package/playwright-tests/footprint-dialog/footprint-insertion.spec.ts +38 -0
  129. package/playwright-tests/footprint-dialog/footprint-preview.spec.ts +34 -0
  130. package/playwright-tests/footprint-dialog/footprint-selection.spec.ts +29 -0
  131. package/playwright-tests/handle-manual-edits.spec.ts +55 -0
  132. package/playwright-tests/home-page.spec.ts +10 -0
  133. package/playwright-tests/images.spec.ts +17 -0
  134. package/playwright-tests/manual-edits.spec.ts +89 -0
  135. package/playwright-tests/preview-page.spec.ts +14 -0
  136. package/playwright-tests/quickstart-page.spec.ts +10 -0
  137. package/playwright-tests/search-links.spec.ts +21 -0
  138. package/playwright-tests/search.spec.ts +27 -0
  139. package/playwright-tests/snapshots/ai-page.spec.ts-AI-Page-lg.png +0 -0
  140. package/playwright-tests/snapshots/ai-page.spec.ts-AI-Page-md.png +0 -0
  141. package/playwright-tests/snapshots/ai-page.spec.ts-AI-Page-xs.png +0 -0
  142. package/playwright-tests/snapshots/cmd-click.spec.ts-underlined-imports.png +0 -0
  143. package/playwright-tests/snapshots/dashboard-page.spec.ts-Dashboard-page-lg.png +0 -0
  144. package/playwright-tests/snapshots/dashboard-page.spec.ts-Dashboard-page-md.png +0 -0
  145. package/playwright-tests/snapshots/dashboard-page.spec.ts-Dashboard-page-xs.png +0 -0
  146. package/playwright-tests/snapshots/editor-page.spec.ts-editor-with-snippet.png +0 -0
  147. package/playwright-tests/snapshots/error-fallback.spec.ts-error-fallback-lg.png +0 -0
  148. package/playwright-tests/snapshots/error-fallback.spec.ts-error-fallback-md.png +0 -0
  149. package/playwright-tests/snapshots/error-fallback.spec.ts-error-fallback-xs.png +0 -0
  150. package/playwright-tests/snapshots/files-dialog.spec.ts-view-snippet-files.png +0 -0
  151. package/playwright-tests/snapshots/footprint-dialog/footprint-dialog.spec.ts-footprint-preview-lg.png +0 -0
  152. package/playwright-tests/snapshots/footprint-dialog/footprint-dialog.spec.ts-footprint-preview-md.png +0 -0
  153. package/playwright-tests/snapshots/footprint-dialog/footprint-dialog.spec.ts-footprint-preview-xs.png +0 -0
  154. package/playwright-tests/snapshots/footprint-dialog/footprint-insertion.spec.ts-footprint-insertion-lg.png +0 -0
  155. package/playwright-tests/snapshots/footprint-dialog/footprint-insertion.spec.ts-footprint-insertion-md.png +0 -0
  156. package/playwright-tests/snapshots/footprint-dialog/footprint-insertion.spec.ts-footprint-insertion-xs.png +0 -0
  157. package/playwright-tests/snapshots/footprint-dialog/footprint-preview.spec.ts-footprint-preview-lg.png +0 -0
  158. package/playwright-tests/snapshots/footprint-dialog/footprint-preview.spec.ts-footprint-preview-md.png +0 -0
  159. package/playwright-tests/snapshots/footprint-dialog/footprint-preview.spec.ts-footprint-preview-xs.png +0 -0
  160. package/playwright-tests/snapshots/footprint-dialog/footprint-selection.spec.ts-footprint-preview-lg.png +0 -0
  161. package/playwright-tests/snapshots/footprint-dialog/footprint-selection.spec.ts-footprint-preview-md.png +0 -0
  162. package/playwright-tests/snapshots/footprint-dialog/footprint-selection.spec.ts-footprint-preview-xs.png +0 -0
  163. package/playwright-tests/snapshots/handle-manual-edits.spec.ts-handle-manual-edits.png +0 -0
  164. package/playwright-tests/snapshots/home-page.spec.ts-Home-page-lg.png +0 -0
  165. package/playwright-tests/snapshots/home-page.spec.ts-Home-page-md.png +0 -0
  166. package/playwright-tests/snapshots/home-page.spec.ts-Home-page-xs.png +0 -0
  167. package/playwright-tests/snapshots/images.spec.ts-pcb-image.png +0 -0
  168. package/playwright-tests/snapshots/images.spec.ts-schematic-image.png +0 -0
  169. package/playwright-tests/snapshots/manual-edits.spec.ts-editor-manual-edits.png +0 -0
  170. package/playwright-tests/snapshots/manual-edits.spec.ts-manual-edits-view.png +0 -0
  171. package/playwright-tests/snapshots/preview-page.spec.ts-preview-snippet-pcb.png +0 -0
  172. package/playwright-tests/snapshots/preview-page.spec.ts-preview-snippet-schematic.png +0 -0
  173. package/playwright-tests/snapshots/quickstart-page.spec.ts-Quickstart-Pagelg.png +0 -0
  174. package/playwright-tests/snapshots/quickstart-page.spec.ts-Quickstart-Pagemd.png +0 -0
  175. package/playwright-tests/snapshots/quickstart-page.spec.ts-Quickstart-Pagexs.png +0 -0
  176. package/playwright-tests/snapshots/search-links.spec.ts-search-links.png +0 -0
  177. package/playwright-tests/snapshots/search.spec.ts-search-lg.png +0 -0
  178. package/playwright-tests/snapshots/search.spec.ts-search-md.png +0 -0
  179. package/playwright-tests/snapshots/search.spec.ts-search-xs.png +0 -0
  180. package/playwright-tests/snapshots/star.spec.ts-remove-star-button.png +0 -0
  181. package/playwright-tests/snapshots/star.spec.ts-star-button.png +0 -0
  182. package/playwright-tests/snapshots/update-description.spec.ts-update-description.png +0 -0
  183. package/playwright-tests/snapshots/view-snippet.spec.ts-view-snippet-after-lg.png +0 -0
  184. package/playwright-tests/snapshots/view-snippet.spec.ts-view-snippet-after-md.png +0 -0
  185. package/playwright-tests/snapshots/view-snippet.spec.ts-view-snippet-after-xs.png +0 -0
  186. package/playwright-tests/snapshots/view-snippet.spec.ts-view-snippet-before-lg.png +0 -0
  187. package/playwright-tests/snapshots/view-snippet.spec.ts-view-snippet-before-md.png +0 -0
  188. package/playwright-tests/snapshots/view-snippet.spec.ts-view-snippet-before-xs.png +0 -0
  189. package/playwright-tests/snapshots/view-snippet.spec.ts-view-snippet-files.png +0 -0
  190. package/playwright-tests/star.spec.ts +40 -0
  191. package/playwright-tests/update-description.spec.ts +18 -0
  192. package/playwright-tests/view-snippet.spec.ts +35 -0
  193. package/playwright-tests/viewports.ts +5 -0
  194. package/playwright.config.ts +27 -0
  195. package/postcss.config.js +6 -0
  196. package/public/robots.txt +9 -0
  197. package/renovate.json +24 -0
  198. package/scripts/generate-image-sizes.ts +58 -0
  199. package/scripts/generate-sitemap.ts +103 -0
  200. package/scripts/generate_bundle_stats.js +192 -0
  201. package/scripts/snapshot.ts +35 -0
  202. package/src/App.tsx +113 -0
  203. package/src/ContextProviders.tsx +9 -0
  204. package/src/assets/originals/editor_example_1.webp +0 -0
  205. package/src/assets/originals/editor_example_1_more_square.webp +0 -0
  206. package/src/assets/originals/editor_example_2.webp +0 -0
  207. package/src/assets/originals/example_schematic.webp +0 -0
  208. package/src/components/AiChatInterface.tsx +221 -0
  209. package/src/components/AiChatMessage.tsx +86 -0
  210. package/src/components/Analytics.tsx +30 -0
  211. package/src/components/BomTable.tsx +69 -0
  212. package/src/components/ChatInput.tsx +53 -0
  213. package/src/components/CircuitToSvgWithMouseControl.tsx +78 -0
  214. package/src/components/CmdKMenu.tsx +301 -0
  215. package/src/components/CodeAndPreview.tsx +258 -0
  216. package/src/components/CodeEditor.tsx +460 -0
  217. package/src/components/CodeEditorHeader.tsx +160 -0
  218. package/src/components/CreateNewSnippetWithAiHero.tsx +77 -0
  219. package/src/components/DownloadButtonAndMenu.tsx +212 -0
  220. package/src/components/EditorNav.tsx +428 -0
  221. package/src/components/ErrorFallback.tsx +25 -0
  222. package/src/components/ErrorTabContent.tsx +122 -0
  223. package/src/components/FAQ.tsx +113 -0
  224. package/src/components/Footer.tsx +134 -0
  225. package/src/components/Footer2.tsx +100 -0
  226. package/src/components/FootprintDialog.tsx +342 -0
  227. package/src/components/Header.tsx +187 -0
  228. package/src/components/Header2.tsx +135 -0
  229. package/src/components/HeaderDropdown.tsx +83 -0
  230. package/src/components/HeaderLogin.tsx +94 -0
  231. package/src/components/JLCPCBImportDialog.tsx +155 -0
  232. package/src/components/LandingHero.tsx +175 -0
  233. package/src/components/LatestSnippets.tsx +39 -0
  234. package/src/components/OptimizedImage.tsx +96 -0
  235. package/src/components/OrderPreviewContent.tsx +61 -0
  236. package/src/components/ParametersEditor.tsx +140 -0
  237. package/src/components/PcbViewerWithContainerHeight.tsx +47 -0
  238. package/src/components/PrefetchPageLink.tsx +45 -0
  239. package/src/components/PreviewContent.tsx +375 -0
  240. package/src/components/PreviewEmptyState.tsx +16 -0
  241. package/src/components/RunButton.tsx +27 -0
  242. package/src/components/SearchComponent.tsx +163 -0
  243. package/src/components/ShippingInformationForm.tsx +423 -0
  244. package/src/components/SnippetLink.tsx +37 -0
  245. package/src/components/StaticPreviewContent.tsx +89 -0
  246. package/src/components/StaticViewSnippetHeader.tsx +70 -0
  247. package/src/components/StaticViewSnippetSidebar.tsx +100 -0
  248. package/src/components/TableViewer/CircuitJsonTableViewer.tsx +316 -0
  249. package/src/components/TableViewer/ClickableText.tsx +21 -0
  250. package/src/components/TableViewer/HeaderCell.tsx +27 -0
  251. package/src/components/TableViewer/Modal.tsx +39 -0
  252. package/src/components/TrendingSnippetCarousel.tsx +77 -0
  253. package/src/components/TypeBadge.tsx +31 -0
  254. package/src/components/ViewSnippetHeader.tsx +144 -0
  255. package/src/components/ViewSnippetSidebar.tsx +162 -0
  256. package/src/components/dialogs/confirm-delete-snippet-dialog.tsx +80 -0
  257. package/src/components/dialogs/create-order-dialog.tsx +146 -0
  258. package/src/components/dialogs/create-use-dialog.tsx +34 -0
  259. package/src/components/dialogs/edit-description-dialog.tsx +96 -0
  260. package/src/components/dialogs/files-dialog.tsx +70 -0
  261. package/src/components/dialogs/import-snippet-dialog.tsx +84 -0
  262. package/src/components/dialogs/rename-snippet-dialog.tsx +81 -0
  263. package/src/components/dialogs/view-ts-files-dialog.tsx +89 -0
  264. package/src/components/ui/accordion.tsx +55 -0
  265. package/src/components/ui/alert-dialog.tsx +139 -0
  266. package/src/components/ui/alert.tsx +59 -0
  267. package/src/components/ui/aspect-ratio.tsx +5 -0
  268. package/src/components/ui/avatar.tsx +48 -0
  269. package/src/components/ui/badge.tsx +36 -0
  270. package/src/components/ui/breadcrumb.tsx +118 -0
  271. package/src/components/ui/button.tsx +58 -0
  272. package/src/components/ui/calendar.tsx +73 -0
  273. package/src/components/ui/card.tsx +76 -0
  274. package/src/components/ui/carousel.tsx +260 -0
  275. package/src/components/ui/chart.tsx +363 -0
  276. package/src/components/ui/checkbox.tsx +28 -0
  277. package/src/components/ui/collapsible.tsx +9 -0
  278. package/src/components/ui/combobox.tsx +178 -0
  279. package/src/components/ui/command.tsx +151 -0
  280. package/src/components/ui/context-menu.tsx +202 -0
  281. package/src/components/ui/dialog.tsx +120 -0
  282. package/src/components/ui/drawer.tsx +116 -0
  283. package/src/components/ui/dropdown-menu.tsx +203 -0
  284. package/src/components/ui/form.tsx +182 -0
  285. package/src/components/ui/hover-card.tsx +27 -0
  286. package/src/components/ui/input-otp.tsx +69 -0
  287. package/src/components/ui/input.tsx +25 -0
  288. package/src/components/ui/label.tsx +24 -0
  289. package/src/components/ui/menubar.tsx +238 -0
  290. package/src/components/ui/navigation-menu.tsx +129 -0
  291. package/src/components/ui/pagination.tsx +121 -0
  292. package/src/components/ui/popover.tsx +31 -0
  293. package/src/components/ui/progress.tsx +26 -0
  294. package/src/components/ui/radio-group.tsx +42 -0
  295. package/src/components/ui/resizable.tsx +43 -0
  296. package/src/components/ui/scroll-area.tsx +46 -0
  297. package/src/components/ui/searchable-select.tsx +94 -0
  298. package/src/components/ui/select.tsx +162 -0
  299. package/src/components/ui/separator.tsx +29 -0
  300. package/src/components/ui/sheet.tsx +141 -0
  301. package/src/components/ui/skeleton.tsx +18 -0
  302. package/src/components/ui/slider.tsx +26 -0
  303. package/src/components/ui/sonner.tsx +30 -0
  304. package/src/components/ui/switch.tsx +27 -0
  305. package/src/components/ui/table.tsx +120 -0
  306. package/src/components/ui/tabs.tsx +53 -0
  307. package/src/components/ui/textarea.tsx +24 -0
  308. package/src/components/ui/toast.tsx +128 -0
  309. package/src/components/ui/toaster.tsx +33 -0
  310. package/src/components/ui/toggle-group.tsx +59 -0
  311. package/src/components/ui/toggle.tsx +43 -0
  312. package/src/components/ui/tooltip.tsx +28 -0
  313. package/src/entry-server.tsx +12 -0
  314. package/src/hooks/use-account-balance.ts +24 -0
  315. package/src/hooks/use-ai-api.ts +35 -0
  316. package/src/hooks/use-axios.ts +20 -0
  317. package/src/hooks/use-code-completion-ai-api.ts +11 -0
  318. package/src/hooks/use-compiled-tsx.ts +37 -0
  319. package/src/hooks/use-copy-to-clipboard.ts +26 -0
  320. package/src/hooks/use-create-snippet-mutation.ts +62 -0
  321. package/src/hooks/use-current-snippet-id.ts +75 -0
  322. package/src/hooks/use-current-snippet.ts +24 -0
  323. package/src/hooks/use-global-store.ts +33 -0
  324. package/src/hooks/use-is-using-fake-api.ts +3 -0
  325. package/src/hooks/use-run-tsx/construct-circuit.tsx +64 -0
  326. package/src/hooks/use-run-tsx/eval-compiled-js.ts +9 -0
  327. package/src/hooks/use-run-tsx/index.tsx +251 -0
  328. package/src/hooks/use-save-snippet.ts +66 -0
  329. package/src/hooks/use-sign-in.ts +22 -0
  330. package/src/hooks/use-snippet-by-name.ts +25 -0
  331. package/src/hooks/use-snippet.ts +23 -0
  332. package/src/hooks/use-snippets-base-api-url.ts +3 -0
  333. package/src/hooks/use-toast.tsx +208 -0
  334. package/src/hooks/use-typecheck.ts +54 -0
  335. package/src/hooks/use-url-params.ts +31 -0
  336. package/src/hooks/use-warn-user-on-page-change.ts +23 -0
  337. package/src/hooks/useForkSnippetMutation.ts +52 -0
  338. package/src/index.css +21 -0
  339. package/src/lib/__tests__/constants.test.ts +66 -0
  340. package/src/lib/base64ToBytes.ts +5 -0
  341. package/src/lib/bytesToBase64.ts +4 -0
  342. package/src/lib/codemirror/basic-setup.ts +51 -0
  343. package/src/lib/constants.ts +12 -0
  344. package/src/lib/decodeUrlHashToText.ts +15 -0
  345. package/src/lib/defaultCodeForBlankCode.tsx +7 -0
  346. package/src/lib/download-fns/createBlobURL.ts +4 -0
  347. package/src/lib/download-fns/download-assembly-svg.ts +12 -0
  348. package/src/lib/download-fns/download-circuit-json-fn.ts +12 -0
  349. package/src/lib/download-fns/download-dsn-file-fn.ts +12 -0
  350. package/src/lib/download-fns/download-fabrication-files.ts +233 -0
  351. package/src/lib/download-fns/download-gltf.ts +49 -0
  352. package/src/lib/download-fns/download-kicad-files.ts +27 -0
  353. package/src/lib/download-fns/download-readable-netlist.ts +12 -0
  354. package/src/lib/download-fns/download-schematic-svg.ts +12 -0
  355. package/src/lib/download-fns/download-simple-route-json.ts +17 -0
  356. package/src/lib/encodeTextToUrlHash.ts +17 -0
  357. package/src/lib/get-snippet-template.ts +26 -0
  358. package/src/lib/handleManualEditsImport.tsx +65 -0
  359. package/src/lib/jlc-parts-engine.ts +69 -0
  360. package/src/lib/templates/blank-3d-model-template.ts +12 -0
  361. package/src/lib/templates/blank-circuit-board-template.ts +24 -0
  362. package/src/lib/templates/blank-footprint-template.ts +29 -0
  363. package/src/lib/templates/blank-package-template.ts +22 -0
  364. package/src/lib/templates/blinking-led-board-template.ts +63 -0
  365. package/src/lib/templates/manual-edits-template.ts +5 -0
  366. package/src/lib/templates/usb-c-led-flashlight-template.ts +22 -0
  367. package/src/lib/utils/checkIfManualEditsImported.ts +6 -0
  368. package/src/lib/utils/getSyntaxError.ts +13 -0
  369. package/src/lib/utils/index.ts +6 -0
  370. package/src/lib/utils/load-prettier.ts +18 -0
  371. package/src/lib/utils/parseFootprintParams.ts +52 -0
  372. package/src/lib/utils/parseJsonOrNull.ts +8 -0
  373. package/src/lib/utils/pcbManualEditEventHandler.ts +156 -0
  374. package/src/main.tsx +10 -0
  375. package/src/pages/ai.tsx +92 -0
  376. package/src/pages/authorize.tsx +54 -0
  377. package/src/pages/dashboard.tsx +165 -0
  378. package/src/pages/dev-login.tsx +68 -0
  379. package/src/pages/editor.tsx +28 -0
  380. package/src/pages/landing.tsx +236 -0
  381. package/src/pages/my-orders.tsx +54 -0
  382. package/src/pages/newest.tsx +16 -0
  383. package/src/pages/preview.tsx +70 -0
  384. package/src/pages/quickstart.tsx +198 -0
  385. package/src/pages/search.tsx +26 -0
  386. package/src/pages/settings.tsx +25 -0
  387. package/src/pages/user-profile.tsx +97 -0
  388. package/src/pages/view-order.tsx +123 -0
  389. package/src/pages/view-snippet.tsx +149 -0
  390. package/src/prettier.ts +9 -0
  391. package/src/vite-env.d.ts +1 -0
  392. package/tailwind.config.js +47 -0
  393. package/tsconfig.json +30 -0
  394. package/vercel.json +50 -0
  395. package/vite.config.ts +146 -0
  396. package/winterspec.config.ts +6 -0
@@ -0,0 +1,208 @@
1
+ import * as React from "react"
2
+
3
+ import type { ToastActionElement, ToastProps } from "@/components/ui/toast"
4
+
5
+ const TOAST_LIMIT = 1
6
+ const TOAST_REMOVE_DELAY = 1000000
7
+
8
+ type ToasterToast = ToastProps & {
9
+ id: string
10
+ title?: React.ReactNode
11
+ description?: React.ReactNode
12
+ action?: ToastActionElement
13
+ }
14
+
15
+ const actionTypes = {
16
+ ADD_TOAST: "ADD_TOAST",
17
+ UPDATE_TOAST: "UPDATE_TOAST",
18
+ DISMISS_TOAST: "DISMISS_TOAST",
19
+ REMOVE_TOAST: "REMOVE_TOAST",
20
+ } as const
21
+
22
+ let count = 0
23
+
24
+ function genId() {
25
+ count = (count + 1) % Number.MAX_SAFE_INTEGER
26
+ return count.toString()
27
+ }
28
+
29
+ type ActionType = typeof actionTypes
30
+
31
+ type Action =
32
+ | {
33
+ type: ActionType["ADD_TOAST"]
34
+ toast: ToasterToast
35
+ }
36
+ | {
37
+ type: ActionType["UPDATE_TOAST"]
38
+ toast: Partial<ToasterToast>
39
+ }
40
+ | {
41
+ type: ActionType["DISMISS_TOAST"]
42
+ toastId?: ToasterToast["id"]
43
+ }
44
+ | {
45
+ type: ActionType["REMOVE_TOAST"]
46
+ toastId?: ToasterToast["id"]
47
+ }
48
+
49
+ interface State {
50
+ toasts: ToasterToast[]
51
+ }
52
+
53
+ const toastTimeouts = new Map<string, ReturnType<typeof setTimeout>>()
54
+
55
+ const addToRemoveQueue = (toastId: string) => {
56
+ if (toastTimeouts.has(toastId)) {
57
+ return
58
+ }
59
+
60
+ const timeout = setTimeout(() => {
61
+ toastTimeouts.delete(toastId)
62
+ dispatch({
63
+ type: "REMOVE_TOAST",
64
+ toastId: toastId,
65
+ })
66
+ }, TOAST_REMOVE_DELAY)
67
+
68
+ toastTimeouts.set(toastId, timeout)
69
+ }
70
+
71
+ export const reducer = (state: State, action: Action): State => {
72
+ switch (action.type) {
73
+ case "ADD_TOAST":
74
+ return {
75
+ ...state,
76
+ toasts: [action.toast, ...state.toasts].slice(0, TOAST_LIMIT),
77
+ }
78
+
79
+ case "UPDATE_TOAST":
80
+ return {
81
+ ...state,
82
+ toasts: state.toasts.map((t) =>
83
+ t.id === action.toast.id ? { ...t, ...action.toast } : t,
84
+ ),
85
+ }
86
+
87
+ case "DISMISS_TOAST": {
88
+ const { toastId } = action
89
+
90
+ // ! Side effects ! - This could be extracted into a dismissToast() action,
91
+ // but I'll keep it here for simplicity
92
+ if (toastId) {
93
+ addToRemoveQueue(toastId)
94
+ } else {
95
+ state.toasts.forEach((toast) => {
96
+ addToRemoveQueue(toast.id)
97
+ })
98
+ }
99
+
100
+ return {
101
+ ...state,
102
+ toasts: state.toasts.map((t) =>
103
+ t.id === toastId || toastId === undefined
104
+ ? {
105
+ ...t,
106
+ open: false,
107
+ }
108
+ : t,
109
+ ),
110
+ }
111
+ }
112
+ case "REMOVE_TOAST":
113
+ if (action.toastId === undefined) {
114
+ return {
115
+ ...state,
116
+ toasts: [],
117
+ }
118
+ }
119
+ return {
120
+ ...state,
121
+ toasts: state.toasts.filter((t) => t.id !== action.toastId),
122
+ }
123
+ }
124
+ }
125
+
126
+ const listeners: Array<(state: State) => void> = []
127
+
128
+ let memoryState: State = { toasts: [] }
129
+
130
+ function dispatch(action: Action) {
131
+ memoryState = reducer(memoryState, action)
132
+ listeners.forEach((listener) => {
133
+ listener(memoryState)
134
+ })
135
+ }
136
+
137
+ type Toast = Omit<ToasterToast, "id">
138
+
139
+ function toast({ ...props }: Toast) {
140
+ const id = genId()
141
+
142
+ const update = (props: ToasterToast) =>
143
+ dispatch({
144
+ type: "UPDATE_TOAST",
145
+ toast: { ...props, id },
146
+ })
147
+ const dismiss = () => dispatch({ type: "DISMISS_TOAST", toastId: id })
148
+
149
+ dispatch({
150
+ type: "ADD_TOAST",
151
+ toast: {
152
+ ...props,
153
+ id,
154
+ open: true,
155
+ onOpenChange: (open) => {
156
+ if (!open) dismiss()
157
+ },
158
+ },
159
+ })
160
+
161
+ return {
162
+ id: id,
163
+ dismiss,
164
+ update,
165
+ }
166
+ }
167
+
168
+ function useToast() {
169
+ const [state, setState] = React.useState<State>(memoryState)
170
+
171
+ React.useEffect(() => {
172
+ listeners.push(setState)
173
+ return () => {
174
+ const index = listeners.indexOf(setState)
175
+ if (index > -1) {
176
+ listeners.splice(index, 1)
177
+ }
178
+ }
179
+ }, [state])
180
+
181
+ return {
182
+ ...state,
183
+ toast,
184
+ dismiss: (toastId?: string) => dispatch({ type: "DISMISS_TOAST", toastId }),
185
+ }
186
+ }
187
+
188
+ export { useToast, toast }
189
+
190
+ export function useNotImplementedToast() {
191
+ const { toast } = useToast()
192
+ return (feature: string) => {
193
+ toast({
194
+ title: "Not Implemented",
195
+ description: (
196
+ <div>
197
+ The {feature} feature is not implemented yet. Help us out!{" "}
198
+ <a
199
+ className="text-blue-500 hover:underline font-semibold"
200
+ href="https://github.com/tscircuit/snippets"
201
+ >
202
+ Check out our Github
203
+ </a>
204
+ </div>
205
+ ),
206
+ })
207
+ }
208
+ }
@@ -0,0 +1,54 @@
1
+ // import * as ts from "typescript"
2
+
3
+ // export function checkTypeScriptErrors(code: string): ts.Diagnostic[] {
4
+ // // Create a virtual TypeScript compiler
5
+ // const compilerOptions: ts.CompilerOptions = {
6
+ // noEmit: true,
7
+ // allowJs: true,
8
+ // checkJs: true,
9
+ // strict: true,
10
+ // }
11
+
12
+ // // Create a virtual file system
13
+ // const fileSystem = {
14
+ // "example.ts": code,
15
+ // }
16
+
17
+ // // Create a compiler host
18
+ // const compilerHost = ts.createCompilerHost(compilerOptions)
19
+ // compilerHost.getSourceFile = (fileName: string) => {
20
+ // return ts.createSourceFile(
21
+ // fileName,
22
+ // fileSystem[fileName] || "",
23
+ // ts.ScriptTarget.Latest,
24
+ // )
25
+ // }
26
+ // compilerHost.writeFile = () => {}
27
+ // compilerHost.getCurrentDirectory = () => "/"
28
+ // compilerHost.getDefaultLibFileName = () => "lib.d.ts"
29
+
30
+ // // Create a program
31
+ // const program = ts.createProgram(
32
+ // ["example.ts"],
33
+ // compilerOptions,
34
+ // compilerHost,
35
+ // )
36
+
37
+ // // Get the diagnostics (errors)
38
+ // const diagnostics = ts.getPreEmitDiagnostics(program)
39
+
40
+ // return diagnostics
41
+ // }
42
+
43
+ // // Example usage
44
+ // const code = `
45
+ // let x: number = "string"; // Type error
46
+ // let y: string = 5; // Another type error
47
+ // let z: number = 10; // No error
48
+ // `
49
+
50
+ // const errors = checkTypeScriptErrors(code)
51
+ // console.log("Errors found:", errors.length)
52
+ // errors.forEach((error) => {
53
+ // console.log(`Error on line ${error.start}: ${error.messageText}`)
54
+ // })
@@ -0,0 +1,31 @@
1
+ import { useState, useEffect } from "react"
2
+
3
+ const getUrlParams = () => {
4
+ const searchParams = new URLSearchParams(window.location.search)
5
+ const newParams: Record<string, string> = {}
6
+
7
+ for (const [key, value] of searchParams.entries()) {
8
+ newParams[key] = value
9
+ }
10
+ return newParams
11
+ }
12
+
13
+ export const useUrlParams = () => {
14
+ const [params, setParams] = useState<Record<string, string>>(getUrlParams())
15
+
16
+ useEffect(() => {
17
+ const updateParams = () => {
18
+ setParams(getUrlParams())
19
+ }
20
+
21
+ updateParams()
22
+
23
+ window.addEventListener("popstate", updateParams)
24
+
25
+ return () => {
26
+ window.removeEventListener("popstate", updateParams)
27
+ }
28
+ }, [])
29
+
30
+ return params
31
+ }
@@ -0,0 +1,23 @@
1
+ import { useEffect } from "react"
2
+ export default function useWarnUserOnPageChange({
3
+ hasUnsavedChanges,
4
+ }: {
5
+ hasUnsavedChanges: boolean
6
+ }) {
7
+ useEffect(() => {
8
+ const handleBeforeUnload = (event: BeforeUnloadEvent) => {
9
+ if (hasUnsavedChanges) {
10
+ event.preventDefault()
11
+ event.returnValue = "" // Shows the confirmation dialog on reload or close if there are unsaved changes
12
+ }
13
+ }
14
+
15
+ // Attach event listeners
16
+ window.addEventListener("beforeunload", handleBeforeUnload)
17
+
18
+ // Cleanup event listeners on component unmount
19
+ return () => {
20
+ window.removeEventListener("beforeunload", handleBeforeUnload)
21
+ }
22
+ }, [hasUnsavedChanges])
23
+ }
@@ -0,0 +1,52 @@
1
+ import { useMutation } from "react-query"
2
+ import { useGlobalStore } from "@/hooks/use-global-store"
3
+ import { useAxios } from "@/hooks/use-axios"
4
+ import { useToast } from "@/hooks/use-toast"
5
+ import { Snippet } from "fake-snippets-api/lib/db/schema"
6
+
7
+ export const useForkSnippetMutation = ({
8
+ snippet,
9
+ currentCode,
10
+ onSuccess,
11
+ }: {
12
+ snippet: Snippet
13
+ currentCode?: string
14
+ onSuccess?: (forkedSnippet: Snippet) => void
15
+ }) => {
16
+ const axios = useAxios()
17
+ const session = useGlobalStore((s) => s.session)
18
+ const { toast } = useToast()
19
+
20
+ return useMutation(
21
+ ["createForkSnippet"],
22
+ async () => {
23
+ if (!session) throw new Error("No session")
24
+ if (!snippet) throw new Error("No snippet to fork")
25
+
26
+ const { data } = await axios.post("/snippets/create", {
27
+ unscoped_name: snippet.unscoped_name,
28
+ snippet_type: snippet.snippet_type,
29
+ owner_name: session.github_username,
30
+ code: currentCode ?? snippet.code, // Use currentCode if provided, fall back to snippet.code
31
+ })
32
+ return data.snippet
33
+ },
34
+ {
35
+ onSuccess: (forkedSnippet: Snippet) => {
36
+ toast({
37
+ title: `Forked snippet`,
38
+ description: `You have successfully forked the snippet. Redirecting...`,
39
+ })
40
+ onSuccess?.(forkedSnippet)
41
+ },
42
+ onError: (error: any) => {
43
+ console.error("Error forking snippet:", error)
44
+ toast({
45
+ title: "Error",
46
+ description: "Failed to fork snippet. Please try again.",
47
+ variant: "destructive",
48
+ })
49
+ },
50
+ },
51
+ )
52
+ }
package/src/index.css ADDED
@@ -0,0 +1,21 @@
1
+ @tailwind base;
2
+ @tailwind components;
3
+ @tailwind utilities;
4
+ @layer base {
5
+ :root {
6
+ --radius: 0.5rem;
7
+ }
8
+ }
9
+
10
+ @keyframes fadeIn {
11
+ from {
12
+ opacity: 0;
13
+ }
14
+ to {
15
+ opacity: 1;
16
+ }
17
+ }
18
+
19
+ .animate-fadeIn {
20
+ animation: fadeIn 0.3s ease-in-out;
21
+ }
@@ -0,0 +1,66 @@
1
+ import { describe, expect, test } from "bun:test"
2
+ import { TSCI_PACKAGE_PATTERN } from "../constants"
3
+
4
+ describe("TSCI_PACKAGE_PATTERN", () => {
5
+ // Test valid patterns
6
+ const validPackages = [
7
+ "@tsci/basic",
8
+ "@tsci/package-name",
9
+ "@tsci/multiple-dashes-here",
10
+ "@tsci/package.subpackage",
11
+ "@tsci/complex-package.complex-subpackage",
12
+ "@tsci/a.b.c",
13
+ "@tsci/package-with-numbers123",
14
+ "@tsci/package123-with-numbers",
15
+ "@tsci/core.utils-helper",
16
+ "@tsci/double--dash",
17
+ "@tsci/seveibar.ABM8_272_T3",
18
+ ]
19
+
20
+ validPackages.forEach((pkg) => {
21
+ test(`should match valid package: ${pkg}`, () => {
22
+ const matches = Array.from(pkg.matchAll(TSCI_PACKAGE_PATTERN))
23
+ expect(matches).toHaveLength(1)
24
+ expect(matches[0][0]).toBe(pkg)
25
+ })
26
+ })
27
+
28
+ // Test invalid patterns
29
+ const invalidPackages = [
30
+ "@tsci/", // Empty
31
+ "@tsci/123start-with-number", // Starts with number
32
+ "@tsci/-start-with-dash", // Starts with dash
33
+ "@tsci/.start-with-dot", // Starts with dot
34
+
35
+ "@tsci/double..dot", // Double dot
36
+ "@tsci/end-with-dash-", // Ends with dash
37
+ "@tsci/end-with-dot.", // Ends with dot
38
+ "@tsci/invalid@chars", // Invalid characters
39
+ "@tsci/dot.dash.-mixed", // Invalid dot-dash combination
40
+ "@tsci/dash-.dot-mixed", // Invalid dash-dot combination
41
+ "@tsci/space in name", // Contains space
42
+ "@tsci/!", // Invalid character
43
+ "@tsci/pkg#123", // Invalid character
44
+ "@tsci/@invalid", // Invalid character
45
+ ]
46
+
47
+ invalidPackages.forEach((pkg) => {
48
+ test(`should not match invalid package: ${pkg}`, () => {
49
+ const matches = Array.from(pkg.matchAll(TSCI_PACKAGE_PATTERN))
50
+ console.log(Array.from(pkg.matchAll(TSCI_PACKAGE_PATTERN)))
51
+ expect(matches).toHaveLength(0)
52
+ })
53
+ })
54
+
55
+ // Test import statements
56
+ test("should match package names in import statements", () => {
57
+ const importStatement =
58
+ 'import { something } from "@tsci/valid-package" import { other } from "@tsci/other-package.sub"'
59
+ const matches = Array.from(importStatement.matchAll(TSCI_PACKAGE_PATTERN))
60
+ expect(matches).toHaveLength(2)
61
+ expect(matches.map((m) => m[0])).toEqual([
62
+ "@tsci/valid-package",
63
+ "@tsci/other-package.sub",
64
+ ])
65
+ })
66
+ })
@@ -0,0 +1,5 @@
1
+ // Base64 encoding and decoding functions
2
+ export function base64ToBytes(base64: string): Uint8Array {
3
+ const binString = atob(base64)
4
+ return Uint8Array.from(binString, (m) => m.codePointAt(0)!)
5
+ }
@@ -0,0 +1,4 @@
1
+ export function bytesToBase64(bytes: Uint8Array): string {
2
+ const binString = String.fromCodePoint(...bytes)
3
+ return btoa(binString)
4
+ }
@@ -0,0 +1,51 @@
1
+ import {
2
+ keymap,
3
+ highlightSpecialChars,
4
+ drawSelection,
5
+ highlightActiveLine,
6
+ dropCursor,
7
+ rectangularSelection,
8
+ crosshairCursor,
9
+ lineNumbers,
10
+ highlightActiveLineGutter,
11
+ } from "@codemirror/view"
12
+ import { Extension, EditorState } from "@codemirror/state"
13
+ import {
14
+ defaultHighlightStyle,
15
+ syntaxHighlighting,
16
+ indentOnInput,
17
+ bracketMatching,
18
+ foldGutter,
19
+ foldKeymap,
20
+ } from "@codemirror/language"
21
+ import { defaultKeymap, history, historyKeymap } from "@codemirror/commands"
22
+ import { searchKeymap, highlightSelectionMatches } from "@codemirror/search"
23
+ import { autocompletion, completionKeymap } from "@codemirror/autocomplete"
24
+ import { lintKeymap } from "@codemirror/lint"
25
+
26
+ export const basicSetup: Extension = (() => [
27
+ lineNumbers(),
28
+ highlightActiveLineGutter(),
29
+ highlightSpecialChars(),
30
+ history(),
31
+ foldGutter(),
32
+ drawSelection(),
33
+ dropCursor(),
34
+ EditorState.allowMultipleSelections.of(false),
35
+ indentOnInput(),
36
+ syntaxHighlighting(defaultHighlightStyle, { fallback: true }),
37
+ bracketMatching(),
38
+ autocompletion(),
39
+ rectangularSelection(),
40
+ crosshairCursor(),
41
+ highlightActiveLine(),
42
+ highlightSelectionMatches(),
43
+ keymap.of([
44
+ ...defaultKeymap,
45
+ ...searchKeymap,
46
+ ...historyKeymap,
47
+ ...foldKeymap,
48
+ ...completionKeymap,
49
+ ...lintKeymap,
50
+ ]),
51
+ ])()
@@ -0,0 +1,12 @@
1
+ /**
2
+ * Regular expression pattern for matching @tsci package imports
3
+ * Rules:
4
+ * - Must start with @tsci/
5
+ * - First character after @tsci/ must be a letter
6
+ * - Can contain letters, numbers, single dashes between alphanumeric characters
7
+ * - Can have subpackages separated by single dots
8
+ * - Cannot end with dots or dashes
9
+ * - Cannot have consecutive dots or dashes
10
+ */
11
+ export const TSCI_PACKAGE_PATTERN =
12
+ /@tsci\/[a-zA-Z][a-zA-Z0-9]*(?:--?[a-zA-Z0-9]+)*(?:\.[a-zA-Z][a-zA-Z0-9_]*(?:--?[a-zA-Z0-9_]+)*)*/g
@@ -0,0 +1,15 @@
1
+ import { gunzipSync, strFromU8 } from "fflate"
2
+ import { base64ToBytes } from "./base64ToBytes"
3
+
4
+ export function decodeUrlHashToText(url: string): string | null {
5
+ // Extract the base64 part from the URL
6
+ const base64Data = url.split("#data:application/gzip;base64,")?.[1]
7
+ if (!base64Data) {
8
+ return null
9
+ }
10
+
11
+ // Decode base64, decompress, and convert to string
12
+ const compressedData = base64ToBytes(base64Data)
13
+ const decompressedData = gunzipSync(compressedData)
14
+ return strFromU8(decompressedData)
15
+ }
@@ -0,0 +1,7 @@
1
+ export const defaultCodeForBlankPage = `
2
+ export default () => (
3
+ <board width="10mm" height="10mm">
4
+ {/* write your code here! */}
5
+ </board>
6
+ )
7
+ `.trim()
@@ -0,0 +1,4 @@
1
+ export const createBlobURL = (content: string) => {
2
+ const blob = new Blob([content], { type: "text/plain" })
3
+ return URL.createObjectURL(blob)
4
+ }
@@ -0,0 +1,12 @@
1
+ import { AnyCircuitElement } from "circuit-json"
2
+ import { convertCircuitJsonToAssemblySvg } from "circuit-to-svg"
3
+ import { saveAs } from "file-saver"
4
+
5
+ export const downloadAssemblySvg = (
6
+ circuitJson: AnyCircuitElement[],
7
+ fileName: string,
8
+ ) => {
9
+ const svg = convertCircuitJsonToAssemblySvg(circuitJson)
10
+ const blob = new Blob([svg], { type: "image/svg" })
11
+ saveAs(blob, fileName + ".svg")
12
+ }
@@ -0,0 +1,12 @@
1
+ import { saveAs } from "file-saver"
2
+ import { createBlobURL } from "./createBlobURL"
3
+ import { AnyCircuitElement } from "circuit-json"
4
+
5
+ export const downloadCircuitJson = (
6
+ circuitJson: AnyCircuitElement[],
7
+ fileName: string,
8
+ ) => {
9
+ const stringifiedCircuitJson = JSON.stringify(circuitJson, null, 2)
10
+ const blob = new Blob([stringifiedCircuitJson], { type: "application/json" })
11
+ saveAs(blob, fileName)
12
+ }
@@ -0,0 +1,12 @@
1
+ import { AnyCircuitElement } from "circuit-json"
2
+ import { saveAs } from "file-saver"
3
+ import { convertCircuitJsonToDsnString } from "dsn-converter"
4
+
5
+ export const downloadDsnFile = (
6
+ circuitJson: AnyCircuitElement[],
7
+ fileName: string,
8
+ ) => {
9
+ const dsnString = convertCircuitJsonToDsnString(circuitJson)
10
+ const blob = new Blob([dsnString], { type: "text/plain" })
11
+ saveAs(blob, fileName + ".dsn")
12
+ }