@nastechai-research/ui 0.13.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.
- package/CHANGELOG.md +5 -0
- package/README.md +21 -0
- package/dist/assets/filler-bg0.webp +0 -0
- package/dist/assets.d.ts +38 -0
- package/dist/fonts/Collapse-Bold.woff2 +0 -0
- package/dist/fonts/Collapse-BoldItalic.woff2 +0 -0
- package/dist/fonts/Collapse-Italic.woff2 +0 -0
- package/dist/fonts/Collapse-Light.woff2 +0 -0
- package/dist/fonts/Collapse-LightItalic.woff2 +0 -0
- package/dist/fonts/Collapse-Regular.woff2 +0 -0
- package/dist/fonts/Collapse-Thin.woff2 +0 -0
- package/dist/fonts/Collapse-ThinItalic.woff2 +0 -0
- package/dist/fonts/Mondwest-Regular.woff2 +0 -0
- package/dist/fonts/Neuebit-Bold.woff2 +0 -0
- package/dist/fonts/RulesCompressed-Medium.woff2 +0 -0
- package/dist/fonts/RulesCompressed-Regular.woff2 +0 -0
- package/dist/fonts/RulesExpanded-Bold.woff2 +0 -0
- package/dist/fonts/RulesExpanded-Regular.woff2 +0 -0
- package/dist/fonts.d.ts +6 -0
- package/dist/fonts.js +6 -0
- package/dist/hooks/use-below-breakpoint.d.ts +2 -0
- package/dist/hooks/use-below-breakpoint.js +17 -0
- package/dist/hooks/use-capped-frame.d.ts +2 -0
- package/dist/hooks/use-capped-frame.js +15 -0
- package/dist/hooks/use-confirm-delete.d.ts +10 -0
- package/dist/hooks/use-confirm-delete.js +35 -0
- package/dist/hooks/use-css-var-dims.d.ts +1 -0
- package/dist/hooks/use-css-var-dims.js +29 -0
- package/dist/hooks/use-gpu-tier.d.ts +34 -0
- package/dist/hooks/use-gpu-tier.js +111 -0
- package/dist/hooks/use-render-loop.d.ts +41 -0
- package/dist/hooks/use-render-loop.js +63 -0
- package/dist/hooks/use-smooth-controls.d.ts +16 -0
- package/dist/hooks/use-smooth-controls.js +217 -0
- package/dist/hooks/use-toast.d.ts +7 -0
- package/dist/hooks/use-toast.js +21 -0
- package/dist/index.d.ts +79 -0
- package/dist/index.js +107 -0
- package/dist/ui/basic-page.d.ts +7 -0
- package/dist/ui/basic-page.js +18 -0
- package/dist/ui/build.css +4 -0
- package/dist/ui/components/animated-count.d.ts +10 -0
- package/dist/ui/components/animated-count.js +113 -0
- package/dist/ui/components/ascii.d.ts +10 -0
- package/dist/ui/components/ascii.js +79 -0
- package/dist/ui/components/badge.d.ts +6 -0
- package/dist/ui/components/badge.js +40 -0
- package/dist/ui/components/badges/nous-girl.d.ts +2 -0
- package/dist/ui/components/badges/nous-girl.js +83 -0
- package/dist/ui/components/blend-mode.d.ts +28 -0
- package/dist/ui/components/blend-mode.js +69 -0
- package/dist/ui/components/blink.d.ts +6 -0
- package/dist/ui/components/blink.js +17 -0
- package/dist/ui/components/bottom-sheet.d.ts +15 -0
- package/dist/ui/components/bottom-sheet.js +192 -0
- package/dist/ui/components/button.d.ts +14 -0
- package/dist/ui/components/button.js +147 -0
- package/dist/ui/components/card.d.ts +5 -0
- package/dist/ui/components/card.js +74 -0
- package/dist/ui/components/checkbox.d.ts +2 -0
- package/dist/ui/components/checkbox.js +27 -0
- package/dist/ui/components/command-block.d.ts +24 -0
- package/dist/ui/components/command-block.js +56 -0
- package/dist/ui/components/confirm-dialog.d.ts +13 -0
- package/dist/ui/components/confirm-dialog.js +113 -0
- package/dist/ui/components/cursor.d.ts +3 -0
- package/dist/ui/components/cursor.js +97 -0
- package/dist/ui/components/dialog.d.ts +15 -0
- package/dist/ui/components/dialog.js +171 -0
- package/dist/ui/components/dropdown-menu.d.ts +12 -0
- package/dist/ui/components/dropdown-menu.js +102 -0
- package/dist/ui/components/fit-text/fit-text.css +42 -0
- package/dist/ui/components/fit-text/index.d.ts +9 -0
- package/dist/ui/components/fit-text/index.js +25 -0
- package/dist/ui/components/graphs/bar-chart.d.ts +12 -0
- package/dist/ui/components/graphs/bar-chart.js +129 -0
- package/dist/ui/components/graphs/index.d.ts +3 -0
- package/dist/ui/components/graphs/index.js +4 -0
- package/dist/ui/components/graphs/line-chart.d.ts +14 -0
- package/dist/ui/components/graphs/line-chart.js +175 -0
- package/dist/ui/components/graphs/utils.d.ts +52 -0
- package/dist/ui/components/graphs/utils.js +162 -0
- package/dist/ui/components/grid/grid.css +79 -0
- package/dist/ui/components/grid/index.d.ts +2 -0
- package/dist/ui/components/grid/index.js +17 -0
- package/dist/ui/components/hover-bg.d.ts +1 -0
- package/dist/ui/components/hover-bg.js +14 -0
- package/dist/ui/components/icons/arrow.d.ts +6 -0
- package/dist/ui/components/icons/arrow.js +44 -0
- package/dist/ui/components/icons/check.d.ts +2 -0
- package/dist/ui/components/icons/check.js +13 -0
- package/dist/ui/components/icons/chevron.d.ts +6 -0
- package/dist/ui/components/icons/chevron.js +51 -0
- package/dist/ui/components/icons/discord.d.ts +2 -0
- package/dist/ui/components/icons/discord.js +15 -0
- package/dist/ui/components/icons/eye.d.ts +2 -0
- package/dist/ui/components/icons/eye.js +8 -0
- package/dist/ui/components/icons/gear.d.ts +6 -0
- package/dist/ui/components/icons/gear.js +30 -0
- package/dist/ui/components/icons/github.d.ts +2 -0
- package/dist/ui/components/icons/github.js +15 -0
- package/dist/ui/components/icons/hamburger.d.ts +6 -0
- package/dist/ui/components/icons/hamburger.js +56 -0
- package/dist/ui/components/icons/heart.d.ts +2 -0
- package/dist/ui/components/icons/heart.js +11 -0
- package/dist/ui/components/icons/index.d.ts +12 -0
- package/dist/ui/components/icons/index.js +13 -0
- package/dist/ui/components/icons/link.d.ts +2 -0
- package/dist/ui/components/icons/link.js +13 -0
- package/dist/ui/components/icons/minus.d.ts +2 -0
- package/dist/ui/components/icons/minus.js +13 -0
- package/dist/ui/components/icons/search.d.ts +2 -0
- package/dist/ui/components/icons/search.js +33 -0
- package/dist/ui/components/image-distortion.d.ts +21 -0
- package/dist/ui/components/image-distortion.js +398 -0
- package/dist/ui/components/input.d.ts +1 -0
- package/dist/ui/components/input.js +21 -0
- package/dist/ui/components/label.d.ts +1 -0
- package/dist/ui/components/label.js +18 -0
- package/dist/ui/components/leva-client.d.ts +1 -0
- package/dist/ui/components/leva-client.js +12 -0
- package/dist/ui/components/list-item.d.ts +6 -0
- package/dist/ui/components/list-item.js +27 -0
- package/dist/ui/components/overlays/blend-modes.d.ts +1 -0
- package/dist/ui/components/overlays/blend-modes.js +14 -0
- package/dist/ui/components/overlays/glitch.d.ts +6 -0
- package/dist/ui/components/overlays/glitch.js +209 -0
- package/dist/ui/components/overlays/greys.d.ts +6 -0
- package/dist/ui/components/overlays/greys.js +339 -0
- package/dist/ui/components/overlays/index.d.ts +14 -0
- package/dist/ui/components/overlays/index.js +34 -0
- package/dist/ui/components/overlays/lens-layers.d.ts +14 -0
- package/dist/ui/components/overlays/lens-layers.js +95 -0
- package/dist/ui/components/overlays/lens.d.ts +44 -0
- package/dist/ui/components/overlays/lens.js +60 -0
- package/dist/ui/components/overlays/noise.d.ts +6 -0
- package/dist/ui/components/overlays/noise.js +136 -0
- package/dist/ui/components/overlays/vignette.d.ts +6 -0
- package/dist/ui/components/overlays/vignette.js +47 -0
- package/dist/ui/components/poster.d.ts +62 -0
- package/dist/ui/components/poster.js +256 -0
- package/dist/ui/components/progress.d.ts +9 -0
- package/dist/ui/components/progress.js +53 -0
- package/dist/ui/components/scene-canvas.d.ts +23 -0
- package/dist/ui/components/scene-canvas.js +179 -0
- package/dist/ui/components/scramble.d.ts +9 -0
- package/dist/ui/components/scramble.js +63 -0
- package/dist/ui/components/segmented.d.ts +20 -0
- package/dist/ui/components/segmented.js +51 -0
- package/dist/ui/components/select.d.ts +18 -0
- package/dist/ui/components/select.js +215 -0
- package/dist/ui/components/selection-switcher.d.ts +1 -0
- package/dist/ui/components/selection-switcher.js +34 -0
- package/dist/ui/components/separator.d.ts +5 -0
- package/dist/ui/components/separator.js +22 -0
- package/dist/ui/components/shader.d.ts +7 -0
- package/dist/ui/components/shader.js +60 -0
- package/dist/ui/components/socials.d.ts +20 -0
- package/dist/ui/components/socials.js +21 -0
- package/dist/ui/components/spinner.d.ts +20 -0
- package/dist/ui/components/spinner.js +38 -0
- package/dist/ui/components/stats.d.ts +16 -0
- package/dist/ui/components/stats.js +36 -0
- package/dist/ui/components/switch.d.ts +7 -0
- package/dist/ui/components/switch.js +37 -0
- package/dist/ui/components/tabs.d.ts +14 -0
- package/dist/ui/components/tabs.js +44 -0
- package/dist/ui/components/terminal-demo.d.ts +32 -0
- package/dist/ui/components/terminal-demo.js +125 -0
- package/dist/ui/components/theme-toggle.d.ts +6 -0
- package/dist/ui/components/theme-toggle.js +66 -0
- package/dist/ui/components/tier-card.d.ts +53 -0
- package/dist/ui/components/tier-card.js +146 -0
- package/dist/ui/components/toast.d.ts +8 -0
- package/dist/ui/components/toast.js +39 -0
- package/dist/ui/components/tv.d.ts +3 -0
- package/dist/ui/components/tv.js +239 -0
- package/dist/ui/components/typography/h1.d.ts +11 -0
- package/dist/ui/components/typography/h1.js +18 -0
- package/dist/ui/components/typography/h2.d.ts +11 -0
- package/dist/ui/components/typography/h2.js +18 -0
- package/dist/ui/components/typography/index.d.ts +15 -0
- package/dist/ui/components/typography/index.js +41 -0
- package/dist/ui/components/typography/legend.d.ts +6 -0
- package/dist/ui/components/typography/legend.js +20 -0
- package/dist/ui/components/typography/small.d.ts +2 -0
- package/dist/ui/components/typography/small.js +9 -0
- package/dist/ui/components/watchlist.d.ts +11 -0
- package/dist/ui/components/watchlist.js +80 -0
- package/dist/ui/fonts.css +63 -0
- package/dist/ui/footer.d.ts +20 -0
- package/dist/ui/footer.js +65 -0
- package/dist/ui/globals.css +395 -0
- package/dist/ui/header.d.ts +41 -0
- package/dist/ui/header.js +270 -0
- package/dist/ui/layout-wrapper.d.ts +1 -0
- package/dist/ui/layout-wrapper.js +7 -0
- package/dist/utils/color.d.ts +4 -0
- package/dist/utils/color.js +14 -0
- package/dist/utils/index.d.ts +15 -0
- package/dist/utils/index.js +48 -0
- package/dist/utils/poly.d.ts +8 -0
- package/dist/utils/poly.js +3 -0
- package/package.json +120 -0
- package/src/assets/filler-bg0.webp +0 -0
- package/src/assets.d.ts +38 -0
- package/src/fonts/Collapse-Bold.woff2 +0 -0
- package/src/fonts/Collapse-BoldItalic.woff2 +0 -0
- package/src/fonts/Collapse-Italic.woff2 +0 -0
- package/src/fonts/Collapse-Light.woff2 +0 -0
- package/src/fonts/Collapse-LightItalic.woff2 +0 -0
- package/src/fonts/Collapse-Regular.woff2 +0 -0
- package/src/fonts/Collapse-Thin.woff2 +0 -0
- package/src/fonts/Collapse-ThinItalic.woff2 +0 -0
- package/src/fonts/Mondwest-Regular.woff2 +0 -0
- package/src/fonts/Neuebit-Bold.woff2 +0 -0
- package/src/fonts/RulesCompressed-Medium.woff2 +0 -0
- package/src/fonts/RulesCompressed-Regular.woff2 +0 -0
- package/src/fonts/RulesExpanded-Bold.woff2 +0 -0
- package/src/fonts/RulesExpanded-Regular.woff2 +0 -0
- package/src/fonts.ts +6 -0
- package/src/hooks/use-below-breakpoint.ts +21 -0
- package/src/hooks/use-capped-frame.ts +18 -0
- package/src/hooks/use-confirm-delete.ts +43 -0
- package/src/hooks/use-css-var-dims.ts +39 -0
- package/src/hooks/use-gpu-tier.ts +190 -0
- package/src/hooks/use-render-loop.ts +121 -0
- package/src/hooks/use-smooth-controls.ts +318 -0
- package/src/hooks/use-toast.ts +29 -0
- package/src/index.ts +130 -0
- package/src/ui/basic-page.tsx +34 -0
- package/src/ui/build.css +4 -0
- package/src/ui/components/animated-count.stories.tsx +67 -0
- package/src/ui/components/animated-count.tsx +168 -0
- package/src/ui/components/ascii.stories.tsx +30 -0
- package/src/ui/components/ascii.tsx +110 -0
- package/src/ui/components/badge.stories.tsx +31 -0
- package/src/ui/components/badge.tsx +60 -0
- package/src/ui/components/badges/nous-girl.tsx +52 -0
- package/src/ui/components/blend-mode.stories.tsx +33 -0
- package/src/ui/components/blend-mode.tsx +129 -0
- package/src/ui/components/blink.stories.tsx +32 -0
- package/src/ui/components/blink.tsx +21 -0
- package/src/ui/components/bottom-sheet.stories.tsx +43 -0
- package/src/ui/components/bottom-sheet.tsx +227 -0
- package/src/ui/components/button.stories.tsx +68 -0
- package/src/ui/components/button.tsx +170 -0
- package/src/ui/components/card.stories.tsx +63 -0
- package/src/ui/components/card.tsx +85 -0
- package/src/ui/components/checkbox.stories.tsx +113 -0
- package/src/ui/components/checkbox.tsx +36 -0
- package/src/ui/components/command-block.stories.tsx +52 -0
- package/src/ui/components/command-block.tsx +86 -0
- package/src/ui/components/confirm-dialog.stories.tsx +91 -0
- package/src/ui/components/confirm-dialog.tsx +130 -0
- package/src/ui/components/cursor.tsx +115 -0
- package/src/ui/components/dialog.stories.tsx +169 -0
- package/src/ui/components/dialog.tsx +177 -0
- package/src/ui/components/dropdown-menu.stories.tsx +52 -0
- package/src/ui/components/dropdown-menu.tsx +117 -0
- package/src/ui/components/fit-text/fit-text.css +42 -0
- package/src/ui/components/fit-text/index.stories.tsx +33 -0
- package/src/ui/components/fit-text/index.tsx +45 -0
- package/src/ui/components/forms.stories.tsx +173 -0
- package/src/ui/components/graphs/bar-chart.tsx +153 -0
- package/src/ui/components/graphs/index.stories.tsx +64 -0
- package/src/ui/components/graphs/index.tsx +4 -0
- package/src/ui/components/graphs/line-chart.tsx +213 -0
- package/src/ui/components/graphs/utils.tsx +265 -0
- package/src/ui/components/grid/grid.css +79 -0
- package/src/ui/components/grid/index.tsx +19 -0
- package/src/ui/components/hover-bg.stories.tsx +29 -0
- package/src/ui/components/hover-bg.tsx +15 -0
- package/src/ui/components/icons/arrow.tsx +42 -0
- package/src/ui/components/icons/check.tsx +14 -0
- package/src/ui/components/icons/chevron.tsx +45 -0
- package/src/ui/components/icons/discord.tsx +16 -0
- package/src/ui/components/icons/eye.tsx +12 -0
- package/src/ui/components/icons/gear.tsx +51 -0
- package/src/ui/components/icons/github.tsx +16 -0
- package/src/ui/components/icons/hamburger.tsx +52 -0
- package/src/ui/components/icons/heart.tsx +12 -0
- package/src/ui/components/icons/index.ts +12 -0
- package/src/ui/components/icons/link.tsx +14 -0
- package/src/ui/components/icons/minus.tsx +14 -0
- package/src/ui/components/icons/search.tsx +28 -0
- package/src/ui/components/image-distortion.stories.tsx +120 -0
- package/src/ui/components/image-distortion.tsx +499 -0
- package/src/ui/components/input.stories.tsx +39 -0
- package/src/ui/components/input.tsx +20 -0
- package/src/ui/components/label.stories.tsx +26 -0
- package/src/ui/components/label.tsx +16 -0
- package/src/ui/components/leva-client.tsx +14 -0
- package/src/ui/components/list-item.stories.tsx +83 -0
- package/src/ui/components/list-item.tsx +37 -0
- package/src/ui/components/overlays/blend-modes.ts +13 -0
- package/src/ui/components/overlays/glitch.tsx +243 -0
- package/src/ui/components/overlays/greys.tsx +386 -0
- package/src/ui/components/overlays/index.tsx +47 -0
- package/src/ui/components/overlays/lens-layers.tsx +121 -0
- package/src/ui/components/overlays/lens.ts +91 -0
- package/src/ui/components/overlays/noise.tsx +174 -0
- package/src/ui/components/overlays/vignette.tsx +60 -0
- package/src/ui/components/poster.stories.tsx +513 -0
- package/src/ui/components/poster.tsx +411 -0
- package/src/ui/components/progress.stories.tsx +48 -0
- package/src/ui/components/progress.tsx +56 -0
- package/src/ui/components/scene-canvas.tsx +254 -0
- package/src/ui/components/scramble.stories.tsx +49 -0
- package/src/ui/components/scramble.tsx +95 -0
- package/src/ui/components/segmented.stories.tsx +101 -0
- package/src/ui/components/segmented.tsx +81 -0
- package/src/ui/components/select.stories.tsx +88 -0
- package/src/ui/components/select.tsx +267 -0
- package/src/ui/components/selection-switcher.tsx +44 -0
- package/src/ui/components/separator.stories.tsx +33 -0
- package/src/ui/components/separator.tsx +24 -0
- package/src/ui/components/shader.tsx +83 -0
- package/src/ui/components/socials.tsx +42 -0
- package/src/ui/components/spinner.stories.tsx +101 -0
- package/src/ui/components/spinner.tsx +60 -0
- package/src/ui/components/stats.stories.tsx +24 -0
- package/src/ui/components/stats.tsx +53 -0
- package/src/ui/components/switch.stories.tsx +77 -0
- package/src/ui/components/switch.tsx +48 -0
- package/src/ui/components/tabs.stories.tsx +101 -0
- package/src/ui/components/tabs.tsx +66 -0
- package/src/ui/components/terminal-demo.stories.tsx +67 -0
- package/src/ui/components/terminal-demo.tsx +189 -0
- package/src/ui/components/theme-toggle.stories.tsx +47 -0
- package/src/ui/components/theme-toggle.tsx +66 -0
- package/src/ui/components/tier-card.stories.tsx +217 -0
- package/src/ui/components/tier-card.tsx +190 -0
- package/src/ui/components/toast.stories.tsx +55 -0
- package/src/ui/components/toast.tsx +49 -0
- package/src/ui/components/tv.stories.tsx +37 -0
- package/src/ui/components/tv.tsx +257 -0
- package/src/ui/components/typography/h1.tsx +18 -0
- package/src/ui/components/typography/h2.tsx +18 -0
- package/src/ui/components/typography/index.tsx +54 -0
- package/src/ui/components/typography/legend.tsx +24 -0
- package/src/ui/components/typography/small.tsx +11 -0
- package/src/ui/components/watchlist.stories.tsx +33 -0
- package/src/ui/components/watchlist.tsx +105 -0
- package/src/ui/fonts.css +63 -0
- package/src/ui/footer.tsx +111 -0
- package/src/ui/globals.css +395 -0
- package/src/ui/header.tsx +398 -0
- package/src/ui/layout-wrapper.tsx +11 -0
- package/src/utils/color.ts +21 -0
- package/src/utils/index.ts +62 -0
- package/src/utils/poly.ts +26 -0
package/CHANGELOG.md
ADDED
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
|
package/dist/assets.d.ts
ADDED
|
@@ -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
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
package/dist/fonts.d.ts
ADDED
|
@@ -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,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,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 {};
|