@nastechai-research/ui 0.18.2

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 (352) hide show
  1. package/CHANGELOG.md +5 -0
  2. package/README.md +21 -0
  3. package/dist/assets/filler-bg0.webp +0 -0
  4. package/dist/assets.d.ts +38 -0
  5. package/dist/fonts/Collapse-Bold.woff2 +0 -0
  6. package/dist/fonts/Collapse-BoldItalic.woff2 +0 -0
  7. package/dist/fonts/Collapse-Italic.woff2 +0 -0
  8. package/dist/fonts/Collapse-Light.woff2 +0 -0
  9. package/dist/fonts/Collapse-LightItalic.woff2 +0 -0
  10. package/dist/fonts/Collapse-Regular.woff2 +0 -0
  11. package/dist/fonts/Collapse-Thin.woff2 +0 -0
  12. package/dist/fonts/Collapse-ThinItalic.woff2 +0 -0
  13. package/dist/fonts/Mondwest-Regular.woff2 +0 -0
  14. package/dist/fonts/Neuebit-Bold.woff2 +0 -0
  15. package/dist/fonts/RulesCompressed-Medium.woff2 +0 -0
  16. package/dist/fonts/RulesCompressed-Regular.woff2 +0 -0
  17. package/dist/fonts/RulesExpanded-Bold.woff2 +0 -0
  18. package/dist/fonts/RulesExpanded-Regular.woff2 +0 -0
  19. package/dist/fonts.d.ts +6 -0
  20. package/dist/fonts.js +6 -0
  21. package/dist/hooks/use-below-breakpoint.d.ts +2 -0
  22. package/dist/hooks/use-below-breakpoint.js +17 -0
  23. package/dist/hooks/use-capped-frame.d.ts +2 -0
  24. package/dist/hooks/use-capped-frame.js +15 -0
  25. package/dist/hooks/use-confirm-delete.d.ts +10 -0
  26. package/dist/hooks/use-confirm-delete.js +35 -0
  27. package/dist/hooks/use-css-var-dims.d.ts +1 -0
  28. package/dist/hooks/use-css-var-dims.js +29 -0
  29. package/dist/hooks/use-gpu-tier.d.ts +34 -0
  30. package/dist/hooks/use-gpu-tier.js +111 -0
  31. package/dist/hooks/use-render-loop.d.ts +41 -0
  32. package/dist/hooks/use-render-loop.js +63 -0
  33. package/dist/hooks/use-smooth-controls.d.ts +16 -0
  34. package/dist/hooks/use-smooth-controls.js +217 -0
  35. package/dist/hooks/use-toast.d.ts +7 -0
  36. package/dist/hooks/use-toast.js +21 -0
  37. package/dist/index.d.ts +79 -0
  38. package/dist/index.js +107 -0
  39. package/dist/ui/basic-page.d.ts +7 -0
  40. package/dist/ui/basic-page.js +18 -0
  41. package/dist/ui/build.css +4 -0
  42. package/dist/ui/components/animated-count.d.ts +10 -0
  43. package/dist/ui/components/animated-count.js +113 -0
  44. package/dist/ui/components/ascii.d.ts +10 -0
  45. package/dist/ui/components/ascii.js +79 -0
  46. package/dist/ui/components/badge.d.ts +6 -0
  47. package/dist/ui/components/badge.js +40 -0
  48. package/dist/ui/components/badges/nous-girl.d.ts +2 -0
  49. package/dist/ui/components/badges/nous-girl.js +83 -0
  50. package/dist/ui/components/blend-mode.d.ts +28 -0
  51. package/dist/ui/components/blend-mode.js +69 -0
  52. package/dist/ui/components/blink.d.ts +6 -0
  53. package/dist/ui/components/blink.js +17 -0
  54. package/dist/ui/components/bottom-sheet.d.ts +15 -0
  55. package/dist/ui/components/bottom-sheet.js +192 -0
  56. package/dist/ui/components/button.d.ts +14 -0
  57. package/dist/ui/components/button.js +147 -0
  58. package/dist/ui/components/card.d.ts +5 -0
  59. package/dist/ui/components/card.js +74 -0
  60. package/dist/ui/components/checkbox.d.ts +2 -0
  61. package/dist/ui/components/checkbox.js +27 -0
  62. package/dist/ui/components/command-block.d.ts +24 -0
  63. package/dist/ui/components/command-block.js +56 -0
  64. package/dist/ui/components/confirm-dialog.d.ts +13 -0
  65. package/dist/ui/components/confirm-dialog.js +113 -0
  66. package/dist/ui/components/cursor.d.ts +3 -0
  67. package/dist/ui/components/cursor.js +97 -0
  68. package/dist/ui/components/dialog.d.ts +15 -0
  69. package/dist/ui/components/dialog.js +171 -0
  70. package/dist/ui/components/dropdown-menu.d.ts +12 -0
  71. package/dist/ui/components/dropdown-menu.js +102 -0
  72. package/dist/ui/components/fit-text/fit-text.css +42 -0
  73. package/dist/ui/components/fit-text/index.d.ts +9 -0
  74. package/dist/ui/components/fit-text/index.js +25 -0
  75. package/dist/ui/components/graphs/bar-chart.d.ts +12 -0
  76. package/dist/ui/components/graphs/bar-chart.js +129 -0
  77. package/dist/ui/components/graphs/index.d.ts +3 -0
  78. package/dist/ui/components/graphs/index.js +4 -0
  79. package/dist/ui/components/graphs/line-chart.d.ts +14 -0
  80. package/dist/ui/components/graphs/line-chart.js +175 -0
  81. package/dist/ui/components/graphs/utils.d.ts +52 -0
  82. package/dist/ui/components/graphs/utils.js +162 -0
  83. package/dist/ui/components/grid/grid.css +79 -0
  84. package/dist/ui/components/grid/index.d.ts +2 -0
  85. package/dist/ui/components/grid/index.js +17 -0
  86. package/dist/ui/components/hover-bg.d.ts +1 -0
  87. package/dist/ui/components/hover-bg.js +14 -0
  88. package/dist/ui/components/icons/arrow.d.ts +6 -0
  89. package/dist/ui/components/icons/arrow.js +44 -0
  90. package/dist/ui/components/icons/check.d.ts +2 -0
  91. package/dist/ui/components/icons/check.js +13 -0
  92. package/dist/ui/components/icons/chevron.d.ts +6 -0
  93. package/dist/ui/components/icons/chevron.js +51 -0
  94. package/dist/ui/components/icons/discord.d.ts +2 -0
  95. package/dist/ui/components/icons/discord.js +15 -0
  96. package/dist/ui/components/icons/eye.d.ts +2 -0
  97. package/dist/ui/components/icons/eye.js +8 -0
  98. package/dist/ui/components/icons/gear.d.ts +6 -0
  99. package/dist/ui/components/icons/gear.js +30 -0
  100. package/dist/ui/components/icons/github.d.ts +2 -0
  101. package/dist/ui/components/icons/github.js +15 -0
  102. package/dist/ui/components/icons/hamburger.d.ts +6 -0
  103. package/dist/ui/components/icons/hamburger.js +56 -0
  104. package/dist/ui/components/icons/heart.d.ts +2 -0
  105. package/dist/ui/components/icons/heart.js +11 -0
  106. package/dist/ui/components/icons/index.d.ts +12 -0
  107. package/dist/ui/components/icons/index.js +13 -0
  108. package/dist/ui/components/icons/link.d.ts +2 -0
  109. package/dist/ui/components/icons/link.js +13 -0
  110. package/dist/ui/components/icons/minus.d.ts +2 -0
  111. package/dist/ui/components/icons/minus.js +13 -0
  112. package/dist/ui/components/icons/search.d.ts +2 -0
  113. package/dist/ui/components/icons/search.js +33 -0
  114. package/dist/ui/components/image-distortion.d.ts +21 -0
  115. package/dist/ui/components/image-distortion.js +398 -0
  116. package/dist/ui/components/input.d.ts +1 -0
  117. package/dist/ui/components/input.js +21 -0
  118. package/dist/ui/components/label.d.ts +1 -0
  119. package/dist/ui/components/label.js +18 -0
  120. package/dist/ui/components/leva-client.d.ts +1 -0
  121. package/dist/ui/components/leva-client.js +12 -0
  122. package/dist/ui/components/list-item.d.ts +6 -0
  123. package/dist/ui/components/list-item.js +27 -0
  124. package/dist/ui/components/overlays/blend-modes.d.ts +1 -0
  125. package/dist/ui/components/overlays/blend-modes.js +14 -0
  126. package/dist/ui/components/overlays/glitch.d.ts +6 -0
  127. package/dist/ui/components/overlays/glitch.js +209 -0
  128. package/dist/ui/components/overlays/greys.d.ts +6 -0
  129. package/dist/ui/components/overlays/greys.js +339 -0
  130. package/dist/ui/components/overlays/index.d.ts +14 -0
  131. package/dist/ui/components/overlays/index.js +34 -0
  132. package/dist/ui/components/overlays/lens-layers.d.ts +14 -0
  133. package/dist/ui/components/overlays/lens-layers.js +95 -0
  134. package/dist/ui/components/overlays/lens.d.ts +44 -0
  135. package/dist/ui/components/overlays/lens.js +60 -0
  136. package/dist/ui/components/overlays/noise.d.ts +6 -0
  137. package/dist/ui/components/overlays/noise.js +136 -0
  138. package/dist/ui/components/overlays/vignette.d.ts +6 -0
  139. package/dist/ui/components/overlays/vignette.js +47 -0
  140. package/dist/ui/components/poster.d.ts +62 -0
  141. package/dist/ui/components/poster.js +256 -0
  142. package/dist/ui/components/progress.d.ts +9 -0
  143. package/dist/ui/components/progress.js +53 -0
  144. package/dist/ui/components/scene-canvas.d.ts +23 -0
  145. package/dist/ui/components/scene-canvas.js +179 -0
  146. package/dist/ui/components/scramble.d.ts +9 -0
  147. package/dist/ui/components/scramble.js +63 -0
  148. package/dist/ui/components/segmented.d.ts +20 -0
  149. package/dist/ui/components/segmented.js +51 -0
  150. package/dist/ui/components/select.d.ts +18 -0
  151. package/dist/ui/components/select.js +215 -0
  152. package/dist/ui/components/selection-switcher.d.ts +1 -0
  153. package/dist/ui/components/selection-switcher.js +34 -0
  154. package/dist/ui/components/separator.d.ts +5 -0
  155. package/dist/ui/components/separator.js +22 -0
  156. package/dist/ui/components/shader.d.ts +7 -0
  157. package/dist/ui/components/shader.js +60 -0
  158. package/dist/ui/components/socials.d.ts +20 -0
  159. package/dist/ui/components/socials.js +21 -0
  160. package/dist/ui/components/spinner.d.ts +20 -0
  161. package/dist/ui/components/spinner.js +38 -0
  162. package/dist/ui/components/stats.d.ts +16 -0
  163. package/dist/ui/components/stats.js +36 -0
  164. package/dist/ui/components/switch.d.ts +7 -0
  165. package/dist/ui/components/switch.js +37 -0
  166. package/dist/ui/components/tabs.d.ts +14 -0
  167. package/dist/ui/components/tabs.js +44 -0
  168. package/dist/ui/components/terminal-demo.d.ts +32 -0
  169. package/dist/ui/components/terminal-demo.js +125 -0
  170. package/dist/ui/components/theme-toggle.d.ts +6 -0
  171. package/dist/ui/components/theme-toggle.js +66 -0
  172. package/dist/ui/components/tier-card.d.ts +53 -0
  173. package/dist/ui/components/tier-card.js +146 -0
  174. package/dist/ui/components/toast.d.ts +8 -0
  175. package/dist/ui/components/toast.js +39 -0
  176. package/dist/ui/components/tv.d.ts +3 -0
  177. package/dist/ui/components/tv.js +239 -0
  178. package/dist/ui/components/typography/h1.d.ts +11 -0
  179. package/dist/ui/components/typography/h1.js +18 -0
  180. package/dist/ui/components/typography/h2.d.ts +11 -0
  181. package/dist/ui/components/typography/h2.js +18 -0
  182. package/dist/ui/components/typography/index.d.ts +15 -0
  183. package/dist/ui/components/typography/index.js +41 -0
  184. package/dist/ui/components/typography/legend.d.ts +6 -0
  185. package/dist/ui/components/typography/legend.js +20 -0
  186. package/dist/ui/components/typography/small.d.ts +2 -0
  187. package/dist/ui/components/typography/small.js +9 -0
  188. package/dist/ui/components/watchlist.d.ts +11 -0
  189. package/dist/ui/components/watchlist.js +80 -0
  190. package/dist/ui/fonts.css +63 -0
  191. package/dist/ui/footer.d.ts +20 -0
  192. package/dist/ui/footer.js +65 -0
  193. package/dist/ui/globals.css +395 -0
  194. package/dist/ui/header.d.ts +41 -0
  195. package/dist/ui/header.js +270 -0
  196. package/dist/ui/layout-wrapper.d.ts +1 -0
  197. package/dist/ui/layout-wrapper.js +7 -0
  198. package/dist/utils/color.d.ts +4 -0
  199. package/dist/utils/color.js +14 -0
  200. package/dist/utils/index.d.ts +15 -0
  201. package/dist/utils/index.js +48 -0
  202. package/dist/utils/poly.d.ts +8 -0
  203. package/dist/utils/poly.js +3 -0
  204. package/package.json +120 -0
  205. package/src/assets/filler-bg0.webp +0 -0
  206. package/src/assets.d.ts +38 -0
  207. package/src/fonts/Collapse-Bold.woff2 +0 -0
  208. package/src/fonts/Collapse-BoldItalic.woff2 +0 -0
  209. package/src/fonts/Collapse-Italic.woff2 +0 -0
  210. package/src/fonts/Collapse-Light.woff2 +0 -0
  211. package/src/fonts/Collapse-LightItalic.woff2 +0 -0
  212. package/src/fonts/Collapse-Regular.woff2 +0 -0
  213. package/src/fonts/Collapse-Thin.woff2 +0 -0
  214. package/src/fonts/Collapse-ThinItalic.woff2 +0 -0
  215. package/src/fonts/Mondwest-Regular.woff2 +0 -0
  216. package/src/fonts/Neuebit-Bold.woff2 +0 -0
  217. package/src/fonts/RulesCompressed-Medium.woff2 +0 -0
  218. package/src/fonts/RulesCompressed-Regular.woff2 +0 -0
  219. package/src/fonts/RulesExpanded-Bold.woff2 +0 -0
  220. package/src/fonts/RulesExpanded-Regular.woff2 +0 -0
  221. package/src/fonts.ts +6 -0
  222. package/src/hooks/use-below-breakpoint.ts +21 -0
  223. package/src/hooks/use-capped-frame.ts +18 -0
  224. package/src/hooks/use-confirm-delete.ts +43 -0
  225. package/src/hooks/use-css-var-dims.ts +39 -0
  226. package/src/hooks/use-gpu-tier.ts +190 -0
  227. package/src/hooks/use-render-loop.ts +121 -0
  228. package/src/hooks/use-smooth-controls.ts +318 -0
  229. package/src/hooks/use-toast.ts +29 -0
  230. package/src/index.ts +130 -0
  231. package/src/ui/basic-page.tsx +34 -0
  232. package/src/ui/build.css +4 -0
  233. package/src/ui/components/animated-count.stories.tsx +67 -0
  234. package/src/ui/components/animated-count.tsx +168 -0
  235. package/src/ui/components/ascii.stories.tsx +30 -0
  236. package/src/ui/components/ascii.tsx +110 -0
  237. package/src/ui/components/badge.stories.tsx +31 -0
  238. package/src/ui/components/badge.tsx +60 -0
  239. package/src/ui/components/badges/nous-girl.tsx +52 -0
  240. package/src/ui/components/blend-mode.stories.tsx +33 -0
  241. package/src/ui/components/blend-mode.tsx +129 -0
  242. package/src/ui/components/blink.stories.tsx +32 -0
  243. package/src/ui/components/blink.tsx +21 -0
  244. package/src/ui/components/bottom-sheet.stories.tsx +43 -0
  245. package/src/ui/components/bottom-sheet.tsx +227 -0
  246. package/src/ui/components/button.stories.tsx +68 -0
  247. package/src/ui/components/button.tsx +170 -0
  248. package/src/ui/components/card.stories.tsx +63 -0
  249. package/src/ui/components/card.tsx +85 -0
  250. package/src/ui/components/checkbox.stories.tsx +113 -0
  251. package/src/ui/components/checkbox.tsx +36 -0
  252. package/src/ui/components/command-block.stories.tsx +52 -0
  253. package/src/ui/components/command-block.tsx +86 -0
  254. package/src/ui/components/confirm-dialog.stories.tsx +91 -0
  255. package/src/ui/components/confirm-dialog.tsx +130 -0
  256. package/src/ui/components/cursor.tsx +115 -0
  257. package/src/ui/components/dialog.stories.tsx +169 -0
  258. package/src/ui/components/dialog.tsx +177 -0
  259. package/src/ui/components/dropdown-menu.stories.tsx +52 -0
  260. package/src/ui/components/dropdown-menu.tsx +117 -0
  261. package/src/ui/components/fit-text/fit-text.css +42 -0
  262. package/src/ui/components/fit-text/index.stories.tsx +33 -0
  263. package/src/ui/components/fit-text/index.tsx +45 -0
  264. package/src/ui/components/forms.stories.tsx +173 -0
  265. package/src/ui/components/graphs/bar-chart.tsx +153 -0
  266. package/src/ui/components/graphs/index.stories.tsx +64 -0
  267. package/src/ui/components/graphs/index.tsx +4 -0
  268. package/src/ui/components/graphs/line-chart.tsx +213 -0
  269. package/src/ui/components/graphs/utils.tsx +265 -0
  270. package/src/ui/components/grid/grid.css +79 -0
  271. package/src/ui/components/grid/index.tsx +19 -0
  272. package/src/ui/components/hover-bg.stories.tsx +29 -0
  273. package/src/ui/components/hover-bg.tsx +15 -0
  274. package/src/ui/components/icons/arrow.tsx +42 -0
  275. package/src/ui/components/icons/check.tsx +14 -0
  276. package/src/ui/components/icons/chevron.tsx +45 -0
  277. package/src/ui/components/icons/discord.tsx +16 -0
  278. package/src/ui/components/icons/eye.tsx +12 -0
  279. package/src/ui/components/icons/gear.tsx +51 -0
  280. package/src/ui/components/icons/github.tsx +16 -0
  281. package/src/ui/components/icons/hamburger.tsx +52 -0
  282. package/src/ui/components/icons/heart.tsx +12 -0
  283. package/src/ui/components/icons/index.ts +12 -0
  284. package/src/ui/components/icons/link.tsx +14 -0
  285. package/src/ui/components/icons/minus.tsx +14 -0
  286. package/src/ui/components/icons/search.tsx +28 -0
  287. package/src/ui/components/image-distortion.stories.tsx +120 -0
  288. package/src/ui/components/image-distortion.tsx +499 -0
  289. package/src/ui/components/input.stories.tsx +39 -0
  290. package/src/ui/components/input.tsx +20 -0
  291. package/src/ui/components/label.stories.tsx +26 -0
  292. package/src/ui/components/label.tsx +16 -0
  293. package/src/ui/components/leva-client.tsx +14 -0
  294. package/src/ui/components/list-item.stories.tsx +83 -0
  295. package/src/ui/components/list-item.tsx +37 -0
  296. package/src/ui/components/overlays/blend-modes.ts +13 -0
  297. package/src/ui/components/overlays/glitch.tsx +243 -0
  298. package/src/ui/components/overlays/greys.tsx +386 -0
  299. package/src/ui/components/overlays/index.tsx +47 -0
  300. package/src/ui/components/overlays/lens-layers.tsx +121 -0
  301. package/src/ui/components/overlays/lens.ts +91 -0
  302. package/src/ui/components/overlays/noise.tsx +174 -0
  303. package/src/ui/components/overlays/vignette.tsx +60 -0
  304. package/src/ui/components/poster.stories.tsx +513 -0
  305. package/src/ui/components/poster.tsx +411 -0
  306. package/src/ui/components/progress.stories.tsx +48 -0
  307. package/src/ui/components/progress.tsx +56 -0
  308. package/src/ui/components/scene-canvas.tsx +254 -0
  309. package/src/ui/components/scramble.stories.tsx +49 -0
  310. package/src/ui/components/scramble.tsx +95 -0
  311. package/src/ui/components/segmented.stories.tsx +101 -0
  312. package/src/ui/components/segmented.tsx +81 -0
  313. package/src/ui/components/select.stories.tsx +88 -0
  314. package/src/ui/components/select.tsx +267 -0
  315. package/src/ui/components/selection-switcher.tsx +44 -0
  316. package/src/ui/components/separator.stories.tsx +33 -0
  317. package/src/ui/components/separator.tsx +24 -0
  318. package/src/ui/components/shader.tsx +83 -0
  319. package/src/ui/components/socials.tsx +42 -0
  320. package/src/ui/components/spinner.stories.tsx +101 -0
  321. package/src/ui/components/spinner.tsx +60 -0
  322. package/src/ui/components/stats.stories.tsx +24 -0
  323. package/src/ui/components/stats.tsx +53 -0
  324. package/src/ui/components/switch.stories.tsx +77 -0
  325. package/src/ui/components/switch.tsx +48 -0
  326. package/src/ui/components/tabs.stories.tsx +101 -0
  327. package/src/ui/components/tabs.tsx +66 -0
  328. package/src/ui/components/terminal-demo.stories.tsx +67 -0
  329. package/src/ui/components/terminal-demo.tsx +189 -0
  330. package/src/ui/components/theme-toggle.stories.tsx +47 -0
  331. package/src/ui/components/theme-toggle.tsx +66 -0
  332. package/src/ui/components/tier-card.stories.tsx +217 -0
  333. package/src/ui/components/tier-card.tsx +190 -0
  334. package/src/ui/components/toast.stories.tsx +55 -0
  335. package/src/ui/components/toast.tsx +49 -0
  336. package/src/ui/components/tv.stories.tsx +37 -0
  337. package/src/ui/components/tv.tsx +257 -0
  338. package/src/ui/components/typography/h1.tsx +18 -0
  339. package/src/ui/components/typography/h2.tsx +18 -0
  340. package/src/ui/components/typography/index.tsx +54 -0
  341. package/src/ui/components/typography/legend.tsx +24 -0
  342. package/src/ui/components/typography/small.tsx +11 -0
  343. package/src/ui/components/watchlist.stories.tsx +33 -0
  344. package/src/ui/components/watchlist.tsx +105 -0
  345. package/src/ui/fonts.css +63 -0
  346. package/src/ui/footer.tsx +111 -0
  347. package/src/ui/globals.css +395 -0
  348. package/src/ui/header.tsx +398 -0
  349. package/src/ui/layout-wrapper.tsx +11 -0
  350. package/src/utils/color.ts +21 -0
  351. package/src/utils/index.ts +62 -0
  352. package/src/utils/poly.ts +26 -0
package/CHANGELOG.md ADDED
@@ -0,0 +1,5 @@
1
+ # @nastechai-research/ui Changelog
2
+
3
+ ## 0.18.2
4
+
5
+ Initial release under the `@nastechai-research` scope.
package/README.md ADDED
@@ -0,0 +1,21 @@
1
+ # @nastechai-research/ui
2
+
3
+ The shared design system for React UI @ Nastechai. Components, hooks, utils, fonts, overlays, and more.
4
+
5
+ ## Installation
6
+
7
+ ```bash
8
+ npm install @nastechai-research/ui
9
+ ```
10
+
11
+ ## Usage
12
+
13
+ ```tsx
14
+ import { Button } from '@nastechai-research/ui/ui/components/button'
15
+ import { Spinner } from '@nastechai-research/ui/ui/components/spinner'
16
+ import { Typography } from '@nastechai-research/ui/ui/components/typography/index'
17
+ ```
18
+
19
+ ## License
20
+
21
+ MIT © Nastechai Research
Binary file
@@ -0,0 +1,38 @@
1
+ interface StaticImageData {
2
+ src: string
3
+ height: number
4
+ width: number
5
+ blurDataURL?: string
6
+ blurWidth?: number
7
+ blurHeight?: number
8
+ }
9
+
10
+ declare module '*.jpg' {
11
+ const content: StaticImageData
12
+ export default content
13
+ }
14
+
15
+ declare module '*.jpeg' {
16
+ const content: StaticImageData
17
+ export default content
18
+ }
19
+
20
+ declare module '*.png' {
21
+ const content: StaticImageData
22
+ export default content
23
+ }
24
+
25
+ declare module '*.svg' {
26
+ const content: StaticImageData
27
+ export default content
28
+ }
29
+
30
+ declare module '*.gif' {
31
+ const content: StaticImageData
32
+ export default content
33
+ }
34
+
35
+ declare module '*.webp' {
36
+ const content: StaticImageData
37
+ export default content
38
+ }
Binary file
Binary file
Binary file
Binary file
Binary file
@@ -0,0 +1,6 @@
1
+ /** CSS variable names for the design language fonts. Set automatically by fonts.css. */
2
+ export declare const FONT_SANS = "--font-sans";
3
+ export declare const FONT_MONO = "--font-mono";
4
+ export declare const FONT_RULES_COMPRESSED = "--font-rules-compressed";
5
+ export declare const FONT_RULES_EXPANDED = "--font-rules-expanded";
6
+ export declare const FONT_MONDWEST = "--font-mondwest";
package/dist/fonts.js ADDED
@@ -0,0 +1,6 @@
1
+ export const FONT_SANS = "--font-sans";
2
+ export const FONT_MONO = "--font-mono";
3
+ export const FONT_RULES_COMPRESSED = "--font-rules-compressed";
4
+ export const FONT_RULES_EXPANDED = "--font-rules-expanded";
5
+ export const FONT_MONDWEST = "--font-mondwest";
6
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiPHN0ZGluPiJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiLyoqIENTUyB2YXJpYWJsZSBuYW1lcyBmb3IgdGhlIGRlc2lnbiBsYW5ndWFnZSBmb250cy4gU2V0IGF1dG9tYXRpY2FsbHkgYnkgZm9udHMuY3NzLiAqL1xuZXhwb3J0IGNvbnN0IEZPTlRfU0FOUyA9ICctLWZvbnQtc2FucydcbmV4cG9ydCBjb25zdCBGT05UX01PTk8gPSAnLS1mb250LW1vbm8nXG5leHBvcnQgY29uc3QgRk9OVF9SVUxFU19DT01QUkVTU0VEID0gJy0tZm9udC1ydWxlcy1jb21wcmVzc2VkJ1xuZXhwb3J0IGNvbnN0IEZPTlRfUlVMRVNfRVhQQU5ERUQgPSAnLS1mb250LXJ1bGVzLWV4cGFuZGVkJ1xuZXhwb3J0IGNvbnN0IEZPTlRfTU9ORFdFU1QgPSAnLS1mb250LW1vbmR3ZXN0J1xuIl0sCiAgIm1hcHBpbmdzIjogIkFBQ08sYUFBTSxZQUFZO0FBQ2xCLGFBQU0sWUFBWTtBQUNsQixhQUFNLHdCQUF3QjtBQUM5QixhQUFNLHNCQUFzQjtBQUM1QixhQUFNLGdCQUFnQjsiLAogICJuYW1lcyI6IFtdCn0K
@@ -0,0 +1,2 @@
1
+ /** True when viewport width is strictly below `px`. */
2
+ export declare function useBelowBreakpoint(px: number): boolean;
@@ -0,0 +1,17 @@
1
+ "use client";
2
+ import { useEffect, useState } from "react";
3
+ export function useBelowBreakpoint(px) {
4
+ const query = `(max-width: ${px - 1}px)`;
5
+ const [matches, setMatches] = useState(
6
+ () => typeof window !== "undefined" ? window.matchMedia(query).matches : false
7
+ );
8
+ useEffect(() => {
9
+ const mql = window.matchMedia(query);
10
+ const sync = () => setMatches(mql.matches);
11
+ sync();
12
+ mql.addEventListener("change", sync);
13
+ return () => mql.removeEventListener("change", sync);
14
+ }, [query]);
15
+ return matches;
16
+ }
17
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiPHN0ZGluPiJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiJ3VzZSBjbGllbnQnXG5cbmltcG9ydCB7IHVzZUVmZmVjdCwgdXNlU3RhdGUgfSBmcm9tICdyZWFjdCdcblxuLyoqIFRydWUgd2hlbiB2aWV3cG9ydCB3aWR0aCBpcyBzdHJpY3RseSBiZWxvdyBgcHhgLiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHVzZUJlbG93QnJlYWtwb2ludChweDogbnVtYmVyKSB7XG4gIGNvbnN0IHF1ZXJ5ID0gYChtYXgtd2lkdGg6ICR7cHggLSAxfXB4KWBcbiAgY29uc3QgW21hdGNoZXMsIHNldE1hdGNoZXNdID0gdXNlU3RhdGUoKCkgPT5cbiAgICB0eXBlb2Ygd2luZG93ICE9PSAndW5kZWZpbmVkJyA/IHdpbmRvdy5tYXRjaE1lZGlhKHF1ZXJ5KS5tYXRjaGVzIDogZmFsc2VcbiAgKVxuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgY29uc3QgbXFsID0gd2luZG93Lm1hdGNoTWVkaWEocXVlcnkpXG4gICAgY29uc3Qgc3luYyA9ICgpID0+IHNldE1hdGNoZXMobXFsLm1hdGNoZXMpXG4gICAgc3luYygpXG4gICAgbXFsLmFkZEV2ZW50TGlzdGVuZXIoJ2NoYW5nZScsIHN5bmMpXG4gICAgcmV0dXJuICgpID0+IG1xbC5yZW1vdmVFdmVudExpc3RlbmVyKCdjaGFuZ2UnLCBzeW5jKVxuICB9LCBbcXVlcnldKVxuXG4gIHJldHVybiBtYXRjaGVzXG59XG4iXSwKICAibWFwcGluZ3MiOiAiO0FBRUEsU0FBUyxXQUFXLGdCQUFnQjtBQUc3QixnQkFBUyxtQkFBbUIsSUFBWTtBQUM3QyxRQUFNLFFBQVEsZUFBZSxLQUFLLENBQUM7QUFDbkMsUUFBTSxDQUFDLFNBQVMsVUFBVSxJQUFJO0FBQUEsSUFBUyxNQUNyQyxPQUFPLFdBQVcsY0FBYyxPQUFPLFdBQVcsS0FBSyxFQUFFLFVBQVU7QUFBQSxFQUNyRTtBQUVBLFlBQVUsTUFBTTtBQUNkLFVBQU0sTUFBTSxPQUFPLFdBQVcsS0FBSztBQUNuQyxVQUFNLE9BQU8sTUFBTSxXQUFXLElBQUksT0FBTztBQUN6QyxTQUFLO0FBQ0wsUUFBSSxpQkFBaUIsVUFBVSxJQUFJO0FBQ25DLFdBQU8sTUFBTSxJQUFJLG9CQUFvQixVQUFVLElBQUk7QUFBQSxFQUNyRCxHQUFHLENBQUMsS0FBSyxDQUFDO0FBRVYsU0FBTztBQUNUOyIsCiAgIm5hbWVzIjogW10KfQo=
@@ -0,0 +1,2 @@
1
+ import type { RenderCallback } from '@react-three/fiber';
2
+ export declare function useCappedFrame(cb: RenderCallback, max?: number): void;
@@ -0,0 +1,15 @@
1
+ "use client";
2
+ import { useFrame, useThree } from "@react-three/fiber";
3
+ import { useRef } from "react";
4
+ export function useCappedFrame(cb, max) {
5
+ const last = useRef(performance.now());
6
+ const { size } = useThree();
7
+ const interval = 1e3 / (max ?? (size.width < 1024 ? 60 : 120));
8
+ useFrame((st, delta) => {
9
+ if (performance.now() - last.current > interval) {
10
+ last.current = performance.now();
11
+ cb(st, delta);
12
+ }
13
+ });
14
+ }
15
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiPHN0ZGluPiJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiJ3VzZSBjbGllbnQnXG5cbmltcG9ydCB7IHVzZUZyYW1lLCB1c2VUaHJlZSB9IGZyb20gJ0ByZWFjdC10aHJlZS9maWJlcidcbmltcG9ydCB0eXBlIHsgUmVuZGVyQ2FsbGJhY2sgfSBmcm9tICdAcmVhY3QtdGhyZWUvZmliZXInXG5pbXBvcnQgeyB1c2VSZWYgfSBmcm9tICdyZWFjdCdcblxuZXhwb3J0IGZ1bmN0aW9uIHVzZUNhcHBlZEZyYW1lKGNiOiBSZW5kZXJDYWxsYmFjaywgbWF4PzogbnVtYmVyKSB7XG4gIGNvbnN0IGxhc3QgPSB1c2VSZWYocGVyZm9ybWFuY2Uubm93KCkpXG4gIGNvbnN0IHsgc2l6ZSB9ID0gdXNlVGhyZWUoKVxuICBjb25zdCBpbnRlcnZhbCA9IDFlMyAvIChtYXggPz8gKHNpemUud2lkdGggPCAxMDI0ID8gNjAgOiAxMjApKVxuXG4gIHVzZUZyYW1lKChzdCwgZGVsdGEpID0+IHtcbiAgICBpZiAocGVyZm9ybWFuY2Uubm93KCkgLSBsYXN0LmN1cnJlbnQgPiBpbnRlcnZhbCkge1xuICAgICAgbGFzdC5jdXJyZW50ID0gcGVyZm9ybWFuY2Uubm93KClcbiAgICAgIGNiKHN0LCBkZWx0YSlcbiAgICB9XG4gIH0pXG59XG4iXSwKICAibWFwcGluZ3MiOiAiO0FBRUEsU0FBUyxVQUFVLGdCQUFnQjtBQUVuQyxTQUFTLGNBQWM7QUFFaEIsZ0JBQVMsZUFBZSxJQUFvQixLQUFjO0FBQy9ELFFBQU0sT0FBTyxPQUFPLFlBQVksSUFBSSxDQUFDO0FBQ3JDLFFBQU0sRUFBRSxLQUFLLElBQUksU0FBUztBQUMxQixRQUFNLFdBQVcsT0FBTyxRQUFRLEtBQUssUUFBUSxPQUFPLEtBQUs7QUFFekQsV0FBUyxDQUFDLElBQUksVUFBVTtBQUN0QixRQUFJLFlBQVksSUFBSSxJQUFJLEtBQUssVUFBVSxVQUFVO0FBQy9DLFdBQUssVUFBVSxZQUFZLElBQUk7QUFDL0IsU0FBRyxJQUFJLEtBQUs7QUFBQSxJQUNkO0FBQUEsRUFDRixDQUFDO0FBQ0g7IiwKICAibmFtZXMiOiBbXQp9Cg==
@@ -0,0 +1,10 @@
1
+ export declare function useConfirmDelete<TId>({ onDelete }: {
2
+ onDelete: (id: TId) => Promise<void>;
3
+ }): {
4
+ readonly cancel: () => void;
5
+ readonly confirm: () => Promise<void>;
6
+ readonly isDeleting: boolean;
7
+ readonly isOpen: boolean;
8
+ readonly pendingId: TId | null;
9
+ readonly requestDelete: (id: TId) => void;
10
+ };
@@ -0,0 +1,35 @@
1
+ "use client";
2
+ import { useCallback, useState } from "react";
3
+ export function useConfirmDelete({
4
+ onDelete
5
+ }) {
6
+ const [pendingId, setPendingId] = useState(null);
7
+ const [isDeleting, setIsDeleting] = useState(false);
8
+ const requestDelete = useCallback((id) => {
9
+ setPendingId(id);
10
+ }, []);
11
+ const cancel = useCallback(() => {
12
+ if (!isDeleting) setPendingId(null);
13
+ }, [isDeleting]);
14
+ const confirm = useCallback(async () => {
15
+ if (pendingId === null) return;
16
+ const id = pendingId;
17
+ setIsDeleting(true);
18
+ try {
19
+ await onDelete(id);
20
+ setPendingId(null);
21
+ } catch {
22
+ } finally {
23
+ setIsDeleting(false);
24
+ }
25
+ }, [pendingId, onDelete]);
26
+ return {
27
+ cancel,
28
+ confirm,
29
+ isDeleting,
30
+ isOpen: pendingId !== null,
31
+ pendingId,
32
+ requestDelete
33
+ };
34
+ }
35
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiPHN0ZGluPiJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiJ3VzZSBjbGllbnQnXG5cbmltcG9ydCB7IHVzZUNhbGxiYWNrLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuXG5leHBvcnQgZnVuY3Rpb24gdXNlQ29uZmlybURlbGV0ZTxUSWQ+KHtcbiAgb25EZWxldGVcbn06IHtcbiAgb25EZWxldGU6IChpZDogVElkKSA9PiBQcm9taXNlPHZvaWQ+XG59KSB7XG4gIGNvbnN0IFtwZW5kaW5nSWQsIHNldFBlbmRpbmdJZF0gPSB1c2VTdGF0ZTxUSWQgfCBudWxsPihudWxsKVxuICBjb25zdCBbaXNEZWxldGluZywgc2V0SXNEZWxldGluZ10gPSB1c2VTdGF0ZShmYWxzZSlcblxuICBjb25zdCByZXF1ZXN0RGVsZXRlID0gdXNlQ2FsbGJhY2soKGlkOiBUSWQpID0+IHtcbiAgICBzZXRQZW5kaW5nSWQoaWQpXG4gIH0sIFtdKVxuXG4gIGNvbnN0IGNhbmNlbCA9IHVzZUNhbGxiYWNrKCgpID0+IHtcbiAgICBpZiAoIWlzRGVsZXRpbmcpIHNldFBlbmRpbmdJZChudWxsKVxuICB9LCBbaXNEZWxldGluZ10pXG5cbiAgY29uc3QgY29uZmlybSA9IHVzZUNhbGxiYWNrKGFzeW5jICgpID0+IHtcbiAgICBpZiAocGVuZGluZ0lkID09PSBudWxsKSByZXR1cm5cbiAgICBjb25zdCBpZCA9IHBlbmRpbmdJZFxuICAgIHNldElzRGVsZXRpbmcodHJ1ZSlcbiAgICB0cnkge1xuICAgICAgYXdhaXQgb25EZWxldGUoaWQpXG4gICAgICBzZXRQZW5kaW5nSWQobnVsbClcbiAgICB9IGNhdGNoIHtcbiAgICAgIC8vIERpYWxvZyBzdGF5cyBvcGVuOyBjYWxsZXIgY2FuIHN1cmZhY2UgZXJyb3JzIGluIG9uRGVsZXRlXG4gICAgfSBmaW5hbGx5IHtcbiAgICAgIHNldElzRGVsZXRpbmcoZmFsc2UpXG4gICAgfVxuICB9LCBbcGVuZGluZ0lkLCBvbkRlbGV0ZV0pXG5cbiAgcmV0dXJuIHtcbiAgICBjYW5jZWwsXG4gICAgY29uZmlybSxcbiAgICBpc0RlbGV0aW5nLFxuICAgIGlzT3BlbjogcGVuZGluZ0lkICE9PSBudWxsLFxuICAgIHBlbmRpbmdJZCxcbiAgICByZXF1ZXN0RGVsZXRlXG4gIH0gYXMgY29uc3Rcbn1cbiJdLAogICJtYXBwaW5ncyI6ICI7QUFFQSxTQUFTLGFBQWEsZ0JBQWdCO0FBRS9CLGdCQUFTLGlCQUFzQjtBQUFBLEVBQ3BDO0FBQ0YsR0FFRztBQUNELFFBQU0sQ0FBQyxXQUFXLFlBQVksSUFBSSxTQUFxQixJQUFJO0FBQzNELFFBQU0sQ0FBQyxZQUFZLGFBQWEsSUFBSSxTQUFTLEtBQUs7QUFFbEQsUUFBTSxnQkFBZ0IsWUFBWSxDQUFDLE9BQVk7QUFDN0MsaUJBQWEsRUFBRTtBQUFBLEVBQ2pCLEdBQUcsQ0FBQyxDQUFDO0FBRUwsUUFBTSxTQUFTLFlBQVksTUFBTTtBQUMvQixRQUFJLENBQUMsV0FBWSxjQUFhLElBQUk7QUFBQSxFQUNwQyxHQUFHLENBQUMsVUFBVSxDQUFDO0FBRWYsUUFBTSxVQUFVLFlBQVksWUFBWTtBQUN0QyxRQUFJLGNBQWMsS0FBTTtBQUN4QixVQUFNLEtBQUs7QUFDWCxrQkFBYyxJQUFJO0FBQ2xCLFFBQUk7QUFDRixZQUFNLFNBQVMsRUFBRTtBQUNqQixtQkFBYSxJQUFJO0FBQUEsSUFDbkIsUUFBUTtBQUFBLElBRVIsVUFBRTtBQUNBLG9CQUFjLEtBQUs7QUFBQSxJQUNyQjtBQUFBLEVBQ0YsR0FBRyxDQUFDLFdBQVcsUUFBUSxDQUFDO0FBRXhCLFNBQU87QUFBQSxJQUNMO0FBQUEsSUFDQTtBQUFBLElBQ0E7QUFBQSxJQUNBLFFBQVEsY0FBYztBQUFBLElBQ3RCO0FBQUEsSUFDQTtBQUFBLEVBQ0Y7QUFDRjsiLAogICJuYW1lcyI6IFtdCn0K
@@ -0,0 +1 @@
1
+ export declare function useCssVarDims(name: string, ref: React.RefObject<HTMLElement | null>): void;
@@ -0,0 +1,29 @@
1
+ "use client";
2
+ import { useEffect } from "react";
3
+ export function useCssVarDims(name, ref) {
4
+ useEffect(() => {
5
+ if (!ref.current) {
6
+ return;
7
+ }
8
+ const update = (width2, height2) => {
9
+ document.documentElement.style.setProperty(
10
+ `--${name}-width`,
11
+ `${width2}px`
12
+ );
13
+ document.documentElement.style.setProperty(
14
+ `--${name}-height`,
15
+ `${height2}px`
16
+ );
17
+ };
18
+ const { height, width } = ref.current.getBoundingClientRect();
19
+ update(width, height);
20
+ const ro = new ResizeObserver((entries) => {
21
+ for (const entry of entries) {
22
+ update(entry.contentRect.width, entry.contentRect.height);
23
+ }
24
+ });
25
+ ro.observe(ref.current);
26
+ return () => ro.disconnect();
27
+ }, [name, ref]);
28
+ }
29
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiPHN0ZGluPiJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiJ3VzZSBjbGllbnQnXG5cbmltcG9ydCB7IHVzZUVmZmVjdCB9IGZyb20gJ3JlYWN0J1xuXG5leHBvcnQgZnVuY3Rpb24gdXNlQ3NzVmFyRGltcyhcbiAgbmFtZTogc3RyaW5nLFxuICByZWY6IFJlYWN0LlJlZk9iamVjdDxIVE1MRWxlbWVudCB8IG51bGw+XG4pIHtcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAoIXJlZi5jdXJyZW50KSB7XG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICBjb25zdCB1cGRhdGUgPSAod2lkdGg6IG51bWJlciwgaGVpZ2h0OiBudW1iZXIpID0+IHtcbiAgICAgIGRvY3VtZW50LmRvY3VtZW50RWxlbWVudC5zdHlsZS5zZXRQcm9wZXJ0eShcbiAgICAgICAgYC0tJHtuYW1lfS13aWR0aGAsXG4gICAgICAgIGAke3dpZHRofXB4YFxuICAgICAgKVxuXG4gICAgICBkb2N1bWVudC5kb2N1bWVudEVsZW1lbnQuc3R5bGUuc2V0UHJvcGVydHkoXG4gICAgICAgIGAtLSR7bmFtZX0taGVpZ2h0YCxcbiAgICAgICAgYCR7aGVpZ2h0fXB4YFxuICAgICAgKVxuICAgIH1cblxuICAgIGNvbnN0IHsgaGVpZ2h0LCB3aWR0aCB9ID0gcmVmLmN1cnJlbnQuZ2V0Qm91bmRpbmdDbGllbnRSZWN0KClcbiAgICB1cGRhdGUod2lkdGgsIGhlaWdodClcblxuICAgIGNvbnN0IHJvID0gbmV3IFJlc2l6ZU9ic2VydmVyKGVudHJpZXMgPT4ge1xuICAgICAgZm9yIChjb25zdCBlbnRyeSBvZiBlbnRyaWVzKSB7XG4gICAgICAgIHVwZGF0ZShlbnRyeS5jb250ZW50UmVjdC53aWR0aCwgZW50cnkuY29udGVudFJlY3QuaGVpZ2h0KVxuICAgICAgfVxuICAgIH0pXG5cbiAgICByby5vYnNlcnZlKHJlZi5jdXJyZW50KVxuXG4gICAgcmV0dXJuICgpID0+IHJvLmRpc2Nvbm5lY3QoKVxuICB9LCBbbmFtZSwgcmVmXSlcbn1cbiJdLAogICJtYXBwaW5ncyI6ICI7QUFFQSxTQUFTLGlCQUFpQjtBQUVuQixnQkFBUyxjQUNkLE1BQ0EsS0FDQTtBQUNBLFlBQVUsTUFBTTtBQUNkLFFBQUksQ0FBQyxJQUFJLFNBQVM7QUFDaEI7QUFBQSxJQUNGO0FBRUEsVUFBTSxTQUFTLENBQUNBLFFBQWVDLFlBQW1CO0FBQ2hELGVBQVMsZ0JBQWdCLE1BQU07QUFBQSxRQUM3QixLQUFLLElBQUk7QUFBQSxRQUNULEdBQUdELE1BQUs7QUFBQSxNQUNWO0FBRUEsZUFBUyxnQkFBZ0IsTUFBTTtBQUFBLFFBQzdCLEtBQUssSUFBSTtBQUFBLFFBQ1QsR0FBR0MsT0FBTTtBQUFBLE1BQ1g7QUFBQSxJQUNGO0FBRUEsVUFBTSxFQUFFLFFBQVEsTUFBTSxJQUFJLElBQUksUUFBUSxzQkFBc0I7QUFDNUQsV0FBTyxPQUFPLE1BQU07QUFFcEIsVUFBTSxLQUFLLElBQUksZUFBZSxhQUFXO0FBQ3ZDLGlCQUFXLFNBQVMsU0FBUztBQUMzQixlQUFPLE1BQU0sWUFBWSxPQUFPLE1BQU0sWUFBWSxNQUFNO0FBQUEsTUFDMUQ7QUFBQSxJQUNGLENBQUM7QUFFRCxPQUFHLFFBQVEsSUFBSSxPQUFPO0FBRXRCLFdBQU8sTUFBTSxHQUFHLFdBQVc7QUFBQSxFQUM3QixHQUFHLENBQUMsTUFBTSxHQUFHLENBQUM7QUFDaEI7IiwKICAibmFtZXMiOiBbIndpZHRoIiwgImhlaWdodCJdCn0K
@@ -0,0 +1,34 @@
1
+ /**
2
+ * Tiers:
3
+ * 0 — no WebGL / software renderer / prefers-reduced-motion / WebGL ctx
4
+ * creation failed / detection not yet run
5
+ * 1 — low-end GPU (integrated, mobile, or failed perf benchmark)
6
+ * 2 — capable GPU (discrete / high-end integrated)
7
+ *
8
+ * The atom starts **pessimistic** (`0`) and the probe is scheduled to run
9
+ * *after the first paint* (see `scheduleDetection` below), never synchronously
10
+ * at module-evaluation time.
11
+ *
12
+ * Why pessimistic + deferred:
13
+ * - Every consumer gates its `<canvas>` / `THREE.WebGLRenderer` on `tier > 0`
14
+ * (or `=== 2`). Starting at `0` means a consumer reading `$gpuTier` during
15
+ * its first render never attempts to create a renderer before detection has
16
+ * run, so the `THREE.WebGLRenderer: Error creating WebGL context` crash on
17
+ * hardware where context creation fails cannot happen — the component
18
+ * renders its fallback and upgrades once detection resolves to a capable
19
+ * tier. (A previous version made the probe synchronous-at-module-load to
20
+ * dodge this crash; the pessimistic default removes the crash without that
21
+ * cost.)
22
+ * - Probing WebGL is *expensive* on software renderers: creating a context
23
+ * under SwiftShader / llvmpipe can block the main thread for hundreds of
24
+ * milliseconds. Running it synchronously at module load stalled first paint
25
+ * and produced a visible boot-time flash in apps that merely import this
26
+ * hook (e.g. the Hermes dashboard backdrop). Deferring past first paint
27
+ * keeps boot smooth; the tier just upgrades a frame or two later.
28
+ * - For SSR the server keeps the default `0` and the client's first render
29
+ * also reads `0`, so there is no hydration mismatch.
30
+ */
31
+ export declare const $gpuTier: import("nanostores").PreinitializedWritableAtom<GpuTier> & object;
32
+ export declare function useGpuTier(): GpuTier;
33
+ type GpuTier = 0 | 1 | 2;
34
+ export {};
@@ -0,0 +1,111 @@
1
+ "use client";
2
+ import { useStore } from "@nanostores/react";
3
+ import { atom } from "nanostores";
4
+ export const $gpuTier = atom(0);
5
+ const SOFTWARE_PATTERNS = /swiftshader|llvmpipe|softpipe|software|microsoft basic/i;
6
+ const LOW_END_PATTERNS = /intel.*hd|intel.*uhd|intel.*iris|mali|adreno\s?[1-5]|powervr|apple gpu/i;
7
+ let detected = false;
8
+ function detectGpuTier() {
9
+ if (detected || typeof window === "undefined") {
10
+ return;
11
+ }
12
+ detected = true;
13
+ if (window.matchMedia("(prefers-reduced-motion: reduce)").matches) {
14
+ return;
15
+ }
16
+ let gl = null;
17
+ try {
18
+ const canvas = document.createElement("canvas");
19
+ gl = canvas.getContext("webgl") || canvas.getContext("experimental-webgl");
20
+ } catch {
21
+ return;
22
+ }
23
+ if (!gl) {
24
+ return;
25
+ }
26
+ const ext = gl.getExtension("WEBGL_debug_renderer_info");
27
+ const renderer = String(
28
+ ext ? gl.getParameter(ext.UNMASKED_RENDERER_WEBGL) : gl.getParameter(gl.RENDERER)
29
+ );
30
+ if (SOFTWARE_PATTERNS.test(renderer)) {
31
+ gl.getExtension("WEBGL_lose_context")?.loseContext();
32
+ return;
33
+ }
34
+ if (LOW_END_PATTERNS.test(renderer)) {
35
+ $gpuTier.set(1);
36
+ gl.getExtension("WEBGL_lose_context")?.loseContext();
37
+ return;
38
+ }
39
+ $gpuTier.set(2);
40
+ runBenchmark(gl).then((fps) => $gpuTier.set(fps < 30 ? 1 : 2)).catch(() => $gpuTier.set(1)).finally(() => gl?.getExtension("WEBGL_lose_context")?.loseContext());
41
+ }
42
+ function scheduleDetection() {
43
+ if (typeof window === "undefined") {
44
+ return;
45
+ }
46
+ if (typeof window.requestIdleCallback === "function") {
47
+ window.requestIdleCallback(() => detectGpuTier(), { timeout: 1e3 });
48
+ } else if (typeof window.requestAnimationFrame === "function") {
49
+ window.requestAnimationFrame(
50
+ () => window.requestAnimationFrame(() => detectGpuTier())
51
+ );
52
+ } else {
53
+ setTimeout(() => detectGpuTier(), 0);
54
+ }
55
+ }
56
+ scheduleDetection();
57
+ function runBenchmark(gl) {
58
+ return new Promise((resolve) => {
59
+ const vs = gl.createShader(gl.VERTEX_SHADER);
60
+ const fs = gl.createShader(gl.FRAGMENT_SHADER);
61
+ gl.shaderSource(
62
+ vs,
63
+ "attribute vec2 a;void main(){gl_Position=vec4(a,0,1);}"
64
+ );
65
+ gl.shaderSource(
66
+ fs,
67
+ "precision highp float;uniform float t;void main(){float v=0.;for(int i=0;i<64;i++)v+=sin(float(i)*t*.01);gl_FragColor=vec4(v*.001);}"
68
+ );
69
+ gl.compileShader(vs);
70
+ gl.compileShader(fs);
71
+ const prog = gl.createProgram();
72
+ gl.attachShader(prog, vs);
73
+ gl.attachShader(prog, fs);
74
+ gl.linkProgram(prog);
75
+ gl.useProgram(prog);
76
+ const buf = gl.createBuffer();
77
+ gl.bindBuffer(gl.ARRAY_BUFFER, buf);
78
+ gl.bufferData(
79
+ gl.ARRAY_BUFFER,
80
+ new Float32Array([-1, -1, 1, -1, -1, 1, 1, 1]),
81
+ gl.STATIC_DRAW
82
+ );
83
+ const a = gl.getAttribLocation(prog, "a");
84
+ gl.enableVertexAttribArray(a);
85
+ gl.vertexAttribPointer(a, 2, gl.FLOAT, false, 0, 0);
86
+ const uT = gl.getUniformLocation(prog, "t");
87
+ let frames = 0;
88
+ const start = performance.now();
89
+ const tick = () => {
90
+ gl.uniform1f(uT, frames);
91
+ gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4);
92
+ gl.finish();
93
+ frames++;
94
+ if (performance.now() - start < 200) {
95
+ requestAnimationFrame(tick);
96
+ } else {
97
+ const elapsed = performance.now() - start;
98
+ gl.deleteProgram(prog);
99
+ gl.deleteShader(vs);
100
+ gl.deleteShader(fs);
101
+ gl.deleteBuffer(buf);
102
+ resolve(frames / elapsed * 1e3);
103
+ }
104
+ };
105
+ requestAnimationFrame(tick);
106
+ });
107
+ }
108
+ export function useGpuTier() {
109
+ return useStore($gpuTier);
110
+ }
111
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiPHN0ZGluPiJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiJ3VzZSBjbGllbnQnXG5cbmltcG9ydCB7IHVzZVN0b3JlIH0gZnJvbSAnQG5hbm9zdG9yZXMvcmVhY3QnXG5pbXBvcnQgeyBhdG9tIH0gZnJvbSAnbmFub3N0b3JlcydcblxuLyoqXG4gKiBUaWVyczpcbiAqICAwIFx1MjAxNCBubyBXZWJHTCAvIHNvZnR3YXJlIHJlbmRlcmVyIC8gcHJlZmVycy1yZWR1Y2VkLW1vdGlvbiAvIFdlYkdMIGN0eFxuICogICAgICBjcmVhdGlvbiBmYWlsZWQgLyBkZXRlY3Rpb24gbm90IHlldCBydW5cbiAqICAxIFx1MjAxNCBsb3ctZW5kIEdQVSAoaW50ZWdyYXRlZCwgbW9iaWxlLCBvciBmYWlsZWQgcGVyZiBiZW5jaG1hcmspXG4gKiAgMiBcdTIwMTQgY2FwYWJsZSBHUFUgKGRpc2NyZXRlIC8gaGlnaC1lbmQgaW50ZWdyYXRlZClcbiAqXG4gKiBUaGUgYXRvbSBzdGFydHMgKipwZXNzaW1pc3RpYyoqIChgMGApIGFuZCB0aGUgcHJvYmUgaXMgc2NoZWR1bGVkIHRvIHJ1blxuICogKmFmdGVyIHRoZSBmaXJzdCBwYWludCogKHNlZSBgc2NoZWR1bGVEZXRlY3Rpb25gIGJlbG93KSwgbmV2ZXIgc3luY2hyb25vdXNseVxuICogYXQgbW9kdWxlLWV2YWx1YXRpb24gdGltZS5cbiAqXG4gKiBXaHkgcGVzc2ltaXN0aWMgKyBkZWZlcnJlZDpcbiAqICAtIEV2ZXJ5IGNvbnN1bWVyIGdhdGVzIGl0cyBgPGNhbnZhcz5gIC8gYFRIUkVFLldlYkdMUmVuZGVyZXJgIG9uIGB0aWVyID4gMGBcbiAqICAgIChvciBgPT09IDJgKS4gU3RhcnRpbmcgYXQgYDBgIG1lYW5zIGEgY29uc3VtZXIgcmVhZGluZyBgJGdwdVRpZXJgIGR1cmluZ1xuICogICAgaXRzIGZpcnN0IHJlbmRlciBuZXZlciBhdHRlbXB0cyB0byBjcmVhdGUgYSByZW5kZXJlciBiZWZvcmUgZGV0ZWN0aW9uIGhhc1xuICogICAgcnVuLCBzbyB0aGUgYFRIUkVFLldlYkdMUmVuZGVyZXI6IEVycm9yIGNyZWF0aW5nIFdlYkdMIGNvbnRleHRgIGNyYXNoIG9uXG4gKiAgICBoYXJkd2FyZSB3aGVyZSBjb250ZXh0IGNyZWF0aW9uIGZhaWxzIGNhbm5vdCBoYXBwZW4gXHUyMDE0IHRoZSBjb21wb25lbnRcbiAqICAgIHJlbmRlcnMgaXRzIGZhbGxiYWNrIGFuZCB1cGdyYWRlcyBvbmNlIGRldGVjdGlvbiByZXNvbHZlcyB0byBhIGNhcGFibGVcbiAqICAgIHRpZXIuIChBIHByZXZpb3VzIHZlcnNpb24gbWFkZSB0aGUgcHJvYmUgc3luY2hyb25vdXMtYXQtbW9kdWxlLWxvYWQgdG9cbiAqICAgIGRvZGdlIHRoaXMgY3Jhc2g7IHRoZSBwZXNzaW1pc3RpYyBkZWZhdWx0IHJlbW92ZXMgdGhlIGNyYXNoIHdpdGhvdXQgdGhhdFxuICogICAgY29zdC4pXG4gKiAgLSBQcm9iaW5nIFdlYkdMIGlzICpleHBlbnNpdmUqIG9uIHNvZnR3YXJlIHJlbmRlcmVyczogY3JlYXRpbmcgYSBjb250ZXh0XG4gKiAgICB1bmRlciBTd2lmdFNoYWRlciAvIGxsdm1waXBlIGNhbiBibG9jayB0aGUgbWFpbiB0aHJlYWQgZm9yIGh1bmRyZWRzIG9mXG4gKiAgICBtaWxsaXNlY29uZHMuIFJ1bm5pbmcgaXQgc3luY2hyb25vdXNseSBhdCBtb2R1bGUgbG9hZCBzdGFsbGVkIGZpcnN0IHBhaW50XG4gKiAgICBhbmQgcHJvZHVjZWQgYSB2aXNpYmxlIGJvb3QtdGltZSBmbGFzaCBpbiBhcHBzIHRoYXQgbWVyZWx5IGltcG9ydCB0aGlzXG4gKiAgICBob29rIChlLmcuIHRoZSBIZXJtZXMgZGFzaGJvYXJkIGJhY2tkcm9wKS4gRGVmZXJyaW5nIHBhc3QgZmlyc3QgcGFpbnRcbiAqICAgIGtlZXBzIGJvb3Qgc21vb3RoOyB0aGUgdGllciBqdXN0IHVwZ3JhZGVzIGEgZnJhbWUgb3IgdHdvIGxhdGVyLlxuICogIC0gRm9yIFNTUiB0aGUgc2VydmVyIGtlZXBzIHRoZSBkZWZhdWx0IGAwYCBhbmQgdGhlIGNsaWVudCdzIGZpcnN0IHJlbmRlclxuICogICAgYWxzbyByZWFkcyBgMGAsIHNvIHRoZXJlIGlzIG5vIGh5ZHJhdGlvbiBtaXNtYXRjaC5cbiAqL1xuZXhwb3J0IGNvbnN0ICRncHVUaWVyID0gYXRvbTxHcHVUaWVyPigwKVxuXG5jb25zdCBTT0ZUV0FSRV9QQVRURVJOUyA9XG4gIC9zd2lmdHNoYWRlcnxsbHZtcGlwZXxzb2Z0cGlwZXxzb2Z0d2FyZXxtaWNyb3NvZnQgYmFzaWMvaVxuXG5jb25zdCBMT1dfRU5EX1BBVFRFUk5TID1cbiAgL2ludGVsLipoZHxpbnRlbC4qdWhkfGludGVsLippcmlzfG1hbGl8YWRyZW5vXFxzP1sxLTVdfHBvd2VydnJ8YXBwbGUgZ3B1L2lcblxubGV0IGRldGVjdGVkID0gZmFsc2VcblxuZnVuY3Rpb24gZGV0ZWN0R3B1VGllcigpIHtcbiAgaWYgKGRldGVjdGVkIHx8IHR5cGVvZiB3aW5kb3cgPT09ICd1bmRlZmluZWQnKSB7XG4gICAgcmV0dXJuXG4gIH1cblxuICBkZXRlY3RlZCA9IHRydWVcblxuICAvLyBUaGUgYXRvbSBhbHJlYWR5IGhvbGRzIDA7IHRoZSBlYXJseSByZXR1cm5zIGJlbG93IHNpbXBseSBsZWF2ZSBpdCB0aGVyZS5cbiAgaWYgKHdpbmRvdy5tYXRjaE1lZGlhKCcocHJlZmVycy1yZWR1Y2VkLW1vdGlvbjogcmVkdWNlKScpLm1hdGNoZXMpIHtcbiAgICByZXR1cm5cbiAgfVxuXG4gIGxldCBnbDogbnVsbCB8IFdlYkdMUmVuZGVyaW5nQ29udGV4dCA9IG51bGxcblxuICB0cnkge1xuICAgIGNvbnN0IGNhbnZhcyA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoJ2NhbnZhcycpXG4gICAgZ2wgPSAoY2FudmFzLmdldENvbnRleHQoJ3dlYmdsJykgfHxcbiAgICAgIGNhbnZhcy5nZXRDb250ZXh0KCdleHBlcmltZW50YWwtd2ViZ2wnKSkgYXMgbnVsbCB8IFdlYkdMUmVuZGVyaW5nQ29udGV4dFxuICB9IGNhdGNoIHtcbiAgICAvLyBTb21lIHNhbmRib3hlZCAvIGhhcmRlbmVkIGNvbnRleHRzIHRocm93IG9uIGdldENvbnRleHQgcmF0aGVyIHRoYW5cbiAgICAvLyByZXR1cm5pbmcgbnVsbCAoZS5nLiBjZXJ0YWluIGNvcnBvcmF0ZSBicm93c2VyIHBvbGljaWVzKS4gVHJlYXQgYXNcbiAgICAvLyBcIm5vIFdlYkdMIGF2YWlsYWJsZVwiLlxuICAgIHJldHVyblxuICB9XG5cbiAgaWYgKCFnbCkge1xuICAgIHJldHVyblxuICB9XG5cbiAgY29uc3QgZXh0ID0gZ2wuZ2V0RXh0ZW5zaW9uKCdXRUJHTF9kZWJ1Z19yZW5kZXJlcl9pbmZvJylcbiAgY29uc3QgcmVuZGVyZXIgPSBTdHJpbmcoXG4gICAgZXh0XG4gICAgICA/IGdsLmdldFBhcmFtZXRlcihleHQuVU5NQVNLRURfUkVOREVSRVJfV0VCR0wpXG4gICAgICA6IGdsLmdldFBhcmFtZXRlcihnbC5SRU5ERVJFUilcbiAgKVxuXG4gIGlmIChTT0ZUV0FSRV9QQVRURVJOUy50ZXN0KHJlbmRlcmVyKSkge1xuICAgIGdsLmdldEV4dGVuc2lvbignV0VCR0xfbG9zZV9jb250ZXh0Jyk/Lmxvc2VDb250ZXh0KClcblxuICAgIHJldHVyblxuICB9XG5cbiAgaWYgKExPV19FTkRfUEFUVEVSTlMudGVzdChyZW5kZXJlcikpIHtcbiAgICAkZ3B1VGllci5zZXQoMSlcbiAgICBnbC5nZXRFeHRlbnNpb24oJ1dFQkdMX2xvc2VfY29udGV4dCcpPy5sb3NlQ29udGV4dCgpXG5cbiAgICByZXR1cm5cbiAgfVxuXG4gICRncHVUaWVyLnNldCgyKVxuXG4gIHJ1bkJlbmNobWFyayhnbClcbiAgICAudGhlbihmcHMgPT4gJGdwdVRpZXIuc2V0KGZwcyA8IDMwID8gMSA6IDIpKVxuICAgIC5jYXRjaCgoKSA9PiAkZ3B1VGllci5zZXQoMSkpXG4gICAgLmZpbmFsbHkoKCkgPT4gZ2w/LmdldEV4dGVuc2lvbignV0VCR0xfbG9zZV9jb250ZXh0Jyk/Lmxvc2VDb250ZXh0KCkpXG59XG5cbi8qKlxuICogUnVuIGRldGVjdGlvbiBvbmNlLCAqYWZ0ZXIqIHRoZSBmaXJzdCBwYWludCBzbyB0aGUgKHBvdGVudGlhbGx5IGV4cGVuc2l2ZSlcbiAqIFdlYkdMIHByb2JlIG5ldmVyIGJsb2NrcyBpbml0aWFsIHJlbmRlci4gUHJlZmVyIGByZXF1ZXN0SWRsZUNhbGxiYWNrYCBzbyB0aGVcbiAqIHByb2JlIHlpZWxkcyB0byByZW5kZXJpbmcgYW5kIGlucHV0OyBmYWxsIGJhY2sgdG8gYSBkb3VibGVcbiAqIGByZXF1ZXN0QW5pbWF0aW9uRnJhbWVgIChndWFyYW50ZWVzIGF0IGxlYXN0IG9uZSBwYWludGVkIGZyYW1lKSBhbmQgZmluYWxseVxuICogYHNldFRpbWVvdXRgIHdoZXJlIG5laXRoZXIgZXhpc3RzLlxuICovXG5mdW5jdGlvbiBzY2hlZHVsZURldGVjdGlvbigpIHtcbiAgaWYgKHR5cGVvZiB3aW5kb3cgPT09ICd1bmRlZmluZWQnKSB7XG4gICAgcmV0dXJuXG4gIH1cblxuICBpZiAodHlwZW9mIHdpbmRvdy5yZXF1ZXN0SWRsZUNhbGxiYWNrID09PSAnZnVuY3Rpb24nKSB7XG4gICAgd2luZG93LnJlcXVlc3RJZGxlQ2FsbGJhY2soKCkgPT4gZGV0ZWN0R3B1VGllcigpLCB7IHRpbWVvdXQ6IDEwMDAgfSlcbiAgfSBlbHNlIGlmICh0eXBlb2Ygd2luZG93LnJlcXVlc3RBbmltYXRpb25GcmFtZSA9PT0gJ2Z1bmN0aW9uJykge1xuICAgIHdpbmRvdy5yZXF1ZXN0QW5pbWF0aW9uRnJhbWUoKCkgPT5cbiAgICAgIHdpbmRvdy5yZXF1ZXN0QW5pbWF0aW9uRnJhbWUoKCkgPT4gZGV0ZWN0R3B1VGllcigpKVxuICAgIClcbiAgfSBlbHNlIHtcbiAgICBzZXRUaW1lb3V0KCgpID0+IGRldGVjdEdwdVRpZXIoKSwgMClcbiAgfVxufVxuXG5zY2hlZHVsZURldGVjdGlvbigpXG5cbmZ1bmN0aW9uIHJ1bkJlbmNobWFyayhnbDogV2ViR0xSZW5kZXJpbmdDb250ZXh0KTogUHJvbWlzZTxudW1iZXI+IHtcbiAgcmV0dXJuIG5ldyBQcm9taXNlKHJlc29sdmUgPT4ge1xuICAgIGNvbnN0IHZzID0gZ2wuY3JlYXRlU2hhZGVyKGdsLlZFUlRFWF9TSEFERVIpIVxuICAgIGNvbnN0IGZzID0gZ2wuY3JlYXRlU2hhZGVyKGdsLkZSQUdNRU5UX1NIQURFUikhXG4gICAgZ2wuc2hhZGVyU291cmNlKFxuICAgICAgdnMsXG4gICAgICAnYXR0cmlidXRlIHZlYzIgYTt2b2lkIG1haW4oKXtnbF9Qb3NpdGlvbj12ZWM0KGEsMCwxKTt9J1xuICAgIClcbiAgICBnbC5zaGFkZXJTb3VyY2UoXG4gICAgICBmcyxcbiAgICAgICdwcmVjaXNpb24gaGlnaHAgZmxvYXQ7dW5pZm9ybSBmbG9hdCB0O3ZvaWQgbWFpbigpe2Zsb2F0IHY9MC47Zm9yKGludCBpPTA7aTw2NDtpKyspdis9c2luKGZsb2F0KGkpKnQqLjAxKTtnbF9GcmFnQ29sb3I9dmVjNCh2Ki4wMDEpO30nXG4gICAgKVxuICAgIGdsLmNvbXBpbGVTaGFkZXIodnMpXG4gICAgZ2wuY29tcGlsZVNoYWRlcihmcylcblxuICAgIGNvbnN0IHByb2cgPSBnbC5jcmVhdGVQcm9ncmFtKCkhXG4gICAgZ2wuYXR0YWNoU2hhZGVyKHByb2csIHZzKVxuICAgIGdsLmF0dGFjaFNoYWRlcihwcm9nLCBmcylcbiAgICBnbC5saW5rUHJvZ3JhbShwcm9nKVxuICAgIGdsLnVzZVByb2dyYW0ocHJvZylcblxuICAgIGNvbnN0IGJ1ZiA9IGdsLmNyZWF0ZUJ1ZmZlcigpXG4gICAgZ2wuYmluZEJ1ZmZlcihnbC5BUlJBWV9CVUZGRVIsIGJ1ZilcbiAgICBnbC5idWZmZXJEYXRhKFxuICAgICAgZ2wuQVJSQVlfQlVGRkVSLFxuICAgICAgbmV3IEZsb2F0MzJBcnJheShbLTEsIC0xLCAxLCAtMSwgLTEsIDEsIDEsIDFdKSxcbiAgICAgIGdsLlNUQVRJQ19EUkFXXG4gICAgKVxuICAgIGNvbnN0IGEgPSBnbC5nZXRBdHRyaWJMb2NhdGlvbihwcm9nLCAnYScpXG4gICAgZ2wuZW5hYmxlVmVydGV4QXR0cmliQXJyYXkoYSlcbiAgICBnbC52ZXJ0ZXhBdHRyaWJQb2ludGVyKGEsIDIsIGdsLkZMT0FULCBmYWxzZSwgMCwgMClcblxuICAgIGNvbnN0IHVUID0gZ2wuZ2V0VW5pZm9ybUxvY2F0aW9uKHByb2csICd0JylcbiAgICBsZXQgZnJhbWVzID0gMFxuICAgIGNvbnN0IHN0YXJ0ID0gcGVyZm9ybWFuY2Uubm93KClcblxuICAgIGNvbnN0IHRpY2sgPSAoKSA9PiB7XG4gICAgICBnbC51bmlmb3JtMWYodVQsIGZyYW1lcylcbiAgICAgIGdsLmRyYXdBcnJheXMoZ2wuVFJJQU5HTEVfU1RSSVAsIDAsIDQpXG4gICAgICBnbC5maW5pc2goKVxuICAgICAgZnJhbWVzKytcblxuICAgICAgaWYgKHBlcmZvcm1hbmNlLm5vdygpIC0gc3RhcnQgPCAyMDApIHtcbiAgICAgICAgcmVxdWVzdEFuaW1hdGlvbkZyYW1lKHRpY2spXG4gICAgICB9IGVsc2Uge1xuICAgICAgICBjb25zdCBlbGFwc2VkID0gcGVyZm9ybWFuY2Uubm93KCkgLSBzdGFydFxuICAgICAgICBnbC5kZWxldGVQcm9ncmFtKHByb2cpXG4gICAgICAgIGdsLmRlbGV0ZVNoYWRlcih2cylcbiAgICAgICAgZ2wuZGVsZXRlU2hhZGVyKGZzKVxuICAgICAgICBnbC5kZWxldGVCdWZmZXIoYnVmKVxuICAgICAgICByZXNvbHZlKChmcmFtZXMgLyBlbGFwc2VkKSAqIDEwMDApXG4gICAgICB9XG4gICAgfVxuXG4gICAgcmVxdWVzdEFuaW1hdGlvbkZyYW1lKHRpY2spXG4gIH0pXG59XG5cbmV4cG9ydCBmdW5jdGlvbiB1c2VHcHVUaWVyKCkge1xuICByZXR1cm4gdXNlU3RvcmUoJGdwdVRpZXIpXG59XG5cbnR5cGUgR3B1VGllciA9IDAgfCAxIHwgMlxuIl0sCiAgIm1hcHBpbmdzIjogIjtBQUVBLFNBQVMsZ0JBQWdCO0FBQ3pCLFNBQVMsWUFBWTtBQWdDZCxhQUFNLFdBQVcsS0FBYyxD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
@@ -0,0 +1,41 @@
1
+ /**
2
+ * Visibility- and intersection-aware render-loop helper for the WebGL
3
+ * overlays.
4
+ *
5
+ * The overlays were previously running fragment shaders at 60fps for the
6
+ * entire lifetime of the page — including when the tab was hidden, the
7
+ * canvas had been scrolled out of view, or the user had been idle for
8
+ * hours. On retina laptops the compositor cost of mix-blend-mode on a
9
+ * full-viewport canvas plus continuous WebGL rasterisation is enough to
10
+ * keep the GPU hot indefinitely, which is what manifests as "fans go
11
+ * crazy after 2 hours of idle".
12
+ *
13
+ * `runRenderLoop` wraps a frame callback so that it:
14
+ *
15
+ * 1. Pauses entirely when `document.hidden` is true (background tab,
16
+ * minimised window, screen locked).
17
+ * 2. Pauses when the canvas's bounding rect is offscreen (we tell
18
+ * `IntersectionObserver` to look at the canvas itself).
19
+ * 3. Optionally caps the frame rate via a min-interval — the previous
20
+ * `gpuTier === 1 ? setTimeout(loop, 100) : raf` trick is preserved
21
+ * and extended so even tier-2 GPUs cap at e.g. 30fps for overlays
22
+ * that don't need 60.
23
+ *
24
+ * The callback receives the *delta* time in seconds since the last call
25
+ * (so `uTime` advances correctly across pauses without ever skipping
26
+ * forward by hours).
27
+ */
28
+ interface RunRenderLoopOptions {
29
+ /** Element to observe with IntersectionObserver. When fully out of
30
+ * view, the loop pauses. Pass the canvas element itself. */
31
+ el: Element;
32
+ /** Min ms between frames. 0 = no cap (uses requestAnimationFrame).
33
+ * Anything > 0 uses setTimeout-driven scheduling. */
34
+ minIntervalMs?: number;
35
+ /** Frame callback. Receives the elapsed seconds since the previous
36
+ * *executed* frame (not since the previous scheduled frame), so
37
+ * uniforms keyed off this value will not jump after a long pause. */
38
+ onFrame: (deltaSeconds: number) => void;
39
+ }
40
+ export declare function runRenderLoop({ el, minIntervalMs, onFrame }: RunRenderLoopOptions): () => void;
41
+ export {};
@@ -0,0 +1,63 @@
1
+ "use client";
2
+ export function runRenderLoop({
3
+ el,
4
+ minIntervalMs = 0,
5
+ onFrame
6
+ }) {
7
+ let running = true;
8
+ let visible = !document.hidden;
9
+ let inView = true;
10
+ let last = performance.now();
11
+ let raf = 0;
12
+ let timer;
13
+ const onVisibility = () => {
14
+ visible = !document.hidden;
15
+ if (visible) {
16
+ last = performance.now();
17
+ schedule();
18
+ }
19
+ };
20
+ const io = new IntersectionObserver(
21
+ (entries) => {
22
+ const wasInView = inView;
23
+ inView = entries.some((e) => e.isIntersecting);
24
+ if (!wasInView && inView) {
25
+ last = performance.now();
26
+ schedule();
27
+ }
28
+ },
29
+ { threshold: 0 }
30
+ );
31
+ io.observe(el);
32
+ document.addEventListener("visibilitychange", onVisibility);
33
+ const tick = () => {
34
+ if (!running) return;
35
+ if (!visible || !inView) {
36
+ return;
37
+ }
38
+ const now = performance.now();
39
+ const delta = (now - last) / 1e3;
40
+ last = now;
41
+ onFrame(delta);
42
+ schedule();
43
+ };
44
+ function schedule() {
45
+ if (!running || !visible || !inView) return;
46
+ if (minIntervalMs > 0) {
47
+ timer = setTimeout(tick, minIntervalMs);
48
+ } else {
49
+ raf = requestAnimationFrame(tick);
50
+ }
51
+ }
52
+ schedule();
53
+ return () => {
54
+ running = false;
55
+ io.disconnect();
56
+ document.removeEventListener("visibilitychange", onVisibility);
57
+ cancelAnimationFrame(raf);
58
+ if (timer !== void 0) {
59
+ clearTimeout(timer);
60
+ }
61
+ };
62
+ }
63
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiPHN0ZGluPiJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiJ3VzZSBjbGllbnQnXG5cbi8qKlxuICogVmlzaWJpbGl0eS0gYW5kIGludGVyc2VjdGlvbi1hd2FyZSByZW5kZXItbG9vcCBoZWxwZXIgZm9yIHRoZSBXZWJHTFxuICogb3ZlcmxheXMuXG4gKlxuICogVGhlIG92ZXJsYXlzIHdlcmUgcHJldmlvdXNseSBydW5uaW5nIGZyYWdtZW50IHNoYWRlcnMgYXQgNjBmcHMgZm9yIHRoZVxuICogZW50aXJlIGxpZmV0aW1lIG9mIHRoZSBwYWdlIFx1MjAxNCBpbmNsdWRpbmcgd2hlbiB0aGUgdGFiIHdhcyBoaWRkZW4sIHRoZVxuICogY2FudmFzIGhhZCBiZWVuIHNjcm9sbGVkIG91dCBvZiB2aWV3LCBvciB0aGUgdXNlciBoYWQgYmVlbiBpZGxlIGZvclxuICogaG91cnMuIE9uIHJldGluYSBsYXB0b3BzIHRoZSBjb21wb3NpdG9yIGNvc3Qgb2YgbWl4LWJsZW5kLW1vZGUgb24gYVxuICogZnVsbC12aWV3cG9ydCBjYW52YXMgcGx1cyBjb250aW51b3VzIFdlYkdMIHJhc3RlcmlzYXRpb24gaXMgZW5vdWdoIHRvXG4gKiBrZWVwIHRoZSBHUFUgaG90IGluZGVmaW5pdGVseSwgd2hpY2ggaXMgd2hhdCBtYW5pZmVzdHMgYXMgXCJmYW5zIGdvXG4gKiBjcmF6eSBhZnRlciAyIGhvdXJzIG9mIGlkbGVcIi5cbiAqXG4gKiBgcnVuUmVuZGVyTG9vcGAgd3JhcHMgYSBmcmFtZSBjYWxsYmFjayBzbyB0aGF0IGl0OlxuICpcbiAqICAgMS4gUGF1c2VzIGVudGlyZWx5IHdoZW4gYGRvY3VtZW50LmhpZGRlbmAgaXMgdHJ1ZSAoYmFja2dyb3VuZCB0YWIsXG4gKiAgICAgIG1pbmltaXNlZCB3aW5kb3csIHNjcmVlbiBsb2NrZWQpLlxuICogICAyLiBQYXVzZXMgd2hlbiB0aGUgY2FudmFzJ3MgYm91bmRpbmcgcmVjdCBpcyBvZmZzY3JlZW4gKHdlIHRlbGxcbiAqICAgICAgYEludGVyc2VjdGlvbk9ic2VydmVyYCB0byBsb29rIGF0IHRoZSBjYW52YXMgaXRzZWxmKS5cbiAqICAgMy4gT3B0aW9uYWxseSBjYXBzIHRoZSBmcmFtZSByYXRlIHZpYSBhIG1pbi1pbnRlcnZhbCBcdTIwMTQgdGhlIHByZXZpb3VzXG4gKiAgICAgIGBncHVUaWVyID09PSAxID8gc2V0VGltZW91dChsb29wLCAxMDApIDogcmFmYCB0cmljayBpcyBwcmVzZXJ2ZWRcbiAqICAgICAgYW5kIGV4dGVuZGVkIHNvIGV2ZW4gdGllci0yIEdQVXMgY2FwIGF0IGUuZy4gMzBmcHMgZm9yIG92ZXJsYXlzXG4gKiAgICAgIHRoYXQgZG9uJ3QgbmVlZCA2MC5cbiAqXG4gKiBUaGUgY2FsbGJhY2sgcmVjZWl2ZXMgdGhlICpkZWx0YSogdGltZSBpbiBzZWNvbmRzIHNpbmNlIHRoZSBsYXN0IGNhbGxcbiAqIChzbyBgdVRpbWVgIGFkdmFuY2VzIGNvcnJlY3RseSBhY3Jvc3MgcGF1c2VzIHdpdGhvdXQgZXZlciBza2lwcGluZ1xuICogZm9yd2FyZCBieSBob3VycykuXG4gKi9cblxuaW50ZXJmYWNlIFJ1blJlbmRlckxvb3BPcHRpb25zIHtcbiAgLyoqIEVsZW1lbnQgdG8gb2JzZXJ2ZSB3aXRoIEludGVyc2VjdGlvbk9ic2VydmVyLiBXaGVuIGZ1bGx5IG91dCBvZlxuICAgKiAgdmlldywgdGhlIGxvb3AgcGF1c2VzLiBQYXNzIHRoZSBjYW52YXMgZWxlbWVudCBpdHNlbGYuICovXG4gIGVsOiBFbGVtZW50XG4gIC8qKiBNaW4gbXMgYmV0d2VlbiBmcmFtZXMuIDAgPSBubyBjYXAgKHVzZXMgcmVxdWVzdEFuaW1hdGlvbkZyYW1lKS5cbiAgICogIEFueXRoaW5nID4gMCB1c2VzIHNldFRpbWVvdXQtZHJpdmVuIHNjaGVkdWxpbmcuICovXG4gIG1pbkludGVydmFsTXM/OiBudW1iZXJcbiAgLyoqIEZyYW1lIGNhbGxiYWNrLiBSZWNlaXZlcyB0aGUgZWxhcHNlZCBzZWNvbmRzIHNpbmNlIHRoZSBwcmV2aW91c1xuICAgKiAgKmV4ZWN1dGVkKiBmcmFtZSAobm90IHNpbmNlIHRoZSBwcmV2aW91cyBzY2hlZHVsZWQgZnJhbWUpLCBzb1xuICAgKiAgdW5pZm9ybXMga2V5ZWQgb2ZmIHRoaXMgdmFsdWUgd2lsbCBub3QganVtcCBhZnRlciBhIGxvbmcgcGF1c2UuICovXG4gIG9uRnJhbWU6IChkZWx0YVNlY29uZHM6IG51bWJlcikgPT4gdm9pZFxufVxuXG5leHBvcnQgZnVuY3Rpb24gcnVuUmVuZGVyTG9vcCh7XG4gIGVsLFxuICBtaW5JbnRlcnZhbE1zID0gMCxcbiAgb25GcmFtZVxufTogUnVuUmVuZGVyTG9vcE9wdGlvbnMpIHtcbiAgbGV0IHJ1bm5pbmcgPSB0cnVlXG4gIGxldCB2aXNpYmxlID0gIWRvY3VtZW50LmhpZGRlblxuICBsZXQgaW5WaWV3ID0gdHJ1ZVxuICBsZXQgbGFzdCA9IHBlcmZvcm1hbmNlLm5vdygpXG4gIGxldCByYWYgPSAwXG4gIGxldCB0aW1lcjogUmV0dXJuVHlwZTx0eXBlb2Ygc2V0VGltZW91dD4gfCB1bmRlZmluZWRcblxuICBjb25zdCBvblZpc2liaWxpdHkgPSAoKSA9PiB7XG4gICAgdmlzaWJsZSA9ICFkb2N1bWVudC5oaWRkZW5cblxuICAgIC8vIFdoZW4gd2UgY29tZSBiYWNrIGZyb20gYSBoaWRkZW4gdGFiLCByZXNldCB0aGUgY2xvY2sgc28gdGhlIG5leHRcbiAgICAvLyBmcmFtZSdzIGRlbHRhIGlzIH5vbmUgZnJhbWUsIG5vdCBcImhvdXJzIHNpbmNlIEkgd2FzIGhpZGRlblwiLlxuICAgIGlmICh2aXNpYmxlKSB7XG4gICAgICBsYXN0ID0gcGVyZm9ybWFuY2Uubm93KClcbiAgICAgIHNjaGVkdWxlKClcbiAgICB9XG4gIH1cblxuICBjb25zdCBpbyA9IG5ldyBJbnRlcnNlY3Rpb25PYnNlcnZlcihcbiAgICBlbnRyaWVzID0+IHtcbiAgICAgIGNvbnN0IHdhc0luVmlldyA9IGluVmlld1xuICAgICAgaW5WaWV3ID0gZW50cmllcy5zb21lKGUgPT4gZS5pc0ludGVyc2VjdGluZylcblxuICAgICAgaWYgKCF3YXNJblZpZXcgJiYgaW5WaWV3KSB7XG4gICAgICAgIGxhc3QgPSBwZXJmb3JtYW5jZS5ub3coKVxuICAgICAgICBzY2hlZHVsZSgpXG4gICAgICB9XG4gICAgfSxcbiAgICB7IHRocmVzaG9sZDogMCB9XG4gIClcblxuICBpby5vYnNlcnZlKGVsKVxuICBkb2N1bWVudC5hZGRFdmVudExpc3RlbmVyKCd2aXNpYmlsaXR5Y2hhbmdlJywgb25WaXNpYmlsaXR5KVxuXG4gIGNvbnN0IHRpY2sgPSAoKSA9PiB7XG4gICAgaWYgKCFydW5uaW5nKSByZXR1cm5cblxuICAgIGlmICghdmlzaWJsZSB8fCAhaW5WaWV3KSB7XG4gICAgICAvLyBEb24ndCByZXNjaGVkdWxlIFx1MjAxNCB3ZSdsbCBiZSByZS1raWNrZWQgYnkgdmlzaWJpbGl0eWNoYW5nZSBvciBJTy5cbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIGNvbnN0IG5vdyA9IHBlcmZvcm1hbmNlLm5vdygpXG4gICAgY29uc3QgZGVsdGEgPSAobm93IC0gbGFzdCkgLyAxMDAwXG4gICAgbGFzdCA9IG5vd1xuXG4gICAgb25GcmFtZShkZWx0YSlcbiAgICBzY2hlZHVsZSgpXG4gIH1cblxuICBmdW5jdGlvbiBzY2hlZHVsZSgpIHtcbiAgICBpZiAoIXJ1bm5pbmcgfHwgIXZpc2libGUgfHwgIWluVmlldykgcmV0dXJuXG5cbiAgICBpZiAobWluSW50ZXJ2YWxNcyA+IDApIHtcbiAgICAgIHRpbWVyID0gc2V0VGltZW91dCh0aWNrLCBtaW5JbnRlcnZhbE1zKVxuICAgIH0gZWxzZSB7XG4gICAgICByYWYgPSByZXF1ZXN0QW5pbWF0aW9uRnJhbWUodGljaylcbiAgICB9XG4gIH1cblxuICBzY2hlZHVsZSgpXG5cbiAgcmV0dXJuICgpID0+IHtcbiAgICBydW5uaW5nID0gZmFsc2VcbiAgICBpby5kaXNjb25uZWN0KClcbiAgICBkb2N1bWVudC5yZW1vdmVFdmVudExpc3RlbmVyKCd2aXNpYmlsaXR5Y2hhbmdlJywgb25WaXNpYmlsaXR5KVxuICAgIGNhbmNlbEFuaW1hdGlvbkZyYW1lKHJhZilcblxuICAgIGlmICh0aW1lciAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICBjbGVhclRpbWVvdXQodGltZXIpXG4gICAgfVxuICB9XG59XG4iXSwKICAibWFwcGluZ3MiOiAiO0FBMkNPLGdCQUFTLGNBQWM7QUFBQSxFQUM1QjtBQUFBLEVBQ0EsZ0JBQWdCO0FBQUEsRUFDaEI7QUFDRixHQUF5QjtBQUN2QixNQUFJLFVBQVU7QUFDZCxNQUFJLFVBQVUsQ0FBQyxTQUFTO0FBQ3hCLE1BQUksU0FBUztBQUNiLE1BQUksT0FBTyxZQUFZLElBQUk7QUFDM0IsTUFBSSxNQUFNO0FBQ1YsTUFBSTtBQUVKLFFBQU0sZUFBZSxNQUFNO0FBQ3pCLGNBQVUsQ0FBQyxTQUFTO0FBSXBCLFFBQUksU0FBUztBQUNYLGFBQU8sWUFBWSxJQUFJO0FBQ3ZCLGVBQVM7QUFBQSxJQUNYO0FBQUEsRUFDRjtBQUVBLFFBQU0sS0FBSyxJQUFJO0FBQUEsSUFDYixhQUFXO0FBQ1QsWUFBTSxZQUFZO0FBQ2xCLGVBQVMsUUFBUSxLQUFLLE9BQUssRUFBRSxjQUFjO0FBRTNDLFVBQUksQ0FBQyxhQUFhLFFBQVE7QUFDeEIsZUFBTyxZQUFZLElBQUk7QUFDdkIsaUJBQVM7QUFBQSxNQUNYO0FBQUEsSUFDRjtBQUFBLElBQ0EsRUFBRSxXQUFXLEVBQUU7QUFBQSxFQUNqQjtBQUVBLEtBQUcsUUFBUSxFQUFFO0FBQ2IsV0FBUyxpQkFBaUIsb0JBQW9CLFlBQVk7QUFFMUQsUUFBTSxPQUFPLE1BQU07QUFDakIsUUFBSSxDQUFDLFFBQVM7QUFFZCxRQUFJLENBQUMsV0FBVyxDQUFDLFFBQVE7QUFFdkI7QUFBQSxJQUNGO0FBRUEsVUFBTSxNQUFNLFlBQVksSUFBSTtBQUM1QixVQUFNLFNBQVMsTUFBTSxRQUFRO0FBQzdCLFdBQU87QUFFUCxZQUFRLEtBQUs7QUFDYixhQUFTO0FBQUEsRUFDWDtBQUVBLFdBQVMsV0FBVztBQUNsQixRQUFJLENBQUMsV0FBVyxDQUFDLFdBQVcsQ0FBQyxPQUFRO0FBRXJDLFFBQUksZ0JBQWdCLEdBQUc7QUFDckIsY0FBUSxXQUFXLE1BQU0sYUFBYTtBQUFBLElBQ3hDLE9BQU87QUFDTCxZQUFNLHNCQUFzQixJQUFJO0FBQUEsSUFDbEM7QUFBQSxFQUNGO0FBRUEsV0FBUztBQUVULFNBQU8sTUFBTTtBQUNYLGNBQVU7QUFDVixPQUFHLFdBQVc7QUFDZCxhQUFTLG9CQUFvQixvQkFBb0IsWUFBWTtBQUM3RCx5QkFBcUIsR0FBRztBQUV4QixRQUFJLFVBQVUsUUFBVztBQUN2QixtQkFBYSxLQUFLO0FBQUEsSUFDcEI7QUFBQSxFQUNGO0FBQ0Y7IiwKICAibmFtZXMiOiBbXQp9Cg==
@@ -0,0 +1,16 @@
1
+ import { useControls } from 'leva';
2
+ import { type WritableAtom } from 'nanostores';
3
+ export declare function useSmoothControls<T extends Record<string, any>>(label: string, initialArgs: T, options?: UseSmoothControlsOptions, dependencies?: Parameters<typeof useControls>[3]): { [K in keyof T]: T[K] extends {
4
+ value: infer V;
5
+ } ? V : never; };
6
+ export declare const getControlAtom: <T = any>(label: string, key: string) => undefined | WritableAtom<T>;
7
+ export declare const setControlValue: <T = any>(label: string, key: string, value: T, options?: {
8
+ animate?: boolean;
9
+ duration?: number;
10
+ }) => void;
11
+ type UseSmoothControlsOptions = Parameters<typeof useControls>[2] & {
12
+ duration?: number;
13
+ onRandomize?: () => void;
14
+ onReset?: () => void;
15
+ };
16
+ export {};