The response has been limited to 50k tokens of the smallest files in the repo. You can remove this limitation by removing the max tokens filter.
├── .gitignore
├── LICENSE
├── README.md
├── assets
    ├── button.png
    ├── card.png
    ├── project-liquid.gif
    └── video.mov
├── bun.lock
├── esbuild.config.js
├── liquid-glass-example
    ├── .gitignore
    ├── README.md
    ├── bun.lock
    ├── bun.lockb
    ├── next.config.ts
    ├── package.json
    ├── postcss.config.mjs
    ├── public
    │   └── favicon.ico
    ├── src
    │   ├── pages
    │   │   ├── _app.tsx
    │   │   ├── _document.tsx
    │   │   ├── api
    │   │   │   └── hello.ts
    │   │   └── index.tsx
    │   └── styles
    │   │   └── globals.css
    └── tsconfig.json
├── package-lock.json
├── package.json
├── src
    ├── index.tsx
    ├── shader-utils.ts
    └── utils.ts
└── tsconfig.json


/.gitignore:
--------------------------------------------------------------------------------
  1 | # Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore
  2 | 
  3 | # Logs
  4 | 
  5 | logs
  6 | _.log
  7 | npm-debug.log_
  8 | yarn-debug.log*
  9 | yarn-error.log*
 10 | lerna-debug.log*
 11 | .pnpm-debug.log*
 12 | 
 13 | # Caches
 14 | 
 15 | .cache
 16 | 
 17 | # Diagnostic reports (https://nodejs.org/api/report.html)
 18 | 
 19 | report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json
 20 | 
 21 | # Runtime data
 22 | 
 23 | pids
 24 | _.pid
 25 | _.seed
 26 | *.pid.lock
 27 | 
 28 | # Directory for instrumented libs generated by jscoverage/JSCover
 29 | 
 30 | lib-cov
 31 | 
 32 | # Coverage directory used by tools like istanbul
 33 | 
 34 | coverage
 35 | *.lcov
 36 | 
 37 | # nyc test coverage
 38 | 
 39 | .nyc_output
 40 | 
 41 | # Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
 42 | 
 43 | .grunt
 44 | 
 45 | # Bower dependency directory (https://bower.io/)
 46 | 
 47 | bower_components
 48 | 
 49 | # node-waf configuration
 50 | 
 51 | .lock-wscript
 52 | 
 53 | # Compiled binary addons (https://nodejs.org/api/addons.html)
 54 | 
 55 | build/Release
 56 | 
 57 | # Dependency directories
 58 | 
 59 | node_modules/
 60 | jspm_packages/
 61 | 
 62 | # Snowpack dependency directory (https://snowpack.dev/)
 63 | 
 64 | web_modules/
 65 | 
 66 | # TypeScript cache
 67 | 
 68 | *.tsbuildinfo
 69 | 
 70 | # Optional npm cache directory
 71 | 
 72 | .npm
 73 | 
 74 | # Optional eslint cache
 75 | 
 76 | .eslintcache
 77 | 
 78 | # Optional stylelint cache
 79 | 
 80 | .stylelintcache
 81 | 
 82 | # Microbundle cache
 83 | 
 84 | .rpt2_cache/
 85 | .rts2_cache_cjs/
 86 | .rts2_cache_es/
 87 | .rts2_cache_umd/
 88 | 
 89 | # Optional REPL history
 90 | 
 91 | .node_repl_history
 92 | 
 93 | # Output of 'npm pack'
 94 | 
 95 | *.tgz
 96 | 
 97 | # Yarn Integrity file
 98 | 
 99 | .yarn-integrity
100 | 
101 | # dotenv environment variable files
102 | 
103 | .env
104 | .env.development.local
105 | .env.test.local
106 | .env.production.local
107 | .env.local
108 | 
109 | # parcel-bundler cache (https://parceljs.org/)
110 | 
111 | .parcel-cache
112 | 
113 | # Next.js build output
114 | 
115 | .next
116 | out
117 | 
118 | # Nuxt.js build / generate output
119 | 
120 | .nuxt
121 | dist
122 | 
123 | # Build output
124 | dist/
125 | 
126 | # Gatsby files
127 | 
128 | # Comment in the public line in if your project uses Gatsby and not Next.js
129 | 
130 | # https://nextjs.org/blog/next-9-1#public-directory-support
131 | 
132 | # public
133 | 
134 | # vuepress build output
135 | 
136 | .vuepress/dist
137 | 
138 | # vuepress v2.x temp and cache directory
139 | 
140 | .temp
141 | 
142 | # Docusaurus cache and generated files
143 | 
144 | .docusaurus
145 | 
146 | # Serverless directories
147 | 
148 | .serverless/
149 | 
150 | # FuseBox cache
151 | 
152 | .fusebox/
153 | 
154 | # DynamoDB Local files
155 | 
156 | .dynamodb/
157 | 
158 | # TernJS port file
159 | 
160 | .tern-port
161 | 
162 | # Stores VSCode versions used for testing VSCode extensions
163 | 
164 | .vscode-test
165 | 
166 | # yarn v2
167 | 
168 | .yarn/cache
169 | .yarn/unplugged
170 | .yarn/build-state.yml
171 | .yarn/install-state.gz
172 | .pnp.*
173 | 
174 | # IntelliJ based IDEs
175 | .idea
176 | 
177 | # Finder (MacOS) folder config
178 | .DS_Store
179 | 
180 | liquid-glass-example/src/components
181 | 


--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | Copyright 2025 MAX ROVENSKY
2 | 
3 | Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the “Software”), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
4 | 
5 | The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
6 | 
7 | THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
8 | 


--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
  1 | # Liquid Glass React
  2 | 
  3 | Apple's Liquid Glass effect for React.
  4 | 
  5 | Card Example              |  Button Example
  6 | :-------------------------:|:-------------------------:
  7 | ![](https://github.com/rdev/liquid-glass-react/raw/master/assets/card.png)  |  ![](https://github.com/rdev/liquid-glass-react/raw/master/assets/button.png)
  8 | 
  9 | ## 🎬  Demo
 10 | 
 11 | [Click here](https://liquid-glass.maxrovensky.com) to see it in action!
 12 | 
 13 | ![project liquid gif](./assets/project-liquid.gif)
 14 | 
 15 | ## ✨ Features
 16 | 
 17 | - Proper edgy bending and refraction
 18 | - Multiple refraction modes
 19 | - Configurable frosty level
 20 | - Supports arbitrary child elements
 21 | - Configurable paddings
 22 | - Correct hover and click effects
 23 | - Edges and highlights take on the underlying light like Apple's does
 24 | - Configurable chromatic aberration
 25 | - Configurable elasticity, to mimic Apple's "liquid" feel
 26 | 
 27 | > **⚠️ NOTE:** Safari and Firefox only partially support the effect (displacement will not be visible)
 28 | 
 29 | ## 🚀 Usage
 30 | 
 31 | ### Installation
 32 | 
 33 | ```bash
 34 | npm install liquid-glass-react
 35 | ```
 36 | 
 37 | ### Basic Usage
 38 | 
 39 | ```tsx
 40 | import LiquidGlass from 'liquid-glass-react'
 41 | 
 42 | function App() {
 43 |   return (
 44 |     <LiquidGlass>
 45 |       <div className="p-6">
 46 |         <h2>Your content here</h2>
 47 |         <p>This will have the liquid glass effect</p>
 48 |       </div>
 49 |     </LiquidGlass>
 50 |   )
 51 | }
 52 | ```
 53 | 
 54 | ### Button Example
 55 | 
 56 | ```tsx
 57 | <LiquidGlass
 58 |   displacementScale={64}
 59 |   blurAmount={0.1}
 60 |   saturation={130}
 61 |   aberrationIntensity={2}
 62 |   elasticity={0.35}
 63 |   cornerRadius={100}
 64 |   padding="8px 16px"
 65 |   onClick={() => console.log('Button clicked!')}
 66 | >
 67 |   <span className="text-white font-medium">Click Me</span>
 68 | </LiquidGlass>
 69 | ```
 70 | 
 71 | ### Mouse Container Example
 72 | 
 73 | When you want the glass effect to respond to mouse movement over a larger area (like a parent container), use the `mouseContainer` prop:
 74 | 
 75 | ```tsx
 76 | function App() {
 77 |   const containerRef = useRef<HTMLDivElement>(null)
 78 | 
 79 |   return (
 80 |     <div ref={containerRef} className="w-full h-screen bg-image">
 81 |       <LiquidGlass
 82 |         mouseContainer={containerRef}
 83 |         elasticity={0.3}
 84 |         style={{ position: 'fixed', top: '50%', left: '50%' }}
 85 |       >
 86 |         <div className="p-6">
 87 |           <h2>Glass responds to mouse anywhere in the container</h2>
 88 |         </div>
 89 |       </LiquidGlass>
 90 |     </div>
 91 |   )
 92 | }
 93 | ```
 94 | 
 95 | ## Props
 96 | 
 97 | | Prop | Type | Default | Description |
 98 | |------|------|---------|-------------|
 99 | | `children` | `React.ReactNode` | - | The content to render inside the glass container |
100 | | `displacementScale` | `number` | `70` | Controls the intensity of the displacement effect |
101 | | `blurAmount` | `number` | `0.0625` | Controls the blur/frosting level |
102 | | `saturation` | `number` | `140` | Controls color saturation of the glass effect |
103 | | `aberrationIntensity` | `number` | `2` | Controls chromatic aberration intensity |
104 | | `elasticity` | `number` | `0.15` | Controls the "liquid" elastic feel (0 = rigid, higher = more elastic) |
105 | | `cornerRadius` | `number` | `999` | Border radius in pixels |
106 | | `className` | `string` | `""` | Additional CSS classes |
107 | | `padding` | `string` | - | CSS padding value |
108 | | `style` | `React.CSSProperties` | - | Additional inline styles |
109 | | `overLight` | `boolean` | `false` | Whether the glass is over a light background |
110 | | `onClick` | `() => void` | - | Click handler |
111 | | `mouseContainer` | `React.RefObject<HTMLElement \| null> \| null` | `null` | Container element to track mouse movement on (defaults to the glass component itself) |
112 | | `mode` | `"standard" \| "polar" \| "prominent" \| "shader"` | `"standard"` | Refraction mode for different visual effects. `shader` is the most accurate but not the most stable. |
113 | | `globalMousePos` | `{ x: number; y: number }` | - | Global mouse position coordinates for manual control |
114 | | `mouseOffset` | `{ x: number; y: number }` | - | Mouse position offset for fine-tuning positioning |
115 | 


--------------------------------------------------------------------------------
/assets/button.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/rdev/liquid-glass-react/ac48eab18d1f7f444ae30002d240cae29c863a21/assets/button.png


--------------------------------------------------------------------------------
/assets/card.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/rdev/liquid-glass-react/ac48eab18d1f7f444ae30002d240cae29c863a21/assets/card.png


--------------------------------------------------------------------------------
/assets/project-liquid.gif:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/rdev/liquid-glass-react/ac48eab18d1f7f444ae30002d240cae29c863a21/assets/project-liquid.gif


--------------------------------------------------------------------------------
/assets/video.mov:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/rdev/liquid-glass-react/ac48eab18d1f7f444ae30002d240cae29c863a21/assets/video.mov


--------------------------------------------------------------------------------
/bun.lock:
--------------------------------------------------------------------------------
  1 | {
  2 |   "lockfileVersion": 1,
  3 |   "workspaces": {
  4 |     "": {
  5 |       "name": "liquid-glass-react",
  6 |       "devDependencies": {
  7 |         "@biomejs/biome": "^1.9.4",
  8 |         "@types/react": "^18.2.0",
  9 |         "@types/react-dom": "^18.2.0",
 10 |         "esbuild": "^0.19.0",
 11 |         "react": "^18.2.0",
 12 |         "react-dom": "^18.2.0",
 13 |         "typescript": "^5.0.0",
 14 |       },
 15 |       "peerDependencies": {
 16 |         "react": ">=16.8.0",
 17 |         "react-dom": ">=16.8.0",
 18 |       },
 19 |     },
 20 |   },
 21 |   "packages": {
 22 |     "@biomejs/biome": ["@biomejs/biome@1.9.4", "", { "optionalDependencies": { "@biomejs/cli-darwin-arm64": "1.9.4", "@biomejs/cli-darwin-x64": "1.9.4", "@biomejs/cli-linux-arm64": "1.9.4", "@biomejs/cli-linux-arm64-musl": "1.9.4", "@biomejs/cli-linux-x64": "1.9.4", "@biomejs/cli-linux-x64-musl": "1.9.4", "@biomejs/cli-win32-arm64": "1.9.4", "@biomejs/cli-win32-x64": "1.9.4" }, "bin": { "biome": "bin/biome" } }, "sha512-1rkd7G70+o9KkTn5KLmDYXihGoTaIGO9PIIN2ZB7UJxFrWw04CZHPYiMRjYsaDvVV7hP1dYNRLxSANLaBFGpog=="],
 23 | 
 24 |     "@biomejs/cli-darwin-arm64": ["@biomejs/cli-darwin-arm64@1.9.4", "", { "os": "darwin", "cpu": "arm64" }, "sha512-bFBsPWrNvkdKrNCYeAp+xo2HecOGPAy9WyNyB/jKnnedgzl4W4Hb9ZMzYNbf8dMCGmUdSavlYHiR01QaYR58cw=="],
 25 | 
 26 |     "@biomejs/cli-darwin-x64": ["@biomejs/cli-darwin-x64@1.9.4", "", { "os": "darwin", "cpu": "x64" }, "sha512-ngYBh/+bEedqkSevPVhLP4QfVPCpb+4BBe2p7Xs32dBgs7rh9nY2AIYUL6BgLw1JVXV8GlpKmb/hNiuIxfPfZg=="],
 27 | 
 28 |     "@biomejs/cli-linux-arm64": ["@biomejs/cli-linux-arm64@1.9.4", "", { "os": "linux", "cpu": "arm64" }, "sha512-fJIW0+LYujdjUgJJuwesP4EjIBl/N/TcOX3IvIHJQNsAqvV2CHIogsmA94BPG6jZATS4Hi+xv4SkBBQSt1N4/g=="],
 29 | 
 30 |     "@biomejs/cli-linux-arm64-musl": ["@biomejs/cli-linux-arm64-musl@1.9.4", "", { "os": "linux", "cpu": "arm64" }, "sha512-v665Ct9WCRjGa8+kTr0CzApU0+XXtRgwmzIf1SeKSGAv+2scAlW6JR5PMFo6FzqqZ64Po79cKODKf3/AAmECqA=="],
 31 | 
 32 |     "@biomejs/cli-linux-x64": ["@biomejs/cli-linux-x64@1.9.4", "", { "os": "linux", "cpu": "x64" }, "sha512-lRCJv/Vi3Vlwmbd6K+oQ0KhLHMAysN8lXoCI7XeHlxaajk06u7G+UsFSO01NAs5iYuWKmVZjmiOzJ0OJmGsMwg=="],
 33 | 
 34 |     "@biomejs/cli-linux-x64-musl": ["@biomejs/cli-linux-x64-musl@1.9.4", "", { "os": "linux", "cpu": "x64" }, "sha512-gEhi/jSBhZ2m6wjV530Yy8+fNqG8PAinM3oV7CyO+6c3CEh16Eizm21uHVsyVBEB6RIM8JHIl6AGYCv6Q6Q9Tg=="],
 35 | 
 36 |     "@biomejs/cli-win32-arm64": ["@biomejs/cli-win32-arm64@1.9.4", "", { "os": "win32", "cpu": "arm64" }, "sha512-tlbhLk+WXZmgwoIKwHIHEBZUwxml7bRJgk0X2sPyNR3S93cdRq6XulAZRQJ17FYGGzWne0fgrXBKpl7l4M87Hg=="],
 37 | 
 38 |     "@biomejs/cli-win32-x64": ["@biomejs/cli-win32-x64@1.9.4", "", { "os": "win32", "cpu": "x64" }, "sha512-8Y5wMhVIPaWe6jw2H+KlEm4wP/f7EW3810ZLmDlrEEy5KvBsb9ECEfu/kMWD484ijfQ8+nIi0giMgu9g1UAuuA=="],
 39 | 
 40 |     "@esbuild/aix-ppc64": ["@esbuild/aix-ppc64@0.19.12", "", { "os": "aix", "cpu": "ppc64" }, "sha512-bmoCYyWdEL3wDQIVbcyzRyeKLgk2WtWLTWz1ZIAZF/EGbNOwSA6ew3PftJ1PqMiOOGu0OyFMzG53L0zqIpPeNA=="],
 41 | 
 42 |     "@esbuild/android-arm": ["@esbuild/android-arm@0.19.12", "", { "os": "android", "cpu": "arm" }, "sha512-qg/Lj1mu3CdQlDEEiWrlC4eaPZ1KztwGJ9B6J+/6G+/4ewxJg7gqj8eVYWvao1bXrqGiW2rsBZFSX3q2lcW05w=="],
 43 | 
 44 |     "@esbuild/android-arm64": ["@esbuild/android-arm64@0.19.12", "", { "os": "android", "cpu": "arm64" }, "sha512-P0UVNGIienjZv3f5zq0DP3Nt2IE/3plFzuaS96vihvD0Hd6H/q4WXUGpCxD/E8YrSXfNyRPbpTq+T8ZQioSuPA=="],
 45 | 
 46 |     "@esbuild/android-x64": ["@esbuild/android-x64@0.19.12", "", { "os": "android", "cpu": "x64" }, "sha512-3k7ZoUW6Q6YqhdhIaq/WZ7HwBpnFBlW905Fa4s4qWJyiNOgT1dOqDiVAQFwBH7gBRZr17gLrlFCRzF6jFh7Kew=="],
 47 | 
 48 |     "@esbuild/darwin-arm64": ["@esbuild/darwin-arm64@0.19.12", "", { "os": "darwin", "cpu": "arm64" }, "sha512-B6IeSgZgtEzGC42jsI+YYu9Z3HKRxp8ZT3cqhvliEHovq8HSX2YX8lNocDn79gCKJXOSaEot9MVYky7AKjCs8g=="],
 49 | 
 50 |     "@esbuild/darwin-x64": ["@esbuild/darwin-x64@0.19.12", "", { "os": "darwin", "cpu": "x64" }, "sha512-hKoVkKzFiToTgn+41qGhsUJXFlIjxI/jSYeZf3ugemDYZldIXIxhvwN6erJGlX4t5h417iFuheZ7l+YVn05N3A=="],
 51 | 
 52 |     "@esbuild/freebsd-arm64": ["@esbuild/freebsd-arm64@0.19.12", "", { "os": "freebsd", "cpu": "arm64" }, "sha512-4aRvFIXmwAcDBw9AueDQ2YnGmz5L6obe5kmPT8Vd+/+x/JMVKCgdcRwH6APrbpNXsPz+K653Qg8HB/oXvXVukA=="],
 53 | 
 54 |     "@esbuild/freebsd-x64": ["@esbuild/freebsd-x64@0.19.12", "", { "os": "freebsd", "cpu": "x64" }, "sha512-EYoXZ4d8xtBoVN7CEwWY2IN4ho76xjYXqSXMNccFSx2lgqOG/1TBPW0yPx1bJZk94qu3tX0fycJeeQsKovA8gg=="],
 55 | 
 56 |     "@esbuild/linux-arm": ["@esbuild/linux-arm@0.19.12", "", { "os": "linux", "cpu": "arm" }, "sha512-J5jPms//KhSNv+LO1S1TX1UWp1ucM6N6XuL6ITdKWElCu8wXP72l9MM0zDTzzeikVyqFE6U8YAV9/tFyj0ti+w=="],
 57 | 
 58 |     "@esbuild/linux-arm64": ["@esbuild/linux-arm64@0.19.12", "", { "os": "linux", "cpu": "arm64" }, "sha512-EoTjyYyLuVPfdPLsGVVVC8a0p1BFFvtpQDB/YLEhaXyf/5bczaGeN15QkR+O4S5LeJ92Tqotve7i1jn35qwvdA=="],
 59 | 
 60 |     "@esbuild/linux-ia32": ["@esbuild/linux-ia32@0.19.12", "", { "os": "linux", "cpu": "ia32" }, "sha512-Thsa42rrP1+UIGaWz47uydHSBOgTUnwBwNq59khgIwktK6x60Hivfbux9iNR0eHCHzOLjLMLfUMLCypBkZXMHA=="],
 61 | 
 62 |     "@esbuild/linux-loong64": ["@esbuild/linux-loong64@0.19.12", "", { "os": "linux", "cpu": "none" }, "sha512-LiXdXA0s3IqRRjm6rV6XaWATScKAXjI4R4LoDlvO7+yQqFdlr1Bax62sRwkVvRIrwXxvtYEHHI4dm50jAXkuAA=="],
 63 | 
 64 |     "@esbuild/linux-mips64el": ["@esbuild/linux-mips64el@0.19.12", "", { "os": "linux", "cpu": "none" }, "sha512-fEnAuj5VGTanfJ07ff0gOA6IPsvrVHLVb6Lyd1g2/ed67oU1eFzL0r9WL7ZzscD+/N6i3dWumGE1Un4f7Amf+w=="],
 65 | 
 66 |     "@esbuild/linux-ppc64": ["@esbuild/linux-ppc64@0.19.12", "", { "os": "linux", "cpu": "ppc64" }, "sha512-nYJA2/QPimDQOh1rKWedNOe3Gfc8PabU7HT3iXWtNUbRzXS9+vgB0Fjaqr//XNbd82mCxHzik2qotuI89cfixg=="],
 67 | 
 68 |     "@esbuild/linux-riscv64": ["@esbuild/linux-riscv64@0.19.12", "", { "os": "linux", "cpu": "none" }, "sha512-2MueBrlPQCw5dVJJpQdUYgeqIzDQgw3QtiAHUC4RBz9FXPrskyyU3VI1hw7C0BSKB9OduwSJ79FTCqtGMWqJHg=="],
 69 | 
 70 |     "@esbuild/linux-s390x": ["@esbuild/linux-s390x@0.19.12", "", { "os": "linux", "cpu": "s390x" }, "sha512-+Pil1Nv3Umes4m3AZKqA2anfhJiVmNCYkPchwFJNEJN5QxmTs1uzyy4TvmDrCRNT2ApwSari7ZIgrPeUx4UZDg=="],
 71 | 
 72 |     "@esbuild/linux-x64": ["@esbuild/linux-x64@0.19.12", "", { "os": "linux", "cpu": "x64" }, "sha512-B71g1QpxfwBvNrfyJdVDexenDIt1CiDN1TIXLbhOw0KhJzE78KIFGX6OJ9MrtC0oOqMWf+0xop4qEU8JrJTwCg=="],
 73 | 
 74 |     "@esbuild/netbsd-x64": ["@esbuild/netbsd-x64@0.19.12", "", { "os": "none", "cpu": "x64" }, "sha512-3ltjQ7n1owJgFbuC61Oj++XhtzmymoCihNFgT84UAmJnxJfm4sYCiSLTXZtE00VWYpPMYc+ZQmB6xbSdVh0JWA=="],
 75 | 
 76 |     "@esbuild/openbsd-x64": ["@esbuild/openbsd-x64@0.19.12", "", { "os": "openbsd", "cpu": "x64" }, "sha512-RbrfTB9SWsr0kWmb9srfF+L933uMDdu9BIzdA7os2t0TXhCRjrQyCeOt6wVxr79CKD4c+p+YhCj31HBkYcXebw=="],
 77 | 
 78 |     "@esbuild/sunos-x64": ["@esbuild/sunos-x64@0.19.12", "", { "os": "sunos", "cpu": "x64" }, "sha512-HKjJwRrW8uWtCQnQOz9qcU3mUZhTUQvi56Q8DPTLLB+DawoiQdjsYq+j+D3s9I8VFtDr+F9CjgXKKC4ss89IeA=="],
 79 | 
 80 |     "@esbuild/win32-arm64": ["@esbuild/win32-arm64@0.19.12", "", { "os": "win32", "cpu": "arm64" }, "sha512-URgtR1dJnmGvX864pn1B2YUYNzjmXkuJOIqG2HdU62MVS4EHpU2946OZoTMnRUHklGtJdJZ33QfzdjGACXhn1A=="],
 81 | 
 82 |     "@esbuild/win32-ia32": ["@esbuild/win32-ia32@0.19.12", "", { "os": "win32", "cpu": "ia32" }, "sha512-+ZOE6pUkMOJfmxmBZElNOx72NKpIa/HFOMGzu8fqzQJ5kgf6aTGrcJaFsNiVMH4JKpMipyK+7k0n2UXN7a8YKQ=="],
 83 | 
 84 |     "@esbuild/win32-x64": ["@esbuild/win32-x64@0.19.12", "", { "os": "win32", "cpu": "x64" }, "sha512-T1QyPSDCyMXaO3pzBkF96E8xMkiRYbUEZADd29SyPGabqxMViNoii+NcK7eWJAEoU6RZyEm5lVSIjTmcdoB9HA=="],
 85 | 
 86 |     "@types/prop-types": ["@types/prop-types@15.7.15", "", {}, "sha512-F6bEyamV9jKGAFBEmlQnesRPGOQqS2+Uwi0Em15xenOxHaf2hv6L8YCVn3rPdPJOiJfPiCnLIRyvwVaqMY3MIw=="],
 87 | 
 88 |     "@types/react": ["@types/react@18.3.23", "", { "dependencies": { "@types/prop-types": "*", "csstype": "^3.0.2" } }, "sha512-/LDXMQh55EzZQ0uVAZmKKhfENivEvWz6E+EYzh+/MCjMhNsotd+ZHhBGIjFDTi6+fz0OhQQQLbTgdQIxxCsC0w=="],
 89 | 
 90 |     "@types/react-dom": ["@types/react-dom@18.3.7", "", { "peerDependencies": { "@types/react": "^18.0.0" } }, "sha512-MEe3UeoENYVFXzoXEWsvcpg6ZvlrFNlOQ7EOsvhI3CfAXwzPfO8Qwuxd40nepsYKqyyVQnTdEfv68q91yLcKrQ=="],
 91 | 
 92 |     "csstype": ["csstype@3.1.3", "", {}, "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw=="],
 93 | 
 94 |     "esbuild": ["esbuild@0.19.12", "", { "optionalDependencies": { "@esbuild/aix-ppc64": "0.19.12", "@esbuild/android-arm": "0.19.12", "@esbuild/android-arm64": "0.19.12", "@esbuild/android-x64": "0.19.12", "@esbuild/darwin-arm64": "0.19.12", "@esbuild/darwin-x64": "0.19.12", "@esbuild/freebsd-arm64": "0.19.12", "@esbuild/freebsd-x64": "0.19.12", "@esbuild/linux-arm": "0.19.12", "@esbuild/linux-arm64": "0.19.12", "@esbuild/linux-ia32": "0.19.12", "@esbuild/linux-loong64": "0.19.12", "@esbuild/linux-mips64el": "0.19.12", "@esbuild/linux-ppc64": "0.19.12", "@esbuild/linux-riscv64": "0.19.12", "@esbuild/linux-s390x": "0.19.12", "@esbuild/linux-x64": "0.19.12", "@esbuild/netbsd-x64": "0.19.12", "@esbuild/openbsd-x64": "0.19.12", "@esbuild/sunos-x64": "0.19.12", "@esbuild/win32-arm64": "0.19.12", "@esbuild/win32-ia32": "0.19.12", "@esbuild/win32-x64": "0.19.12" }, "bin": "bin/esbuild" }, "sha512-aARqgq8roFBj054KvQr5f1sFu0D65G+miZRCuJyJ0G13Zwx7vRar5Zhn2tkQNzIXcBrNVsv/8stehpj+GAjgbg=="],
 95 | 
 96 |     "js-tokens": ["js-tokens@4.0.0", "", {}, "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ=="],
 97 | 
 98 |     "loose-envify": ["loose-envify@1.4.0", "", { "dependencies": { "js-tokens": "^3.0.0 || ^4.0.0" }, "bin": "cli.js" }, "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q=="],
 99 | 
100 |     "react": ["react@18.3.1", "", { "dependencies": { "loose-envify": "^1.1.0" } }, "sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ=="],
101 | 
102 |     "react-dom": ["react-dom@18.3.1", "", { "dependencies": { "loose-envify": "^1.1.0", "scheduler": "^0.23.2" }, "peerDependencies": { "react": "^18.3.1" } }, "sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw=="],
103 | 
104 |     "scheduler": ["scheduler@0.23.2", "", { "dependencies": { "loose-envify": "^1.1.0" } }, "sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ=="],
105 | 
106 |     "typescript": ["typescript@5.8.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-p1diW6TqL9L07nNxvRMM7hMMw4c5XOo/1ibL4aAIGmSAt9slTE1Xgw5KWuof2uTOvCg9BY7ZRi+GaF+7sfgPeQ=="],
107 |   }
108 | }
109 | 


--------------------------------------------------------------------------------
/esbuild.config.js:
--------------------------------------------------------------------------------
 1 | const esbuild = require('esbuild');
 2 | const path = require('path');
 3 | 
 4 | const createBuildConfig = (format) => ({
 5 |   entryPoints: ['src/index.ts'],
 6 |   bundle: true,
 7 |   format,
 8 |   outfile: format === 'esm' ? 'dist/index.esm.js' : 'dist/index.js',
 9 |   external: ['react', 'react-dom'],
10 |   target: ['es2020'],
11 |   jsx: 'automatic',
12 |   jsxImportSource: 'react',
13 |   minify: process.env.NODE_ENV === 'production',
14 |   sourcemap: true,
15 |   platform: 'browser',
16 |   splitting: false,
17 |   loader: {
18 |     '.tsx': 'tsx',
19 |     '.ts': 'ts',
20 |     '.js': 'js',
21 |     '.jsx': 'jsx'
22 |   },
23 |   define: {
24 |     'process.env.NODE_ENV': JSON.stringify(process.env.NODE_ENV || 'development')
25 |   }
26 | });
27 | 
28 | const buildAll = async () => {
29 |   try {
30 |     console.log('🚀 Building ESM bundle...');
31 |     await esbuild.build(createBuildConfig('esm'));
32 |     console.log('✅ ESM bundle built successfully');
33 | 
34 |     console.log('🚀 Building CJS bundle...');
35 |     await esbuild.build(createBuildConfig('cjs'));
36 |     console.log('✅ CJS bundle built successfully');
37 | 
38 |     console.log('🎉 All bundles built successfully!');
39 |   } catch (error) {
40 |     console.error('❌ Build failed:', error);
41 |     process.exit(1);
42 |   }
43 | };
44 | 
45 | const watch = async () => {
46 |   try {
47 |     console.log('👀 Starting watch mode for ESM bundle...');
48 |     const ctx = await esbuild.context(createBuildConfig('esm'));
49 |     await ctx.watch();
50 |     console.log('✅ Watch mode active - listening for changes...');
51 |   } catch (error) {
52 |     console.error('❌ Watch mode failed:', error);
53 |     process.exit(1);
54 |   }
55 | };
56 | 
57 | // Check command line arguments
58 | const args = process.argv.slice(2);
59 | if (args.includes('--watch')) {
60 |   watch();
61 | } else {
62 |   buildAll();
63 | }
64 | 
65 | module.exports = { createBuildConfig, buildAll, watch };


--------------------------------------------------------------------------------
/liquid-glass-example/.gitignore:
--------------------------------------------------------------------------------
 1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
 2 | 
 3 | # dependencies
 4 | /node_modules
 5 | /.pnp
 6 | .pnp.*
 7 | .yarn/*
 8 | !.yarn/patches
 9 | !.yarn/plugins
10 | !.yarn/releases
11 | !.yarn/versions
12 | 
13 | # testing
14 | /coverage
15 | 
16 | # next.js
17 | /.next/
18 | /out/
19 | 
20 | # production
21 | /build
22 | 
23 | # misc
24 | .DS_Store
25 | *.pem
26 | 
27 | # debug
28 | npm-debug.log*
29 | yarn-debug.log*
30 | yarn-error.log*
31 | .pnpm-debug.log*
32 | 
33 | # env files (can opt-in for committing if needed)
34 | .env*
35 | 
36 | # vercel
37 | .vercel
38 | 
39 | # typescript
40 | *.tsbuildinfo
41 | next-env.d.ts
42 | 


--------------------------------------------------------------------------------
/liquid-glass-example/README.md:
--------------------------------------------------------------------------------
 1 | This is a [Next.js](https://nextjs.org) project bootstrapped with [`create-next-app`](https://nextjs.org/docs/pages/api-reference/create-next-app).
 2 | 
 3 | ## Getting Started
 4 | 
 5 | First, run the development server:
 6 | 
 7 | ```bash
 8 | npm run dev
 9 | # or
10 | yarn dev
11 | # or
12 | pnpm dev
13 | # or
14 | bun dev
15 | ```
16 | 
17 | Open [http://localhost:3000](http://localhost:3000) with your browser to see the result.
18 | 
19 | You can start editing the page by modifying `pages/index.tsx`. The page auto-updates as you edit the file.
20 | 
21 | [API routes](https://nextjs.org/docs/pages/building-your-application/routing/api-routes) can be accessed on [http://localhost:3000/api/hello](http://localhost:3000/api/hello). This endpoint can be edited in `pages/api/hello.ts`.
22 | 
23 | The `pages/api` directory is mapped to `/api/*`. Files in this directory are treated as [API routes](https://nextjs.org/docs/pages/building-your-application/routing/api-routes) instead of React pages.
24 | 
25 | This project uses [`next/font`](https://nextjs.org/docs/pages/building-your-application/optimizing/fonts) to automatically optimize and load [Geist](https://vercel.com/font), a new font family for Vercel.
26 | 
27 | ## Learn More
28 | 
29 | To learn more about Next.js, take a look at the following resources:
30 | 
31 | - [Next.js Documentation](https://nextjs.org/docs) - learn about Next.js features and API.
32 | - [Learn Next.js](https://nextjs.org/learn-pages-router) - an interactive Next.js tutorial.
33 | 
34 | You can check out [the Next.js GitHub repository](https://github.com/vercel/next.js) - your feedback and contributions are welcome!
35 | 
36 | ## Deploy on Vercel
37 | 
38 | The easiest way to deploy your Next.js app is to use the [Vercel Platform](https://vercel.com/new?utm_medium=default-template&filter=next.js&utm_source=create-next-app&utm_campaign=create-next-app-readme) from the creators of Next.js.
39 | 
40 | Check out our [Next.js deployment documentation](https://nextjs.org/docs/pages/building-your-application/deploying) for more details.
41 | 


--------------------------------------------------------------------------------
/liquid-glass-example/bun.lock:
--------------------------------------------------------------------------------
  1 | {
  2 |   "lockfileVersion": 1,
  3 |   "workspaces": {
  4 |     "": {
  5 |       "name": "liquid-glass-example",
  6 |       "dependencies": {
  7 |         "liquid-glass-react": "latest",
  8 |         "lucide-react": "^0.514.0",
  9 |         "next": "15.3.3",
 10 |         "react": "^19.0.0",
 11 |         "react-dom": "^19.0.0",
 12 |       },
 13 |       "devDependencies": {
 14 |         "@tailwindcss/postcss": "^4",
 15 |         "@types/node": "^20",
 16 |         "@types/react": "^19",
 17 |         "@types/react-dom": "^19",
 18 |         "tailwindcss": "^4",
 19 |         "typescript": "^5",
 20 |       },
 21 |     },
 22 |   },
 23 |   "packages": {
 24 |     "@alloc/quick-lru": ["@alloc/quick-lru@5.2.0", "", {}, "sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw=="],
 25 | 
 26 |     "@ampproject/remapping": ["@ampproject/remapping@2.3.0", "", { "dependencies": { "@jridgewell/gen-mapping": "^0.3.5", "@jridgewell/trace-mapping": "^0.3.24" } }, "sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw=="],
 27 | 
 28 |     "@emnapi/runtime": ["@emnapi/runtime@1.4.3", "", { "dependencies": { "tslib": "^2.4.0" } }, "sha512-pBPWdu6MLKROBX05wSNKcNb++m5Er+KQ9QkB+WVM+pW2Kx9hoSrVTnu3BdkI5eBLZoKu/J6mW/B6i6bJB2ytXQ=="],
 29 | 
 30 |     "@img/sharp-darwin-arm64": ["@img/sharp-darwin-arm64@0.34.2", "", { "optionalDependencies": { "@img/sharp-libvips-darwin-arm64": "1.1.0" }, "os": "darwin", "cpu": "arm64" }, "sha512-OfXHZPppddivUJnqyKoi5YVeHRkkNE2zUFT2gbpKxp/JZCFYEYubnMg+gOp6lWfasPrTS+KPosKqdI+ELYVDtg=="],
 31 | 
 32 |     "@img/sharp-darwin-x64": ["@img/sharp-darwin-x64@0.34.2", "", { "optionalDependencies": { "@img/sharp-libvips-darwin-x64": "1.1.0" }, "os": "darwin", "cpu": "x64" }, "sha512-dYvWqmjU9VxqXmjEtjmvHnGqF8GrVjM2Epj9rJ6BUIXvk8slvNDJbhGFvIoXzkDhrJC2jUxNLz/GUjjvSzfw+g=="],
 33 | 
 34 |     "@img/sharp-libvips-darwin-arm64": ["@img/sharp-libvips-darwin-arm64@1.1.0", "", { "os": "darwin", "cpu": "arm64" }, "sha512-HZ/JUmPwrJSoM4DIQPv/BfNh9yrOA8tlBbqbLz4JZ5uew2+o22Ik+tHQJcih7QJuSa0zo5coHTfD5J8inqj9DA=="],
 35 | 
 36 |     "@img/sharp-libvips-darwin-x64": ["@img/sharp-libvips-darwin-x64@1.1.0", "", { "os": "darwin", "cpu": "x64" }, "sha512-Xzc2ToEmHN+hfvsl9wja0RlnXEgpKNmftriQp6XzY/RaSfwD9th+MSh0WQKzUreLKKINb3afirxW7A0fz2YWuQ=="],
 37 | 
 38 |     "@img/sharp-libvips-linux-arm": ["@img/sharp-libvips-linux-arm@1.1.0", "", { "os": "linux", "cpu": "arm" }, "sha512-s8BAd0lwUIvYCJyRdFqvsj+BJIpDBSxs6ivrOPm/R7piTs5UIwY5OjXrP2bqXC9/moGsyRa37eYWYCOGVXxVrA=="],
 39 | 
 40 |     "@img/sharp-libvips-linux-arm64": ["@img/sharp-libvips-linux-arm64@1.1.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-IVfGJa7gjChDET1dK9SekxFFdflarnUB8PwW8aGwEoF3oAsSDuNUTYS+SKDOyOJxQyDC1aPFMuRYLoDInyV9Ew=="],
 41 | 
 42 |     "@img/sharp-libvips-linux-ppc64": ["@img/sharp-libvips-linux-ppc64@1.1.0", "", { "os": "linux", "cpu": "ppc64" }, "sha512-tiXxFZFbhnkWE2LA8oQj7KYR+bWBkiV2nilRldT7bqoEZ4HiDOcePr9wVDAZPi/Id5fT1oY9iGnDq20cwUz8lQ=="],
 43 | 
 44 |     "@img/sharp-libvips-linux-s390x": ["@img/sharp-libvips-linux-s390x@1.1.0", "", { "os": "linux", "cpu": "s390x" }, "sha512-xukSwvhguw7COyzvmjydRb3x/09+21HykyapcZchiCUkTThEQEOMtBj9UhkaBRLuBrgLFzQ2wbxdeCCJW/jgJA=="],
 45 | 
 46 |     "@img/sharp-libvips-linux-x64": ["@img/sharp-libvips-linux-x64@1.1.0", "", { "os": "linux", "cpu": "x64" }, "sha512-yRj2+reB8iMg9W5sULM3S74jVS7zqSzHG3Ol/twnAAkAhnGQnpjj6e4ayUz7V+FpKypwgs82xbRdYtchTTUB+Q=="],
 47 | 
 48 |     "@img/sharp-libvips-linuxmusl-arm64": ["@img/sharp-libvips-linuxmusl-arm64@1.1.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-jYZdG+whg0MDK+q2COKbYidaqW/WTz0cc1E+tMAusiDygrM4ypmSCjOJPmFTvHHJ8j/6cAGyeDWZOsK06tP33w=="],
 49 | 
 50 |     "@img/sharp-libvips-linuxmusl-x64": ["@img/sharp-libvips-linuxmusl-x64@1.1.0", "", { "os": "linux", "cpu": "x64" }, "sha512-wK7SBdwrAiycjXdkPnGCPLjYb9lD4l6Ze2gSdAGVZrEL05AOUJESWU2lhlC+Ffn5/G+VKuSm6zzbQSzFX/P65A=="],
 51 | 
 52 |     "@img/sharp-linux-arm": ["@img/sharp-linux-arm@0.34.2", "", { "optionalDependencies": { "@img/sharp-libvips-linux-arm": "1.1.0" }, "os": "linux", "cpu": "arm" }, "sha512-0DZzkvuEOqQUP9mo2kjjKNok5AmnOr1jB2XYjkaoNRwpAYMDzRmAqUIa1nRi58S2WswqSfPOWLNOr0FDT3H5RQ=="],
 53 | 
 54 |     "@img/sharp-linux-arm64": ["@img/sharp-linux-arm64@0.34.2", "", { "optionalDependencies": { "@img/sharp-libvips-linux-arm64": "1.1.0" }, "os": "linux", "cpu": "arm64" }, "sha512-D8n8wgWmPDakc83LORcfJepdOSN6MvWNzzz2ux0MnIbOqdieRZwVYY32zxVx+IFUT8er5KPcyU3XXsn+GzG/0Q=="],
 55 | 
 56 |     "@img/sharp-linux-s390x": ["@img/sharp-linux-s390x@0.34.2", "", { "optionalDependencies": { "@img/sharp-libvips-linux-s390x": "1.1.0" }, "os": "linux", "cpu": "s390x" }, "sha512-EGZ1xwhBI7dNISwxjChqBGELCWMGDvmxZXKjQRuqMrakhO8QoMgqCrdjnAqJq/CScxfRn+Bb7suXBElKQpPDiw=="],
 57 | 
 58 |     "@img/sharp-linux-x64": ["@img/sharp-linux-x64@0.34.2", "", { "optionalDependencies": { "@img/sharp-libvips-linux-x64": "1.1.0" }, "os": "linux", "cpu": "x64" }, "sha512-sD7J+h5nFLMMmOXYH4DD9UtSNBD05tWSSdWAcEyzqW8Cn5UxXvsHAxmxSesYUsTOBmUnjtxghKDl15EvfqLFbQ=="],
 59 | 
 60 |     "@img/sharp-linuxmusl-arm64": ["@img/sharp-linuxmusl-arm64@0.34.2", "", { "optionalDependencies": { "@img/sharp-libvips-linuxmusl-arm64": "1.1.0" }, "os": "linux", "cpu": "arm64" }, "sha512-NEE2vQ6wcxYav1/A22OOxoSOGiKnNmDzCYFOZ949xFmrWZOVII1Bp3NqVVpvj+3UeHMFyN5eP/V5hzViQ5CZNA=="],
 61 | 
 62 |     "@img/sharp-linuxmusl-x64": ["@img/sharp-linuxmusl-x64@0.34.2", "", { "optionalDependencies": { "@img/sharp-libvips-linuxmusl-x64": "1.1.0" }, "os": "linux", "cpu": "x64" }, "sha512-DOYMrDm5E6/8bm/yQLCWyuDJwUnlevR8xtF8bs+gjZ7cyUNYXiSf/E8Kp0Ss5xasIaXSHzb888V1BE4i1hFhAA=="],
 63 | 
 64 |     "@img/sharp-wasm32": ["@img/sharp-wasm32@0.34.2", "", { "dependencies": { "@emnapi/runtime": "^1.4.3" }, "cpu": "none" }, "sha512-/VI4mdlJ9zkaq53MbIG6rZY+QRN3MLbR6usYlgITEzi4Rpx5S6LFKsycOQjkOGmqTNmkIdLjEvooFKwww6OpdQ=="],
 65 | 
 66 |     "@img/sharp-win32-arm64": ["@img/sharp-win32-arm64@0.34.2", "", { "os": "win32", "cpu": "arm64" }, "sha512-cfP/r9FdS63VA5k0xiqaNaEoGxBg9k7uE+RQGzuK9fHt7jib4zAVVseR9LsE4gJcNWgT6APKMNnCcnyOtmSEUQ=="],
 67 | 
 68 |     "@img/sharp-win32-ia32": ["@img/sharp-win32-ia32@0.34.2", "", { "os": "win32", "cpu": "ia32" }, "sha512-QLjGGvAbj0X/FXl8n1WbtQ6iVBpWU7JO94u/P2M4a8CFYsvQi4GW2mRy/JqkRx0qpBzaOdKJKw8uc930EX2AHw=="],
 69 | 
 70 |     "@img/sharp-win32-x64": ["@img/sharp-win32-x64@0.34.2", "", { "os": "win32", "cpu": "x64" }, "sha512-aUdT6zEYtDKCaxkofmmJDJYGCf0+pJg3eU9/oBuqvEeoB9dKI6ZLc/1iLJCTuJQDO4ptntAlkUmHgGjyuobZbw=="],
 71 | 
 72 |     "@isaacs/fs-minipass": ["@isaacs/fs-minipass@4.0.1", "", { "dependencies": { "minipass": "^7.0.4" } }, "sha512-wgm9Ehl2jpeqP3zw/7mo3kRHFp5MEDhqAdwy1fTGkHAwnkGOVsgpvQhL8B5n1qlb01jV3n/bI0ZfZp5lWA1k4w=="],
 73 | 
 74 |     "@jridgewell/gen-mapping": ["@jridgewell/gen-mapping@0.3.8", "", { "dependencies": { "@jridgewell/set-array": "^1.2.1", "@jridgewell/sourcemap-codec": "^1.4.10", "@jridgewell/trace-mapping": "^0.3.24" } }, "sha512-imAbBGkb+ebQyxKgzv5Hu2nmROxoDOXHh80evxdoXNOrvAnVx7zimzc1Oo5h9RlfV4vPXaE2iM5pOFbvOCClWA=="],
 75 | 
 76 |     "@jridgewell/resolve-uri": ["@jridgewell/resolve-uri@3.1.2", "", {}, "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw=="],
 77 | 
 78 |     "@jridgewell/set-array": ["@jridgewell/set-array@1.2.1", "", {}, "sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A=="],
 79 | 
 80 |     "@jridgewell/sourcemap-codec": ["@jridgewell/sourcemap-codec@1.5.0", "", {}, "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ=="],
 81 | 
 82 |     "@jridgewell/trace-mapping": ["@jridgewell/trace-mapping@0.3.25", "", { "dependencies": { "@jridgewell/resolve-uri": "^3.1.0", "@jridgewell/sourcemap-codec": "^1.4.14" } }, "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ=="],
 83 | 
 84 |     "@next/env": ["@next/env@15.3.3", "", {}, "sha512-OdiMrzCl2Xi0VTjiQQUK0Xh7bJHnOuET2s+3V+Y40WJBAXrJeGA3f+I8MZJ/YQ3mVGi5XGR1L66oFlgqXhQ4Vw=="],
 85 | 
 86 |     "@next/swc-darwin-arm64": ["@next/swc-darwin-arm64@15.3.3", "", { "os": "darwin", "cpu": "arm64" }, "sha512-WRJERLuH+O3oYB4yZNVahSVFmtxRNjNF1I1c34tYMoJb0Pve+7/RaLAJJizyYiFhjYNGHRAE1Ri2Fd23zgDqhg=="],
 87 | 
 88 |     "@next/swc-darwin-x64": ["@next/swc-darwin-x64@15.3.3", "", { "os": "darwin", "cpu": "x64" }, "sha512-XHdzH/yBc55lu78k/XwtuFR/ZXUTcflpRXcsu0nKmF45U96jt1tsOZhVrn5YH+paw66zOANpOnFQ9i6/j+UYvw=="],
 89 | 
 90 |     "@next/swc-linux-arm64-gnu": ["@next/swc-linux-arm64-gnu@15.3.3", "", { "os": "linux", "cpu": "arm64" }, "sha512-VZ3sYL2LXB8znNGcjhocikEkag/8xiLgnvQts41tq6i+wql63SMS1Q6N8RVXHw5pEUjiof+II3HkDd7GFcgkzw=="],
 91 | 
 92 |     "@next/swc-linux-arm64-musl": ["@next/swc-linux-arm64-musl@15.3.3", "", { "os": "linux", "cpu": "arm64" }, "sha512-h6Y1fLU4RWAp1HPNJWDYBQ+e3G7sLckyBXhmH9ajn8l/RSMnhbuPBV/fXmy3muMcVwoJdHL+UtzRzs0nXOf9SA=="],
 93 | 
 94 |     "@next/swc-linux-x64-gnu": ["@next/swc-linux-x64-gnu@15.3.3", "", { "os": "linux", "cpu": "x64" }, "sha512-jJ8HRiF3N8Zw6hGlytCj5BiHyG/K+fnTKVDEKvUCyiQ/0r5tgwO7OgaRiOjjRoIx2vwLR+Rz8hQoPrnmFbJdfw=="],
 95 | 
 96 |     "@next/swc-linux-x64-musl": ["@next/swc-linux-x64-musl@15.3.3", "", { "os": "linux", "cpu": "x64" }, "sha512-HrUcTr4N+RgiiGn3jjeT6Oo208UT/7BuTr7K0mdKRBtTbT4v9zJqCDKO97DUqqoBK1qyzP1RwvrWTvU6EPh/Cw=="],
 97 | 
 98 |     "@next/swc-win32-arm64-msvc": ["@next/swc-win32-arm64-msvc@15.3.3", "", { "os": "win32", "cpu": "arm64" }, "sha512-SxorONgi6K7ZUysMtRF3mIeHC5aA3IQLmKFQzU0OuhuUYwpOBc1ypaLJLP5Bf3M9k53KUUUj4vTPwzGvl/NwlQ=="],
 99 | 
100 |     "@next/swc-win32-x64-msvc": ["@next/swc-win32-x64-msvc@15.3.3", "", { "os": "win32", "cpu": "x64" }, "sha512-4QZG6F8enl9/S2+yIiOiju0iCTFd93d8VC1q9LZS4p/Xuk81W2QDjCFeoogmrWWkAD59z8ZxepBQap2dKS5ruw=="],
101 | 
102 |     "@swc/counter": ["@swc/counter@0.1.3", "", {}, "sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ=="],
103 | 
104 |     "@swc/helpers": ["@swc/helpers@0.5.15", "", { "dependencies": { "tslib": "^2.8.0" } }, "sha512-JQ5TuMi45Owi4/BIMAJBoSQoOJu12oOk/gADqlcUL9JEdHB8vyjUSsxqeNXnmXHjYKMi2WcYtezGEEhqUI/E2g=="],
105 | 
106 |     "@tailwindcss/node": ["@tailwindcss/node@4.1.8", "", { "dependencies": { "@ampproject/remapping": "^2.3.0", "enhanced-resolve": "^5.18.1", "jiti": "^2.4.2", "lightningcss": "1.30.1", "magic-string": "^0.30.17", "source-map-js": "^1.2.1", "tailwindcss": "4.1.8" } }, "sha512-OWwBsbC9BFAJelmnNcrKuf+bka2ZxCE2A4Ft53Tkg4uoiE67r/PMEYwCsourC26E+kmxfwE0hVzMdxqeW+xu7Q=="],
107 | 
108 |     "@tailwindcss/oxide": ["@tailwindcss/oxide@4.1.8", "", { "dependencies": { "detect-libc": "^2.0.4", "tar": "^7.4.3" }, "optionalDependencies": { "@tailwindcss/oxide-android-arm64": "4.1.8", "@tailwindcss/oxide-darwin-arm64": "4.1.8", "@tailwindcss/oxide-darwin-x64": "4.1.8", "@tailwindcss/oxide-freebsd-x64": "4.1.8", "@tailwindcss/oxide-linux-arm-gnueabihf": "4.1.8", "@tailwindcss/oxide-linux-arm64-gnu": "4.1.8", "@tailwindcss/oxide-linux-arm64-musl": "4.1.8", "@tailwindcss/oxide-linux-x64-gnu": "4.1.8", "@tailwindcss/oxide-linux-x64-musl": "4.1.8", "@tailwindcss/oxide-wasm32-wasi": "4.1.8", "@tailwindcss/oxide-win32-arm64-msvc": "4.1.8", "@tailwindcss/oxide-win32-x64-msvc": "4.1.8" } }, "sha512-d7qvv9PsM5N3VNKhwVUhpK6r4h9wtLkJ6lz9ZY9aeZgrUWk1Z8VPyqyDT9MZlem7GTGseRQHkeB1j3tC7W1P+A=="],
109 | 
110 |     "@tailwindcss/oxide-android-arm64": ["@tailwindcss/oxide-android-arm64@4.1.8", "", { "os": "android", "cpu": "arm64" }, "sha512-Fbz7qni62uKYceWYvUjRqhGfZKwhZDQhlrJKGtnZfuNtHFqa8wmr+Wn74CTWERiW2hn3mN5gTpOoxWKk0jRxjg=="],
111 | 
112 |     "@tailwindcss/oxide-darwin-arm64": ["@tailwindcss/oxide-darwin-arm64@4.1.8", "", { "os": "darwin", "cpu": "arm64" }, "sha512-RdRvedGsT0vwVVDztvyXhKpsU2ark/BjgG0huo4+2BluxdXo8NDgzl77qh0T1nUxmM11eXwR8jA39ibvSTbi7A=="],
113 | 
114 |     "@tailwindcss/oxide-darwin-x64": ["@tailwindcss/oxide-darwin-x64@4.1.8", "", { "os": "darwin", "cpu": "x64" }, "sha512-t6PgxjEMLp5Ovf7uMb2OFmb3kqzVTPPakWpBIFzppk4JE4ix0yEtbtSjPbU8+PZETpaYMtXvss2Sdkx8Vs4XRw=="],
115 | 
116 |     "@tailwindcss/oxide-freebsd-x64": ["@tailwindcss/oxide-freebsd-x64@4.1.8", "", { "os": "freebsd", "cpu": "x64" }, "sha512-g8C8eGEyhHTqwPStSwZNSrOlyx0bhK/V/+zX0Y+n7DoRUzyS8eMbVshVOLJTDDC+Qn9IJnilYbIKzpB9n4aBsg=="],
117 | 
118 |     "@tailwindcss/oxide-linux-arm-gnueabihf": ["@tailwindcss/oxide-linux-arm-gnueabihf@4.1.8", "", { "os": "linux", "cpu": "arm" }, "sha512-Jmzr3FA4S2tHhaC6yCjac3rGf7hG9R6Gf2z9i9JFcuyy0u79HfQsh/thifbYTF2ic82KJovKKkIB6Z9TdNhCXQ=="],
119 | 
120 |     "@tailwindcss/oxide-linux-arm64-gnu": ["@tailwindcss/oxide-linux-arm64-gnu@4.1.8", "", { "os": "linux", "cpu": "arm64" }, "sha512-qq7jXtO1+UEtCmCeBBIRDrPFIVI4ilEQ97qgBGdwXAARrUqSn/L9fUrkb1XP/mvVtoVeR2bt/0L77xx53bPZ/Q=="],
121 | 
122 |     "@tailwindcss/oxide-linux-arm64-musl": ["@tailwindcss/oxide-linux-arm64-musl@4.1.8", "", { "os": "linux", "cpu": "arm64" }, "sha512-O6b8QesPbJCRshsNApsOIpzKt3ztG35gfX9tEf4arD7mwNinsoCKxkj8TgEE0YRjmjtO3r9FlJnT/ENd9EVefQ=="],
123 | 
124 |     "@tailwindcss/oxide-linux-x64-gnu": ["@tailwindcss/oxide-linux-x64-gnu@4.1.8", "", { "os": "linux", "cpu": "x64" }, "sha512-32iEXX/pXwikshNOGnERAFwFSfiltmijMIAbUhnNyjFr3tmWmMJWQKU2vNcFX0DACSXJ3ZWcSkzNbaKTdngH6g=="],
125 | 
126 |     "@tailwindcss/oxide-linux-x64-musl": ["@tailwindcss/oxide-linux-x64-musl@4.1.8", "", { "os": "linux", "cpu": "x64" }, "sha512-s+VSSD+TfZeMEsCaFaHTaY5YNj3Dri8rST09gMvYQKwPphacRG7wbuQ5ZJMIJXN/puxPcg/nU+ucvWguPpvBDg=="],
127 | 
128 |     "@tailwindcss/oxide-wasm32-wasi": ["@tailwindcss/oxide-wasm32-wasi@4.1.8", "", { "dependencies": { "@emnapi/core": "^1.4.3", "@emnapi/runtime": "^1.4.3", "@emnapi/wasi-threads": "^1.0.2", "@napi-rs/wasm-runtime": "^0.2.10", "@tybys/wasm-util": "^0.9.0", "tslib": "^2.8.0" }, "cpu": "none" }, "sha512-CXBPVFkpDjM67sS1psWohZ6g/2/cd+cq56vPxK4JeawelxwK4YECgl9Y9TjkE2qfF+9/s1tHHJqrC4SS6cVvSg=="],
129 | 
130 |     "@tailwindcss/oxide-win32-arm64-msvc": ["@tailwindcss/oxide-win32-arm64-msvc@4.1.8", "", { "os": "win32", "cpu": "arm64" }, "sha512-7GmYk1n28teDHUjPlIx4Z6Z4hHEgvP5ZW2QS9ygnDAdI/myh3HTHjDqtSqgu1BpRoI4OiLx+fThAyA1JePoENA=="],
131 | 
132 |     "@tailwindcss/oxide-win32-x64-msvc": ["@tailwindcss/oxide-win32-x64-msvc@4.1.8", "", { "os": "win32", "cpu": "x64" }, "sha512-fou+U20j+Jl0EHwK92spoWISON2OBnCazIc038Xj2TdweYV33ZRkS9nwqiUi2d/Wba5xg5UoHfvynnb/UB49cQ=="],
133 | 
134 |     "@tailwindcss/postcss": ["@tailwindcss/postcss@4.1.8", "", { "dependencies": { "@alloc/quick-lru": "^5.2.0", "@tailwindcss/node": "4.1.8", "@tailwindcss/oxide": "4.1.8", "postcss": "^8.4.41", "tailwindcss": "4.1.8" } }, "sha512-vB/vlf7rIky+w94aWMw34bWW1ka6g6C3xIOdICKX2GC0VcLtL6fhlLiafF0DVIwa9V6EHz8kbWMkS2s2QvvNlw=="],
135 | 
136 |     "@types/node": ["@types/node@20.19.0", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-hfrc+1tud1xcdVTABC2JiomZJEklMcXYNTVtZLAeqTVWD+qL5jkHKT+1lOtqDdGxt+mB53DTtiz673vfjU8D1Q=="],
137 | 
138 |     "@types/react": ["@types/react@19.1.7", "", { "dependencies": { "csstype": "^3.0.2" } }, "sha512-BnsPLV43ddr05N71gaGzyZ5hzkCmGwhMvYc8zmvI8Ci1bRkkDSzDDVfAXfN2tk748OwI7ediiPX6PfT9p0QGVg=="],
139 | 
140 |     "@types/react-dom": ["@types/react-dom@19.1.6", "", { "peerDependencies": { "@types/react": "^19.0.0" } }, "sha512-4hOiT/dwO8Ko0gV1m/TJZYk3y0KBnY9vzDh7W+DH17b2HFSOGgdj33dhihPeuy3l0q23+4e+hoXHV6hCC4dCXw=="],
141 | 
142 |     "busboy": ["busboy@1.6.0", "", { "dependencies": { "streamsearch": "^1.1.0" } }, "sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA=="],
143 | 
144 |     "caniuse-lite": ["caniuse-lite@1.0.30001721", "", {}, "sha512-cOuvmUVtKrtEaoKiO0rSc29jcjwMwX5tOHDy4MgVFEWiUXj4uBMJkwI8MDySkgXidpMiHUcviogAvFi4pA2hDQ=="],
145 | 
146 |     "chownr": ["chownr@3.0.0", "", {}, "sha512-+IxzY9BZOQd/XuYPRmrvEVjF/nqj5kgT4kEq7VofrDoM1MxoRjEWkrCC3EtLi59TVawxTAn+orJwFQcrqEN1+g=="],
147 | 
148 |     "client-only": ["client-only@0.0.1", "", {}, "sha512-IV3Ou0jSMzZrd3pZ48nLkT9DA7Ag1pnPzaiQhpW7c3RbcqqzvzzVu+L8gfqMp/8IM2MQtSiqaCxrrcfu8I8rMA=="],
149 | 
150 |     "color": ["color@4.2.3", "", { "dependencies": { "color-convert": "^2.0.1", "color-string": "^1.9.0" } }, "sha512-1rXeuUUiGGrykh+CeBdu5Ie7OJwinCgQY0bc7GCRxy5xVHy+moaqkpL/jqQq0MtQOeYcrqEz4abc5f0KtU7W4A=="],
151 | 
152 |     "color-convert": ["color-convert@2.0.1", "", { "dependencies": { "color-name": "~1.1.4" } }, "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ=="],
153 | 
154 |     "color-name": ["color-name@1.1.4", "", {}, "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA=="],
155 | 
156 |     "color-string": ["color-string@1.9.1", "", { "dependencies": { "color-name": "^1.0.0", "simple-swizzle": "^0.2.2" } }, "sha512-shrVawQFojnZv6xM40anx4CkoDP+fZsw/ZerEMsW/pyzsRbElpsL/DBVW7q3ExxwusdNXI3lXpuhEZkzs8p5Eg=="],
157 | 
158 |     "csstype": ["csstype@3.1.3", "", {}, "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw=="],
159 | 
160 |     "detect-libc": ["detect-libc@2.0.4", "", {}, "sha512-3UDv+G9CsCKO1WKMGw9fwq/SWJYbI0c5Y7LU1AXYoDdbhE2AHQ6N6Nb34sG8Fj7T5APy8qXDCKuuIHd1BR0tVA=="],
161 | 
162 |     "enhanced-resolve": ["enhanced-resolve@5.18.1", "", { "dependencies": { "graceful-fs": "^4.2.4", "tapable": "^2.2.0" } }, "sha512-ZSW3ma5GkcQBIpwZTSRAI8N71Uuwgs93IezB7mf7R60tC8ZbJideoDNKjHn2O9KIlx6rkGTTEk1xUCK2E1Y2Yg=="],
163 | 
164 |     "graceful-fs": ["graceful-fs@4.2.11", "", {}, "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ=="],
165 | 
166 |     "is-arrayish": ["is-arrayish@0.3.2", "", {}, "sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ=="],
167 | 
168 |     "jiti": ["jiti@2.4.2", "", { "bin": { "jiti": "lib/jiti-cli.mjs" } }, "sha512-rg9zJN+G4n2nfJl5MW3BMygZX56zKPNVEYYqq7adpmMh4Jn2QNEwhvQlFy6jPVdcod7txZtKHWnyZiA3a0zP7A=="],
169 | 
170 |     "lightningcss": ["lightningcss@1.30.1", "", { "dependencies": { "detect-libc": "^2.0.3" }, "optionalDependencies": { "lightningcss-darwin-arm64": "1.30.1", "lightningcss-darwin-x64": "1.30.1", "lightningcss-freebsd-x64": "1.30.1", "lightningcss-linux-arm-gnueabihf": "1.30.1", "lightningcss-linux-arm64-gnu": "1.30.1", "lightningcss-linux-arm64-musl": "1.30.1", "lightningcss-linux-x64-gnu": "1.30.1", "lightningcss-linux-x64-musl": "1.30.1", "lightningcss-win32-arm64-msvc": "1.30.1", "lightningcss-win32-x64-msvc": "1.30.1" } }, "sha512-xi6IyHML+c9+Q3W0S4fCQJOym42pyurFiJUHEcEyHS0CeKzia4yZDEsLlqOFykxOdHpNy0NmvVO31vcSqAxJCg=="],
171 | 
172 |     "lightningcss-darwin-arm64": ["lightningcss-darwin-arm64@1.30.1", "", { "os": "darwin", "cpu": "arm64" }, "sha512-c8JK7hyE65X1MHMN+Viq9n11RRC7hgin3HhYKhrMyaXflk5GVplZ60IxyoVtzILeKr+xAJwg6zK6sjTBJ0FKYQ=="],
173 | 
174 |     "lightningcss-darwin-x64": ["lightningcss-darwin-x64@1.30.1", "", { "os": "darwin", "cpu": "x64" }, "sha512-k1EvjakfumAQoTfcXUcHQZhSpLlkAuEkdMBsI/ivWw9hL+7FtilQc0Cy3hrx0AAQrVtQAbMI7YjCgYgvn37PzA=="],
175 | 
176 |     "lightningcss-freebsd-x64": ["lightningcss-freebsd-x64@1.30.1", "", { "os": "freebsd", "cpu": "x64" }, "sha512-kmW6UGCGg2PcyUE59K5r0kWfKPAVy4SltVeut+umLCFoJ53RdCUWxcRDzO1eTaxf/7Q2H7LTquFHPL5R+Gjyig=="],
177 | 
178 |     "lightningcss-linux-arm-gnueabihf": ["lightningcss-linux-arm-gnueabihf@1.30.1", "", { "os": "linux", "cpu": "arm" }, "sha512-MjxUShl1v8pit+6D/zSPq9S9dQ2NPFSQwGvxBCYaBYLPlCWuPh9/t1MRS8iUaR8i+a6w7aps+B4N0S1TYP/R+Q=="],
179 | 
180 |     "lightningcss-linux-arm64-gnu": ["lightningcss-linux-arm64-gnu@1.30.1", "", { "os": "linux", "cpu": "arm64" }, "sha512-gB72maP8rmrKsnKYy8XUuXi/4OctJiuQjcuqWNlJQ6jZiWqtPvqFziskH3hnajfvKB27ynbVCucKSm2rkQp4Bw=="],
181 | 
182 |     "lightningcss-linux-arm64-musl": ["lightningcss-linux-arm64-musl@1.30.1", "", { "os": "linux", "cpu": "arm64" }, "sha512-jmUQVx4331m6LIX+0wUhBbmMX7TCfjF5FoOH6SD1CttzuYlGNVpA7QnrmLxrsub43ClTINfGSYyHe2HWeLl5CQ=="],
183 | 
184 |     "lightningcss-linux-x64-gnu": ["lightningcss-linux-x64-gnu@1.30.1", "", { "os": "linux", "cpu": "x64" }, "sha512-piWx3z4wN8J8z3+O5kO74+yr6ze/dKmPnI7vLqfSqI8bccaTGY5xiSGVIJBDd5K5BHlvVLpUB3S2YCfelyJ1bw=="],
185 | 
186 |     "lightningcss-linux-x64-musl": ["lightningcss-linux-x64-musl@1.30.1", "", { "os": "linux", "cpu": "x64" }, "sha512-rRomAK7eIkL+tHY0YPxbc5Dra2gXlI63HL+v1Pdi1a3sC+tJTcFrHX+E86sulgAXeI7rSzDYhPSeHHjqFhqfeQ=="],
187 | 
188 |     "lightningcss-win32-arm64-msvc": ["lightningcss-win32-arm64-msvc@1.30.1", "", { "os": "win32", "cpu": "arm64" }, "sha512-mSL4rqPi4iXq5YVqzSsJgMVFENoa4nGTT/GjO2c0Yl9OuQfPsIfncvLrEW6RbbB24WtZ3xP/2CCmI3tNkNV4oA=="],
189 | 
190 |     "lightningcss-win32-x64-msvc": ["lightningcss-win32-x64-msvc@1.30.1", "", { "os": "win32", "cpu": "x64" }, "sha512-PVqXh48wh4T53F/1CCu8PIPCxLzWyCnn/9T5W1Jpmdy5h9Cwd+0YQS6/LwhHXSafuc61/xg9Lv5OrCby6a++jg=="],
191 | 
192 |     "liquid-glass-react": ["liquid-glass-react@1.0.0", "", { "peerDependencies": { "react": ">=19", "react-dom": ">=19" } }, "sha512-2IKXBoYJAkKZIINSIdRxHbita9BCPooSKlNzEZrs9uSw0retWovYeTl3fqKtXYvOQo0Myza4NL4DoRC28uJGtQ=="],
193 | 
194 |     "lucide-react": ["lucide-react@0.514.0", "", { "peerDependencies": { "react": "^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-HXD0OAMd+JM2xCjlwG1EGW9Nuab64dhjO3+MvdyD+pSUeOTBaVAPhQblKIYmmX4RyBYbdzW0VWnJpjJmxWGr6w=="],
195 | 
196 |     "magic-string": ["magic-string@0.30.17", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.5.0" } }, "sha512-sNPKHvyjVf7gyjwS4xGTaW/mCnF8wnjtifKBEhxfZ7E/S8tQ0rssrwGNn6q8JH/ohItJfSQp9mBtQYuTlH5QnA=="],
197 | 
198 |     "minipass": ["minipass@7.1.2", "", {}, "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw=="],
199 | 
200 |     "minizlib": ["minizlib@3.0.2", "", { "dependencies": { "minipass": "^7.1.2" } }, "sha512-oG62iEk+CYt5Xj2YqI5Xi9xWUeZhDI8jjQmC5oThVH5JGCTgIjr7ciJDzC7MBzYd//WvR1OTmP5Q38Q8ShQtVA=="],
201 | 
202 |     "mkdirp": ["mkdirp@3.0.1", "", { "bin": { "mkdirp": "dist/cjs/src/bin.js" } }, "sha512-+NsyUUAZDmo6YVHzL/stxSu3t9YS1iljliy3BSDrXJ/dkn1KYdmtZODGGjLcc9XLgVVpH4KshHB8XmZgMhaBXg=="],
203 | 
204 |     "nanoid": ["nanoid@3.3.11", "", { "bin": { "nanoid": "bin/nanoid.cjs" } }, "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w=="],
205 | 
206 |     "next": ["next@15.3.3", "", { "dependencies": { "@next/env": "15.3.3", "@swc/counter": "0.1.3", "@swc/helpers": "0.5.15", "busboy": "1.6.0", "caniuse-lite": "^1.0.30001579", "postcss": "8.4.31", "styled-jsx": "5.1.6" }, "optionalDependencies": { "@next/swc-darwin-arm64": "15.3.3", "@next/swc-darwin-x64": "15.3.3", "@next/swc-linux-arm64-gnu": "15.3.3", "@next/swc-linux-arm64-musl": "15.3.3", "@next/swc-linux-x64-gnu": "15.3.3", "@next/swc-linux-x64-musl": "15.3.3", "@next/swc-win32-arm64-msvc": "15.3.3", "@next/swc-win32-x64-msvc": "15.3.3", "sharp": "^0.34.1" }, "peerDependencies": { "@opentelemetry/api": "^1.1.0", "@playwright/test": "^1.41.2", "babel-plugin-react-compiler": "*", "react": "^18.2.0 || 19.0.0-rc-de68d2f4-20241204 || ^19.0.0", "react-dom": "^18.2.0 || 19.0.0-rc-de68d2f4-20241204 || ^19.0.0", "sass": "^1.3.0" }, "optionalPeers": ["@opentelemetry/api", "@playwright/test", "babel-plugin-react-compiler", "sass"], "bin": { "next": "dist/bin/next" } }, "sha512-JqNj29hHNmCLtNvd090SyRbXJiivQ+58XjCcrC50Crb5g5u2zi7Y2YivbsEfzk6AtVI80akdOQbaMZwWB1Hthw=="],
207 | 
208 |     "picocolors": ["picocolors@1.1.1", "", {}, "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA=="],
209 | 
210 |     "postcss": ["postcss@8.5.4", "", { "dependencies": { "nanoid": "^3.3.11", "picocolors": "^1.1.1", "source-map-js": "^1.2.1" } }, "sha512-QSa9EBe+uwlGTFmHsPKokv3B/oEMQZxfqW0QqNCyhpa6mB1afzulwn8hihglqAb2pOw+BJgNlmXQ8la2VeHB7w=="],
211 | 
212 |     "react": ["react@19.1.0", "", {}, "sha512-FS+XFBNvn3GTAWq26joslQgWNoFu08F4kl0J4CgdNKADkdSGXQyTCnKteIAJy96Br6YbpEU1LSzV5dYtjMkMDg=="],
213 | 
214 |     "react-dom": ["react-dom@19.1.0", "", { "dependencies": { "scheduler": "^0.26.0" }, "peerDependencies": { "react": "^19.1.0" } }, "sha512-Xs1hdnE+DyKgeHJeJznQmYMIBG3TKIHJJT95Q58nHLSrElKlGQqDTR2HQ9fx5CN/Gk6Vh/kupBTDLU11/nDk/g=="],
215 | 
216 |     "scheduler": ["scheduler@0.26.0", "", {}, "sha512-NlHwttCI/l5gCPR3D1nNXtWABUmBwvZpEQiD4IXSbIDq8BzLIK/7Ir5gTFSGZDUu37K5cMNp0hFtzO38sC7gWA=="],
217 | 
218 |     "semver": ["semver@7.7.2", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-RF0Fw+rO5AMf9MAyaRXI4AV0Ulj5lMHqVxxdSgiVbixSCXoEmmX/jk0CuJw4+3SqroYO9VoUh+HcuJivvtJemA=="],
219 | 
220 |     "sharp": ["sharp@0.34.2", "", { "dependencies": { "color": "^4.2.3", "detect-libc": "^2.0.4", "semver": "^7.7.2" }, "optionalDependencies": { "@img/sharp-darwin-arm64": "0.34.2", "@img/sharp-darwin-x64": "0.34.2", "@img/sharp-libvips-darwin-arm64": "1.1.0", "@img/sharp-libvips-darwin-x64": "1.1.0", "@img/sharp-libvips-linux-arm": "1.1.0", "@img/sharp-libvips-linux-arm64": "1.1.0", "@img/sharp-libvips-linux-ppc64": "1.1.0", "@img/sharp-libvips-linux-s390x": "1.1.0", "@img/sharp-libvips-linux-x64": "1.1.0", "@img/sharp-libvips-linuxmusl-arm64": "1.1.0", "@img/sharp-libvips-linuxmusl-x64": "1.1.0", "@img/sharp-linux-arm": "0.34.2", "@img/sharp-linux-arm64": "0.34.2", "@img/sharp-linux-s390x": "0.34.2", "@img/sharp-linux-x64": "0.34.2", "@img/sharp-linuxmusl-arm64": "0.34.2", "@img/sharp-linuxmusl-x64": "0.34.2", "@img/sharp-wasm32": "0.34.2", "@img/sharp-win32-arm64": "0.34.2", "@img/sharp-win32-ia32": "0.34.2", "@img/sharp-win32-x64": "0.34.2" } }, "sha512-lszvBmB9QURERtyKT2bNmsgxXK0ShJrL/fvqlonCo7e6xBF8nT8xU6pW+PMIbLsz0RxQk3rgH9kd8UmvOzlMJg=="],
221 | 
222 |     "simple-swizzle": ["simple-swizzle@0.2.2", "", { "dependencies": { "is-arrayish": "^0.3.1" } }, "sha512-JA//kQgZtbuY83m+xT+tXJkmJncGMTFT+C+g2h2R9uxkYIrE2yy9sgmcLhCnw57/WSD+Eh3J97FPEDFnbXnDUg=="],
223 | 
224 |     "source-map-js": ["source-map-js@1.2.1", "", {}, "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA=="],
225 | 
226 |     "streamsearch": ["streamsearch@1.1.0", "", {}, "sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg=="],
227 | 
228 |     "styled-jsx": ["styled-jsx@5.1.6", "", { "dependencies": { "client-only": "0.0.1" }, "peerDependencies": { "react": ">= 16.8.0 || 17.x.x || ^18.0.0-0 || ^19.0.0-0" } }, "sha512-qSVyDTeMotdvQYoHWLNGwRFJHC+i+ZvdBRYosOFgC+Wg1vx4frN2/RG/NA7SYqqvKNLf39P2LSRA2pu6n0XYZA=="],
229 | 
230 |     "tailwindcss": ["tailwindcss@4.1.8", "", {}, "sha512-kjeW8gjdxasbmFKpVGrGd5T4i40mV5J2Rasw48QARfYeQ8YS9x02ON9SFWax3Qf616rt4Cp3nVNIj6Hd1mP3og=="],
231 | 
232 |     "tapable": ["tapable@2.2.2", "", {}, "sha512-Re10+NauLTMCudc7T5WLFLAwDhQ0JWdrMK+9B2M8zR5hRExKmsRDCBA7/aV/pNJFltmBFO5BAMlQFi/vq3nKOg=="],
233 | 
234 |     "tar": ["tar@7.4.3", "", { "dependencies": { "@isaacs/fs-minipass": "^4.0.0", "chownr": "^3.0.0", "minipass": "^7.1.2", "minizlib": "^3.0.1", "mkdirp": "^3.0.1", "yallist": "^5.0.0" } }, "sha512-5S7Va8hKfV7W5U6g3aYxXmlPoZVAwUMy9AOKyF2fVuZa2UD3qZjg578OrLRt8PcNN1PleVaL/5/yYATNL0ICUw=="],
235 | 
236 |     "tslib": ["tslib@2.8.1", "", {}, "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="],
237 | 
238 |     "typescript": ["typescript@5.8.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-p1diW6TqL9L07nNxvRMM7hMMw4c5XOo/1ibL4aAIGmSAt9slTE1Xgw5KWuof2uTOvCg9BY7ZRi+GaF+7sfgPeQ=="],
239 | 
240 |     "undici-types": ["undici-types@6.21.0", "", {}, "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ=="],
241 | 
242 |     "yallist": ["yallist@5.0.0", "", {}, "sha512-YgvUTfwqyc7UXVMrB+SImsVYSmTS8X/tSrtdNZMImM+n7+QTriRXyXim0mBrTXNeqzVF0KWGgHPeiyViFFrNDw=="],
243 | 
244 |     "@tailwindcss/oxide-wasm32-wasi/@emnapi/core": ["@emnapi/core@1.4.3", "", { "dependencies": { "@emnapi/wasi-threads": "1.0.2", "tslib": "^2.4.0" }, "bundled": true }, "sha512-4m62DuCE07lw01soJwPiBGC0nAww0Q+RY70VZ+n49yDIO13yyinhbWCeNnaob0lakDtWQzSdtNWzJeOJt2ma+g=="],
245 | 
246 |     "@tailwindcss/oxide-wasm32-wasi/@emnapi/runtime": ["@emnapi/runtime@1.4.3", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-pBPWdu6MLKROBX05wSNKcNb++m5Er+KQ9QkB+WVM+pW2Kx9hoSrVTnu3BdkI5eBLZoKu/J6mW/B6i6bJB2ytXQ=="],
247 | 
248 |     "@tailwindcss/oxide-wasm32-wasi/@emnapi/wasi-threads": ["@emnapi/wasi-threads@1.0.2", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-5n3nTJblwRi8LlXkJ9eBzu+kZR8Yxcc7ubakyQTFzPMtIhFpUBRbsnc2Dv88IZDIbCDlBiWrknhB4Lsz7mg6BA=="],
249 | 
250 |     "@tailwindcss/oxide-wasm32-wasi/@napi-rs/wasm-runtime": ["@napi-rs/wasm-runtime@0.2.11", "", { "dependencies": { "@emnapi/core": "^1.4.3", "@emnapi/runtime": "^1.4.3", "@tybys/wasm-util": "^0.9.0" }, "bundled": true }, "sha512-9DPkXtvHydrcOsopiYpUgPHpmj0HWZKMUnL2dZqpvC42lsratuBG06V5ipyno0fUek5VlFsNQ+AcFATSrJXgMA=="],
251 | 
252 |     "@tailwindcss/oxide-wasm32-wasi/@tybys/wasm-util": ["@tybys/wasm-util@0.9.0", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-6+7nlbMVX/PVDCwaIQ8nTOPveOcFLSt8GcXdx8hD0bt39uWxYT88uXzqTd4fTvqta7oeUJqudepapKNt2DYJFw=="],
253 | 
254 |     "@tailwindcss/oxide-wasm32-wasi/tslib": ["tslib@2.8.1", "", { "bundled": true }, "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="],
255 | 
256 |     "next/postcss": ["postcss@8.4.31", "", { "dependencies": { "nanoid": "^3.3.6", "picocolors": "^1.0.0", "source-map-js": "^1.0.2" } }, "sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ=="],
257 |   }
258 | }
259 | 


--------------------------------------------------------------------------------
/liquid-glass-example/bun.lockb:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/rdev/liquid-glass-react/ac48eab18d1f7f444ae30002d240cae29c863a21/liquid-glass-example/bun.lockb


--------------------------------------------------------------------------------
/liquid-glass-example/next.config.ts:
--------------------------------------------------------------------------------
 1 | import type { NextConfig } from "next";
 2 | 
 3 | const nextConfig: NextConfig = {
 4 |   /* config options here */
 5 |   reactStrictMode: true,
 6 |   experimental: {
 7 |     externalDir: true,
 8 |   },
 9 | };
10 | 
11 | export default nextConfig;
12 | 


--------------------------------------------------------------------------------
/liquid-glass-example/package.json:
--------------------------------------------------------------------------------
 1 | {
 2 |   "name": "liquid-glass-example",
 3 |   "version": "0.1.0",
 4 |   "private": true,
 5 |   "scripts": {
 6 |     "dev": "next dev --turbopack",
 7 |     "build": "next build",
 8 |     "start": "next start",
 9 |     "lint": "next lint"
10 |   },
11 |   "dependencies": {
12 |     "liquid-glass-react": "^1.0.2",
13 |     "lucide-react": "^0.514.0",
14 |     "next": "15.3.3",
15 |     "react": "^19.0.0",
16 |     "react-dom": "^19.0.0"
17 |   },
18 |   "devDependencies": {
19 |     "typescript": "^5",
20 |     "@types/node": "^20",
21 |     "@types/react": "^19",
22 |     "@types/react-dom": "^19",
23 |     "@tailwindcss/postcss": "^4",
24 |     "tailwindcss": "^4"
25 |   }
26 | }
27 | 


--------------------------------------------------------------------------------
/liquid-glass-example/postcss.config.mjs:
--------------------------------------------------------------------------------
1 | const config = {
2 |   plugins: ["@tailwindcss/postcss"],
3 | };
4 | 
5 | export default config;
6 | 


--------------------------------------------------------------------------------
/liquid-glass-example/public/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/rdev/liquid-glass-react/ac48eab18d1f7f444ae30002d240cae29c863a21/liquid-glass-example/public/favicon.ico


--------------------------------------------------------------------------------
/liquid-glass-example/src/pages/_app.tsx:
--------------------------------------------------------------------------------
1 | import "@/styles/globals.css"
2 | import type { AppProps } from "next/app"
3 | 
4 | export default function App({ Component, pageProps }: AppProps) {
5 |   return <Component {...pageProps} />
6 | }
7 | 


--------------------------------------------------------------------------------
/liquid-glass-example/src/pages/_document.tsx:
--------------------------------------------------------------------------------
 1 | import { Html, Head, Main, NextScript } from "next/document"
 2 | 
 3 | export default function Document() {
 4 |   return (
 5 |     <Html lang="en">
 6 |       <Head />
 7 |       <body className="antialiased">
 8 |         <Main />
 9 |         <NextScript />
10 |       </body>
11 |     </Html>
12 |   )
13 | }
14 | 


--------------------------------------------------------------------------------
/liquid-glass-example/src/pages/api/hello.ts:
--------------------------------------------------------------------------------
 1 | // Next.js API route support: https://nextjs.org/docs/api-routes/introduction
 2 | import type { NextApiRequest, NextApiResponse } from "next"
 3 | 
 4 | type Data = {
 5 |   name: string
 6 | }
 7 | 
 8 | export default function handler(_req: NextApiRequest, res: NextApiResponse<Data>) {
 9 |   res.status(200).json({ name: "John Doe" })
10 | }
11 | 


--------------------------------------------------------------------------------
/liquid-glass-example/src/pages/index.tsx:
--------------------------------------------------------------------------------
  1 | import { Geist } from "next/font/google"
  2 | import { useState, useRef } from "react"
  3 | import LiquidGlass from "liquid-glass-react"
  4 | import { LogOutIcon, Github } from "lucide-react"
  5 | 
  6 | const geistSans = Geist({
  7 |   variable: "--font-geist-sans",
  8 |   subsets: ["latin"],
  9 | })
 10 | 
 11 | export default function Home() {
 12 |   // User Info Card Controls
 13 |   const [displacementScale, setDisplacementScale] = useState(100)
 14 |   const [blurAmount, setBlurAmount] = useState(0.5)
 15 |   const [saturation, setSaturation] = useState(140)
 16 |   const [aberrationIntensity, setAberrationIntensity] = useState(2)
 17 |   const [elasticity, setElasticity] = useState(0)
 18 |   const [cornerRadius, setCornerRadius] = useState(32)
 19 |   const [userInfoOverLight, setUserInfoOverLight] = useState(false)
 20 |   const [userInfoMode, setUserInfoMode] = useState<"standard" | "polar" | "prominent" | "shader">("standard")
 21 | 
 22 |   // Log Out Button Controls
 23 |   const [logoutDisplacementScale, setLogoutDisplacementScale] = useState(64)
 24 |   const [logoutBlurAmount, setLogoutBlurAmount] = useState(0.1)
 25 |   const [logoutSaturation, setLogoutSaturation] = useState(130)
 26 |   const [logoutAberrationIntensity, setLogoutAberrationIntensity] = useState(2)
 27 |   const [logoutElasticity, setLogoutElasticity] = useState(0.35)
 28 |   const [logoutCornerRadius, setLogoutCornerRadius] = useState(100)
 29 |   const [logoutOverLight, setLogoutOverLight] = useState(false)
 30 |   const [logoutMode, setLogoutMode] = useState<"standard" | "polar" | "prominent" | "shader">("standard")
 31 | 
 32 |   // Shared state
 33 |   const [activeTab, setActiveTab] = useState<"userInfo" | "logOut">("userInfo")
 34 |   const containerRef = useRef<HTMLDivElement>(null)
 35 | 
 36 |   const [scroll, setScroll] = useState(0)
 37 | 
 38 |   const handleScroll = (event: React.UIEvent<HTMLDivElement>) => {
 39 |     requestAnimationFrame(() => {
 40 |       setScroll((event?.target as any)?.scrollTop)
 41 |     })
 42 |   }
 43 | 
 44 |   const scrollingOverBrightSection = scroll > 230 && scroll < 500
 45 | 
 46 |   return (
 47 |     <div
 48 |       className={`${geistSans.className} grid grid-cols-1 grid-rows-2 md:grid-rows-1 md:grid-cols-3 shadow-2xl w-full max-w-5xl mx-auto md:my-10 h-screen md:max-h-[calc(100vh-5rem)] md:rounded-3xl overflow-hidden font-[family-name:var(--font-geist-sans)]`}
 49 |     >
 50 |       {/* Left Panel - Glass Effect Demo */}
 51 |       <div className="flex-1 relative overflow-auto min-h-screen md:col-span-2" ref={containerRef} onScroll={handleScroll}>
 52 |         <div className="w-full min-h-[200vh] absolute top-0 left-0 pb-96 mb-96">
 53 |           <img src="https://picsum.photos/2000/2000" className="w-full h-96 object-cover" />
 54 |           <div className="flex flex-col gap-2" id="bright-section">
 55 |             <h2 className="text-2xl font-semibold my-5 text-center">Some Heading</h2>
 56 |             <p className="px-10">
 57 |               Bacon ipsum dolor amet hamburger Bacon ipsum dolor amet hamburger <br />
 58 |               Bacon ipsum dolor amet hamburger Bacon ipsum dolor amet hamburger
 59 |               <br />
 60 |               Bacon ipsum dolor amet hamburger Bacon ipsum dolor amet hamburger
 61 |               <br />
 62 |               Bacon ipsum dolor amet hamburger Bacon ipsum dolor amet hamburger
 63 |               <br />
 64 |               Bacon ipsum dolor amet hamburger Bacon ipsum dolor amet hamburger
 65 |               <br />
 66 |               Bacon ipsum dolor amet hamburger Bacon ipsum dolor amet hamburger
 67 |             </p>
 68 |           </div>
 69 |           <img src="https://picsum.photos/1200/1200" className="w-full h-80 object-cover my-10" />
 70 |           <img src="https://picsum.photos/1400/1300" className="w-full h-72 object-cover my-10" />
 71 |           <img src="https://picsum.photos/1100/1200" className="w-full h-96 object-cover my-10 mb-96" />
 72 |         </div>
 73 | 
 74 |         {activeTab === "userInfo" && (
 75 |           <LiquidGlass
 76 |               displacementScale={displacementScale}
 77 |               blurAmount={blurAmount}
 78 |               saturation={saturation}
 79 |               aberrationIntensity={aberrationIntensity}
 80 |               elasticity={elasticity}
 81 |               cornerRadius={cornerRadius}
 82 |               mouseContainer={containerRef}
 83 |               overLight={scrollingOverBrightSection || userInfoOverLight}
 84 |               mode={userInfoMode}
 85 |               style={{
 86 |                 position: "fixed",
 87 |                 top: "25%",
 88 |                 left: "40%",
 89 |               }}
 90 |             >
 91 |               <div className="w-72 text-shadow-lg">
 92 |                 <h3 className="text-xl font-semibold mb-4">User Info</h3>
 93 |                 <div className="space-y-3">
 94 |                   <div className="flex items-center space-x-3">
 95 |                     <div className="w-12 h-12 bg-black/10 backdrop-blur rounded-full flex items-center justify-center text-white font-semibold">JD</div>
 96 |                     <div>
 97 |                       <p className="font-medium">John Doe</p>
 98 |                       <p className="text-sm text-white">Software Engineer</p>
 99 |                     </div>
100 |                   </div>
101 |                   <div className="pt-2 space-y-2">
102 |                     <div className="flex justify-between">
103 |                       <span className="text-sm text-white">Email:</span>
104 |                       <span className="text-sm">john.doe@example.com</span>
105 |                     </div>
106 |                     <div className="flex justify-between">
107 |                       <span className="text-sm text-white">Location:</span>
108 |                       <span className="text-sm">San Francisco, CA</span>
109 |                     </div>
110 |                     <div className="flex justify-between">
111 |                       <span className="text-sm text-white">Joined:</span>
112 |                       <span className="text-sm">March 2023</span>
113 |                     </div>
114 |                   </div>
115 |                 </div>
116 |               </div>
117 |             </LiquidGlass>
118 |         )}
119 | 
120 |         {activeTab === "logOut" && (
121 |           <LiquidGlass
122 |             displacementScale={logoutDisplacementScale}
123 |             blurAmount={logoutBlurAmount}
124 |             saturation={logoutSaturation}
125 |             aberrationIntensity={logoutAberrationIntensity}
126 |             elasticity={logoutElasticity}
127 |             cornerRadius={logoutCornerRadius}
128 |             mouseContainer={containerRef}
129 |             overLight={scrollingOverBrightSection || logoutOverLight}
130 |             mode={logoutMode}
131 |             padding="8px 16px"
132 |             onClick={() => {
133 |               console.log("Logged out")
134 |             }}
135 |             style={{
136 |               position: "fixed",
137 |               top: "20%",
138 |               left: "40%",
139 |             }}
140 |           >
141 |             <h3 className="text-lg font-medium flex items-center gap-2">
142 |               Log Out
143 |               <LogOutIcon className="w-5 h-5" />
144 |             </h3>
145 |           </LiquidGlass>
146 |         )}
147 |       </div>
148 | 
149 |       {/* Right Panel - Control Panel */}
150 |       <div className="row-start-2 rounded-t-3xl md:rounded-none md:col-start-3 bg-gray-900/80 h-full overflow-y-auto backdrop-blur-md border-l border-white/10 p-8 flex flex-col">
151 |         <div className="mb-8">
152 |           <div className="flex items-center justify-between mb-4">
153 |             <h2 className="text-2xl font-bold text-white">Glassy Boi but Web</h2>
154 |             <a href="https://github.com/rdev/liquid-glass-react" target="_blank" rel="noopener noreferrer" className="text-white/70 hover:text-white transition-colors p-2 hover:bg-white/10 rounded-lg" title="View on GitHub">
155 |               <Github className="w-6 h-6" />
156 |             </a>
157 |           </div>
158 |           <p className="text-white/60 text-sm">Liquid Glass container effect for React. With settings and effects and stuff.</p>
159 | 
160 |           <p className="font-semibold text-yellow-300 text-xs mt-2 leading-snug">⚠️ This doesn't fully work in Safari and Firefox. You will not see edge refraction on non-chromium browsers.</p>
161 |         </div>
162 | 
163 |         {/* Tab Switcher */}
164 |         <div className="flex mb-6 bg-white/5 rounded-lg p-1">
165 |           <button
166 |             onClick={() => setActiveTab("userInfo")}
167 |             className={`flex-1 px-4 py-2 text-sm font-medium rounded-md transition-all ${activeTab === "userInfo" ? "bg-blue-500 text-white shadow-lg" : "text-white/70 hover:text-white hover:bg-white/10"}`}
168 |           >
169 |             User Info Card
170 |           </button>
171 |           <button
172 |             onClick={() => setActiveTab("logOut")}
173 |             className={`flex-1 px-4 py-2 text-sm font-medium rounded-md transition-all ${activeTab === "logOut" ? "bg-blue-500 text-white shadow-lg" : "text-white/70 hover:text-white hover:bg-white/10"}`}
174 |           >
175 |             Log Out Button
176 |           </button>
177 |         </div>
178 | 
179 |         <div className="space-y-8 flex-1">
180 |           {activeTab === "userInfo" && (
181 |             <>
182 |               <div>
183 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Refraction Mode</span>
184 |                 <div className="space-y-2">
185 |                   <div className="flex items-center space-x-3">
186 |                     <input
187 |                       type="radio"
188 |                       id="userInfoModeStandard"
189 |                       name="userInfoMode"
190 |                       value="standard"
191 |                       checked={userInfoMode === "standard"}
192 |                       onChange={(e) => setUserInfoMode(e.target.value as "standard" | "polar" | "prominent" | "shader")}
193 |                       className="w-4 h-4 accent-blue-500"
194 |                     />
195 |                     <label htmlFor="userInfoModeStandard" className="text-sm text-white/90">
196 |                       Standard
197 |                     </label>
198 |                   </div>
199 |                   <div className="flex items-center space-x-3">
200 |                     <input
201 |                       type="radio"
202 |                       id="userInfoModePolar"
203 |                       name="userInfoMode"
204 |                       value="polar"
205 |                       checked={userInfoMode === "polar"}
206 |                       onChange={(e) => setUserInfoMode(e.target.value as "standard" | "polar" | "prominent" | "shader")}
207 |                       className="w-4 h-4 accent-blue-500"
208 |                     />
209 |                     <label htmlFor="userInfoModePolar" className="text-sm text-white/90">
210 |                       Polar
211 |                     </label>
212 |                   </div>
213 |                   <div className="flex items-center space-x-3">
214 |                     <input
215 |                       type="radio"
216 |                       id="userInfoModeProminent"
217 |                       name="userInfoMode"
218 |                       value="prominent"
219 |                       checked={userInfoMode === "prominent"}
220 |                       onChange={(e) => setUserInfoMode(e.target.value as "standard" | "polar" | "prominent" | "shader")}
221 |                       className="w-4 h-4 accent-blue-500"
222 |                     />
223 |                     <label htmlFor="userInfoModeProminent" className="text-sm text-white/90">
224 |                       Prominent
225 |                     </label>
226 |                   </div>
227 |                   <div className="flex items-center space-x-3">
228 |                     <input
229 |                       type="radio"
230 |                       id="userInfoModeShader"
231 |                       name="userInfoMode"
232 |                       value="shader"
233 |                       checked={userInfoMode === "shader"}
234 |                       onChange={(e) => setUserInfoMode(e.target.value as "standard" | "polar" | "prominent" | "shader")}
235 |                       className="w-4 h-4 accent-blue-500"
236 |                     />
237 |                     <label htmlFor="userInfoModeShader" className="text-sm text-white/90">
238 |                       Shader (Experimental)
239 |                     </label>
240 |                   </div>
241 |                 </div>
242 |                 <p className="text-xs text-white/50 mt-2">Controls the refraction calculation method</p>
243 |               </div>
244 | 
245 |               <div>
246 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Displacement Scale</span>
247 |                 <div className="mb-2">
248 |                   <span className="text-xl font-mono text-blue-300">{displacementScale}</span>
249 |                 </div>
250 |                 <input type="range" min="0" max="200" step="1" value={displacementScale} onChange={(e) => setDisplacementScale(Number(e.target.value))} className="w-full" />
251 |                 <p className="text-xs text-white/50 mt-2">Controls the intensity of edge distortion</p>
252 |               </div>
253 | 
254 |               <div>
255 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Blur Amount</span>
256 |                 <div className="mb-2">
257 |                   <span className="text-xl font-mono text-green-300">{blurAmount.toFixed(1)}</span>
258 |                 </div>
259 |                 <input type="range" min="0" max="1" step="0.01" value={blurAmount} onChange={(e) => setBlurAmount(Number(e.target.value))} className="w-full" />
260 |                 <p className="text-xs text-white/50 mt-2">Controls backdrop blur intensity</p>
261 |               </div>
262 | 
263 |               <div>
264 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Saturation</span>
265 |                 <div className="mb-2">
266 |                   <span className="text-xl font-mono text-purple-300">{saturation}%</span>
267 |                 </div>
268 |                 <input type="range" min="100" max="300" step="10" value={saturation} onChange={(e) => setSaturation(Number(e.target.value))} className="w-full" />
269 |                 <p className="text-xs text-white/50 mt-2">Controls color saturation of the backdrop</p>
270 |               </div>
271 | 
272 |               <div>
273 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Chromatic Aberration</span>
274 |                 <div className="mb-2">
275 |                   <span className="text-xl font-mono text-cyan-300">{aberrationIntensity}</span>
276 |                 </div>
277 |                 <input type="range" min="0" max="20" step="1" value={aberrationIntensity} onChange={(e) => setAberrationIntensity(Number(e.target.value))} className="w-full" />
278 |                 <p className="text-xs text-white/50 mt-2">Controls RGB channel separation intensity</p>
279 |               </div>
280 | 
281 |               <div>
282 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Elasticity</span>
283 |                 <div className="mb-2">
284 |                   <span className="text-xl font-mono text-orange-300">{elasticity.toFixed(2)}</span>
285 |                 </div>
286 |                 <input type="range" min="0" max="1" step="0.05" value={elasticity} onChange={(e) => setElasticity(Number(e.target.value))} className="w-full" />
287 |                 <p className="text-xs text-white/50 mt-2">Controls how much the glass reaches toward the cursor</p>
288 |               </div>
289 | 
290 |               <div>
291 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Corner Radius</span>
292 |                 <div className="mb-2">
293 |                   <span className="text-xl font-mono text-pink-300">{cornerRadius === 999 ? "Full" : `${cornerRadius}px`}</span>
294 |                 </div>
295 |                 <input type="range" min="0" max="100" step="1" value={cornerRadius} onChange={(e) => setCornerRadius(Number(e.target.value))} className="w-full" />
296 |                 <p className="text-xs text-white/50 mt-2">Controls the roundness of the glass corners</p>
297 |               </div>
298 | 
299 |               <div>
300 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Over Light</span>
301 |                 <div className="flex items-center space-x-3">
302 |                   <input type="checkbox" id="userInfoOverLight" checked={userInfoOverLight} onChange={(e) => setUserInfoOverLight(e.target.checked)} className="w-5 h-5 accent-blue-500" />
303 |                   <label htmlFor="userInfoOverLight" className="text-sm text-white/90">
304 |                     Tint liquid glass dark (use for bright backgrounds)
305 |                   </label>
306 |                 </div>
307 |                 <p className="text-xs text-white/50 mt-2">Makes the glass darker for better visibility on light backgrounds</p>
308 |               </div>
309 |             </>
310 |           )}
311 | 
312 |           {activeTab === "logOut" && (
313 |             <>
314 |               <div>
315 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Refraction Mode</span>
316 |                 <div className="space-y-2">
317 |                   <div className="flex items-center space-x-3">
318 |                     <input
319 |                       type="radio"
320 |                       id="logoutModeStandard"
321 |                       name="logoutMode"
322 |                       value="standard"
323 |                       checked={logoutMode === "standard"}
324 |                       onChange={(e) => setLogoutMode(e.target.value as "standard" | "polar" | "prominent" | "shader")}
325 |                       className="w-4 h-4 accent-blue-500"
326 |                     />
327 |                     <label htmlFor="logoutModeStandard" className="text-sm text-white/90">
328 |                       Standard
329 |                     </label>
330 |                   </div>
331 |                   <div className="flex items-center space-x-3">
332 |                     <input
333 |                       type="radio"
334 |                       id="logoutModePolar"
335 |                       name="logoutMode"
336 |                       value="polar"
337 |                       checked={logoutMode === "polar"}
338 |                       onChange={(e) => setLogoutMode(e.target.value as "standard" | "polar" | "prominent" | "shader")}
339 |                       className="w-4 h-4 accent-blue-500"
340 |                     />
341 |                     <label htmlFor="logoutModePolar" className="text-sm text-white/90">
342 |                       Polar
343 |                     </label>
344 |                   </div>
345 |                   <div className="flex items-center space-x-3">
346 |                     <input
347 |                       type="radio"
348 |                       id="logoutModeProminent"
349 |                       name="logoutMode"
350 |                       value="prominent"
351 |                       checked={logoutMode === "prominent"}
352 |                       onChange={(e) => setLogoutMode(e.target.value as "standard" | "polar" | "prominent" | "shader")}
353 |                       className="w-4 h-4 accent-blue-500"
354 |                     />
355 |                     <label htmlFor="logoutModeProminent" className="text-sm text-white/90">
356 |                       Prominent
357 |                     </label>
358 |                   </div>
359 |                   <div className="flex items-center space-x-3">
360 |                     <input
361 |                       type="radio"
362 |                       id="logoutModeShader"
363 |                       name="logoutMode"
364 |                       value="shader"
365 |                       checked={logoutMode === "shader"}
366 |                       onChange={(e) => setLogoutMode(e.target.value as "standard" | "polar" | "prominent" | "shader")}
367 |                       className="w-4 h-4 accent-blue-500"
368 |                     />
369 |                     <label htmlFor="logoutModeShader" className="text-sm text-white/90">
370 |                       Shader
371 |                     </label>
372 |                   </div>
373 |                 </div>
374 |                 <p className="text-xs text-white/50 mt-2">Controls the refraction calculation method</p>
375 |               </div>
376 | 
377 |               <div>
378 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Displacement Scale</span>
379 |                 <div className="mb-2">
380 |                   <span className="text-xl font-mono text-blue-300">{logoutDisplacementScale}</span>
381 |                 </div>
382 |                 <input type="range" min="0" max="200" step="1" value={logoutDisplacementScale} onChange={(e) => setLogoutDisplacementScale(Number(e.target.value))} className="w-full" />
383 |                 <p className="text-xs text-white/50 mt-2">Controls the intensity of edge distortion</p>
384 |               </div>
385 | 
386 |               <div>
387 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Blur Amount</span>
388 |                 <div className="mb-2">
389 |                   <span className="text-xl font-mono text-green-300">{logoutBlurAmount.toFixed(1)}</span>
390 |                 </div>
391 |                 <input type="range" min="0" max="1" step="0.01" value={logoutBlurAmount} onChange={(e) => setLogoutBlurAmount(Number(e.target.value))} className="w-full" />
392 |                 <p className="text-xs text-white/50 mt-2">Controls backdrop blur intensity</p>
393 |               </div>
394 | 
395 |               <div>
396 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Saturation</span>
397 |                 <div className="mb-2">
398 |                   <span className="text-xl font-mono text-purple-300">{logoutSaturation}%</span>
399 |                 </div>
400 |                 <input type="range" min="100" max="300" step="10" value={logoutSaturation} onChange={(e) => setLogoutSaturation(Number(e.target.value))} className="w-full" />
401 |                 <p className="text-xs text-white/50 mt-2">Controls color saturation of the backdrop</p>
402 |               </div>
403 | 
404 |               <div>
405 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Chromatic Aberration</span>
406 |                 <div className="mb-2">
407 |                   <span className="text-xl font-mono text-cyan-300">{logoutAberrationIntensity}</span>
408 |                 </div>
409 |                 <input type="range" min="0" max="20" step="1" value={logoutAberrationIntensity} onChange={(e) => setLogoutAberrationIntensity(Number(e.target.value))} className="w-full" />
410 |                 <p className="text-xs text-white/50 mt-2">Controls RGB channel separation intensity</p>
411 |               </div>
412 | 
413 |               <div>
414 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Elasticity</span>
415 |                 <div className="mb-2">
416 |                   <span className="text-xl font-mono text-orange-300">{logoutElasticity.toFixed(2)}</span>
417 |                 </div>
418 |                 <input type="range" min="0" max="1" step="0.05" value={logoutElasticity} onChange={(e) => setLogoutElasticity(Number(e.target.value))} className="w-full" />
419 |                 <p className="text-xs text-white/50 mt-2">Controls how much the glass reaches toward the cursor</p>
420 |               </div>
421 | 
422 |               <div>
423 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Corner Radius</span>
424 |                 <div className="mb-2">
425 |                   <span className="text-xl font-mono text-pink-300">{logoutCornerRadius === 999 ? "Full" : `${logoutCornerRadius}px`}</span>
426 |                 </div>
427 |                 <input type="range" min="0" max="100" step="1" value={logoutCornerRadius} onChange={(e) => setLogoutCornerRadius(Number(e.target.value))} className="w-full" />
428 |                 <p className="text-xs text-white/50 mt-2">Controls the roundness of the glass corners</p>
429 |               </div>
430 | 
431 |               <div>
432 |                 <span className="block text-sm font-semibold text-white/90 mb-3">Over Light</span>
433 |                 <div className="flex items-center space-x-3">
434 |                   <input type="checkbox" id="logoutOverLight" checked={logoutOverLight} onChange={(e) => setLogoutOverLight(e.target.checked)} className="w-5 h-5 accent-blue-500" />
435 |                   <label htmlFor="logoutOverLight" className="text-sm text-white/90">
436 |                     Tint liquid glass dark (use for bright backgrounds)
437 |                   </label>
438 |                 </div>
439 |                 <p className="text-xs text-white/50 mt-2">Makes the glass darker for better visibility on light backgrounds</p>
440 |               </div>
441 |             </>
442 |           )}
443 |         </div>
444 |       </div>
445 |     </div>
446 |   )
447 | }
448 | 


--------------------------------------------------------------------------------
/liquid-glass-example/src/styles/globals.css:
--------------------------------------------------------------------------------
  1 | @import "tailwindcss";
  2 | 
  3 | :root {
  4 |   --background: #ffffff;
  5 |   --foreground: #171717;
  6 | }
  7 | 
  8 | @theme inline {
  9 |   --color-background: var(--background);
 10 |   --color-foreground: var(--foreground);
 11 |   --font-sans: var(--font-geist-sans);
 12 |   --font-mono: var(--font-geist-mono);
 13 | }
 14 | 
 15 | @media (prefers-color-scheme: dark) {
 16 |   :root {
 17 |     --background: #0a0a0a;
 18 |     --foreground: #ededed;
 19 |   }
 20 | }
 21 | 
 22 | body {
 23 |   background: var(--background);
 24 |   color: var(--foreground);
 25 |   font-family: Arial, Helvetica, sans-serif;
 26 | }
 27 | 
 28 | /* Custom range slider styling */
 29 | input[type="range"] {
 30 |   -webkit-appearance: none;
 31 |   appearance: none;
 32 |   background: #00000033;
 33 |   border-radius: 999px;
 34 |   cursor: pointer;
 35 |   height: 24px;
 36 | }
 37 | 
 38 | input[type="range"]::-webkit-slider-track {
 39 |   background: linear-gradient(90deg, rgba(59, 130, 246, 0.3), rgba(168, 85, 247, 0.3));
 40 |   height: 6px;
 41 |   border-radius: 3px;
 42 |   border: 1px solid rgba(255, 255, 255, 0.1);
 43 | }
 44 | 
 45 | input[type="range"]::-webkit-slider-thumb {
 46 |   -webkit-appearance: none;
 47 |   appearance: none;
 48 |   background: linear-gradient(135deg, #60a5fa, #a78bfa);
 49 |   height: 24px;
 50 |   width: 24px;
 51 |   border-radius: 50%;
 52 |   border: 2px solid rgba(255, 255, 255, 0.8);
 53 |   box-shadow: 0 4px 12px rgba(0, 0, 0, 0.4), 0 0 0 0 rgba(96, 165, 250, 0.4);
 54 |   transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
 55 |   cursor: pointer;
 56 | }
 57 | 
 58 | input[type="range"]::-webkit-slider-thumb:hover {
 59 |   background: linear-gradient(135deg, #3b82f6, #8b5cf6);
 60 |   transform: scale(1.15);
 61 |   box-shadow: 0 6px 16px rgba(0, 0, 0, 0.5), 0 0 0 8px rgba(96, 165, 250, 0.2);
 62 |   border-color: white;
 63 | }
 64 | 
 65 | input[type="range"]::-webkit-slider-thumb:active {
 66 |   transform: scale(1.05);
 67 |   box-shadow: 0 2px 8px rgba(0, 0, 0, 0.6), 0 0 0 12px rgba(96, 165, 250, 0.3);
 68 | }
 69 | 
 70 | input[type="range"]::-moz-range-track {
 71 |   background: linear-gradient(90deg, rgba(59, 130, 246, 0.3), rgba(168, 85, 247, 0.3));
 72 |   height: 6px;
 73 |   border-radius: 3px;
 74 |   border: 1px solid rgba(255, 255, 255, 0.1);
 75 | }
 76 | 
 77 | input[type="range"]::-moz-range-thumb {
 78 |   background: linear-gradient(135deg, #60a5fa, #a78bfa);
 79 |   height: 24px;
 80 |   width: 24px;
 81 |   border-radius: 50%;
 82 |   border: 2px solid rgba(255, 255, 255, 0.8);
 83 |   box-shadow: 0 4px 12px rgba(0, 0, 0, 0.4);
 84 |   cursor: pointer;
 85 |   transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
 86 | }
 87 | 
 88 | input[type="range"]::-moz-range-thumb:hover {
 89 |   background: linear-gradient(135deg, #3b82f6, #8b5cf6);
 90 |   transform: scale(1.15);
 91 |   box-shadow: 0 6px 16px rgba(0, 0, 0, 0.5);
 92 |   border-color: white;
 93 | }
 94 | 
 95 | input[type="range"]:focus {
 96 |   outline: none;
 97 | }
 98 | 
 99 | input[type="range"]:focus::-webkit-slider-thumb {
100 |   box-shadow: 0 4px 12px rgba(0, 0, 0, 0.4), 0 0 0 4px rgba(96, 165, 250, 0.3);
101 | }
102 | 


--------------------------------------------------------------------------------
/liquid-glass-example/tsconfig.json:
--------------------------------------------------------------------------------
 1 | {
 2 |   "compilerOptions": {
 3 |     "target": "ES2017",
 4 |     "lib": ["dom", "dom.iterable", "esnext"],
 5 |     "allowJs": true,
 6 |     "skipLibCheck": true,
 7 |     "strict": true,
 8 |     "noEmit": true,
 9 |     "esModuleInterop": true,
10 |     "module": "esnext",
11 |     "moduleResolution": "bundler",
12 |     "resolveJsonModule": true,
13 |     "isolatedModules": true,
14 |     "jsx": "preserve",
15 |     "incremental": true,
16 |     "paths": {
17 |       "@/*": ["./src/*"]
18 |     }
19 |   },
20 |   "include": ["next-env.d.ts", "**/*.ts", "**/*.tsx"],
21 |   "exclude": ["node_modules"]
22 | }
23 | 


--------------------------------------------------------------------------------
/package-lock.json:
--------------------------------------------------------------------------------
  1 | {
  2 |   "name": "liquid-glass-react",
  3 |   "version": "1.0.0",
  4 |   "lockfileVersion": 3,
  5 |   "requires": true,
  6 |   "packages": {
  7 |     "": {
  8 |       "name": "liquid-glass-react",
  9 |       "version": "1.0.0",
 10 |       "license": "MIT",
 11 |       "devDependencies": {
 12 |         "@types/react": "^18.2.0",
 13 |         "@types/react-dom": "^18.2.0",
 14 |         "esbuild": "^0.19.0",
 15 |         "react": "^18.2.0",
 16 |         "react-dom": "^18.2.0",
 17 |         "typescript": "^5.0.0"
 18 |       },
 19 |       "peerDependencies": {
 20 |         "react": ">=16.8.0",
 21 |         "react-dom": ">=16.8.0"
 22 |       }
 23 |     },
 24 |     "node_modules/@esbuild/aix-ppc64": {
 25 |       "version": "0.19.12",
 26 |       "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.19.12.tgz",
 27 |       "integrity": "sha512-bmoCYyWdEL3wDQIVbcyzRyeKLgk2WtWLTWz1ZIAZF/EGbNOwSA6ew3PftJ1PqMiOOGu0OyFMzG53L0zqIpPeNA==",
 28 |       "cpu": [
 29 |         "ppc64"
 30 |       ],
 31 |       "dev": true,
 32 |       "optional": true,
 33 |       "os": [
 34 |         "aix"
 35 |       ],
 36 |       "engines": {
 37 |         "node": ">=12"
 38 |       }
 39 |     },
 40 |     "node_modules/@esbuild/android-arm": {
 41 |       "version": "0.19.12",
 42 |       "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.19.12.tgz",
 43 |       "integrity": "sha512-qg/Lj1mu3CdQlDEEiWrlC4eaPZ1KztwGJ9B6J+/6G+/4ewxJg7gqj8eVYWvao1bXrqGiW2rsBZFSX3q2lcW05w==",
 44 |       "cpu": [
 45 |         "arm"
 46 |       ],
 47 |       "dev": true,
 48 |       "optional": true,
 49 |       "os": [
 50 |         "android"
 51 |       ],
 52 |       "engines": {
 53 |         "node": ">=12"
 54 |       }
 55 |     },
 56 |     "node_modules/@esbuild/android-arm64": {
 57 |       "version": "0.19.12",
 58 |       "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.19.12.tgz",
 59 |       "integrity": "sha512-P0UVNGIienjZv3f5zq0DP3Nt2IE/3plFzuaS96vihvD0Hd6H/q4WXUGpCxD/E8YrSXfNyRPbpTq+T8ZQioSuPA==",
 60 |       "cpu": [
 61 |         "arm64"
 62 |       ],
 63 |       "dev": true,
 64 |       "optional": true,
 65 |       "os": [
 66 |         "android"
 67 |       ],
 68 |       "engines": {
 69 |         "node": ">=12"
 70 |       }
 71 |     },
 72 |     "node_modules/@esbuild/android-x64": {
 73 |       "version": "0.19.12",
 74 |       "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.19.12.tgz",
 75 |       "integrity": "sha512-3k7ZoUW6Q6YqhdhIaq/WZ7HwBpnFBlW905Fa4s4qWJyiNOgT1dOqDiVAQFwBH7gBRZr17gLrlFCRzF6jFh7Kew==",
 76 |       "cpu": [
 77 |         "x64"
 78 |       ],
 79 |       "dev": true,
 80 |       "optional": true,
 81 |       "os": [
 82 |         "android"
 83 |       ],
 84 |       "engines": {
 85 |         "node": ">=12"
 86 |       }
 87 |     },
 88 |     "node_modules/@esbuild/darwin-arm64": {
 89 |       "version": "0.19.12",
 90 |       "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.19.12.tgz",
 91 |       "integrity": "sha512-B6IeSgZgtEzGC42jsI+YYu9Z3HKRxp8ZT3cqhvliEHovq8HSX2YX8lNocDn79gCKJXOSaEot9MVYky7AKjCs8g==",
 92 |       "cpu": [
 93 |         "arm64"
 94 |       ],
 95 |       "dev": true,
 96 |       "optional": true,
 97 |       "os": [
 98 |         "darwin"
 99 |       ],
100 |       "engines": {
101 |         "node": ">=12"
102 |       }
103 |     },
104 |     "node_modules/@esbuild/darwin-x64": {
105 |       "version": "0.19.12",
106 |       "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.19.12.tgz",
107 |       "integrity": "sha512-hKoVkKzFiToTgn+41qGhsUJXFlIjxI/jSYeZf3ugemDYZldIXIxhvwN6erJGlX4t5h417iFuheZ7l+YVn05N3A==",
108 |       "cpu": [
109 |         "x64"
110 |       ],
111 |       "dev": true,
112 |       "optional": true,
113 |       "os": [
114 |         "darwin"
115 |       ],
116 |       "engines": {
117 |         "node": ">=12"
118 |       }
119 |     },
120 |     "node_modules/@esbuild/freebsd-arm64": {
121 |       "version": "0.19.12",
122 |       "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.19.12.tgz",
123 |       "integrity": "sha512-4aRvFIXmwAcDBw9AueDQ2YnGmz5L6obe5kmPT8Vd+/+x/JMVKCgdcRwH6APrbpNXsPz+K653Qg8HB/oXvXVukA==",
124 |       "cpu": [
125 |         "arm64"
126 |       ],
127 |       "dev": true,
128 |       "optional": true,
129 |       "os": [
130 |         "freebsd"
131 |       ],
132 |       "engines": {
133 |         "node": ">=12"
134 |       }
135 |     },
136 |     "node_modules/@esbuild/freebsd-x64": {
137 |       "version": "0.19.12",
138 |       "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.19.12.tgz",
139 |       "integrity": "sha512-EYoXZ4d8xtBoVN7CEwWY2IN4ho76xjYXqSXMNccFSx2lgqOG/1TBPW0yPx1bJZk94qu3tX0fycJeeQsKovA8gg==",
140 |       "cpu": [
141 |         "x64"
142 |       ],
143 |       "dev": true,
144 |       "optional": true,
145 |       "os": [
146 |         "freebsd"
147 |       ],
148 |       "engines": {
149 |         "node": ">=12"
150 |       }
151 |     },
152 |     "node_modules/@esbuild/linux-arm": {
153 |       "version": "0.19.12",
154 |       "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.19.12.tgz",
155 |       "integrity": "sha512-J5jPms//KhSNv+LO1S1TX1UWp1ucM6N6XuL6ITdKWElCu8wXP72l9MM0zDTzzeikVyqFE6U8YAV9/tFyj0ti+w==",
156 |       "cpu": [
157 |         "arm"
158 |       ],
159 |       "dev": true,
160 |       "optional": true,
161 |       "os": [
162 |         "linux"
163 |       ],
164 |       "engines": {
165 |         "node": ">=12"
166 |       }
167 |     },
168 |     "node_modules/@esbuild/linux-arm64": {
169 |       "version": "0.19.12",
170 |       "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.19.12.tgz",
171 |       "integrity": "sha512-EoTjyYyLuVPfdPLsGVVVC8a0p1BFFvtpQDB/YLEhaXyf/5bczaGeN15QkR+O4S5LeJ92Tqotve7i1jn35qwvdA==",
172 |       "cpu": [
173 |         "arm64"
174 |       ],
175 |       "dev": true,
176 |       "optional": true,
177 |       "os": [
178 |         "linux"
179 |       ],
180 |       "engines": {
181 |         "node": ">=12"
182 |       }
183 |     },
184 |     "node_modules/@esbuild/linux-ia32": {
185 |       "version": "0.19.12",
186 |       "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.19.12.tgz",
187 |       "integrity": "sha512-Thsa42rrP1+UIGaWz47uydHSBOgTUnwBwNq59khgIwktK6x60Hivfbux9iNR0eHCHzOLjLMLfUMLCypBkZXMHA==",
188 |       "cpu": [
189 |         "ia32"
190 |       ],
191 |       "dev": true,
192 |       "optional": true,
193 |       "os": [
194 |         "linux"
195 |       ],
196 |       "engines": {
197 |         "node": ">=12"
198 |       }
199 |     },
200 |     "node_modules/@esbuild/linux-loong64": {
201 |       "version": "0.19.12",
202 |       "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.19.12.tgz",
203 |       "integrity": "sha512-LiXdXA0s3IqRRjm6rV6XaWATScKAXjI4R4LoDlvO7+yQqFdlr1Bax62sRwkVvRIrwXxvtYEHHI4dm50jAXkuAA==",
204 |       "cpu": [
205 |         "loong64"
206 |       ],
207 |       "dev": true,
208 |       "optional": true,
209 |       "os": [
210 |         "linux"
211 |       ],
212 |       "engines": {
213 |         "node": ">=12"
214 |       }
215 |     },
216 |     "node_modules/@esbuild/linux-mips64el": {
217 |       "version": "0.19.12",
218 |       "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.19.12.tgz",
219 |       "integrity": "sha512-fEnAuj5VGTanfJ07ff0gOA6IPsvrVHLVb6Lyd1g2/ed67oU1eFzL0r9WL7ZzscD+/N6i3dWumGE1Un4f7Amf+w==",
220 |       "cpu": [
221 |         "mips64el"
222 |       ],
223 |       "dev": true,
224 |       "optional": true,
225 |       "os": [
226 |         "linux"
227 |       ],
228 |       "engines": {
229 |         "node": ">=12"
230 |       }
231 |     },
232 |     "node_modules/@esbuild/linux-ppc64": {
233 |       "version": "0.19.12",
234 |       "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.19.12.tgz",
235 |       "integrity": "sha512-nYJA2/QPimDQOh1rKWedNOe3Gfc8PabU7HT3iXWtNUbRzXS9+vgB0Fjaqr//XNbd82mCxHzik2qotuI89cfixg==",
236 |       "cpu": [
237 |         "ppc64"
238 |       ],
239 |       "dev": true,
240 |       "optional": true,
241 |       "os": [
242 |         "linux"
243 |       ],
244 |       "engines": {
245 |         "node": ">=12"
246 |       }
247 |     },
248 |     "node_modules/@esbuild/linux-riscv64": {
249 |       "version": "0.19.12",
250 |       "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.19.12.tgz",
251 |       "integrity": "sha512-2MueBrlPQCw5dVJJpQdUYgeqIzDQgw3QtiAHUC4RBz9FXPrskyyU3VI1hw7C0BSKB9OduwSJ79FTCqtGMWqJHg==",
252 |       "cpu": [
253 |         "riscv64"
254 |       ],
255 |       "dev": true,
256 |       "optional": true,
257 |       "os": [
258 |         "linux"
259 |       ],
260 |       "engines": {
261 |         "node": ">=12"
262 |       }
263 |     },
264 |     "node_modules/@esbuild/linux-s390x": {
265 |       "version": "0.19.12",
266 |       "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.19.12.tgz",
267 |       "integrity": "sha512-+Pil1Nv3Umes4m3AZKqA2anfhJiVmNCYkPchwFJNEJN5QxmTs1uzyy4TvmDrCRNT2ApwSari7ZIgrPeUx4UZDg==",
268 |       "cpu": [
269 |         "s390x"
270 |       ],
271 |       "dev": true,
272 |       "optional": true,
273 |       "os": [
274 |         "linux"
275 |       ],
276 |       "engines": {
277 |         "node": ">=12"
278 |       }
279 |     },
280 |     "node_modules/@esbuild/linux-x64": {
281 |       "version": "0.19.12",
282 |       "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.19.12.tgz",
283 |       "integrity": "sha512-B71g1QpxfwBvNrfyJdVDexenDIt1CiDN1TIXLbhOw0KhJzE78KIFGX6OJ9MrtC0oOqMWf+0xop4qEU8JrJTwCg==",
284 |       "cpu": [
285 |         "x64"
286 |       ],
287 |       "dev": true,
288 |       "optional": true,
289 |       "os": [
290 |         "linux"
291 |       ],
292 |       "engines": {
293 |         "node": ">=12"
294 |       }
295 |     },
296 |     "node_modules/@esbuild/netbsd-x64": {
297 |       "version": "0.19.12",
298 |       "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.19.12.tgz",
299 |       "integrity": "sha512-3ltjQ7n1owJgFbuC61Oj++XhtzmymoCihNFgT84UAmJnxJfm4sYCiSLTXZtE00VWYpPMYc+ZQmB6xbSdVh0JWA==",
300 |       "cpu": [
301 |         "x64"
302 |       ],
303 |       "dev": true,
304 |       "optional": true,
305 |       "os": [
306 |         "netbsd"
307 |       ],
308 |       "engines": {
309 |         "node": ">=12"
310 |       }
311 |     },
312 |     "node_modules/@esbuild/openbsd-x64": {
313 |       "version": "0.19.12",
314 |       "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.19.12.tgz",
315 |       "integrity": "sha512-RbrfTB9SWsr0kWmb9srfF+L933uMDdu9BIzdA7os2t0TXhCRjrQyCeOt6wVxr79CKD4c+p+YhCj31HBkYcXebw==",
316 |       "cpu": [
317 |         "x64"
318 |       ],
319 |       "dev": true,
320 |       "optional": true,
321 |       "os": [
322 |         "openbsd"
323 |       ],
324 |       "engines": {
325 |         "node": ">=12"
326 |       }
327 |     },
328 |     "node_modules/@esbuild/sunos-x64": {
329 |       "version": "0.19.12",
330 |       "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.19.12.tgz",
331 |       "integrity": "sha512-HKjJwRrW8uWtCQnQOz9qcU3mUZhTUQvi56Q8DPTLLB+DawoiQdjsYq+j+D3s9I8VFtDr+F9CjgXKKC4ss89IeA==",
332 |       "cpu": [
333 |         "x64"
334 |       ],
335 |       "dev": true,
336 |       "optional": true,
337 |       "os": [
338 |         "sunos"
339 |       ],
340 |       "engines": {
341 |         "node": ">=12"
342 |       }
343 |     },
344 |     "node_modules/@esbuild/win32-arm64": {
345 |       "version": "0.19.12",
346 |       "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.19.12.tgz",
347 |       "integrity": "sha512-URgtR1dJnmGvX864pn1B2YUYNzjmXkuJOIqG2HdU62MVS4EHpU2946OZoTMnRUHklGtJdJZ33QfzdjGACXhn1A==",
348 |       "cpu": [
349 |         "arm64"
350 |       ],
351 |       "dev": true,
352 |       "optional": true,
353 |       "os": [
354 |         "win32"
355 |       ],
356 |       "engines": {
357 |         "node": ">=12"
358 |       }
359 |     },
360 |     "node_modules/@esbuild/win32-ia32": {
361 |       "version": "0.19.12",
362 |       "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.19.12.tgz",
363 |       "integrity": "sha512-+ZOE6pUkMOJfmxmBZElNOx72NKpIa/HFOMGzu8fqzQJ5kgf6aTGrcJaFsNiVMH4JKpMipyK+7k0n2UXN7a8YKQ==",
364 |       "cpu": [
365 |         "ia32"
366 |       ],
367 |       "dev": true,
368 |       "optional": true,
369 |       "os": [
370 |         "win32"
371 |       ],
372 |       "engines": {
373 |         "node": ">=12"
374 |       }
375 |     },
376 |     "node_modules/@esbuild/win32-x64": {
377 |       "version": "0.19.12",
378 |       "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.19.12.tgz",
379 |       "integrity": "sha512-T1QyPSDCyMXaO3pzBkF96E8xMkiRYbUEZADd29SyPGabqxMViNoii+NcK7eWJAEoU6RZyEm5lVSIjTmcdoB9HA==",
380 |       "cpu": [
381 |         "x64"
382 |       ],
383 |       "dev": true,
384 |       "optional": true,
385 |       "os": [
386 |         "win32"
387 |       ],
388 |       "engines": {
389 |         "node": ">=12"
390 |       }
391 |     },
392 |     "node_modules/@types/prop-types": {
393 |       "version": "15.7.15",
394 |       "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.15.tgz",
395 |       "integrity": "sha512-F6bEyamV9jKGAFBEmlQnesRPGOQqS2+Uwi0Em15xenOxHaf2hv6L8YCVn3rPdPJOiJfPiCnLIRyvwVaqMY3MIw==",
396 |       "dev": true
397 |     },
398 |     "node_modules/@types/react": {
399 |       "version": "18.3.23",
400 |       "resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.23.tgz",
401 |       "integrity": "sha512-/LDXMQh55EzZQ0uVAZmKKhfENivEvWz6E+EYzh+/MCjMhNsotd+ZHhBGIjFDTi6+fz0OhQQQLbTgdQIxxCsC0w==",
402 |       "dev": true,
403 |       "dependencies": {
404 |         "@types/prop-types": "*",
405 |         "csstype": "^3.0.2"
406 |       }
407 |     },
408 |     "node_modules/@types/react-dom": {
409 |       "version": "18.3.7",
410 |       "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.3.7.tgz",
411 |       "integrity": "sha512-MEe3UeoENYVFXzoXEWsvcpg6ZvlrFNlOQ7EOsvhI3CfAXwzPfO8Qwuxd40nepsYKqyyVQnTdEfv68q91yLcKrQ==",
412 |       "dev": true,
413 |       "peerDependencies": {
414 |         "@types/react": "^18.0.0"
415 |       }
416 |     },
417 |     "node_modules/csstype": {
418 |       "version": "3.1.3",
419 |       "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz",
420 |       "integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==",
421 |       "dev": true
422 |     },
423 |     "node_modules/esbuild": {
424 |       "version": "0.19.12",
425 |       "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.19.12.tgz",
426 |       "integrity": "sha512-aARqgq8roFBj054KvQr5f1sFu0D65G+miZRCuJyJ0G13Zwx7vRar5Zhn2tkQNzIXcBrNVsv/8stehpj+GAjgbg==",
427 |       "dev": true,
428 |       "hasInstallScript": true,
429 |       "bin": {
430 |         "esbuild": "bin/esbuild"
431 |       },
432 |       "engines": {
433 |         "node": ">=12"
434 |       },
435 |       "optionalDependencies": {
436 |         "@esbuild/aix-ppc64": "0.19.12",
437 |         "@esbuild/android-arm": "0.19.12",
438 |         "@esbuild/android-arm64": "0.19.12",
439 |         "@esbuild/android-x64": "0.19.12",
440 |         "@esbuild/darwin-arm64": "0.19.12",
441 |         "@esbuild/darwin-x64": "0.19.12",
442 |         "@esbuild/freebsd-arm64": "0.19.12",
443 |         "@esbuild/freebsd-x64": "0.19.12",
444 |         "@esbuild/linux-arm": "0.19.12",
445 |         "@esbuild/linux-arm64": "0.19.12",
446 |         "@esbuild/linux-ia32": "0.19.12",
447 |         "@esbuild/linux-loong64": "0.19.12",
448 |         "@esbuild/linux-mips64el": "0.19.12",
449 |         "@esbuild/linux-ppc64": "0.19.12",
450 |         "@esbuild/linux-riscv64": "0.19.12",
451 |         "@esbuild/linux-s390x": "0.19.12",
452 |         "@esbuild/linux-x64": "0.19.12",
453 |         "@esbuild/netbsd-x64": "0.19.12",
454 |         "@esbuild/openbsd-x64": "0.19.12",
455 |         "@esbuild/sunos-x64": "0.19.12",
456 |         "@esbuild/win32-arm64": "0.19.12",
457 |         "@esbuild/win32-ia32": "0.19.12",
458 |         "@esbuild/win32-x64": "0.19.12"
459 |       }
460 |     },
461 |     "node_modules/js-tokens": {
462 |       "version": "4.0.0",
463 |       "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
464 |       "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==",
465 |       "dev": true
466 |     },
467 |     "node_modules/loose-envify": {
468 |       "version": "1.4.0",
469 |       "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz",
470 |       "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==",
471 |       "dev": true,
472 |       "dependencies": {
473 |         "js-tokens": "^3.0.0 || ^4.0.0"
474 |       },
475 |       "bin": {
476 |         "loose-envify": "cli.js"
477 |       }
478 |     },
479 |     "node_modules/react": {
480 |       "version": "18.3.1",
481 |       "resolved": "https://registry.npmjs.org/react/-/react-18.3.1.tgz",
482 |       "integrity": "sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ==",
483 |       "dev": true,
484 |       "dependencies": {
485 |         "loose-envify": "^1.1.0"
486 |       },
487 |       "engines": {
488 |         "node": ">=0.10.0"
489 |       }
490 |     },
491 |     "node_modules/react-dom": {
492 |       "version": "18.3.1",
493 |       "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.3.1.tgz",
494 |       "integrity": "sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==",
495 |       "dev": true,
496 |       "dependencies": {
497 |         "loose-envify": "^1.1.0",
498 |         "scheduler": "^0.23.2"
499 |       },
500 |       "peerDependencies": {
501 |         "react": "^18.3.1"
502 |       }
503 |     },
504 |     "node_modules/scheduler": {
505 |       "version": "0.23.2",
506 |       "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.2.tgz",
507 |       "integrity": "sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==",
508 |       "dev": true,
509 |       "dependencies": {
510 |         "loose-envify": "^1.1.0"
511 |       }
512 |     },
513 |     "node_modules/typescript": {
514 |       "version": "5.8.3",
515 |       "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.8.3.tgz",
516 |       "integrity": "sha512-p1diW6TqL9L07nNxvRMM7hMMw4c5XOo/1ibL4aAIGmSAt9slTE1Xgw5KWuof2uTOvCg9BY7ZRi+GaF+7sfgPeQ==",
517 |       "dev": true,
518 |       "bin": {
519 |         "tsc": "bin/tsc",
520 |         "tsserver": "bin/tsserver"
521 |       },
522 |       "engines": {
523 |         "node": ">=14.17"
524 |       }
525 |     }
526 |   }
527 | }
528 | 


--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
 1 | {
 2 |   "name": "liquid-glass-react",
 3 |   "version": "1.1.1",
 4 |   "description": "Apple's Liquid Glass effect for React",
 5 |   "main": "dist/index.js",
 6 |   "module": "dist/index.esm.js",
 7 |   "types": "dist/index.d.ts",
 8 |   "files": ["dist"],
 9 |   "workspaces": ["liquid-glass"],
10 |   "scripts": {
11 |     "build": "npm run clean && npm run build:esm && npm run build:cjs && npm run build:types",
12 |     "build:esm": "esbuild src/index.tsx --bundle --format=esm --outfile=dist/index.esm.js --external:react --external:react-dom",
13 |     "build:cjs": "esbuild src/index.tsx --bundle --format=cjs --outfile=dist/index.js --external:react --external:react-dom",
14 |     "build:types": "tsc --emitDeclarationOnly --outDir dist",
15 |     "clean": "rm -rf dist",
16 |     "dev": "npm run build:esm -- --watch",
17 |     "prepublishOnly": "npm run build"
18 |   },
19 |   "peerDependencies": {
20 |     "react": ">=18",
21 |     "react-dom": ">=18"
22 |   },
23 |   "devDependencies": {
24 |     "@biomejs/biome": "^1.9.4",
25 |     "@types/react": "^18.2.0",
26 |     "@types/react-dom": "^18.2.0",
27 |     "esbuild": "^0.19.0",
28 |     "react": "^18.2.0",
29 |     "react-dom": "^18.2.0",
30 |     "typescript": "^5.0.0"
31 |   },
32 |   "keywords": ["react", "component", "library", "typescript"],
33 |   "author": "",
34 |   "license": "MIT",
35 |   "repository": {
36 |     "type": "git",
37 |     "url": ""
38 |   }
39 | }
40 | 


--------------------------------------------------------------------------------
/src/index.tsx:
--------------------------------------------------------------------------------
  1 | import { type CSSProperties, forwardRef, useCallback, useEffect, useId, useRef, useState } from "react"
  2 | import { ShaderDisplacementGenerator, fragmentShaders } from "./shader-utils"
  3 | import { displacementMap, polarDisplacementMap, prominentDisplacementMap } from "./utils"
  4 | 
  5 | // Generate shader-based displacement map using shaderUtils
  6 | const generateShaderDisplacementMap = (width: number, height: number): string => {
  7 |   const generator = new ShaderDisplacementGenerator({
  8 |     width,
  9 |     height,
 10 |     fragment: fragmentShaders.liquidGlass,
 11 |   })
 12 | 
 13 |   const dataUrl = generator.updateShader()
 14 |   generator.destroy()
 15 | 
 16 |   return dataUrl
 17 | }
 18 | 
 19 | const getMap = (mode: "standard" | "polar" | "prominent" | "shader", shaderMapUrl?: string) => {
 20 |   switch (mode) {
 21 |     case "standard":
 22 |       return displacementMap
 23 |     case "polar":
 24 |       return polarDisplacementMap
 25 |     case "prominent":
 26 |       return prominentDisplacementMap
 27 |     case "shader":
 28 |       return shaderMapUrl || displacementMap
 29 |     default:
 30 |       throw new Error(`Invalid mode: ${mode}`)
 31 |   }
 32 | }
 33 | 
 34 | /* ---------- SVG filter (edge-only displacement) ---------- */
 35 | const GlassFilter: React.FC<{ id: string; displacementScale: number; aberrationIntensity: number; width: number; height: number; mode: "standard" | "polar" | "prominent" | "shader"; shaderMapUrl?: string }> = ({
 36 |   id,
 37 |   displacementScale,
 38 |   aberrationIntensity,
 39 |   width,
 40 |   height,
 41 |   mode,
 42 |   shaderMapUrl,
 43 | }) => (
 44 |   <svg style={{ position: "absolute", width, height }} aria-hidden="true">
 45 |     <defs>
 46 |       <radialGradient id={`${id}-edge-mask`} cx="50%" cy="50%" r="50%">
 47 |         <stop offset="0%" stopColor="black" stopOpacity="0" />
 48 |         <stop offset={`${Math.max(30, 80 - aberrationIntensity * 2)}%`} stopColor="black" stopOpacity="0" />
 49 |         <stop offset="100%" stopColor="white" stopOpacity="1" />
 50 |       </radialGradient>
 51 |       <filter id={id} x="-35%" y="-35%" width="170%" height="170%" colorInterpolationFilters="sRGB">
 52 |         <feImage id="feimage" x="0" y="0" width="100%" height="100%" result="DISPLACEMENT_MAP" href={getMap(mode, shaderMapUrl)} preserveAspectRatio="xMidYMid slice" />
 53 | 
 54 |         {/* Create edge mask using the displacement map itself */}
 55 |         <feColorMatrix
 56 |           in="DISPLACEMENT_MAP"
 57 |           type="matrix"
 58 |           values="0.3 0.3 0.3 0 0
 59 |                  0.3 0.3 0.3 0 0
 60 |                  0.3 0.3 0.3 0 0
 61 |                  0 0 0 1 0"
 62 |           result="EDGE_INTENSITY"
 63 |         />
 64 |         <feComponentTransfer in="EDGE_INTENSITY" result="EDGE_MASK">
 65 |           <feFuncA type="discrete" tableValues={`0 ${aberrationIntensity * 0.05} 1`} />
 66 |         </feComponentTransfer>
 67 | 
 68 |         {/* Original undisplaced image for center */}
 69 |         <feOffset in="SourceGraphic" dx="0" dy="0" result="CENTER_ORIGINAL" />
 70 | 
 71 |         {/* Red channel displacement with slight offset */}
 72 |         <feDisplacementMap in="SourceGraphic" in2="DISPLACEMENT_MAP" scale={displacementScale * (mode === "shader" ? 1 : -1)} xChannelSelector="R" yChannelSelector="B" result="RED_DISPLACED" />
 73 |         <feColorMatrix
 74 |           in="RED_DISPLACED"
 75 |           type="matrix"
 76 |           values="1 0 0 0 0
 77 |                  0 0 0 0 0
 78 |                  0 0 0 0 0
 79 |                  0 0 0 1 0"
 80 |           result="RED_CHANNEL"
 81 |         />
 82 | 
 83 |         {/* Green channel displacement */}
 84 |         <feDisplacementMap in="SourceGraphic" in2="DISPLACEMENT_MAP" scale={displacementScale * ((mode === "shader" ? 1 : -1) - aberrationIntensity * 0.05)} xChannelSelector="R" yChannelSelector="B" result="GREEN_DISPLACED" />
 85 |         <feColorMatrix
 86 |           in="GREEN_DISPLACED"
 87 |           type="matrix"
 88 |           values="0 0 0 0 0
 89 |                  0 1 0 0 0
 90 |                  0 0 0 0 0
 91 |                  0 0 0 1 0"
 92 |           result="GREEN_CHANNEL"
 93 |         />
 94 | 
 95 |         {/* Blue channel displacement with slight offset */}
 96 |         <feDisplacementMap in="SourceGraphic" in2="DISPLACEMENT_MAP" scale={displacementScale * ((mode === "shader" ? 1 : -1) - aberrationIntensity * 0.1)} xChannelSelector="R" yChannelSelector="B" result="BLUE_DISPLACED" />
 97 |         <feColorMatrix
 98 |           in="BLUE_DISPLACED"
 99 |           type="matrix"
100 |           values="0 0 0 0 0
101 |                  0 0 0 0 0
102 |                  0 0 1 0 0
103 |                  0 0 0 1 0"
104 |           result="BLUE_CHANNEL"
105 |         />
106 | 
107 |         {/* Combine all channels with screen blend mode for chromatic aberration */}
108 |         <feBlend in="GREEN_CHANNEL" in2="BLUE_CHANNEL" mode="screen" result="GB_COMBINED" />
109 |         <feBlend in="RED_CHANNEL" in2="GB_COMBINED" mode="screen" result="RGB_COMBINED" />
110 | 
111 |         {/* Add slight blur to soften the aberration effect */}
112 |         <feGaussianBlur in="RGB_COMBINED" stdDeviation={Math.max(0.1, 0.5 - aberrationIntensity * 0.1)} result="ABERRATED_BLURRED" />
113 | 
114 |         {/* Apply edge mask to aberration effect */}
115 |         <feComposite in="ABERRATED_BLURRED" in2="EDGE_MASK" operator="in" result="EDGE_ABERRATION" />
116 | 
117 |         {/* Create inverted mask for center */}
118 |         <feComponentTransfer in="EDGE_MASK" result="INVERTED_MASK">
119 |           <feFuncA type="table" tableValues="1 0" />
120 |         </feComponentTransfer>
121 |         <feComposite in="CENTER_ORIGINAL" in2="INVERTED_MASK" operator="in" result="CENTER_CLEAN" />
122 | 
123 |         {/* Combine edge aberration with clean center */}
124 |         <feComposite in="EDGE_ABERRATION" in2="CENTER_CLEAN" operator="over" />
125 |       </filter>
126 |     </defs>
127 |   </svg>
128 | )
129 | 
130 | /* ---------- container ---------- */
131 | const GlassContainer = forwardRef<
132 |   HTMLDivElement,
133 |   React.PropsWithChildren<{
134 |     className?: string
135 |     style?: React.CSSProperties
136 |     displacementScale?: number
137 |     blurAmount?: number
138 |     saturation?: number
139 |     aberrationIntensity?: number
140 |     mouseOffset?: { x: number; y: number }
141 |     onMouseLeave?: () => void
142 |     onMouseEnter?: () => void
143 |     onMouseDown?: () => void
144 |     onMouseUp?: () => void
145 |     active?: boolean
146 |     overLight?: boolean
147 |     cornerRadius?: number
148 |     padding?: string
149 |     glassSize?: { width: number; height: number }
150 |     onClick?: () => void
151 |     mode?: "standard" | "polar" | "prominent" | "shader"
152 |   }>
153 | >(
154 |   (
155 |     {
156 |       children,
157 |       className = "",
158 |       style,
159 |       displacementScale = 25,
160 |       blurAmount = 12,
161 |       saturation = 180,
162 |       aberrationIntensity = 2,
163 |       onMouseEnter,
164 |       onMouseLeave,
165 |       onMouseDown,
166 |       onMouseUp,
167 |       active = false,
168 |       overLight = false,
169 |       cornerRadius = 999,
170 |       padding = "24px 32px",
171 |       glassSize = { width: 270, height: 69 },
172 |       onClick,
173 |       mode = "standard",
174 |     },
175 |     ref,
176 |   ) => {
177 |     const filterId = useId()
178 |     const [shaderMapUrl, setShaderMapUrl] = useState<string>("")
179 | 
180 |     const isFirefox = navigator.userAgent.toLowerCase().includes("firefox")
181 | 
182 |     // Generate shader displacement map when in shader mode
183 |     useEffect(() => {
184 |       if (mode === "shader") {
185 |         const url = generateShaderDisplacementMap(glassSize.width, glassSize.height)
186 |         setShaderMapUrl(url)
187 |       }
188 |     }, [mode, glassSize.width, glassSize.height])
189 | 
190 |     const backdropStyle = {
191 |       filter: isFirefox ? null : `url(#${filterId})`,
192 |       backdropFilter: `blur(${(overLight ? 12 : 4) + blurAmount * 32}px) saturate(${saturation}%)`,
193 |     }
194 | 
195 |     return (
196 |       <div ref={ref} className={`relative ${className} ${active ? "active" : ""} ${Boolean(onClick) ? "cursor-pointer" : ""}`} style={style} onClick={onClick}>
197 |         <GlassFilter mode={mode} id={filterId} displacementScale={displacementScale} aberrationIntensity={aberrationIntensity} width={glassSize.width} height={glassSize.height} shaderMapUrl={shaderMapUrl} />
198 | 
199 |         <div
200 |           className="glass"
201 |           style={{
202 |             borderRadius: `${cornerRadius}px`,
203 |             position: "relative",
204 |             display: "inline-flex",
205 |             alignItems: "center",
206 |             gap: "24px",
207 |             padding,
208 |             overflow: "hidden",
209 |             transition: "all 0.2s ease-in-out",
210 |             boxShadow: overLight ? "0px 16px 70px rgba(0, 0, 0, 0.75)" : "0px 12px 40px rgba(0, 0, 0, 0.25)",
211 |           }}
212 |           onMouseEnter={onMouseEnter}
213 |           onMouseLeave={onMouseLeave}
214 |           onMouseDown={onMouseDown}
215 |           onMouseUp={onMouseUp}
216 |         >
217 |           {/* backdrop layer that gets wiggly */}
218 |           <span
219 |             className="glass__warp"
220 |             style={
221 |               {
222 |                 ...backdropStyle,
223 |                 position: "absolute",
224 |                 inset: "0",
225 |               } as CSSProperties
226 |             }
227 |           />
228 | 
229 |           {/* user content stays sharp */}
230 |           <div
231 |             className="transition-all duration-150 ease-in-out text-white"
232 |             style={{
233 |               position: "relative",
234 |               zIndex: 1,
235 |               font: "500 20px/1 system-ui",
236 |               textShadow: overLight ? "0px 2px 12px rgba(0, 0, 0, 0)" : "0px 2px 12px rgba(0, 0, 0, 0.4)",
237 |             }}
238 |           >
239 |             {children}
240 |           </div>
241 |         </div>
242 |       </div>
243 |     )
244 |   },
245 | )
246 | 
247 | GlassContainer.displayName = "GlassContainer"
248 | 
249 | interface LiquidGlassProps {
250 |   children: React.ReactNode
251 |   displacementScale?: number
252 |   blurAmount?: number
253 |   saturation?: number
254 |   aberrationIntensity?: number
255 |   elasticity?: number
256 |   cornerRadius?: number
257 |   globalMousePos?: { x: number; y: number }
258 |   mouseOffset?: { x: number; y: number }
259 |   mouseContainer?: React.RefObject<HTMLElement | null> | null
260 |   className?: string
261 |   padding?: string
262 |   style?: React.CSSProperties
263 |   overLight?: boolean
264 |   mode?: "standard" | "polar" | "prominent" | "shader"
265 |   onClick?: () => void
266 | }
267 | 
268 | export default function LiquidGlass({
269 |   children,
270 |   displacementScale = 70,
271 |   blurAmount = 0.0625,
272 |   saturation = 140,
273 |   aberrationIntensity = 2,
274 |   elasticity = 0.15,
275 |   cornerRadius = 999,
276 |   globalMousePos: externalGlobalMousePos,
277 |   mouseOffset: externalMouseOffset,
278 |   mouseContainer = null,
279 |   className = "",
280 |   padding = "24px 32px",
281 |   overLight = false,
282 |   style = {},
283 |   mode = "standard",
284 |   onClick,
285 | }: LiquidGlassProps) {
286 |   const glassRef = useRef<HTMLDivElement>(null)
287 |   const [isHovered, setIsHovered] = useState(false)
288 |   const [isActive, setIsActive] = useState(false)
289 |   const [glassSize, setGlassSize] = useState({ width: 270, height: 69 })
290 |   const [internalGlobalMousePos, setInternalGlobalMousePos] = useState({ x: 0, y: 0 })
291 |   const [internalMouseOffset, setInternalMouseOffset] = useState({ x: 0, y: 0 })
292 | 
293 |   // Use external mouse position if provided, otherwise use internal
294 |   const globalMousePos = externalGlobalMousePos || internalGlobalMousePos
295 |   const mouseOffset = externalMouseOffset || internalMouseOffset
296 | 
297 |   // Internal mouse tracking
298 |   const handleMouseMove = useCallback(
299 |     (e: MouseEvent) => {
300 |       const container = mouseContainer?.current || glassRef.current
301 |       if (!container) {
302 |         return
303 |       }
304 | 
305 |       const rect = container.getBoundingClientRect()
306 |       const centerX = rect.left + rect.width / 2
307 |       const centerY = rect.top + rect.height / 2
308 | 
309 |       setInternalMouseOffset({
310 |         x: ((e.clientX - centerX) / rect.width) * 100,
311 |         y: ((e.clientY - centerY) / rect.height) * 100,
312 |       })
313 | 
314 |       setInternalGlobalMousePos({
315 |         x: e.clientX,
316 |         y: e.clientY,
317 |       })
318 |     },
319 |     [mouseContainer],
320 |   )
321 | 
322 |   // Set up mouse tracking if no external mouse position is provided
323 |   useEffect(() => {
324 |     if (externalGlobalMousePos && externalMouseOffset) {
325 |       // External mouse tracking is provided, don't set up internal tracking
326 |       return
327 |     }
328 | 
329 |     const container = mouseContainer?.current || glassRef.current
330 |     if (!container) {
331 |       return
332 |     }
333 | 
334 |     container.addEventListener("mousemove", handleMouseMove)
335 | 
336 |     return () => {
337 |       container.removeEventListener("mousemove", handleMouseMove)
338 |     }
339 |   }, [handleMouseMove, mouseContainer, externalGlobalMousePos, externalMouseOffset])
340 | 
341 |   // Calculate directional scaling based on mouse position
342 |   const calculateDirectionalScale = useCallback(() => {
343 |     if (!globalMousePos.x || !globalMousePos.y || !glassRef.current) {
344 |       return "scale(1)"
345 |     }
346 | 
347 |     const rect = glassRef.current.getBoundingClientRect()
348 |     const pillCenterX = rect.left + rect.width / 2
349 |     const pillCenterY = rect.top + rect.height / 2
350 |     const pillWidth = glassSize.width
351 |     const pillHeight = glassSize.height
352 | 
353 |     const deltaX = globalMousePos.x - pillCenterX
354 |     const deltaY = globalMousePos.y - pillCenterY
355 | 
356 |     // Calculate distance from mouse to pill edges (not center)
357 |     const edgeDistanceX = Math.max(0, Math.abs(deltaX) - pillWidth / 2)
358 |     const edgeDistanceY = Math.max(0, Math.abs(deltaY) - pillHeight / 2)
359 |     const edgeDistance = Math.sqrt(edgeDistanceX * edgeDistanceX + edgeDistanceY * edgeDistanceY)
360 | 
361 |     // Activation zone: 200px from edges
362 |     const activationZone = 200
363 | 
364 |     // If outside activation zone, no effect
365 |     if (edgeDistance > activationZone) {
366 |       return "scale(1)"
367 |     }
368 | 
369 |     // Calculate fade-in factor (1 at edge, 0 at activation zone boundary)
370 |     const fadeInFactor = 1 - edgeDistance / activationZone
371 | 
372 |     // Normalize the deltas for direction
373 |     const centerDistance = Math.sqrt(deltaX * deltaX + deltaY * deltaY)
374 |     if (centerDistance === 0) {
375 |       return "scale(1)"
376 |     }
377 | 
378 |     const normalizedX = deltaX / centerDistance
379 |     const normalizedY = deltaY / centerDistance
380 | 
381 |     // Calculate stretch factors with fade-in
382 |     const stretchIntensity = Math.min(centerDistance / 300, 1) * elasticity * fadeInFactor
383 | 
384 |     // X-axis scaling: stretch horizontally when moving left/right, compress when moving up/down
385 |     const scaleX = 1 + Math.abs(normalizedX) * stretchIntensity * 0.3 - Math.abs(normalizedY) * stretchIntensity * 0.15
386 | 
387 |     // Y-axis scaling: stretch vertically when moving up/down, compress when moving left/right
388 |     const scaleY = 1 + Math.abs(normalizedY) * stretchIntensity * 0.3 - Math.abs(normalizedX) * stretchIntensity * 0.15
389 | 
390 |     return `scaleX(${Math.max(0.8, scaleX)}) scaleY(${Math.max(0.8, scaleY)})`
391 |   }, [globalMousePos, elasticity, glassSize])
392 | 
393 |   // Helper function to calculate fade-in factor based on distance from element edges
394 |   const calculateFadeInFactor = useCallback(() => {
395 |     if (!globalMousePos.x || !globalMousePos.y || !glassRef.current) {
396 |       return 0
397 |     }
398 | 
399 |     const rect = glassRef.current.getBoundingClientRect()
400 |     const pillCenterX = rect.left + rect.width / 2
401 |     const pillCenterY = rect.top + rect.height / 2
402 |     const pillWidth = glassSize.width
403 |     const pillHeight = glassSize.height
404 | 
405 |     const edgeDistanceX = Math.max(0, Math.abs(globalMousePos.x - pillCenterX) - pillWidth / 2)
406 |     const edgeDistanceY = Math.max(0, Math.abs(globalMousePos.y - pillCenterY) - pillHeight / 2)
407 |     const edgeDistance = Math.sqrt(edgeDistanceX * edgeDistanceX + edgeDistanceY * edgeDistanceY)
408 | 
409 |     const activationZone = 200
410 |     return edgeDistance > activationZone ? 0 : 1 - edgeDistance / activationZone
411 |   }, [globalMousePos, glassSize])
412 | 
413 |   // Helper function to calculate elastic translation
414 |   const calculateElasticTranslation = useCallback(() => {
415 |     if (!glassRef.current) {
416 |       return { x: 0, y: 0 }
417 |     }
418 | 
419 |     const fadeInFactor = calculateFadeInFactor()
420 |     const rect = glassRef.current.getBoundingClientRect()
421 |     const pillCenterX = rect.left + rect.width / 2
422 |     const pillCenterY = rect.top + rect.height / 2
423 | 
424 |     return {
425 |       x: (globalMousePos.x - pillCenterX) * elasticity * 0.1 * fadeInFactor,
426 |       y: (globalMousePos.y - pillCenterY) * elasticity * 0.1 * fadeInFactor,
427 |     }
428 |   }, [globalMousePos, elasticity, calculateFadeInFactor])
429 | 
430 |   // Update glass size whenever component mounts or window resizes
431 |   useEffect(() => {
432 |     const updateGlassSize = () => {
433 |       if (glassRef.current) {
434 |         const rect = glassRef.current.getBoundingClientRect()
435 |         setGlassSize({ width: rect.width, height: rect.height })
436 |       }
437 |     }
438 | 
439 |     updateGlassSize()
440 |     window.addEventListener("resize", updateGlassSize)
441 |     return () => window.removeEventListener("resize", updateGlassSize)
442 |   }, [])
443 | 
444 |   const transformStyle = `translate(calc(-50% + ${calculateElasticTranslation().x}px), calc(-50% + ${calculateElasticTranslation().y}px)) ${isActive && Boolean(onClick) ? "scale(0.96)" : calculateDirectionalScale()}`
445 | 
446 |   const baseStyle = {
447 |     ...style,
448 |     transform: transformStyle,
449 |     transition: "all ease-out 0.2s",
450 |   }
451 | 
452 |   const positionStyles = {
453 |     position: baseStyle.position || "relative",
454 |     top: baseStyle.top || "50%",
455 |     left: baseStyle.left || "50%",
456 |   }
457 | 
458 |   return (
459 |     <>
460 |       {/* Over light effect */}
461 |       <div
462 |         className={`bg-black transition-all duration-150 ease-in-out pointer-events-none ${overLight ? "opacity-20" : "opacity-0"}`}
463 |         style={{
464 |           ...positionStyles,
465 |           height: glassSize.height,
466 |           width: glassSize.width,
467 |           borderRadius: `${cornerRadius}px`,
468 |           transform: baseStyle.transform,
469 |           transition: baseStyle.transition,
470 |         }}
471 |       />
472 |       <div
473 |         className={`bg-black transition-all duration-150 ease-in-out pointer-events-none mix-blend-overlay ${overLight ? "opacity-100" : "opacity-0"}`}
474 |         style={{
475 |           ...positionStyles,
476 |           height: glassSize.height,
477 |           width: glassSize.width,
478 |           borderRadius: `${cornerRadius}px`,
479 |           transform: baseStyle.transform,
480 |           transition: baseStyle.transition,
481 |         }}
482 |       />
483 | 
484 |       <GlassContainer
485 |         ref={glassRef}
486 |         className={className}
487 |         style={baseStyle}
488 |         cornerRadius={cornerRadius}
489 |         displacementScale={overLight ? displacementScale * 0.5 : displacementScale}
490 |         blurAmount={blurAmount}
491 |         saturation={saturation}
492 |         aberrationIntensity={aberrationIntensity}
493 |         glassSize={glassSize}
494 |         padding={padding}
495 |         mouseOffset={mouseOffset}
496 |         onMouseEnter={() => setIsHovered(true)}
497 |         onMouseLeave={() => setIsHovered(false)}
498 |         onMouseDown={() => setIsActive(true)}
499 |         onMouseUp={() => setIsActive(false)}
500 |         active={isActive}
501 |         overLight={overLight}
502 |         onClick={onClick}
503 |         mode={mode}
504 |       >
505 |         {children}
506 |       </GlassContainer>
507 | 
508 |       {/* Border layer 1 - extracted from glass container */}
509 |       <span
510 |         style={{
511 |           ...positionStyles,
512 |           height: glassSize.height,
513 |           width: glassSize.width,
514 |           borderRadius: `${cornerRadius}px`,
515 |           transform: baseStyle.transform,
516 |           transition: baseStyle.transition,
517 |           pointerEvents: "none",
518 |           mixBlendMode: "screen",
519 |           opacity: 0.2,
520 |           padding: "1.5px",
521 |           WebkitMask: "linear-gradient(#000 0 0) content-box, linear-gradient(#000 0 0)",
522 |           WebkitMaskComposite: "xor",
523 |           maskComposite: "exclude",
524 |           boxShadow: "0 0 0 0.5px rgba(255, 255, 255, 0.5) inset, 0 1px 3px rgba(255, 255, 255, 0.25) inset, 0 1px 4px rgba(0, 0, 0, 0.35)",
525 |           background: `linear-gradient(
526 |           ${135 + mouseOffset.x * 1.2}deg,
527 |           rgba(255, 255, 255, 0.0) 0%,
528 |           rgba(255, 255, 255, ${0.12 + Math.abs(mouseOffset.x) * 0.008}) ${Math.max(10, 33 + mouseOffset.y * 0.3)}%,
529 |           rgba(255, 255, 255, ${0.4 + Math.abs(mouseOffset.x) * 0.012}) ${Math.min(90, 66 + mouseOffset.y * 0.4)}%,
530 |           rgba(255, 255, 255, 0.0) 100%
531 |         )`,
532 |         }}
533 |       />
534 | 
535 |       {/* Border layer 2 - duplicate with mix-blend-overlay */}
536 |       <span
537 |         style={{
538 |           ...positionStyles,
539 |           height: glassSize.height,
540 |           width: glassSize.width,
541 |           borderRadius: `${cornerRadius}px`,
542 |           transform: baseStyle.transform,
543 |           transition: baseStyle.transition,
544 |           pointerEvents: "none",
545 |           mixBlendMode: "overlay",
546 |           padding: "1.5px",
547 |           WebkitMask: "linear-gradient(#000 0 0) content-box, linear-gradient(#000 0 0)",
548 |           WebkitMaskComposite: "xor",
549 |           maskComposite: "exclude",
550 |           boxShadow: "0 0 0 0.5px rgba(255, 255, 255, 0.5) inset, 0 1px 3px rgba(255, 255, 255, 0.25) inset, 0 1px 4px rgba(0, 0, 0, 0.35)",
551 |           background: `linear-gradient(
552 |           ${135 + mouseOffset.x * 1.2}deg,
553 |           rgba(255, 255, 255, 0.0) 0%,
554 |           rgba(255, 255, 255, ${0.32 + Math.abs(mouseOffset.x) * 0.008}) ${Math.max(10, 33 + mouseOffset.y * 0.3)}%,
555 |           rgba(255, 255, 255, ${0.6 + Math.abs(mouseOffset.x) * 0.012}) ${Math.min(90, 66 + mouseOffset.y * 0.4)}%,
556 |           rgba(255, 255, 255, 0.0) 100%
557 |         )`,
558 |         }}
559 |       />
560 | 
561 |       {/* Hover effects */}
562 |       {Boolean(onClick) && (
563 |         <>
564 |           <div
565 |             style={{
566 |               ...positionStyles,
567 |               height: glassSize.height,
568 |               width: glassSize.width + 1,
569 |               borderRadius: `${cornerRadius}px`,
570 |               transform: baseStyle.transform,
571 |               pointerEvents: "none",
572 |               transition: "all 0.2s ease-out",
573 |               opacity: isHovered || isActive ? 0.5 : 0,
574 |               backgroundImage: "radial-gradient(circle at 50% 0%, rgba(255, 255, 255, 0.5) 0%, rgba(255, 255, 255, 0) 50%)",
575 |               mixBlendMode: "overlay",
576 |             }}
577 |           />
578 |           <div
579 |             style={{
580 |               ...positionStyles,
581 |               height: glassSize.height,
582 |               width: glassSize.width + 1,
583 |               borderRadius: `${cornerRadius}px`,
584 |               transform: baseStyle.transform,
585 |               pointerEvents: "none",
586 |               transition: "all 0.2s ease-out",
587 |               opacity: isActive ? 0.5 : 0,
588 |               backgroundImage: "radial-gradient(circle at 50% 0%, rgba(255, 255, 255, 1) 0%, rgba(255, 255, 255, 0) 80%)",
589 |               mixBlendMode: "overlay",
590 |             }}
591 |           />
592 |           <div
593 |             style={{
594 |               ...baseStyle,
595 |               height: glassSize.height,
596 |               width: glassSize.width + 1,
597 |               borderRadius: `${cornerRadius}px`,
598 |               position: baseStyle.position,
599 |               top: baseStyle.top,
600 |               left: baseStyle.left,
601 |               pointerEvents: "none",
602 |               transition: "all 0.2s ease-out",
603 |               opacity: isHovered ? 0.4 : isActive ? 0.8 : 0,
604 |               backgroundImage: "radial-gradient(circle at 50% 0%, rgba(255, 255, 255, 1) 0%, rgba(255, 255, 255, 0) 100%)",
605 |               mixBlendMode: "overlay",
606 |             }}
607 |           />
608 |         </>
609 |       )}
610 |     </>
611 |   )
612 | }
613 | 


--------------------------------------------------------------------------------
/src/shader-utils.ts:
--------------------------------------------------------------------------------
  1 | // Adapted from https://github.com/shuding/liquid-glass
  2 | 
  3 | export interface Vec2 {
  4 |   x: number
  5 |   y: number
  6 | }
  7 | 
  8 | export interface ShaderOptions {
  9 |   width: number
 10 |   height: number
 11 |   fragment: (uv: Vec2, mouse?: Vec2) => Vec2
 12 |   mousePosition?: Vec2
 13 | }
 14 | 
 15 | function smoothStep(a: number, b: number, t: number): number {
 16 |   t = Math.max(0, Math.min(1, (t - a) / (b - a)))
 17 |   return t * t * (3 - 2 * t)
 18 | }
 19 | 
 20 | function length(x: number, y: number): number {
 21 |   return Math.sqrt(x * x + y * y)
 22 | }
 23 | 
 24 | function roundedRectSDF(x: number, y: number, width: number, height: number, radius: number): number {
 25 |   const qx = Math.abs(x) - width + radius
 26 |   const qy = Math.abs(y) - height + radius
 27 |   return Math.min(Math.max(qx, qy), 0) + length(Math.max(qx, 0), Math.max(qy, 0)) - radius
 28 | }
 29 | 
 30 | function texture(x: number, y: number): Vec2 {
 31 |   return { x, y }
 32 | }
 33 | 
 34 | // Shader fragment functions for different effects
 35 | export const fragmentShaders = {
 36 |   liquidGlass: (uv: Vec2): Vec2 => {
 37 |     const ix = uv.x - 0.5
 38 |     const iy = uv.y - 0.5
 39 |     const distanceToEdge = roundedRectSDF(ix, iy, 0.3, 0.2, 0.6)
 40 |     const displacement = smoothStep(0.8, 0, distanceToEdge - 0.15)
 41 |     const scaled = smoothStep(0, 1, displacement)
 42 |     return texture(ix * scaled + 0.5, iy * scaled + 0.5)
 43 |   },
 44 | }
 45 | 
 46 | export type FragmentShaderType = keyof typeof fragmentShaders
 47 | 
 48 | export class ShaderDisplacementGenerator {
 49 |   private canvas: HTMLCanvasElement
 50 |   private context: CanvasRenderingContext2D
 51 |   private canvasDPI = 1
 52 | 
 53 |   constructor(private options: ShaderOptions) {
 54 |     this.canvas = document.createElement("canvas")
 55 |     this.canvas.width = options.width * this.canvasDPI
 56 |     this.canvas.height = options.height * this.canvasDPI
 57 |     this.canvas.style.display = "none"
 58 | 
 59 |     const context = this.canvas.getContext("2d")
 60 |     if (!context) {
 61 |       throw new Error("Could not get 2D context")
 62 |     }
 63 |     this.context = context
 64 |   }
 65 | 
 66 |   updateShader(mousePosition?: Vec2): string {
 67 |     const w = this.options.width * this.canvasDPI
 68 |     const h = this.options.height * this.canvasDPI
 69 | 
 70 |     let maxScale = 0
 71 |     const rawValues: number[] = []
 72 | 
 73 |     // Calculate displacement values
 74 |     for (let y = 0; y < h; y++) {
 75 |       for (let x = 0; x < w; x++) {
 76 |         const uv: Vec2 = { x: x / w, y: y / h }
 77 | 
 78 |         const pos = this.options.fragment(uv, mousePosition)
 79 |         const dx = pos.x * w - x
 80 |         const dy = pos.y * h - y
 81 | 
 82 |         maxScale = Math.max(maxScale, Math.abs(dx), Math.abs(dy))
 83 |         rawValues.push(dx, dy)
 84 |       }
 85 |     }
 86 | 
 87 |     // Improved normalization to prevent artifacts while maintaining intensity
 88 |     if (maxScale > 0) {
 89 |       maxScale = Math.max(maxScale, 1) // Ensure minimum scale to prevent over-normalization
 90 |     } else {
 91 |       maxScale = 1
 92 |     }
 93 | 
 94 |     // Create ImageData and fill it
 95 |     const imageData = this.context.createImageData(w, h)
 96 |     const data = imageData.data
 97 | 
 98 |     // Convert to image data with smoother normalization
 99 |     let rawIndex = 0
100 |     for (let y = 0; y < h; y++) {
101 |       for (let x = 0; x < w; x++) {
102 |         const dx = rawValues[rawIndex++]
103 |         const dy = rawValues[rawIndex++]
104 | 
105 |         // Smooth the displacement values at edges to prevent hard transitions
106 |         const edgeDistance = Math.min(x, y, w - x - 1, h - y - 1)
107 |         const edgeFactor = Math.min(1, edgeDistance / 2) // Smooth within 2 pixels of edge
108 | 
109 |         const smoothedDx = dx * edgeFactor
110 |         const smoothedDy = dy * edgeFactor
111 | 
112 |         const r = smoothedDx / maxScale + 0.5
113 |         const g = smoothedDy / maxScale + 0.5
114 | 
115 |         const pixelIndex = (y * w + x) * 4
116 |         data[pixelIndex] = Math.max(0, Math.min(255, r * 255)) // Red channel (X displacement)
117 |         data[pixelIndex + 1] = Math.max(0, Math.min(255, g * 255)) // Green channel (Y displacement)
118 |         data[pixelIndex + 2] = Math.max(0, Math.min(255, g * 255)) // Blue channel (Y displacement for SVG filter compatibility)
119 |         data[pixelIndex + 3] = 255 // Alpha channel
120 |       }
121 |     }
122 | 
123 |     this.context.putImageData(imageData, 0, 0)
124 |     return this.canvas.toDataURL()
125 |   }
126 | 
127 |   destroy(): void {
128 |     this.canvas.remove()
129 |   }
130 | 
131 |   getScale(): number {
132 |     return this.canvasDPI
133 |   }
134 | }
135 | 


--------------------------------------------------------------------------------
/src/utils.ts:
--------------------------------------------------------------------------------
1 | export const displacementMap =
2 |   ""
3 | 
4 | export const polarDisplacementMap =
5 |   ""
6 | 
7 | export const prominentDisplacementMap =
8 |   ""
9 | 


--------------------------------------------------------------------------------
/tsconfig.json:
--------------------------------------------------------------------------------
 1 | {
 2 |   "compilerOptions": {
 3 |     "target": "ES2020",
 4 |     "lib": ["DOM", "DOM.Iterable", "ES6"],
 5 |     "allowJs": true,
 6 |     "skipLibCheck": true,
 7 |     "esModuleInterop": true,
 8 |     "allowSyntheticDefaultImports": true,
 9 |     "strict": true,
10 |     "forceConsistentCasingInFileNames": true,
11 |     "noFallthroughCasesInSwitch": true,
12 |     "module": "ESNext",
13 |     "moduleResolution": "node",
14 |     "resolveJsonModule": true,
15 |     "isolatedModules": true,
16 |     "noEmit": false,
17 |     "declaration": true,
18 |     "declarationMap": true,
19 |     "outDir": "dist",
20 |     "jsx": "react-jsx"
21 |   },
22 |   "include": [
23 |     "src"
24 |   ],
25 |   "exclude": [
26 |     "node_modules",
27 |     "dist"
28 |   ]
29 | }


--------------------------------------------------------------------------------