├── .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 |  |  8 | 9 | ## 🎬 Demo 10 | 11 | [Click here](https://liquid-glass.maxrovensky.com) to see it in action! 12 | 13 |  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 | } --------------------------------------------------------------------------------