├── .gitignore ├── .prettierrc ├── README.md ├── package-lock.json ├── package.json ├── postcss.config.cjs ├── public ├── meta │ └── vite.svg └── resources │ ├── bunny.glb │ └── matcap_1k.jpg ├── src ├── index.html ├── index.scss ├── index.ts ├── scripts │ └── utils.ts ├── styles │ ├── base.scss │ └── mixins │ │ └── media.scss ├── types │ ├── glsl.d.ts │ └── vite-env.d.ts └── webgl │ ├── Particles.ts │ ├── TCanvas.ts │ ├── core │ └── WebGL.ts │ ├── glsl │ └── matcap.glsl │ ├── shaders │ ├── fragment.glsl │ └── vertex.glsl │ └── utils │ ├── Mouse2D.ts │ ├── OrbitControls.ts │ ├── assetLoader.ts │ └── coveredTexture.ts ├── tsconfig.json └── vite.config.ts /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | pnpm-debug.log* 8 | lerna-debug.log* 9 | 10 | node_modules 11 | dist 12 | dist-ssr 13 | *.local 14 | 15 | # Editor directories and files 16 | .vscode/* 17 | !.vscode/extensions.json 18 | .idea 19 | .DS_Store 20 | *.suo 21 | *.ntvs* 22 | *.njsproj 23 | *.sln 24 | *.sw? 25 | 26 | -------------------------------------------------------------------------------- /.prettierrc: -------------------------------------------------------------------------------- 1 | { 2 | "tabWidth": 2, 3 | "useTabs": false, 4 | "semi": false, 5 | "trailingComma": "all", 6 | "singleQuote": true, 7 | "printWidth": 120 8 | } 9 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # reflect-particles 2 | BVH (Bounding Volume Hierarchy) is used to reflect particles in the mesh. 3 | 4 | https://nemutas.github.io/reflect-particles/ 5 | 6 | 7 | 8 | # reference 9 | 10 | - [webgl_buffergeometry_drawrange](https://threejs.org/examples/#webgl_buffergeometry_drawrange) 11 | - [three-mesh-bvh](https://github.com/gkjohnson/three-mesh-bvh) 12 | - model > [6 Test Objects (Quad-Remeshed)](https://sketchfab.com/3d-models/6-test-objects-quad-remeshed-40bb31eb8ec241b0944be141d6d3dd88) 13 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "reflect-particles", 3 | "version": "0.0.0", 4 | "lockfileVersion": 2, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "reflect-particles", 9 | "version": "0.0.0", 10 | "dependencies": { 11 | "gsap": "^3.11.3", 12 | "three": "^0.147.0", 13 | "three-mesh-bvh": "^0.5.19" 14 | }, 15 | "devDependencies": { 16 | "@types/node": "^18.11.5", 17 | "@types/three": "^0.146.0", 18 | "autoprefixer": "^10.4.12", 19 | "gh-pages": "^4.0.0", 20 | "prettier": "^2.8.0", 21 | "ress": "^5.0.2", 22 | "sass": "^1.54.5", 23 | "typescript": "^4.6.4", 24 | "vite": "^3.2.5", 25 | "vite-plugin-glsl": "^0.5.1" 26 | } 27 | }, 28 | "node_modules/@esbuild/android-arm": { 29 | "version": "0.15.12", 30 | "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.15.12.tgz", 31 | "integrity": "sha512-IC7TqIqiyE0MmvAhWkl/8AEzpOtbhRNDo7aph47We1NbE5w2bt/Q+giAhe0YYeVpYnIhGMcuZY92qDK6dQauvA==", 32 | "cpu": [ 33 | "arm" 34 | ], 35 | "dev": true, 36 | "optional": true, 37 | "os": [ 38 | "android" 39 | ], 40 | "engines": { 41 | "node": ">=12" 42 | } 43 | }, 44 | "node_modules/@esbuild/linux-loong64": { 45 | "version": "0.15.12", 46 | "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.15.12.tgz", 47 | "integrity": "sha512-tZEowDjvU7O7I04GYvWQOS4yyP9E/7YlsB0jjw1Ycukgr2ycEzKyIk5tms5WnLBymaewc6VmRKnn5IJWgK4eFw==", 48 | "cpu": [ 49 | "loong64" 50 | ], 51 | "dev": true, 52 | "optional": true, 53 | "os": [ 54 | "linux" 55 | ], 56 | "engines": { 57 | "node": ">=12" 58 | } 59 | }, 60 | "node_modules/@rollup/pluginutils": { 61 | "version": "4.2.1", 62 | "resolved": "https://registry.npmjs.org/@rollup/pluginutils/-/pluginutils-4.2.1.tgz", 63 | "integrity": "sha512-iKnFXr7NkdZAIHiIWE+BX5ULi/ucVFYWD6TbAV+rZctiRTY2PL6tsIKhoIOaoskiWAkgu+VsbXgUVDNLHf+InQ==", 64 | "dev": true, 65 | "dependencies": { 66 | "estree-walker": "^2.0.1", 67 | "picomatch": "^2.2.2" 68 | }, 69 | "engines": { 70 | "node": ">= 8.0.0" 71 | } 72 | }, 73 | "node_modules/@types/node": { 74 | "version": "18.11.5", 75 | "resolved": "https://registry.npmjs.org/@types/node/-/node-18.11.5.tgz", 76 | "integrity": "sha512-3JRwhbjI+cHLAkUorhf8RnqUbFXajvzX4q6fMn5JwkgtuwfYtRQYI3u4V92vI6NJuTsbBQWWh3RZjFsuevyMGQ==", 77 | "dev": true 78 | }, 79 | "node_modules/@types/three": { 80 | "version": "0.146.0", 81 | "resolved": "https://registry.npmjs.org/@types/three/-/three-0.146.0.tgz", 82 | "integrity": "sha512-75AgysUrIvTCB054eQa2pDVFurfeFW8CrMQjpzjt3yHBfuuknoSvvsESd/3EhQxPrz9si3+P0wiDUVsWUlljfA==", 83 | "dev": true, 84 | "dependencies": { 85 | "@types/webxr": "*" 86 | } 87 | }, 88 | "node_modules/@types/webxr": { 89 | "version": "0.5.0", 90 | "resolved": "https://registry.npmjs.org/@types/webxr/-/webxr-0.5.0.tgz", 91 | "integrity": "sha512-IUMDPSXnYIbEO2IereEFcgcqfDREOgmbGqtrMpVPpACTU6pltYLwHgVkrnYv0XhWEcjio9sYEfIEzgn3c7nDqA==", 92 | "dev": true 93 | }, 94 | "node_modules/anymatch": { 95 | "version": "3.1.2", 96 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", 97 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", 98 | "dev": true, 99 | "dependencies": { 100 | "normalize-path": "^3.0.0", 101 | "picomatch": "^2.0.4" 102 | }, 103 | "engines": { 104 | "node": ">= 8" 105 | } 106 | }, 107 | "node_modules/array-union": { 108 | "version": "1.0.2", 109 | "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", 110 | "integrity": "sha512-Dxr6QJj/RdU/hCaBjOfxW+q6lyuVE6JFWIrAUpuOOhoJJoQ99cUn3igRaHVB5P9WrgFVN0FfArM3x0cueOU8ng==", 111 | "dev": true, 112 | "dependencies": { 113 | "array-uniq": "^1.0.1" 114 | }, 115 | "engines": { 116 | "node": ">=0.10.0" 117 | } 118 | }, 119 | "node_modules/array-uniq": { 120 | "version": "1.0.3", 121 | "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz", 122 | "integrity": "sha512-MNha4BWQ6JbwhFhj03YK552f7cb3AzoE8SzeljgChvL1dl3IcvggXVz1DilzySZkCja+CXuZbdW7yATchWn8/Q==", 123 | "dev": true, 124 | "engines": { 125 | "node": ">=0.10.0" 126 | } 127 | }, 128 | "node_modules/async": { 129 | "version": "2.6.4", 130 | "resolved": "https://registry.npmjs.org/async/-/async-2.6.4.tgz", 131 | "integrity": "sha512-mzo5dfJYwAn29PeiJ0zvwTo04zj8HDJj0Mn8TD7sno7q12prdbnasKJHhkm2c1LgrhlJ0teaea8860oxi51mGA==", 132 | "dev": true, 133 | "dependencies": { 134 | "lodash": "^4.17.14" 135 | } 136 | }, 137 | "node_modules/autoprefixer": { 138 | "version": "10.4.12", 139 | "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-10.4.12.tgz", 140 | "integrity": "sha512-WrCGV9/b97Pa+jtwf5UGaRjgQIg7OK3D06GnoYoZNcG1Xb8Gt3EfuKjlhh9i/VtT16g6PYjZ69jdJ2g8FxSC4Q==", 141 | "dev": true, 142 | "funding": [ 143 | { 144 | "type": "opencollective", 145 | "url": "https://opencollective.com/postcss/" 146 | }, 147 | { 148 | "type": "tidelift", 149 | "url": "https://tidelift.com/funding/github/npm/autoprefixer" 150 | } 151 | ], 152 | "dependencies": { 153 | "browserslist": "^4.21.4", 154 | "caniuse-lite": "^1.0.30001407", 155 | "fraction.js": "^4.2.0", 156 | "normalize-range": "^0.1.2", 157 | "picocolors": "^1.0.0", 158 | "postcss-value-parser": "^4.2.0" 159 | }, 160 | "bin": { 161 | "autoprefixer": "bin/autoprefixer" 162 | }, 163 | "engines": { 164 | "node": "^10 || ^12 || >=14" 165 | }, 166 | "peerDependencies": { 167 | "postcss": "^8.1.0" 168 | } 169 | }, 170 | "node_modules/balanced-match": { 171 | "version": "1.0.2", 172 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 173 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", 174 | "dev": true 175 | }, 176 | "node_modules/binary-extensions": { 177 | "version": "2.2.0", 178 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", 179 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", 180 | "dev": true, 181 | "engines": { 182 | "node": ">=8" 183 | } 184 | }, 185 | "node_modules/brace-expansion": { 186 | "version": "1.1.11", 187 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 188 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 189 | "dev": true, 190 | "dependencies": { 191 | "balanced-match": "^1.0.0", 192 | "concat-map": "0.0.1" 193 | } 194 | }, 195 | "node_modules/braces": { 196 | "version": "3.0.2", 197 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 198 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 199 | "dev": true, 200 | "dependencies": { 201 | "fill-range": "^7.0.1" 202 | }, 203 | "engines": { 204 | "node": ">=8" 205 | } 206 | }, 207 | "node_modules/browserslist": { 208 | "version": "4.21.4", 209 | "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.21.4.tgz", 210 | "integrity": "sha512-CBHJJdDmgjl3daYjN5Cp5kbTf1mUhZoS+beLklHIvkOWscs83YAhLlF3Wsh/lciQYAcbBJgTOD44VtG31ZM4Hw==", 211 | "dev": true, 212 | "funding": [ 213 | { 214 | "type": "opencollective", 215 | "url": "https://opencollective.com/browserslist" 216 | }, 217 | { 218 | "type": "tidelift", 219 | "url": "https://tidelift.com/funding/github/npm/browserslist" 220 | } 221 | ], 222 | "dependencies": { 223 | "caniuse-lite": "^1.0.30001400", 224 | "electron-to-chromium": "^1.4.251", 225 | "node-releases": "^2.0.6", 226 | "update-browserslist-db": "^1.0.9" 227 | }, 228 | "bin": { 229 | "browserslist": "cli.js" 230 | }, 231 | "engines": { 232 | "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" 233 | } 234 | }, 235 | "node_modules/caniuse-lite": { 236 | "version": "1.0.30001418", 237 | "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001418.tgz", 238 | "integrity": "sha512-oIs7+JL3K9JRQ3jPZjlH6qyYDp+nBTCais7hjh0s+fuBwufc7uZ7hPYMXrDOJhV360KGMTcczMRObk0/iMqZRg==", 239 | "dev": true, 240 | "funding": [ 241 | { 242 | "type": "opencollective", 243 | "url": "https://opencollective.com/browserslist" 244 | }, 245 | { 246 | "type": "tidelift", 247 | "url": "https://tidelift.com/funding/github/npm/caniuse-lite" 248 | } 249 | ] 250 | }, 251 | "node_modules/chokidar": { 252 | "version": "3.5.3", 253 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz", 254 | "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", 255 | "dev": true, 256 | "funding": [ 257 | { 258 | "type": "individual", 259 | "url": "https://paulmillr.com/funding/" 260 | } 261 | ], 262 | "dependencies": { 263 | "anymatch": "~3.1.2", 264 | "braces": "~3.0.2", 265 | "glob-parent": "~5.1.2", 266 | "is-binary-path": "~2.1.0", 267 | "is-glob": "~4.0.1", 268 | "normalize-path": "~3.0.0", 269 | "readdirp": "~3.6.0" 270 | }, 271 | "engines": { 272 | "node": ">= 8.10.0" 273 | }, 274 | "optionalDependencies": { 275 | "fsevents": "~2.3.2" 276 | } 277 | }, 278 | "node_modules/commander": { 279 | "version": "2.20.3", 280 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", 281 | "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", 282 | "dev": true 283 | }, 284 | "node_modules/commondir": { 285 | "version": "1.0.1", 286 | "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", 287 | "integrity": "sha512-W9pAhw0ja1Edb5GVdIF1mjZw/ASI0AlShXM83UUGe2DVr5TdAPEA1OA8m/g8zWp9x6On7gqufY+FatDbC3MDQg==", 288 | "dev": true 289 | }, 290 | "node_modules/concat-map": { 291 | "version": "0.0.1", 292 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 293 | "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", 294 | "dev": true 295 | }, 296 | "node_modules/electron-to-chromium": { 297 | "version": "1.4.276", 298 | "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.276.tgz", 299 | "integrity": "sha512-EpuHPqu8YhonqLBXHoU6hDJCD98FCe6KDoet3/gY1qsQ6usjJoHqBH2YIVs8FXaAtHwVL8Uqa/fsYao/vq9VWQ==", 300 | "dev": true 301 | }, 302 | "node_modules/email-addresses": { 303 | "version": "3.1.0", 304 | "resolved": "https://registry.npmjs.org/email-addresses/-/email-addresses-3.1.0.tgz", 305 | "integrity": "sha512-k0/r7GrWVL32kZlGwfPNgB2Y/mMXVTq/decgLczm/j34whdaspNrZO8CnXPf1laaHxI6ptUlsnAxN+UAPw+fzg==", 306 | "dev": true 307 | }, 308 | "node_modules/esbuild": { 309 | "version": "0.15.12", 310 | "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.15.12.tgz", 311 | "integrity": "sha512-PcT+/wyDqJQsRVhaE9uX/Oq4XLrFh0ce/bs2TJh4CSaw9xuvI+xFrH2nAYOADbhQjUgAhNWC5LKoUsakm4dxng==", 312 | "dev": true, 313 | "hasInstallScript": true, 314 | "bin": { 315 | "esbuild": "bin/esbuild" 316 | }, 317 | "engines": { 318 | "node": ">=12" 319 | }, 320 | "optionalDependencies": { 321 | "@esbuild/android-arm": "0.15.12", 322 | "@esbuild/linux-loong64": "0.15.12", 323 | "esbuild-android-64": "0.15.12", 324 | "esbuild-android-arm64": "0.15.12", 325 | "esbuild-darwin-64": "0.15.12", 326 | "esbuild-darwin-arm64": "0.15.12", 327 | "esbuild-freebsd-64": "0.15.12", 328 | "esbuild-freebsd-arm64": "0.15.12", 329 | "esbuild-linux-32": "0.15.12", 330 | "esbuild-linux-64": "0.15.12", 331 | "esbuild-linux-arm": "0.15.12", 332 | "esbuild-linux-arm64": "0.15.12", 333 | "esbuild-linux-mips64le": "0.15.12", 334 | "esbuild-linux-ppc64le": "0.15.12", 335 | "esbuild-linux-riscv64": "0.15.12", 336 | "esbuild-linux-s390x": "0.15.12", 337 | "esbuild-netbsd-64": "0.15.12", 338 | "esbuild-openbsd-64": "0.15.12", 339 | "esbuild-sunos-64": "0.15.12", 340 | "esbuild-windows-32": "0.15.12", 341 | "esbuild-windows-64": "0.15.12", 342 | "esbuild-windows-arm64": "0.15.12" 343 | } 344 | }, 345 | "node_modules/esbuild-android-64": { 346 | "version": "0.15.12", 347 | "resolved": "https://registry.npmjs.org/esbuild-android-64/-/esbuild-android-64-0.15.12.tgz", 348 | "integrity": "sha512-MJKXwvPY9g0rGps0+U65HlTsM1wUs9lbjt5CU19RESqycGFDRijMDQsh68MtbzkqWSRdEtiKS1mtPzKneaAI0Q==", 349 | "cpu": [ 350 | "x64" 351 | ], 352 | "dev": true, 353 | "optional": true, 354 | "os": [ 355 | "android" 356 | ], 357 | "engines": { 358 | "node": ">=12" 359 | } 360 | }, 361 | "node_modules/esbuild-android-arm64": { 362 | "version": "0.15.12", 363 | "resolved": "https://registry.npmjs.org/esbuild-android-arm64/-/esbuild-android-arm64-0.15.12.tgz", 364 | "integrity": "sha512-Hc9SEcZbIMhhLcvhr1DH+lrrec9SFTiRzfJ7EGSBZiiw994gfkVV6vG0sLWqQQ6DD7V4+OggB+Hn0IRUdDUqvA==", 365 | "cpu": [ 366 | "arm64" 367 | ], 368 | "dev": true, 369 | "optional": true, 370 | "os": [ 371 | "android" 372 | ], 373 | "engines": { 374 | "node": ">=12" 375 | } 376 | }, 377 | "node_modules/esbuild-darwin-64": { 378 | "version": "0.15.12", 379 | "resolved": "https://registry.npmjs.org/esbuild-darwin-64/-/esbuild-darwin-64-0.15.12.tgz", 380 | "integrity": "sha512-qkmqrTVYPFiePt5qFjP8w/S+GIUMbt6k8qmiPraECUWfPptaPJUGkCKrWEfYFRWB7bY23FV95rhvPyh/KARP8Q==", 381 | "cpu": [ 382 | "x64" 383 | ], 384 | "dev": true, 385 | "optional": true, 386 | "os": [ 387 | "darwin" 388 | ], 389 | "engines": { 390 | "node": ">=12" 391 | } 392 | }, 393 | "node_modules/esbuild-darwin-arm64": { 394 | "version": "0.15.12", 395 | "resolved": "https://registry.npmjs.org/esbuild-darwin-arm64/-/esbuild-darwin-arm64-0.15.12.tgz", 396 | "integrity": "sha512-z4zPX02tQ41kcXMyN3c/GfZpIjKoI/BzHrdKUwhC/Ki5BAhWv59A9M8H+iqaRbwpzYrYidTybBwiZAIWCLJAkw==", 397 | "cpu": [ 398 | "arm64" 399 | ], 400 | "dev": true, 401 | "optional": true, 402 | "os": [ 403 | "darwin" 404 | ], 405 | "engines": { 406 | "node": ">=12" 407 | } 408 | }, 409 | "node_modules/esbuild-freebsd-64": { 410 | "version": "0.15.12", 411 | "resolved": "https://registry.npmjs.org/esbuild-freebsd-64/-/esbuild-freebsd-64-0.15.12.tgz", 412 | "integrity": "sha512-XFL7gKMCKXLDiAiBjhLG0XECliXaRLTZh6hsyzqUqPUf/PY4C6EJDTKIeqqPKXaVJ8+fzNek88285krSz1QECw==", 413 | "cpu": [ 414 | "x64" 415 | ], 416 | "dev": true, 417 | "optional": true, 418 | "os": [ 419 | "freebsd" 420 | ], 421 | "engines": { 422 | "node": ">=12" 423 | } 424 | }, 425 | "node_modules/esbuild-freebsd-arm64": { 426 | "version": "0.15.12", 427 | "resolved": "https://registry.npmjs.org/esbuild-freebsd-arm64/-/esbuild-freebsd-arm64-0.15.12.tgz", 428 | "integrity": "sha512-jwEIu5UCUk6TjiG1X+KQnCGISI+ILnXzIzt9yDVrhjug2fkYzlLbl0K43q96Q3KB66v6N1UFF0r5Ks4Xo7i72g==", 429 | "cpu": [ 430 | "arm64" 431 | ], 432 | "dev": true, 433 | "optional": true, 434 | "os": [ 435 | "freebsd" 436 | ], 437 | "engines": { 438 | "node": ">=12" 439 | } 440 | }, 441 | "node_modules/esbuild-linux-32": { 442 | "version": "0.15.12", 443 | "resolved": "https://registry.npmjs.org/esbuild-linux-32/-/esbuild-linux-32-0.15.12.tgz", 444 | "integrity": "sha512-uSQuSEyF1kVzGzuIr4XM+v7TPKxHjBnLcwv2yPyCz8riV8VUCnO/C4BF3w5dHiVpCd5Z1cebBtZJNlC4anWpwA==", 445 | "cpu": [ 446 | "ia32" 447 | ], 448 | "dev": true, 449 | "optional": true, 450 | "os": [ 451 | "linux" 452 | ], 453 | "engines": { 454 | "node": ">=12" 455 | } 456 | }, 457 | "node_modules/esbuild-linux-64": { 458 | "version": "0.15.12", 459 | "resolved": "https://registry.npmjs.org/esbuild-linux-64/-/esbuild-linux-64-0.15.12.tgz", 460 | "integrity": "sha512-QcgCKb7zfJxqT9o5z9ZUeGH1k8N6iX1Y7VNsEi5F9+HzN1OIx7ESxtQXDN9jbeUSPiRH1n9cw6gFT3H4qbdvcA==", 461 | "cpu": [ 462 | "x64" 463 | ], 464 | "dev": true, 465 | "optional": true, 466 | "os": [ 467 | "linux" 468 | ], 469 | "engines": { 470 | "node": ">=12" 471 | } 472 | }, 473 | "node_modules/esbuild-linux-arm": { 474 | "version": "0.15.12", 475 | "resolved": "https://registry.npmjs.org/esbuild-linux-arm/-/esbuild-linux-arm-0.15.12.tgz", 476 | "integrity": "sha512-Wf7T0aNylGcLu7hBnzMvsTfEXdEdJY/hY3u36Vla21aY66xR0MS5I1Hw8nVquXjTN0A6fk/vnr32tkC/C2lb0A==", 477 | "cpu": [ 478 | "arm" 479 | ], 480 | "dev": true, 481 | "optional": true, 482 | "os": [ 483 | "linux" 484 | ], 485 | "engines": { 486 | "node": ">=12" 487 | } 488 | }, 489 | "node_modules/esbuild-linux-arm64": { 490 | "version": "0.15.12", 491 | "resolved": "https://registry.npmjs.org/esbuild-linux-arm64/-/esbuild-linux-arm64-0.15.12.tgz", 492 | "integrity": "sha512-HtNq5xm8fUpZKwWKS2/YGwSfTF+339L4aIA8yphNKYJckd5hVdhfdl6GM2P3HwLSCORS++++7++//ApEwXEuAQ==", 493 | "cpu": [ 494 | "arm64" 495 | ], 496 | "dev": true, 497 | "optional": true, 498 | "os": [ 499 | "linux" 500 | ], 501 | "engines": { 502 | "node": ">=12" 503 | } 504 | }, 505 | "node_modules/esbuild-linux-mips64le": { 506 | "version": "0.15.12", 507 | "resolved": "https://registry.npmjs.org/esbuild-linux-mips64le/-/esbuild-linux-mips64le-0.15.12.tgz", 508 | "integrity": "sha512-Qol3+AvivngUZkTVFgLpb0H6DT+N5/zM3V1YgTkryPYFeUvuT5JFNDR3ZiS6LxhyF8EE+fiNtzwlPqMDqVcc6A==", 509 | "cpu": [ 510 | "mips64el" 511 | ], 512 | "dev": true, 513 | "optional": true, 514 | "os": [ 515 | "linux" 516 | ], 517 | "engines": { 518 | "node": ">=12" 519 | } 520 | }, 521 | "node_modules/esbuild-linux-ppc64le": { 522 | "version": "0.15.12", 523 | "resolved": "https://registry.npmjs.org/esbuild-linux-ppc64le/-/esbuild-linux-ppc64le-0.15.12.tgz", 524 | "integrity": "sha512-4D8qUCo+CFKaR0cGXtGyVsOI7w7k93Qxb3KFXWr75An0DHamYzq8lt7TNZKoOq/Gh8c40/aKaxvcZnTgQ0TJNg==", 525 | "cpu": [ 526 | "ppc64" 527 | ], 528 | "dev": true, 529 | "optional": true, 530 | "os": [ 531 | "linux" 532 | ], 533 | "engines": { 534 | "node": ">=12" 535 | } 536 | }, 537 | "node_modules/esbuild-linux-riscv64": { 538 | "version": "0.15.12", 539 | "resolved": "https://registry.npmjs.org/esbuild-linux-riscv64/-/esbuild-linux-riscv64-0.15.12.tgz", 540 | "integrity": "sha512-G9w6NcuuCI6TUUxe6ka0enjZHDnSVK8bO+1qDhMOCtl7Tr78CcZilJj8SGLN00zO5iIlwNRZKHjdMpfFgNn1VA==", 541 | "cpu": [ 542 | "riscv64" 543 | ], 544 | "dev": true, 545 | "optional": true, 546 | "os": [ 547 | "linux" 548 | ], 549 | "engines": { 550 | "node": ">=12" 551 | } 552 | }, 553 | "node_modules/esbuild-linux-s390x": { 554 | "version": "0.15.12", 555 | "resolved": "https://registry.npmjs.org/esbuild-linux-s390x/-/esbuild-linux-s390x-0.15.12.tgz", 556 | "integrity": "sha512-Lt6BDnuXbXeqSlVuuUM5z18GkJAZf3ERskGZbAWjrQoi9xbEIsj/hEzVnSAFLtkfLuy2DE4RwTcX02tZFunXww==", 557 | "cpu": [ 558 | "s390x" 559 | ], 560 | "dev": true, 561 | "optional": true, 562 | "os": [ 563 | "linux" 564 | ], 565 | "engines": { 566 | "node": ">=12" 567 | } 568 | }, 569 | "node_modules/esbuild-netbsd-64": { 570 | "version": "0.15.12", 571 | "resolved": "https://registry.npmjs.org/esbuild-netbsd-64/-/esbuild-netbsd-64-0.15.12.tgz", 572 | "integrity": "sha512-jlUxCiHO1dsqoURZDQts+HK100o0hXfi4t54MNRMCAqKGAV33JCVvMplLAa2FwviSojT/5ZG5HUfG3gstwAG8w==", 573 | "cpu": [ 574 | "x64" 575 | ], 576 | "dev": true, 577 | "optional": true, 578 | "os": [ 579 | "netbsd" 580 | ], 581 | "engines": { 582 | "node": ">=12" 583 | } 584 | }, 585 | "node_modules/esbuild-openbsd-64": { 586 | "version": "0.15.12", 587 | "resolved": "https://registry.npmjs.org/esbuild-openbsd-64/-/esbuild-openbsd-64-0.15.12.tgz", 588 | "integrity": "sha512-1o1uAfRTMIWNOmpf8v7iudND0L6zRBYSH45sofCZywrcf7NcZA+c7aFsS1YryU+yN7aRppTqdUK1PgbZVaB1Dw==", 589 | "cpu": [ 590 | "x64" 591 | ], 592 | "dev": true, 593 | "optional": true, 594 | "os": [ 595 | "openbsd" 596 | ], 597 | "engines": { 598 | "node": ">=12" 599 | } 600 | }, 601 | "node_modules/esbuild-sunos-64": { 602 | "version": "0.15.12", 603 | "resolved": "https://registry.npmjs.org/esbuild-sunos-64/-/esbuild-sunos-64-0.15.12.tgz", 604 | "integrity": "sha512-nkl251DpoWoBO9Eq9aFdoIt2yYmp4I3kvQjba3jFKlMXuqQ9A4q+JaqdkCouG3DHgAGnzshzaGu6xofGcXyPXg==", 605 | "cpu": [ 606 | "x64" 607 | ], 608 | "dev": true, 609 | "optional": true, 610 | "os": [ 611 | "sunos" 612 | ], 613 | "engines": { 614 | "node": ">=12" 615 | } 616 | }, 617 | "node_modules/esbuild-windows-32": { 618 | "version": "0.15.12", 619 | "resolved": "https://registry.npmjs.org/esbuild-windows-32/-/esbuild-windows-32-0.15.12.tgz", 620 | "integrity": "sha512-WlGeBZHgPC00O08luIp5B2SP4cNCp/PcS+3Pcg31kdcJPopHxLkdCXtadLU9J82LCfw4TVls21A6lilQ9mzHrw==", 621 | "cpu": [ 622 | "ia32" 623 | ], 624 | "dev": true, 625 | "optional": true, 626 | "os": [ 627 | "win32" 628 | ], 629 | "engines": { 630 | "node": ">=12" 631 | } 632 | }, 633 | "node_modules/esbuild-windows-64": { 634 | "version": "0.15.12", 635 | "resolved": "https://registry.npmjs.org/esbuild-windows-64/-/esbuild-windows-64-0.15.12.tgz", 636 | "integrity": "sha512-VActO3WnWZSN//xjSfbiGOSyC+wkZtI8I4KlgrTo5oHJM6z3MZZBCuFaZHd8hzf/W9KPhF0lY8OqlmWC9HO5AA==", 637 | "cpu": [ 638 | "x64" 639 | ], 640 | "dev": true, 641 | "optional": true, 642 | "os": [ 643 | "win32" 644 | ], 645 | "engines": { 646 | "node": ">=12" 647 | } 648 | }, 649 | "node_modules/esbuild-windows-arm64": { 650 | "version": "0.15.12", 651 | "resolved": "https://registry.npmjs.org/esbuild-windows-arm64/-/esbuild-windows-arm64-0.15.12.tgz", 652 | "integrity": "sha512-Of3MIacva1OK/m4zCNIvBfz8VVROBmQT+gRX6pFTLPngFYcj6TFH/12VveAqq1k9VB2l28EoVMNMUCcmsfwyuA==", 653 | "cpu": [ 654 | "arm64" 655 | ], 656 | "dev": true, 657 | "optional": true, 658 | "os": [ 659 | "win32" 660 | ], 661 | "engines": { 662 | "node": ">=12" 663 | } 664 | }, 665 | "node_modules/escalade": { 666 | "version": "3.1.1", 667 | "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", 668 | "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", 669 | "dev": true, 670 | "engines": { 671 | "node": ">=6" 672 | } 673 | }, 674 | "node_modules/escape-string-regexp": { 675 | "version": "1.0.5", 676 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 677 | "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", 678 | "dev": true, 679 | "engines": { 680 | "node": ">=0.8.0" 681 | } 682 | }, 683 | "node_modules/estree-walker": { 684 | "version": "2.0.2", 685 | "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz", 686 | "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==", 687 | "dev": true 688 | }, 689 | "node_modules/filename-reserved-regex": { 690 | "version": "2.0.0", 691 | "resolved": "https://registry.npmjs.org/filename-reserved-regex/-/filename-reserved-regex-2.0.0.tgz", 692 | "integrity": "sha512-lc1bnsSr4L4Bdif8Xb/qrtokGbq5zlsms/CYH8PP+WtCkGNF65DPiQY8vG3SakEdRn8Dlnm+gW/qWKKjS5sZzQ==", 693 | "dev": true, 694 | "engines": { 695 | "node": ">=4" 696 | } 697 | }, 698 | "node_modules/filenamify": { 699 | "version": "4.3.0", 700 | "resolved": "https://registry.npmjs.org/filenamify/-/filenamify-4.3.0.tgz", 701 | "integrity": "sha512-hcFKyUG57yWGAzu1CMt/dPzYZuv+jAJUT85bL8mrXvNe6hWj6yEHEc4EdcgiA6Z3oi1/9wXJdZPXF2dZNgwgOg==", 702 | "dev": true, 703 | "dependencies": { 704 | "filename-reserved-regex": "^2.0.0", 705 | "strip-outer": "^1.0.1", 706 | "trim-repeated": "^1.0.0" 707 | }, 708 | "engines": { 709 | "node": ">=8" 710 | }, 711 | "funding": { 712 | "url": "https://github.com/sponsors/sindresorhus" 713 | } 714 | }, 715 | "node_modules/fill-range": { 716 | "version": "7.0.1", 717 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 718 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 719 | "dev": true, 720 | "dependencies": { 721 | "to-regex-range": "^5.0.1" 722 | }, 723 | "engines": { 724 | "node": ">=8" 725 | } 726 | }, 727 | "node_modules/find-cache-dir": { 728 | "version": "3.3.2", 729 | "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-3.3.2.tgz", 730 | "integrity": "sha512-wXZV5emFEjrridIgED11OoUKLxiYjAcqot/NJdAkOhlJ+vGzwhOAfcG5OX1jP+S0PcjEn8bdMJv+g2jwQ3Onig==", 731 | "dev": true, 732 | "dependencies": { 733 | "commondir": "^1.0.1", 734 | "make-dir": "^3.0.2", 735 | "pkg-dir": "^4.1.0" 736 | }, 737 | "engines": { 738 | "node": ">=8" 739 | }, 740 | "funding": { 741 | "url": "https://github.com/avajs/find-cache-dir?sponsor=1" 742 | } 743 | }, 744 | "node_modules/find-up": { 745 | "version": "4.1.0", 746 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", 747 | "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", 748 | "dev": true, 749 | "dependencies": { 750 | "locate-path": "^5.0.0", 751 | "path-exists": "^4.0.0" 752 | }, 753 | "engines": { 754 | "node": ">=8" 755 | } 756 | }, 757 | "node_modules/fraction.js": { 758 | "version": "4.2.0", 759 | "resolved": "https://registry.npmjs.org/fraction.js/-/fraction.js-4.2.0.tgz", 760 | "integrity": "sha512-MhLuK+2gUcnZe8ZHlaaINnQLl0xRIGRfcGk2yl8xoQAfHrSsL3rYu6FCmBdkdbhc9EPlwyGHewaRsvwRMJtAlA==", 761 | "dev": true, 762 | "engines": { 763 | "node": "*" 764 | }, 765 | "funding": { 766 | "type": "patreon", 767 | "url": "https://www.patreon.com/infusion" 768 | } 769 | }, 770 | "node_modules/fs-extra": { 771 | "version": "8.1.0", 772 | "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", 773 | "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", 774 | "dev": true, 775 | "dependencies": { 776 | "graceful-fs": "^4.2.0", 777 | "jsonfile": "^4.0.0", 778 | "universalify": "^0.1.0" 779 | }, 780 | "engines": { 781 | "node": ">=6 <7 || >=8" 782 | } 783 | }, 784 | "node_modules/fs.realpath": { 785 | "version": "1.0.0", 786 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 787 | "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", 788 | "dev": true 789 | }, 790 | "node_modules/fsevents": { 791 | "version": "2.3.2", 792 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 793 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 794 | "dev": true, 795 | "hasInstallScript": true, 796 | "optional": true, 797 | "os": [ 798 | "darwin" 799 | ], 800 | "engines": { 801 | "node": "^8.16.0 || ^10.6.0 || >=11.0.0" 802 | } 803 | }, 804 | "node_modules/function-bind": { 805 | "version": "1.1.1", 806 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", 807 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", 808 | "dev": true 809 | }, 810 | "node_modules/gh-pages": { 811 | "version": "4.0.0", 812 | "resolved": "https://registry.npmjs.org/gh-pages/-/gh-pages-4.0.0.tgz", 813 | "integrity": "sha512-p8S0T3aGJc68MtwOcZusul5qPSNZCalap3NWbhRUZYu1YOdp+EjZ+4kPmRM8h3NNRdqw00yuevRjlkuSzCn7iQ==", 814 | "dev": true, 815 | "dependencies": { 816 | "async": "^2.6.1", 817 | "commander": "^2.18.0", 818 | "email-addresses": "^3.0.1", 819 | "filenamify": "^4.3.0", 820 | "find-cache-dir": "^3.3.1", 821 | "fs-extra": "^8.1.0", 822 | "globby": "^6.1.0" 823 | }, 824 | "bin": { 825 | "gh-pages": "bin/gh-pages.js", 826 | "gh-pages-clean": "bin/gh-pages-clean.js" 827 | }, 828 | "engines": { 829 | "node": ">=10" 830 | } 831 | }, 832 | "node_modules/glob": { 833 | "version": "7.2.3", 834 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", 835 | "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", 836 | "dev": true, 837 | "dependencies": { 838 | "fs.realpath": "^1.0.0", 839 | "inflight": "^1.0.4", 840 | "inherits": "2", 841 | "minimatch": "^3.1.1", 842 | "once": "^1.3.0", 843 | "path-is-absolute": "^1.0.0" 844 | }, 845 | "engines": { 846 | "node": "*" 847 | }, 848 | "funding": { 849 | "url": "https://github.com/sponsors/isaacs" 850 | } 851 | }, 852 | "node_modules/glob-parent": { 853 | "version": "5.1.2", 854 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 855 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 856 | "dev": true, 857 | "dependencies": { 858 | "is-glob": "^4.0.1" 859 | }, 860 | "engines": { 861 | "node": ">= 6" 862 | } 863 | }, 864 | "node_modules/globby": { 865 | "version": "6.1.0", 866 | "resolved": "https://registry.npmjs.org/globby/-/globby-6.1.0.tgz", 867 | "integrity": "sha512-KVbFv2TQtbzCoxAnfD6JcHZTYCzyliEaaeM/gH8qQdkKr5s0OP9scEgvdcngyk7AVdY6YVW/TJHd+lQ/Df3Daw==", 868 | "dev": true, 869 | "dependencies": { 870 | "array-union": "^1.0.1", 871 | "glob": "^7.0.3", 872 | "object-assign": "^4.0.1", 873 | "pify": "^2.0.0", 874 | "pinkie-promise": "^2.0.0" 875 | }, 876 | "engines": { 877 | "node": ">=0.10.0" 878 | } 879 | }, 880 | "node_modules/graceful-fs": { 881 | "version": "4.2.10", 882 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.10.tgz", 883 | "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==", 884 | "dev": true 885 | }, 886 | "node_modules/gsap": { 887 | "version": "3.11.3", 888 | "resolved": "https://registry.npmjs.org/gsap/-/gsap-3.11.3.tgz", 889 | "integrity": "sha512-xc/iIJy+LWiMbRa4IdMtdnnKa/7PXEK6NNzV71gdOYUVeTZN7UWnLU0fB7Hi1iwiz4ZZoYkBZPPYGg+2+zzFHA==" 890 | }, 891 | "node_modules/has": { 892 | "version": "1.0.3", 893 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", 894 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", 895 | "dev": true, 896 | "dependencies": { 897 | "function-bind": "^1.1.1" 898 | }, 899 | "engines": { 900 | "node": ">= 0.4.0" 901 | } 902 | }, 903 | "node_modules/immutable": { 904 | "version": "4.1.0", 905 | "resolved": "https://registry.npmjs.org/immutable/-/immutable-4.1.0.tgz", 906 | "integrity": "sha512-oNkuqVTA8jqG1Q6c+UglTOD1xhC1BtjKI7XkCXRkZHrN5m18/XsnUp8Q89GkQO/z+0WjonSvl0FLhDYftp46nQ==", 907 | "dev": true 908 | }, 909 | "node_modules/inflight": { 910 | "version": "1.0.6", 911 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 912 | "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", 913 | "dev": true, 914 | "dependencies": { 915 | "once": "^1.3.0", 916 | "wrappy": "1" 917 | } 918 | }, 919 | "node_modules/inherits": { 920 | "version": "2.0.4", 921 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 922 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", 923 | "dev": true 924 | }, 925 | "node_modules/is-binary-path": { 926 | "version": "2.1.0", 927 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 928 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 929 | "dev": true, 930 | "dependencies": { 931 | "binary-extensions": "^2.0.0" 932 | }, 933 | "engines": { 934 | "node": ">=8" 935 | } 936 | }, 937 | "node_modules/is-core-module": { 938 | "version": "2.10.0", 939 | "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.10.0.tgz", 940 | "integrity": "sha512-Erxj2n/LDAZ7H8WNJXd9tw38GYM3dv8rk8Zcs+jJuxYTW7sozH+SS8NtrSjVL1/vpLvWi1hxy96IzjJ3EHTJJg==", 941 | "dev": true, 942 | "dependencies": { 943 | "has": "^1.0.3" 944 | }, 945 | "funding": { 946 | "url": "https://github.com/sponsors/ljharb" 947 | } 948 | }, 949 | "node_modules/is-extglob": { 950 | "version": "2.1.1", 951 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 952 | "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", 953 | "dev": true, 954 | "engines": { 955 | "node": ">=0.10.0" 956 | } 957 | }, 958 | "node_modules/is-glob": { 959 | "version": "4.0.3", 960 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", 961 | "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", 962 | "dev": true, 963 | "dependencies": { 964 | "is-extglob": "^2.1.1" 965 | }, 966 | "engines": { 967 | "node": ">=0.10.0" 968 | } 969 | }, 970 | "node_modules/is-number": { 971 | "version": "7.0.0", 972 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 973 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 974 | "dev": true, 975 | "engines": { 976 | "node": ">=0.12.0" 977 | } 978 | }, 979 | "node_modules/jsonfile": { 980 | "version": "4.0.0", 981 | "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", 982 | "integrity": "sha512-m6F1R3z8jjlf2imQHS2Qez5sjKWQzbuuhuJ/FKYFRZvPE3PuHcSMVZzfsLhGVOkfd20obL5SWEBew5ShlquNxg==", 983 | "dev": true, 984 | "optionalDependencies": { 985 | "graceful-fs": "^4.1.6" 986 | } 987 | }, 988 | "node_modules/locate-path": { 989 | "version": "5.0.0", 990 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", 991 | "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", 992 | "dev": true, 993 | "dependencies": { 994 | "p-locate": "^4.1.0" 995 | }, 996 | "engines": { 997 | "node": ">=8" 998 | } 999 | }, 1000 | "node_modules/lodash": { 1001 | "version": "4.17.21", 1002 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", 1003 | "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", 1004 | "dev": true 1005 | }, 1006 | "node_modules/make-dir": { 1007 | "version": "3.1.0", 1008 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", 1009 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", 1010 | "dev": true, 1011 | "dependencies": { 1012 | "semver": "^6.0.0" 1013 | }, 1014 | "engines": { 1015 | "node": ">=8" 1016 | }, 1017 | "funding": { 1018 | "url": "https://github.com/sponsors/sindresorhus" 1019 | } 1020 | }, 1021 | "node_modules/minimatch": { 1022 | "version": "3.1.2", 1023 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 1024 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 1025 | "dev": true, 1026 | "dependencies": { 1027 | "brace-expansion": "^1.1.7" 1028 | }, 1029 | "engines": { 1030 | "node": "*" 1031 | } 1032 | }, 1033 | "node_modules/nanoid": { 1034 | "version": "3.3.4", 1035 | "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.4.tgz", 1036 | "integrity": "sha512-MqBkQh/OHTS2egovRtLk45wEyNXwF+cokD+1YPf9u5VfJiRdAiRwB2froX5Co9Rh20xs4siNPm8naNotSD6RBw==", 1037 | "dev": true, 1038 | "bin": { 1039 | "nanoid": "bin/nanoid.cjs" 1040 | }, 1041 | "engines": { 1042 | "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" 1043 | } 1044 | }, 1045 | "node_modules/node-releases": { 1046 | "version": "2.0.6", 1047 | "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.6.tgz", 1048 | "integrity": "sha512-PiVXnNuFm5+iYkLBNeq5211hvO38y63T0i2KKh2KnUs3RpzJ+JtODFjkD8yjLwnDkTYF1eKXheUwdssR+NRZdg==", 1049 | "dev": true 1050 | }, 1051 | "node_modules/normalize-path": { 1052 | "version": "3.0.0", 1053 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 1054 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 1055 | "dev": true, 1056 | "engines": { 1057 | "node": ">=0.10.0" 1058 | } 1059 | }, 1060 | "node_modules/normalize-range": { 1061 | "version": "0.1.2", 1062 | "resolved": "https://registry.npmjs.org/normalize-range/-/normalize-range-0.1.2.tgz", 1063 | "integrity": "sha512-bdok/XvKII3nUpklnV6P2hxtMNrCboOjAcyBuQnWEhO665FwrSNRxU+AqpsyvO6LgGYPspN+lu5CLtw4jPRKNA==", 1064 | "dev": true, 1065 | "engines": { 1066 | "node": ">=0.10.0" 1067 | } 1068 | }, 1069 | "node_modules/object-assign": { 1070 | "version": "4.1.1", 1071 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 1072 | "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", 1073 | "dev": true, 1074 | "engines": { 1075 | "node": ">=0.10.0" 1076 | } 1077 | }, 1078 | "node_modules/once": { 1079 | "version": "1.4.0", 1080 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1081 | "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", 1082 | "dev": true, 1083 | "dependencies": { 1084 | "wrappy": "1" 1085 | } 1086 | }, 1087 | "node_modules/p-limit": { 1088 | "version": "2.3.0", 1089 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", 1090 | "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", 1091 | "dev": true, 1092 | "dependencies": { 1093 | "p-try": "^2.0.0" 1094 | }, 1095 | "engines": { 1096 | "node": ">=6" 1097 | }, 1098 | "funding": { 1099 | "url": "https://github.com/sponsors/sindresorhus" 1100 | } 1101 | }, 1102 | "node_modules/p-locate": { 1103 | "version": "4.1.0", 1104 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", 1105 | "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", 1106 | "dev": true, 1107 | "dependencies": { 1108 | "p-limit": "^2.2.0" 1109 | }, 1110 | "engines": { 1111 | "node": ">=8" 1112 | } 1113 | }, 1114 | "node_modules/p-try": { 1115 | "version": "2.2.0", 1116 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", 1117 | "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", 1118 | "dev": true, 1119 | "engines": { 1120 | "node": ">=6" 1121 | } 1122 | }, 1123 | "node_modules/path-exists": { 1124 | "version": "4.0.0", 1125 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", 1126 | "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", 1127 | "dev": true, 1128 | "engines": { 1129 | "node": ">=8" 1130 | } 1131 | }, 1132 | "node_modules/path-is-absolute": { 1133 | "version": "1.0.1", 1134 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 1135 | "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", 1136 | "dev": true, 1137 | "engines": { 1138 | "node": ">=0.10.0" 1139 | } 1140 | }, 1141 | "node_modules/path-parse": { 1142 | "version": "1.0.7", 1143 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", 1144 | "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", 1145 | "dev": true 1146 | }, 1147 | "node_modules/picocolors": { 1148 | "version": "1.0.0", 1149 | "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", 1150 | "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==", 1151 | "dev": true 1152 | }, 1153 | "node_modules/picomatch": { 1154 | "version": "2.3.1", 1155 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", 1156 | "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", 1157 | "dev": true, 1158 | "engines": { 1159 | "node": ">=8.6" 1160 | }, 1161 | "funding": { 1162 | "url": "https://github.com/sponsors/jonschlinkert" 1163 | } 1164 | }, 1165 | "node_modules/pify": { 1166 | "version": "2.3.0", 1167 | "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", 1168 | "integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==", 1169 | "dev": true, 1170 | "engines": { 1171 | "node": ">=0.10.0" 1172 | } 1173 | }, 1174 | "node_modules/pinkie": { 1175 | "version": "2.0.4", 1176 | "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz", 1177 | "integrity": "sha512-MnUuEycAemtSaeFSjXKW/aroV7akBbY+Sv+RkyqFjgAe73F+MR0TBWKBRDkmfWq/HiFmdavfZ1G7h4SPZXaCSg==", 1178 | "dev": true, 1179 | "engines": { 1180 | "node": ">=0.10.0" 1181 | } 1182 | }, 1183 | "node_modules/pinkie-promise": { 1184 | "version": "2.0.1", 1185 | "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz", 1186 | "integrity": "sha512-0Gni6D4UcLTbv9c57DfxDGdr41XfgUjqWZu492f0cIGr16zDU06BWP/RAEvOuo7CQ0CNjHaLlM59YJJFm3NWlw==", 1187 | "dev": true, 1188 | "dependencies": { 1189 | "pinkie": "^2.0.0" 1190 | }, 1191 | "engines": { 1192 | "node": ">=0.10.0" 1193 | } 1194 | }, 1195 | "node_modules/pkg-dir": { 1196 | "version": "4.2.0", 1197 | "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", 1198 | "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", 1199 | "dev": true, 1200 | "dependencies": { 1201 | "find-up": "^4.0.0" 1202 | }, 1203 | "engines": { 1204 | "node": ">=8" 1205 | } 1206 | }, 1207 | "node_modules/postcss": { 1208 | "version": "8.4.19", 1209 | "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.19.tgz", 1210 | "integrity": "sha512-h+pbPsyhlYj6N2ozBmHhHrs9DzGmbaarbLvWipMRO7RLS+v4onj26MPFXA5OBYFxyqYhUJK456SwDcY9H2/zsA==", 1211 | "dev": true, 1212 | "funding": [ 1213 | { 1214 | "type": "opencollective", 1215 | "url": "https://opencollective.com/postcss/" 1216 | }, 1217 | { 1218 | "type": "tidelift", 1219 | "url": "https://tidelift.com/funding/github/npm/postcss" 1220 | } 1221 | ], 1222 | "dependencies": { 1223 | "nanoid": "^3.3.4", 1224 | "picocolors": "^1.0.0", 1225 | "source-map-js": "^1.0.2" 1226 | }, 1227 | "engines": { 1228 | "node": "^10 || ^12 || >=14" 1229 | } 1230 | }, 1231 | "node_modules/postcss-value-parser": { 1232 | "version": "4.2.0", 1233 | "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz", 1234 | "integrity": "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==", 1235 | "dev": true 1236 | }, 1237 | "node_modules/prettier": { 1238 | "version": "2.8.0", 1239 | "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.8.0.tgz", 1240 | "integrity": "sha512-9Lmg8hTFZKG0Asr/kW9Bp8tJjRVluO8EJQVfY2T7FMw9T5jy4I/Uvx0Rca/XWf50QQ1/SS48+6IJWnrb+2yemA==", 1241 | "dev": true, 1242 | "bin": { 1243 | "prettier": "bin-prettier.js" 1244 | }, 1245 | "engines": { 1246 | "node": ">=10.13.0" 1247 | }, 1248 | "funding": { 1249 | "url": "https://github.com/prettier/prettier?sponsor=1" 1250 | } 1251 | }, 1252 | "node_modules/readdirp": { 1253 | "version": "3.6.0", 1254 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", 1255 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", 1256 | "dev": true, 1257 | "dependencies": { 1258 | "picomatch": "^2.2.1" 1259 | }, 1260 | "engines": { 1261 | "node": ">=8.10.0" 1262 | } 1263 | }, 1264 | "node_modules/resolve": { 1265 | "version": "1.22.1", 1266 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.1.tgz", 1267 | "integrity": "sha512-nBpuuYuY5jFsli/JIs1oldw6fOQCBioohqWZg/2hiaOybXOft4lonv85uDOKXdf8rhyK159cxU5cDcK/NKk8zw==", 1268 | "dev": true, 1269 | "dependencies": { 1270 | "is-core-module": "^2.9.0", 1271 | "path-parse": "^1.0.7", 1272 | "supports-preserve-symlinks-flag": "^1.0.0" 1273 | }, 1274 | "bin": { 1275 | "resolve": "bin/resolve" 1276 | }, 1277 | "funding": { 1278 | "url": "https://github.com/sponsors/ljharb" 1279 | } 1280 | }, 1281 | "node_modules/ress": { 1282 | "version": "5.0.2", 1283 | "resolved": "https://registry.npmjs.org/ress/-/ress-5.0.2.tgz", 1284 | "integrity": "sha512-oHBtOWo/Uc8SzQMbQNIKTcgi8wKmAs7IlNlRywmXudbOtF+c27FlOIq7tnwLDVcTywe6JXYo1pDXHO6kABwNYA==", 1285 | "dev": true 1286 | }, 1287 | "node_modules/rollup": { 1288 | "version": "2.79.1", 1289 | "resolved": "https://registry.npmjs.org/rollup/-/rollup-2.79.1.tgz", 1290 | "integrity": "sha512-uKxbd0IhMZOhjAiD5oAFp7BqvkA4Dv47qpOCtaNvng4HBwdbWtdOh8f5nZNuk2rp51PMGk3bzfWu5oayNEuYnw==", 1291 | "dev": true, 1292 | "bin": { 1293 | "rollup": "dist/bin/rollup" 1294 | }, 1295 | "engines": { 1296 | "node": ">=10.0.0" 1297 | }, 1298 | "optionalDependencies": { 1299 | "fsevents": "~2.3.2" 1300 | } 1301 | }, 1302 | "node_modules/sass": { 1303 | "version": "1.54.5", 1304 | "resolved": "https://registry.npmjs.org/sass/-/sass-1.54.5.tgz", 1305 | "integrity": "sha512-p7DTOzxkUPa/63FU0R3KApkRHwcVZYC0PLnLm5iyZACyp15qSi32x7zVUhRdABAATmkALqgGrjCJAcWvobmhHw==", 1306 | "dev": true, 1307 | "dependencies": { 1308 | "chokidar": ">=3.0.0 <4.0.0", 1309 | "immutable": "^4.0.0", 1310 | "source-map-js": ">=0.6.2 <2.0.0" 1311 | }, 1312 | "bin": { 1313 | "sass": "sass.js" 1314 | }, 1315 | "engines": { 1316 | "node": ">=12.0.0" 1317 | } 1318 | }, 1319 | "node_modules/semver": { 1320 | "version": "6.3.0", 1321 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 1322 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 1323 | "dev": true, 1324 | "bin": { 1325 | "semver": "bin/semver.js" 1326 | } 1327 | }, 1328 | "node_modules/source-map-js": { 1329 | "version": "1.0.2", 1330 | "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.0.2.tgz", 1331 | "integrity": "sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw==", 1332 | "dev": true, 1333 | "engines": { 1334 | "node": ">=0.10.0" 1335 | } 1336 | }, 1337 | "node_modules/strip-outer": { 1338 | "version": "1.0.1", 1339 | "resolved": "https://registry.npmjs.org/strip-outer/-/strip-outer-1.0.1.tgz", 1340 | "integrity": "sha512-k55yxKHwaXnpYGsOzg4Vl8+tDrWylxDEpknGjhTiZB8dFRU5rTo9CAzeycivxV3s+zlTKwrs6WxMxR95n26kwg==", 1341 | "dev": true, 1342 | "dependencies": { 1343 | "escape-string-regexp": "^1.0.2" 1344 | }, 1345 | "engines": { 1346 | "node": ">=0.10.0" 1347 | } 1348 | }, 1349 | "node_modules/supports-preserve-symlinks-flag": { 1350 | "version": "1.0.0", 1351 | "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", 1352 | "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", 1353 | "dev": true, 1354 | "engines": { 1355 | "node": ">= 0.4" 1356 | }, 1357 | "funding": { 1358 | "url": "https://github.com/sponsors/ljharb" 1359 | } 1360 | }, 1361 | "node_modules/three": { 1362 | "version": "0.147.0", 1363 | "resolved": "https://registry.npmjs.org/three/-/three-0.147.0.tgz", 1364 | "integrity": "sha512-LPTOslYQXFkmvceQjFTNnVVli2LaVF6C99Pv34fJypp8NbQLbTlu3KinZ0zURghS5zEehK+VQyvWuPZ/Sm8fzw==" 1365 | }, 1366 | "node_modules/three-mesh-bvh": { 1367 | "version": "0.5.19", 1368 | "resolved": "https://registry.npmjs.org/three-mesh-bvh/-/three-mesh-bvh-0.5.19.tgz", 1369 | "integrity": "sha512-Q8huT0NautyR4afW0oMlOz8a4qm3b1jQ1ccMLF/TZ0rA4i58HljsnBjCLEDLQtIINZyLtZPdw60mrD81V/TZDA==", 1370 | "peerDependencies": { 1371 | "three": ">= 0.123.0" 1372 | } 1373 | }, 1374 | "node_modules/to-regex-range": { 1375 | "version": "5.0.1", 1376 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 1377 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 1378 | "dev": true, 1379 | "dependencies": { 1380 | "is-number": "^7.0.0" 1381 | }, 1382 | "engines": { 1383 | "node": ">=8.0" 1384 | } 1385 | }, 1386 | "node_modules/trim-repeated": { 1387 | "version": "1.0.0", 1388 | "resolved": "https://registry.npmjs.org/trim-repeated/-/trim-repeated-1.0.0.tgz", 1389 | "integrity": "sha512-pkonvlKk8/ZuR0D5tLW8ljt5I8kmxp2XKymhepUeOdCEfKpZaktSArkLHZt76OB1ZvO9bssUsDty4SWhLvZpLg==", 1390 | "dev": true, 1391 | "dependencies": { 1392 | "escape-string-regexp": "^1.0.2" 1393 | }, 1394 | "engines": { 1395 | "node": ">=0.10.0" 1396 | } 1397 | }, 1398 | "node_modules/typescript": { 1399 | "version": "4.7.4", 1400 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.7.4.tgz", 1401 | "integrity": "sha512-C0WQT0gezHuw6AdY1M2jxUO83Rjf0HP7Sk1DtXj6j1EwkQNZrHAg2XPWlq62oqEhYvONq5pkC2Y9oPljWToLmQ==", 1402 | "dev": true, 1403 | "bin": { 1404 | "tsc": "bin/tsc", 1405 | "tsserver": "bin/tsserver" 1406 | }, 1407 | "engines": { 1408 | "node": ">=4.2.0" 1409 | } 1410 | }, 1411 | "node_modules/universalify": { 1412 | "version": "0.1.2", 1413 | "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", 1414 | "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==", 1415 | "dev": true, 1416 | "engines": { 1417 | "node": ">= 4.0.0" 1418 | } 1419 | }, 1420 | "node_modules/update-browserslist-db": { 1421 | "version": "1.0.10", 1422 | "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.0.10.tgz", 1423 | "integrity": "sha512-OztqDenkfFkbSG+tRxBeAnCVPckDBcvibKd35yDONx6OU8N7sqgwc7rCbkJ/WcYtVRZ4ba68d6byhC21GFh7sQ==", 1424 | "dev": true, 1425 | "funding": [ 1426 | { 1427 | "type": "opencollective", 1428 | "url": "https://opencollective.com/browserslist" 1429 | }, 1430 | { 1431 | "type": "tidelift", 1432 | "url": "https://tidelift.com/funding/github/npm/browserslist" 1433 | } 1434 | ], 1435 | "dependencies": { 1436 | "escalade": "^3.1.1", 1437 | "picocolors": "^1.0.0" 1438 | }, 1439 | "bin": { 1440 | "browserslist-lint": "cli.js" 1441 | }, 1442 | "peerDependencies": { 1443 | "browserslist": ">= 4.21.0" 1444 | } 1445 | }, 1446 | "node_modules/vite": { 1447 | "version": "3.2.5", 1448 | "resolved": "https://registry.npmjs.org/vite/-/vite-3.2.5.tgz", 1449 | "integrity": "sha512-4mVEpXpSOgrssFZAOmGIr85wPHKvaDAcXqxVxVRZhljkJOMZi1ibLibzjLHzJvcok8BMguLc7g1W6W/GqZbLdQ==", 1450 | "dev": true, 1451 | "dependencies": { 1452 | "esbuild": "^0.15.9", 1453 | "postcss": "^8.4.18", 1454 | "resolve": "^1.22.1", 1455 | "rollup": "^2.79.1" 1456 | }, 1457 | "bin": { 1458 | "vite": "bin/vite.js" 1459 | }, 1460 | "engines": { 1461 | "node": "^14.18.0 || >=16.0.0" 1462 | }, 1463 | "optionalDependencies": { 1464 | "fsevents": "~2.3.2" 1465 | }, 1466 | "peerDependencies": { 1467 | "@types/node": ">= 14", 1468 | "less": "*", 1469 | "sass": "*", 1470 | "stylus": "*", 1471 | "sugarss": "*", 1472 | "terser": "^5.4.0" 1473 | }, 1474 | "peerDependenciesMeta": { 1475 | "@types/node": { 1476 | "optional": true 1477 | }, 1478 | "less": { 1479 | "optional": true 1480 | }, 1481 | "sass": { 1482 | "optional": true 1483 | }, 1484 | "stylus": { 1485 | "optional": true 1486 | }, 1487 | "sugarss": { 1488 | "optional": true 1489 | }, 1490 | "terser": { 1491 | "optional": true 1492 | } 1493 | } 1494 | }, 1495 | "node_modules/vite-plugin-glsl": { 1496 | "version": "0.5.1", 1497 | "resolved": "https://registry.npmjs.org/vite-plugin-glsl/-/vite-plugin-glsl-0.5.1.tgz", 1498 | "integrity": "sha512-r4zKN8dlOVIB30whl4rlGgiUTnDk4DHDpDMfUwAERkkoJVf7yXqiNCcGCe4SRgi/GLekAyP4ShW4dSyfycvwfA==", 1499 | "dev": true, 1500 | "dependencies": { 1501 | "@rollup/pluginutils": "^4.2.1" 1502 | }, 1503 | "engines": { 1504 | "node": ">= 14.18.0", 1505 | "npm": ">= 6.14.17" 1506 | }, 1507 | "peerDependencies": { 1508 | "vite": "^3.0.0" 1509 | } 1510 | }, 1511 | "node_modules/wrappy": { 1512 | "version": "1.0.2", 1513 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 1514 | "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", 1515 | "dev": true 1516 | } 1517 | }, 1518 | "dependencies": { 1519 | "@esbuild/android-arm": { 1520 | "version": "0.15.12", 1521 | "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.15.12.tgz", 1522 | "integrity": "sha512-IC7TqIqiyE0MmvAhWkl/8AEzpOtbhRNDo7aph47We1NbE5w2bt/Q+giAhe0YYeVpYnIhGMcuZY92qDK6dQauvA==", 1523 | "dev": true, 1524 | "optional": true 1525 | }, 1526 | "@esbuild/linux-loong64": { 1527 | "version": "0.15.12", 1528 | "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.15.12.tgz", 1529 | "integrity": "sha512-tZEowDjvU7O7I04GYvWQOS4yyP9E/7YlsB0jjw1Ycukgr2ycEzKyIk5tms5WnLBymaewc6VmRKnn5IJWgK4eFw==", 1530 | "dev": true, 1531 | "optional": true 1532 | }, 1533 | "@rollup/pluginutils": { 1534 | "version": "4.2.1", 1535 | "resolved": "https://registry.npmjs.org/@rollup/pluginutils/-/pluginutils-4.2.1.tgz", 1536 | "integrity": "sha512-iKnFXr7NkdZAIHiIWE+BX5ULi/ucVFYWD6TbAV+rZctiRTY2PL6tsIKhoIOaoskiWAkgu+VsbXgUVDNLHf+InQ==", 1537 | "dev": true, 1538 | "requires": { 1539 | "estree-walker": "^2.0.1", 1540 | "picomatch": "^2.2.2" 1541 | } 1542 | }, 1543 | "@types/node": { 1544 | "version": "18.11.5", 1545 | "resolved": "https://registry.npmjs.org/@types/node/-/node-18.11.5.tgz", 1546 | "integrity": "sha512-3JRwhbjI+cHLAkUorhf8RnqUbFXajvzX4q6fMn5JwkgtuwfYtRQYI3u4V92vI6NJuTsbBQWWh3RZjFsuevyMGQ==", 1547 | "dev": true 1548 | }, 1549 | "@types/three": { 1550 | "version": "0.146.0", 1551 | "resolved": "https://registry.npmjs.org/@types/three/-/three-0.146.0.tgz", 1552 | "integrity": "sha512-75AgysUrIvTCB054eQa2pDVFurfeFW8CrMQjpzjt3yHBfuuknoSvvsESd/3EhQxPrz9si3+P0wiDUVsWUlljfA==", 1553 | "dev": true, 1554 | "requires": { 1555 | "@types/webxr": "*" 1556 | } 1557 | }, 1558 | "@types/webxr": { 1559 | "version": "0.5.0", 1560 | "resolved": "https://registry.npmjs.org/@types/webxr/-/webxr-0.5.0.tgz", 1561 | "integrity": "sha512-IUMDPSXnYIbEO2IereEFcgcqfDREOgmbGqtrMpVPpACTU6pltYLwHgVkrnYv0XhWEcjio9sYEfIEzgn3c7nDqA==", 1562 | "dev": true 1563 | }, 1564 | "anymatch": { 1565 | "version": "3.1.2", 1566 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", 1567 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", 1568 | "dev": true, 1569 | "requires": { 1570 | "normalize-path": "^3.0.0", 1571 | "picomatch": "^2.0.4" 1572 | } 1573 | }, 1574 | "array-union": { 1575 | "version": "1.0.2", 1576 | "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", 1577 | "integrity": "sha512-Dxr6QJj/RdU/hCaBjOfxW+q6lyuVE6JFWIrAUpuOOhoJJoQ99cUn3igRaHVB5P9WrgFVN0FfArM3x0cueOU8ng==", 1578 | "dev": true, 1579 | "requires": { 1580 | "array-uniq": "^1.0.1" 1581 | } 1582 | }, 1583 | "array-uniq": { 1584 | "version": "1.0.3", 1585 | "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz", 1586 | "integrity": "sha512-MNha4BWQ6JbwhFhj03YK552f7cb3AzoE8SzeljgChvL1dl3IcvggXVz1DilzySZkCja+CXuZbdW7yATchWn8/Q==", 1587 | "dev": true 1588 | }, 1589 | "async": { 1590 | "version": "2.6.4", 1591 | "resolved": "https://registry.npmjs.org/async/-/async-2.6.4.tgz", 1592 | "integrity": "sha512-mzo5dfJYwAn29PeiJ0zvwTo04zj8HDJj0Mn8TD7sno7q12prdbnasKJHhkm2c1LgrhlJ0teaea8860oxi51mGA==", 1593 | "dev": true, 1594 | "requires": { 1595 | "lodash": "^4.17.14" 1596 | } 1597 | }, 1598 | "autoprefixer": { 1599 | "version": "10.4.12", 1600 | "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-10.4.12.tgz", 1601 | "integrity": "sha512-WrCGV9/b97Pa+jtwf5UGaRjgQIg7OK3D06GnoYoZNcG1Xb8Gt3EfuKjlhh9i/VtT16g6PYjZ69jdJ2g8FxSC4Q==", 1602 | "dev": true, 1603 | "requires": { 1604 | "browserslist": "^4.21.4", 1605 | "caniuse-lite": "^1.0.30001407", 1606 | "fraction.js": "^4.2.0", 1607 | "normalize-range": "^0.1.2", 1608 | "picocolors": "^1.0.0", 1609 | "postcss-value-parser": "^4.2.0" 1610 | } 1611 | }, 1612 | "balanced-match": { 1613 | "version": "1.0.2", 1614 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 1615 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", 1616 | "dev": true 1617 | }, 1618 | "binary-extensions": { 1619 | "version": "2.2.0", 1620 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", 1621 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", 1622 | "dev": true 1623 | }, 1624 | "brace-expansion": { 1625 | "version": "1.1.11", 1626 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 1627 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 1628 | "dev": true, 1629 | "requires": { 1630 | "balanced-match": "^1.0.0", 1631 | "concat-map": "0.0.1" 1632 | } 1633 | }, 1634 | "braces": { 1635 | "version": "3.0.2", 1636 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 1637 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 1638 | "dev": true, 1639 | "requires": { 1640 | "fill-range": "^7.0.1" 1641 | } 1642 | }, 1643 | "browserslist": { 1644 | "version": "4.21.4", 1645 | "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.21.4.tgz", 1646 | "integrity": "sha512-CBHJJdDmgjl3daYjN5Cp5kbTf1mUhZoS+beLklHIvkOWscs83YAhLlF3Wsh/lciQYAcbBJgTOD44VtG31ZM4Hw==", 1647 | "dev": true, 1648 | "requires": { 1649 | "caniuse-lite": "^1.0.30001400", 1650 | "electron-to-chromium": "^1.4.251", 1651 | "node-releases": "^2.0.6", 1652 | "update-browserslist-db": "^1.0.9" 1653 | } 1654 | }, 1655 | "caniuse-lite": { 1656 | "version": "1.0.30001418", 1657 | "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001418.tgz", 1658 | "integrity": "sha512-oIs7+JL3K9JRQ3jPZjlH6qyYDp+nBTCais7hjh0s+fuBwufc7uZ7hPYMXrDOJhV360KGMTcczMRObk0/iMqZRg==", 1659 | "dev": true 1660 | }, 1661 | "chokidar": { 1662 | "version": "3.5.3", 1663 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz", 1664 | "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", 1665 | "dev": true, 1666 | "requires": { 1667 | "anymatch": "~3.1.2", 1668 | "braces": "~3.0.2", 1669 | "fsevents": "~2.3.2", 1670 | "glob-parent": "~5.1.2", 1671 | "is-binary-path": "~2.1.0", 1672 | "is-glob": "~4.0.1", 1673 | "normalize-path": "~3.0.0", 1674 | "readdirp": "~3.6.0" 1675 | } 1676 | }, 1677 | "commander": { 1678 | "version": "2.20.3", 1679 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", 1680 | "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", 1681 | "dev": true 1682 | }, 1683 | "commondir": { 1684 | "version": "1.0.1", 1685 | "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", 1686 | "integrity": "sha512-W9pAhw0ja1Edb5GVdIF1mjZw/ASI0AlShXM83UUGe2DVr5TdAPEA1OA8m/g8zWp9x6On7gqufY+FatDbC3MDQg==", 1687 | "dev": true 1688 | }, 1689 | "concat-map": { 1690 | "version": "0.0.1", 1691 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 1692 | "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", 1693 | "dev": true 1694 | }, 1695 | "electron-to-chromium": { 1696 | "version": "1.4.276", 1697 | "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.276.tgz", 1698 | "integrity": "sha512-EpuHPqu8YhonqLBXHoU6hDJCD98FCe6KDoet3/gY1qsQ6usjJoHqBH2YIVs8FXaAtHwVL8Uqa/fsYao/vq9VWQ==", 1699 | "dev": true 1700 | }, 1701 | "email-addresses": { 1702 | "version": "3.1.0", 1703 | "resolved": "https://registry.npmjs.org/email-addresses/-/email-addresses-3.1.0.tgz", 1704 | "integrity": "sha512-k0/r7GrWVL32kZlGwfPNgB2Y/mMXVTq/decgLczm/j34whdaspNrZO8CnXPf1laaHxI6ptUlsnAxN+UAPw+fzg==", 1705 | "dev": true 1706 | }, 1707 | "esbuild": { 1708 | "version": "0.15.12", 1709 | "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.15.12.tgz", 1710 | "integrity": "sha512-PcT+/wyDqJQsRVhaE9uX/Oq4XLrFh0ce/bs2TJh4CSaw9xuvI+xFrH2nAYOADbhQjUgAhNWC5LKoUsakm4dxng==", 1711 | "dev": true, 1712 | "requires": { 1713 | "@esbuild/android-arm": "0.15.12", 1714 | "@esbuild/linux-loong64": "0.15.12", 1715 | "esbuild-android-64": "0.15.12", 1716 | "esbuild-android-arm64": "0.15.12", 1717 | "esbuild-darwin-64": "0.15.12", 1718 | "esbuild-darwin-arm64": "0.15.12", 1719 | "esbuild-freebsd-64": "0.15.12", 1720 | "esbuild-freebsd-arm64": "0.15.12", 1721 | "esbuild-linux-32": "0.15.12", 1722 | "esbuild-linux-64": "0.15.12", 1723 | "esbuild-linux-arm": "0.15.12", 1724 | "esbuild-linux-arm64": "0.15.12", 1725 | "esbuild-linux-mips64le": "0.15.12", 1726 | "esbuild-linux-ppc64le": "0.15.12", 1727 | "esbuild-linux-riscv64": "0.15.12", 1728 | "esbuild-linux-s390x": "0.15.12", 1729 | "esbuild-netbsd-64": "0.15.12", 1730 | "esbuild-openbsd-64": "0.15.12", 1731 | "esbuild-sunos-64": "0.15.12", 1732 | "esbuild-windows-32": "0.15.12", 1733 | "esbuild-windows-64": "0.15.12", 1734 | "esbuild-windows-arm64": "0.15.12" 1735 | } 1736 | }, 1737 | "esbuild-android-64": { 1738 | "version": "0.15.12", 1739 | "resolved": "https://registry.npmjs.org/esbuild-android-64/-/esbuild-android-64-0.15.12.tgz", 1740 | "integrity": "sha512-MJKXwvPY9g0rGps0+U65HlTsM1wUs9lbjt5CU19RESqycGFDRijMDQsh68MtbzkqWSRdEtiKS1mtPzKneaAI0Q==", 1741 | "dev": true, 1742 | "optional": true 1743 | }, 1744 | "esbuild-android-arm64": { 1745 | "version": "0.15.12", 1746 | "resolved": "https://registry.npmjs.org/esbuild-android-arm64/-/esbuild-android-arm64-0.15.12.tgz", 1747 | "integrity": "sha512-Hc9SEcZbIMhhLcvhr1DH+lrrec9SFTiRzfJ7EGSBZiiw994gfkVV6vG0sLWqQQ6DD7V4+OggB+Hn0IRUdDUqvA==", 1748 | "dev": true, 1749 | "optional": true 1750 | }, 1751 | "esbuild-darwin-64": { 1752 | "version": "0.15.12", 1753 | "resolved": "https://registry.npmjs.org/esbuild-darwin-64/-/esbuild-darwin-64-0.15.12.tgz", 1754 | "integrity": "sha512-qkmqrTVYPFiePt5qFjP8w/S+GIUMbt6k8qmiPraECUWfPptaPJUGkCKrWEfYFRWB7bY23FV95rhvPyh/KARP8Q==", 1755 | "dev": true, 1756 | "optional": true 1757 | }, 1758 | "esbuild-darwin-arm64": { 1759 | "version": "0.15.12", 1760 | "resolved": "https://registry.npmjs.org/esbuild-darwin-arm64/-/esbuild-darwin-arm64-0.15.12.tgz", 1761 | "integrity": "sha512-z4zPX02tQ41kcXMyN3c/GfZpIjKoI/BzHrdKUwhC/Ki5BAhWv59A9M8H+iqaRbwpzYrYidTybBwiZAIWCLJAkw==", 1762 | "dev": true, 1763 | "optional": true 1764 | }, 1765 | "esbuild-freebsd-64": { 1766 | "version": "0.15.12", 1767 | "resolved": "https://registry.npmjs.org/esbuild-freebsd-64/-/esbuild-freebsd-64-0.15.12.tgz", 1768 | "integrity": "sha512-XFL7gKMCKXLDiAiBjhLG0XECliXaRLTZh6hsyzqUqPUf/PY4C6EJDTKIeqqPKXaVJ8+fzNek88285krSz1QECw==", 1769 | "dev": true, 1770 | "optional": true 1771 | }, 1772 | "esbuild-freebsd-arm64": { 1773 | "version": "0.15.12", 1774 | "resolved": "https://registry.npmjs.org/esbuild-freebsd-arm64/-/esbuild-freebsd-arm64-0.15.12.tgz", 1775 | "integrity": "sha512-jwEIu5UCUk6TjiG1X+KQnCGISI+ILnXzIzt9yDVrhjug2fkYzlLbl0K43q96Q3KB66v6N1UFF0r5Ks4Xo7i72g==", 1776 | "dev": true, 1777 | "optional": true 1778 | }, 1779 | "esbuild-linux-32": { 1780 | "version": "0.15.12", 1781 | "resolved": "https://registry.npmjs.org/esbuild-linux-32/-/esbuild-linux-32-0.15.12.tgz", 1782 | "integrity": "sha512-uSQuSEyF1kVzGzuIr4XM+v7TPKxHjBnLcwv2yPyCz8riV8VUCnO/C4BF3w5dHiVpCd5Z1cebBtZJNlC4anWpwA==", 1783 | "dev": true, 1784 | "optional": true 1785 | }, 1786 | "esbuild-linux-64": { 1787 | "version": "0.15.12", 1788 | "resolved": "https://registry.npmjs.org/esbuild-linux-64/-/esbuild-linux-64-0.15.12.tgz", 1789 | "integrity": "sha512-QcgCKb7zfJxqT9o5z9ZUeGH1k8N6iX1Y7VNsEi5F9+HzN1OIx7ESxtQXDN9jbeUSPiRH1n9cw6gFT3H4qbdvcA==", 1790 | "dev": true, 1791 | "optional": true 1792 | }, 1793 | "esbuild-linux-arm": { 1794 | "version": "0.15.12", 1795 | "resolved": "https://registry.npmjs.org/esbuild-linux-arm/-/esbuild-linux-arm-0.15.12.tgz", 1796 | "integrity": "sha512-Wf7T0aNylGcLu7hBnzMvsTfEXdEdJY/hY3u36Vla21aY66xR0MS5I1Hw8nVquXjTN0A6fk/vnr32tkC/C2lb0A==", 1797 | "dev": true, 1798 | "optional": true 1799 | }, 1800 | "esbuild-linux-arm64": { 1801 | "version": "0.15.12", 1802 | "resolved": "https://registry.npmjs.org/esbuild-linux-arm64/-/esbuild-linux-arm64-0.15.12.tgz", 1803 | "integrity": "sha512-HtNq5xm8fUpZKwWKS2/YGwSfTF+339L4aIA8yphNKYJckd5hVdhfdl6GM2P3HwLSCORS++++7++//ApEwXEuAQ==", 1804 | "dev": true, 1805 | "optional": true 1806 | }, 1807 | "esbuild-linux-mips64le": { 1808 | "version": "0.15.12", 1809 | "resolved": "https://registry.npmjs.org/esbuild-linux-mips64le/-/esbuild-linux-mips64le-0.15.12.tgz", 1810 | "integrity": "sha512-Qol3+AvivngUZkTVFgLpb0H6DT+N5/zM3V1YgTkryPYFeUvuT5JFNDR3ZiS6LxhyF8EE+fiNtzwlPqMDqVcc6A==", 1811 | "dev": true, 1812 | "optional": true 1813 | }, 1814 | "esbuild-linux-ppc64le": { 1815 | "version": "0.15.12", 1816 | "resolved": "https://registry.npmjs.org/esbuild-linux-ppc64le/-/esbuild-linux-ppc64le-0.15.12.tgz", 1817 | "integrity": "sha512-4D8qUCo+CFKaR0cGXtGyVsOI7w7k93Qxb3KFXWr75An0DHamYzq8lt7TNZKoOq/Gh8c40/aKaxvcZnTgQ0TJNg==", 1818 | "dev": true, 1819 | "optional": true 1820 | }, 1821 | "esbuild-linux-riscv64": { 1822 | "version": "0.15.12", 1823 | "resolved": "https://registry.npmjs.org/esbuild-linux-riscv64/-/esbuild-linux-riscv64-0.15.12.tgz", 1824 | "integrity": "sha512-G9w6NcuuCI6TUUxe6ka0enjZHDnSVK8bO+1qDhMOCtl7Tr78CcZilJj8SGLN00zO5iIlwNRZKHjdMpfFgNn1VA==", 1825 | "dev": true, 1826 | "optional": true 1827 | }, 1828 | "esbuild-linux-s390x": { 1829 | "version": "0.15.12", 1830 | "resolved": "https://registry.npmjs.org/esbuild-linux-s390x/-/esbuild-linux-s390x-0.15.12.tgz", 1831 | "integrity": "sha512-Lt6BDnuXbXeqSlVuuUM5z18GkJAZf3ERskGZbAWjrQoi9xbEIsj/hEzVnSAFLtkfLuy2DE4RwTcX02tZFunXww==", 1832 | "dev": true, 1833 | "optional": true 1834 | }, 1835 | "esbuild-netbsd-64": { 1836 | "version": "0.15.12", 1837 | "resolved": "https://registry.npmjs.org/esbuild-netbsd-64/-/esbuild-netbsd-64-0.15.12.tgz", 1838 | "integrity": "sha512-jlUxCiHO1dsqoURZDQts+HK100o0hXfi4t54MNRMCAqKGAV33JCVvMplLAa2FwviSojT/5ZG5HUfG3gstwAG8w==", 1839 | "dev": true, 1840 | "optional": true 1841 | }, 1842 | "esbuild-openbsd-64": { 1843 | "version": "0.15.12", 1844 | "resolved": "https://registry.npmjs.org/esbuild-openbsd-64/-/esbuild-openbsd-64-0.15.12.tgz", 1845 | "integrity": "sha512-1o1uAfRTMIWNOmpf8v7iudND0L6zRBYSH45sofCZywrcf7NcZA+c7aFsS1YryU+yN7aRppTqdUK1PgbZVaB1Dw==", 1846 | "dev": true, 1847 | "optional": true 1848 | }, 1849 | "esbuild-sunos-64": { 1850 | "version": "0.15.12", 1851 | "resolved": "https://registry.npmjs.org/esbuild-sunos-64/-/esbuild-sunos-64-0.15.12.tgz", 1852 | "integrity": "sha512-nkl251DpoWoBO9Eq9aFdoIt2yYmp4I3kvQjba3jFKlMXuqQ9A4q+JaqdkCouG3DHgAGnzshzaGu6xofGcXyPXg==", 1853 | "dev": true, 1854 | "optional": true 1855 | }, 1856 | "esbuild-windows-32": { 1857 | "version": "0.15.12", 1858 | "resolved": "https://registry.npmjs.org/esbuild-windows-32/-/esbuild-windows-32-0.15.12.tgz", 1859 | "integrity": "sha512-WlGeBZHgPC00O08luIp5B2SP4cNCp/PcS+3Pcg31kdcJPopHxLkdCXtadLU9J82LCfw4TVls21A6lilQ9mzHrw==", 1860 | "dev": true, 1861 | "optional": true 1862 | }, 1863 | "esbuild-windows-64": { 1864 | "version": "0.15.12", 1865 | "resolved": "https://registry.npmjs.org/esbuild-windows-64/-/esbuild-windows-64-0.15.12.tgz", 1866 | "integrity": "sha512-VActO3WnWZSN//xjSfbiGOSyC+wkZtI8I4KlgrTo5oHJM6z3MZZBCuFaZHd8hzf/W9KPhF0lY8OqlmWC9HO5AA==", 1867 | "dev": true, 1868 | "optional": true 1869 | }, 1870 | "esbuild-windows-arm64": { 1871 | "version": "0.15.12", 1872 | "resolved": "https://registry.npmjs.org/esbuild-windows-arm64/-/esbuild-windows-arm64-0.15.12.tgz", 1873 | "integrity": "sha512-Of3MIacva1OK/m4zCNIvBfz8VVROBmQT+gRX6pFTLPngFYcj6TFH/12VveAqq1k9VB2l28EoVMNMUCcmsfwyuA==", 1874 | "dev": true, 1875 | "optional": true 1876 | }, 1877 | "escalade": { 1878 | "version": "3.1.1", 1879 | "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", 1880 | "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", 1881 | "dev": true 1882 | }, 1883 | "escape-string-regexp": { 1884 | "version": "1.0.5", 1885 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 1886 | "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", 1887 | "dev": true 1888 | }, 1889 | "estree-walker": { 1890 | "version": "2.0.2", 1891 | "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz", 1892 | "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==", 1893 | "dev": true 1894 | }, 1895 | "filename-reserved-regex": { 1896 | "version": "2.0.0", 1897 | "resolved": "https://registry.npmjs.org/filename-reserved-regex/-/filename-reserved-regex-2.0.0.tgz", 1898 | "integrity": "sha512-lc1bnsSr4L4Bdif8Xb/qrtokGbq5zlsms/CYH8PP+WtCkGNF65DPiQY8vG3SakEdRn8Dlnm+gW/qWKKjS5sZzQ==", 1899 | "dev": true 1900 | }, 1901 | "filenamify": { 1902 | "version": "4.3.0", 1903 | "resolved": "https://registry.npmjs.org/filenamify/-/filenamify-4.3.0.tgz", 1904 | "integrity": "sha512-hcFKyUG57yWGAzu1CMt/dPzYZuv+jAJUT85bL8mrXvNe6hWj6yEHEc4EdcgiA6Z3oi1/9wXJdZPXF2dZNgwgOg==", 1905 | "dev": true, 1906 | "requires": { 1907 | "filename-reserved-regex": "^2.0.0", 1908 | "strip-outer": "^1.0.1", 1909 | "trim-repeated": "^1.0.0" 1910 | } 1911 | }, 1912 | "fill-range": { 1913 | "version": "7.0.1", 1914 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 1915 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 1916 | "dev": true, 1917 | "requires": { 1918 | "to-regex-range": "^5.0.1" 1919 | } 1920 | }, 1921 | "find-cache-dir": { 1922 | "version": "3.3.2", 1923 | "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-3.3.2.tgz", 1924 | "integrity": "sha512-wXZV5emFEjrridIgED11OoUKLxiYjAcqot/NJdAkOhlJ+vGzwhOAfcG5OX1jP+S0PcjEn8bdMJv+g2jwQ3Onig==", 1925 | "dev": true, 1926 | "requires": { 1927 | "commondir": "^1.0.1", 1928 | "make-dir": "^3.0.2", 1929 | "pkg-dir": "^4.1.0" 1930 | } 1931 | }, 1932 | "find-up": { 1933 | "version": "4.1.0", 1934 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", 1935 | "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", 1936 | "dev": true, 1937 | "requires": { 1938 | "locate-path": "^5.0.0", 1939 | "path-exists": "^4.0.0" 1940 | } 1941 | }, 1942 | "fraction.js": { 1943 | "version": "4.2.0", 1944 | "resolved": "https://registry.npmjs.org/fraction.js/-/fraction.js-4.2.0.tgz", 1945 | "integrity": "sha512-MhLuK+2gUcnZe8ZHlaaINnQLl0xRIGRfcGk2yl8xoQAfHrSsL3rYu6FCmBdkdbhc9EPlwyGHewaRsvwRMJtAlA==", 1946 | "dev": true 1947 | }, 1948 | "fs-extra": { 1949 | "version": "8.1.0", 1950 | "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", 1951 | "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", 1952 | "dev": true, 1953 | "requires": { 1954 | "graceful-fs": "^4.2.0", 1955 | "jsonfile": "^4.0.0", 1956 | "universalify": "^0.1.0" 1957 | } 1958 | }, 1959 | "fs.realpath": { 1960 | "version": "1.0.0", 1961 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 1962 | "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", 1963 | "dev": true 1964 | }, 1965 | "fsevents": { 1966 | "version": "2.3.2", 1967 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 1968 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 1969 | "dev": true, 1970 | "optional": true 1971 | }, 1972 | "function-bind": { 1973 | "version": "1.1.1", 1974 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", 1975 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", 1976 | "dev": true 1977 | }, 1978 | "gh-pages": { 1979 | "version": "4.0.0", 1980 | "resolved": "https://registry.npmjs.org/gh-pages/-/gh-pages-4.0.0.tgz", 1981 | "integrity": "sha512-p8S0T3aGJc68MtwOcZusul5qPSNZCalap3NWbhRUZYu1YOdp+EjZ+4kPmRM8h3NNRdqw00yuevRjlkuSzCn7iQ==", 1982 | "dev": true, 1983 | "requires": { 1984 | "async": "^2.6.1", 1985 | "commander": "^2.18.0", 1986 | "email-addresses": "^3.0.1", 1987 | "filenamify": "^4.3.0", 1988 | "find-cache-dir": "^3.3.1", 1989 | "fs-extra": "^8.1.0", 1990 | "globby": "^6.1.0" 1991 | } 1992 | }, 1993 | "glob": { 1994 | "version": "7.2.3", 1995 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", 1996 | "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", 1997 | "dev": true, 1998 | "requires": { 1999 | "fs.realpath": "^1.0.0", 2000 | "inflight": "^1.0.4", 2001 | "inherits": "2", 2002 | "minimatch": "^3.1.1", 2003 | "once": "^1.3.0", 2004 | "path-is-absolute": "^1.0.0" 2005 | } 2006 | }, 2007 | "glob-parent": { 2008 | "version": "5.1.2", 2009 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 2010 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 2011 | "dev": true, 2012 | "requires": { 2013 | "is-glob": "^4.0.1" 2014 | } 2015 | }, 2016 | "globby": { 2017 | "version": "6.1.0", 2018 | "resolved": "https://registry.npmjs.org/globby/-/globby-6.1.0.tgz", 2019 | "integrity": "sha512-KVbFv2TQtbzCoxAnfD6JcHZTYCzyliEaaeM/gH8qQdkKr5s0OP9scEgvdcngyk7AVdY6YVW/TJHd+lQ/Df3Daw==", 2020 | "dev": true, 2021 | "requires": { 2022 | "array-union": "^1.0.1", 2023 | "glob": "^7.0.3", 2024 | "object-assign": "^4.0.1", 2025 | "pify": "^2.0.0", 2026 | "pinkie-promise": "^2.0.0" 2027 | } 2028 | }, 2029 | "graceful-fs": { 2030 | "version": "4.2.10", 2031 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.10.tgz", 2032 | "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==", 2033 | "dev": true 2034 | }, 2035 | "gsap": { 2036 | "version": "3.11.3", 2037 | "resolved": "https://registry.npmjs.org/gsap/-/gsap-3.11.3.tgz", 2038 | "integrity": "sha512-xc/iIJy+LWiMbRa4IdMtdnnKa/7PXEK6NNzV71gdOYUVeTZN7UWnLU0fB7Hi1iwiz4ZZoYkBZPPYGg+2+zzFHA==" 2039 | }, 2040 | "has": { 2041 | "version": "1.0.3", 2042 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", 2043 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", 2044 | "dev": true, 2045 | "requires": { 2046 | "function-bind": "^1.1.1" 2047 | } 2048 | }, 2049 | "immutable": { 2050 | "version": "4.1.0", 2051 | "resolved": "https://registry.npmjs.org/immutable/-/immutable-4.1.0.tgz", 2052 | "integrity": "sha512-oNkuqVTA8jqG1Q6c+UglTOD1xhC1BtjKI7XkCXRkZHrN5m18/XsnUp8Q89GkQO/z+0WjonSvl0FLhDYftp46nQ==", 2053 | "dev": true 2054 | }, 2055 | "inflight": { 2056 | "version": "1.0.6", 2057 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 2058 | "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", 2059 | "dev": true, 2060 | "requires": { 2061 | "once": "^1.3.0", 2062 | "wrappy": "1" 2063 | } 2064 | }, 2065 | "inherits": { 2066 | "version": "2.0.4", 2067 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 2068 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", 2069 | "dev": true 2070 | }, 2071 | "is-binary-path": { 2072 | "version": "2.1.0", 2073 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 2074 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 2075 | "dev": true, 2076 | "requires": { 2077 | "binary-extensions": "^2.0.0" 2078 | } 2079 | }, 2080 | "is-core-module": { 2081 | "version": "2.10.0", 2082 | "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.10.0.tgz", 2083 | "integrity": "sha512-Erxj2n/LDAZ7H8WNJXd9tw38GYM3dv8rk8Zcs+jJuxYTW7sozH+SS8NtrSjVL1/vpLvWi1hxy96IzjJ3EHTJJg==", 2084 | "dev": true, 2085 | "requires": { 2086 | "has": "^1.0.3" 2087 | } 2088 | }, 2089 | "is-extglob": { 2090 | "version": "2.1.1", 2091 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 2092 | "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", 2093 | "dev": true 2094 | }, 2095 | "is-glob": { 2096 | "version": "4.0.3", 2097 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", 2098 | "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", 2099 | "dev": true, 2100 | "requires": { 2101 | "is-extglob": "^2.1.1" 2102 | } 2103 | }, 2104 | "is-number": { 2105 | "version": "7.0.0", 2106 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 2107 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 2108 | "dev": true 2109 | }, 2110 | "jsonfile": { 2111 | "version": "4.0.0", 2112 | "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", 2113 | "integrity": "sha512-m6F1R3z8jjlf2imQHS2Qez5sjKWQzbuuhuJ/FKYFRZvPE3PuHcSMVZzfsLhGVOkfd20obL5SWEBew5ShlquNxg==", 2114 | "dev": true, 2115 | "requires": { 2116 | "graceful-fs": "^4.1.6" 2117 | } 2118 | }, 2119 | "locate-path": { 2120 | "version": "5.0.0", 2121 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", 2122 | "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", 2123 | "dev": true, 2124 | "requires": { 2125 | "p-locate": "^4.1.0" 2126 | } 2127 | }, 2128 | "lodash": { 2129 | "version": "4.17.21", 2130 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", 2131 | "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", 2132 | "dev": true 2133 | }, 2134 | "make-dir": { 2135 | "version": "3.1.0", 2136 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", 2137 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", 2138 | "dev": true, 2139 | "requires": { 2140 | "semver": "^6.0.0" 2141 | } 2142 | }, 2143 | "minimatch": { 2144 | "version": "3.1.2", 2145 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 2146 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 2147 | "dev": true, 2148 | "requires": { 2149 | "brace-expansion": "^1.1.7" 2150 | } 2151 | }, 2152 | "nanoid": { 2153 | "version": "3.3.4", 2154 | "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.4.tgz", 2155 | "integrity": "sha512-MqBkQh/OHTS2egovRtLk45wEyNXwF+cokD+1YPf9u5VfJiRdAiRwB2froX5Co9Rh20xs4siNPm8naNotSD6RBw==", 2156 | "dev": true 2157 | }, 2158 | "node-releases": { 2159 | "version": "2.0.6", 2160 | "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.6.tgz", 2161 | "integrity": "sha512-PiVXnNuFm5+iYkLBNeq5211hvO38y63T0i2KKh2KnUs3RpzJ+JtODFjkD8yjLwnDkTYF1eKXheUwdssR+NRZdg==", 2162 | "dev": true 2163 | }, 2164 | "normalize-path": { 2165 | "version": "3.0.0", 2166 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 2167 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 2168 | "dev": true 2169 | }, 2170 | "normalize-range": { 2171 | "version": "0.1.2", 2172 | "resolved": "https://registry.npmjs.org/normalize-range/-/normalize-range-0.1.2.tgz", 2173 | "integrity": "sha512-bdok/XvKII3nUpklnV6P2hxtMNrCboOjAcyBuQnWEhO665FwrSNRxU+AqpsyvO6LgGYPspN+lu5CLtw4jPRKNA==", 2174 | "dev": true 2175 | }, 2176 | "object-assign": { 2177 | "version": "4.1.1", 2178 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 2179 | "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", 2180 | "dev": true 2181 | }, 2182 | "once": { 2183 | "version": "1.4.0", 2184 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 2185 | "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", 2186 | "dev": true, 2187 | "requires": { 2188 | "wrappy": "1" 2189 | } 2190 | }, 2191 | "p-limit": { 2192 | "version": "2.3.0", 2193 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", 2194 | "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", 2195 | "dev": true, 2196 | "requires": { 2197 | "p-try": "^2.0.0" 2198 | } 2199 | }, 2200 | "p-locate": { 2201 | "version": "4.1.0", 2202 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", 2203 | "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", 2204 | "dev": true, 2205 | "requires": { 2206 | "p-limit": "^2.2.0" 2207 | } 2208 | }, 2209 | "p-try": { 2210 | "version": "2.2.0", 2211 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", 2212 | "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", 2213 | "dev": true 2214 | }, 2215 | "path-exists": { 2216 | "version": "4.0.0", 2217 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", 2218 | "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", 2219 | "dev": true 2220 | }, 2221 | "path-is-absolute": { 2222 | "version": "1.0.1", 2223 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 2224 | "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", 2225 | "dev": true 2226 | }, 2227 | "path-parse": { 2228 | "version": "1.0.7", 2229 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", 2230 | "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", 2231 | "dev": true 2232 | }, 2233 | "picocolors": { 2234 | "version": "1.0.0", 2235 | "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", 2236 | "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==", 2237 | "dev": true 2238 | }, 2239 | "picomatch": { 2240 | "version": "2.3.1", 2241 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", 2242 | "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", 2243 | "dev": true 2244 | }, 2245 | "pify": { 2246 | "version": "2.3.0", 2247 | "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", 2248 | "integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==", 2249 | "dev": true 2250 | }, 2251 | "pinkie": { 2252 | "version": "2.0.4", 2253 | "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz", 2254 | "integrity": "sha512-MnUuEycAemtSaeFSjXKW/aroV7akBbY+Sv+RkyqFjgAe73F+MR0TBWKBRDkmfWq/HiFmdavfZ1G7h4SPZXaCSg==", 2255 | "dev": true 2256 | }, 2257 | "pinkie-promise": { 2258 | "version": "2.0.1", 2259 | "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz", 2260 | "integrity": "sha512-0Gni6D4UcLTbv9c57DfxDGdr41XfgUjqWZu492f0cIGr16zDU06BWP/RAEvOuo7CQ0CNjHaLlM59YJJFm3NWlw==", 2261 | "dev": true, 2262 | "requires": { 2263 | "pinkie": "^2.0.0" 2264 | } 2265 | }, 2266 | "pkg-dir": { 2267 | "version": "4.2.0", 2268 | "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", 2269 | "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", 2270 | "dev": true, 2271 | "requires": { 2272 | "find-up": "^4.0.0" 2273 | } 2274 | }, 2275 | "postcss": { 2276 | "version": "8.4.19", 2277 | "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.19.tgz", 2278 | "integrity": "sha512-h+pbPsyhlYj6N2ozBmHhHrs9DzGmbaarbLvWipMRO7RLS+v4onj26MPFXA5OBYFxyqYhUJK456SwDcY9H2/zsA==", 2279 | "dev": true, 2280 | "requires": { 2281 | "nanoid": "^3.3.4", 2282 | "picocolors": "^1.0.0", 2283 | "source-map-js": "^1.0.2" 2284 | } 2285 | }, 2286 | "postcss-value-parser": { 2287 | "version": "4.2.0", 2288 | "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz", 2289 | "integrity": "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==", 2290 | "dev": true 2291 | }, 2292 | "prettier": { 2293 | "version": "2.8.0", 2294 | "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.8.0.tgz", 2295 | "integrity": "sha512-9Lmg8hTFZKG0Asr/kW9Bp8tJjRVluO8EJQVfY2T7FMw9T5jy4I/Uvx0Rca/XWf50QQ1/SS48+6IJWnrb+2yemA==", 2296 | "dev": true 2297 | }, 2298 | "readdirp": { 2299 | "version": "3.6.0", 2300 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", 2301 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", 2302 | "dev": true, 2303 | "requires": { 2304 | "picomatch": "^2.2.1" 2305 | } 2306 | }, 2307 | "resolve": { 2308 | "version": "1.22.1", 2309 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.1.tgz", 2310 | "integrity": "sha512-nBpuuYuY5jFsli/JIs1oldw6fOQCBioohqWZg/2hiaOybXOft4lonv85uDOKXdf8rhyK159cxU5cDcK/NKk8zw==", 2311 | "dev": true, 2312 | "requires": { 2313 | "is-core-module": "^2.9.0", 2314 | "path-parse": "^1.0.7", 2315 | "supports-preserve-symlinks-flag": "^1.0.0" 2316 | } 2317 | }, 2318 | "ress": { 2319 | "version": "5.0.2", 2320 | "resolved": "https://registry.npmjs.org/ress/-/ress-5.0.2.tgz", 2321 | "integrity": "sha512-oHBtOWo/Uc8SzQMbQNIKTcgi8wKmAs7IlNlRywmXudbOtF+c27FlOIq7tnwLDVcTywe6JXYo1pDXHO6kABwNYA==", 2322 | "dev": true 2323 | }, 2324 | "rollup": { 2325 | "version": "2.79.1", 2326 | "resolved": "https://registry.npmjs.org/rollup/-/rollup-2.79.1.tgz", 2327 | "integrity": "sha512-uKxbd0IhMZOhjAiD5oAFp7BqvkA4Dv47qpOCtaNvng4HBwdbWtdOh8f5nZNuk2rp51PMGk3bzfWu5oayNEuYnw==", 2328 | "dev": true, 2329 | "requires": { 2330 | "fsevents": "~2.3.2" 2331 | } 2332 | }, 2333 | "sass": { 2334 | "version": "1.54.5", 2335 | "resolved": "https://registry.npmjs.org/sass/-/sass-1.54.5.tgz", 2336 | "integrity": "sha512-p7DTOzxkUPa/63FU0R3KApkRHwcVZYC0PLnLm5iyZACyp15qSi32x7zVUhRdABAATmkALqgGrjCJAcWvobmhHw==", 2337 | "dev": true, 2338 | "requires": { 2339 | "chokidar": ">=3.0.0 <4.0.0", 2340 | "immutable": "^4.0.0", 2341 | "source-map-js": ">=0.6.2 <2.0.0" 2342 | } 2343 | }, 2344 | "semver": { 2345 | "version": "6.3.0", 2346 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 2347 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 2348 | "dev": true 2349 | }, 2350 | "source-map-js": { 2351 | "version": "1.0.2", 2352 | "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.0.2.tgz", 2353 | "integrity": "sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw==", 2354 | "dev": true 2355 | }, 2356 | "strip-outer": { 2357 | "version": "1.0.1", 2358 | "resolved": "https://registry.npmjs.org/strip-outer/-/strip-outer-1.0.1.tgz", 2359 | "integrity": "sha512-k55yxKHwaXnpYGsOzg4Vl8+tDrWylxDEpknGjhTiZB8dFRU5rTo9CAzeycivxV3s+zlTKwrs6WxMxR95n26kwg==", 2360 | "dev": true, 2361 | "requires": { 2362 | "escape-string-regexp": "^1.0.2" 2363 | } 2364 | }, 2365 | "supports-preserve-symlinks-flag": { 2366 | "version": "1.0.0", 2367 | "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", 2368 | "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", 2369 | "dev": true 2370 | }, 2371 | "three": { 2372 | "version": "0.147.0", 2373 | "resolved": "https://registry.npmjs.org/three/-/three-0.147.0.tgz", 2374 | "integrity": "sha512-LPTOslYQXFkmvceQjFTNnVVli2LaVF6C99Pv34fJypp8NbQLbTlu3KinZ0zURghS5zEehK+VQyvWuPZ/Sm8fzw==" 2375 | }, 2376 | "three-mesh-bvh": { 2377 | "version": "0.5.19", 2378 | "resolved": "https://registry.npmjs.org/three-mesh-bvh/-/three-mesh-bvh-0.5.19.tgz", 2379 | "integrity": "sha512-Q8huT0NautyR4afW0oMlOz8a4qm3b1jQ1ccMLF/TZ0rA4i58HljsnBjCLEDLQtIINZyLtZPdw60mrD81V/TZDA==", 2380 | "requires": {} 2381 | }, 2382 | "to-regex-range": { 2383 | "version": "5.0.1", 2384 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 2385 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 2386 | "dev": true, 2387 | "requires": { 2388 | "is-number": "^7.0.0" 2389 | } 2390 | }, 2391 | "trim-repeated": { 2392 | "version": "1.0.0", 2393 | "resolved": "https://registry.npmjs.org/trim-repeated/-/trim-repeated-1.0.0.tgz", 2394 | "integrity": "sha512-pkonvlKk8/ZuR0D5tLW8ljt5I8kmxp2XKymhepUeOdCEfKpZaktSArkLHZt76OB1ZvO9bssUsDty4SWhLvZpLg==", 2395 | "dev": true, 2396 | "requires": { 2397 | "escape-string-regexp": "^1.0.2" 2398 | } 2399 | }, 2400 | "typescript": { 2401 | "version": "4.7.4", 2402 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.7.4.tgz", 2403 | "integrity": "sha512-C0WQT0gezHuw6AdY1M2jxUO83Rjf0HP7Sk1DtXj6j1EwkQNZrHAg2XPWlq62oqEhYvONq5pkC2Y9oPljWToLmQ==", 2404 | "dev": true 2405 | }, 2406 | "universalify": { 2407 | "version": "0.1.2", 2408 | "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", 2409 | "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==", 2410 | "dev": true 2411 | }, 2412 | "update-browserslist-db": { 2413 | "version": "1.0.10", 2414 | "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.0.10.tgz", 2415 | "integrity": "sha512-OztqDenkfFkbSG+tRxBeAnCVPckDBcvibKd35yDONx6OU8N7sqgwc7rCbkJ/WcYtVRZ4ba68d6byhC21GFh7sQ==", 2416 | "dev": true, 2417 | "requires": { 2418 | "escalade": "^3.1.1", 2419 | "picocolors": "^1.0.0" 2420 | } 2421 | }, 2422 | "vite": { 2423 | "version": "3.2.5", 2424 | "resolved": "https://registry.npmjs.org/vite/-/vite-3.2.5.tgz", 2425 | "integrity": "sha512-4mVEpXpSOgrssFZAOmGIr85wPHKvaDAcXqxVxVRZhljkJOMZi1ibLibzjLHzJvcok8BMguLc7g1W6W/GqZbLdQ==", 2426 | "dev": true, 2427 | "requires": { 2428 | "esbuild": "^0.15.9", 2429 | "fsevents": "~2.3.2", 2430 | "postcss": "^8.4.18", 2431 | "resolve": "^1.22.1", 2432 | "rollup": "^2.79.1" 2433 | } 2434 | }, 2435 | "vite-plugin-glsl": { 2436 | "version": "0.5.1", 2437 | "resolved": "https://registry.npmjs.org/vite-plugin-glsl/-/vite-plugin-glsl-0.5.1.tgz", 2438 | "integrity": "sha512-r4zKN8dlOVIB30whl4rlGgiUTnDk4DHDpDMfUwAERkkoJVf7yXqiNCcGCe4SRgi/GLekAyP4ShW4dSyfycvwfA==", 2439 | "dev": true, 2440 | "requires": { 2441 | "@rollup/pluginutils": "^4.2.1" 2442 | } 2443 | }, 2444 | "wrappy": { 2445 | "version": "1.0.2", 2446 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 2447 | "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", 2448 | "dev": true 2449 | } 2450 | } 2451 | } 2452 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "reflect-particles", 3 | "private": true, 4 | "version": "0.0.0", 5 | "type": "module", 6 | "homepage": "https://nemutas.github.io/reflect-particles/", 7 | "scripts": { 8 | "start": "vite", 9 | "build": "tsc && vite build", 10 | "preview": "vite preview", 11 | "deploy": "npm run build && gh-pages -d dist" 12 | }, 13 | "devDependencies": { 14 | "@types/node": "^18.11.5", 15 | "@types/three": "^0.146.0", 16 | "autoprefixer": "^10.4.12", 17 | "gh-pages": "^4.0.0", 18 | "prettier": "^2.8.0", 19 | "ress": "^5.0.2", 20 | "sass": "^1.54.5", 21 | "typescript": "^4.6.4", 22 | "vite": "^3.2.5", 23 | "vite-plugin-glsl": "^0.5.1" 24 | }, 25 | "dependencies": { 26 | "gsap": "^3.11.3", 27 | "three": "^0.147.0", 28 | "three-mesh-bvh": "^0.5.19" 29 | } 30 | } 31 | -------------------------------------------------------------------------------- /postcss.config.cjs: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | plugins: [require('autoprefixer')], 3 | } 4 | -------------------------------------------------------------------------------- /public/meta/vite.svg: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /public/resources/bunny.glb: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nemutas/reflect-particles/1ca299ffd0551edaeb71a5dd6a85faee8d978f99/public/resources/bunny.glb -------------------------------------------------------------------------------- /public/resources/matcap_1k.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nemutas/reflect-particles/1ca299ffd0551edaeb71a5dd6a85faee8d978f99/public/resources/matcap_1k.jpg -------------------------------------------------------------------------------- /src/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | Vite + TS 8 | 9 | 10 | 11 | 12 | 13 |
14 | 15 | GitHub 16 | 17 | 18 | 19 | -------------------------------------------------------------------------------- /src/index.scss: -------------------------------------------------------------------------------- 1 | @use './styles/mixins/media.scss' as *; 2 | 3 | .three-container { 4 | position: absolute; 5 | top: 0; 6 | width: 100%; 7 | height: 100%; 8 | } 9 | 10 | .github { 11 | position: absolute; 12 | bottom: 1rem; 13 | right: 2rem; 14 | font-size: 15px; 15 | color: #000; 16 | opacity: 0.5; 17 | @include sp { 18 | font-size: 20px; 19 | } 20 | 21 | @include hoverable { 22 | transition: opacity 0.3s ease; 23 | 24 | &::after { 25 | position: absolute; 26 | bottom: 0; 27 | left: 0; 28 | content: ''; 29 | width: 100%; 30 | height: 1px; 31 | background: #000; 32 | transform: scale(0, 1); 33 | transform-origin: right top; 34 | transition: transform 0.3s; 35 | } 36 | 37 | &:hover { 38 | opacity: 1; 39 | 40 | &::after { 41 | transform-origin: left top; 42 | transform: scale(1, 1); 43 | } 44 | } 45 | } 46 | } 47 | -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | import { TCanvas } from './webgl/TCanvas' 2 | 3 | const canvas = new TCanvas(document.body) 4 | 5 | window.addEventListener('beforeunload', () => { 6 | canvas.dispose() 7 | }) 8 | -------------------------------------------------------------------------------- /src/scripts/utils.ts: -------------------------------------------------------------------------------- 1 | export const resolvePath = (path: string) => { 2 | const p = path.startsWith('/') ? path.substring(1) : path 3 | return import.meta.env.BASE_URL + p 4 | } 5 | -------------------------------------------------------------------------------- /src/styles/base.scss: -------------------------------------------------------------------------------- 1 | @use 'ress'; 2 | @use './mixins/media.scss' as *; 3 | 4 | @import url('https://fonts.googleapis.com/css2?family=Poppins:wght@300;400;800&display=swap'); 5 | 6 | :root { 7 | --font-poppins: 'Poppins', sans-serif; 8 | --font-weight-regular: 300; 9 | --font-weight-mediam: 400; 10 | --font-weight-exbold: 800; 11 | } 12 | 13 | html { 14 | width: 100%; 15 | height: 100%; 16 | font-size: calc(1000vw / 750); 17 | @include pc { 18 | font-size: calc(1000vw / 1500); 19 | } 20 | font-family: var(--font-poppins); 21 | font-weight: var(--font-weight-regular); 22 | } 23 | 24 | body { 25 | position: relative; 26 | width: 100%; 27 | height: 100%; 28 | } 29 | 30 | a { 31 | color: inherit; 32 | text-decoration: none; 33 | } 34 | 35 | ul { 36 | list-style: none; 37 | } 38 | 39 | h1, 40 | h2, 41 | h3, 42 | h4, 43 | h5, 44 | h6 { 45 | font-size: inherit; 46 | font-weight: inherit; 47 | } 48 | -------------------------------------------------------------------------------- /src/styles/mixins/media.scss: -------------------------------------------------------------------------------- 1 | $--pc-width: 751px; 2 | $--sp-width: 750px; 3 | 4 | @mixin pc { 5 | @media screen and (min-width: $--pc-width) { 6 | @content; 7 | } 8 | } 9 | 10 | @mixin sp { 11 | @media screen and (max-width: $--sp-width) { 12 | @content; 13 | } 14 | } 15 | 16 | @mixin hoverable { 17 | @media (hover: hover) and (pointer: fine) { 18 | @content; 19 | } 20 | } 21 | -------------------------------------------------------------------------------- /src/types/glsl.d.ts: -------------------------------------------------------------------------------- 1 | // https://github.com/UstymUkhman/vite-plugin-glsl/blob/main/test/shaders.d.ts 2 | 3 | declare module '*.vs' { 4 | const value: string 5 | export default value 6 | } 7 | 8 | declare module '*.fs' { 9 | const value: string 10 | export default value 11 | } 12 | 13 | declare module '*.vert' { 14 | const value: string 15 | export default value 16 | } 17 | 18 | declare module '*.frag' { 19 | const value: string 20 | export default value 21 | } 22 | 23 | declare module '*.glsl' { 24 | const value: string 25 | export default value 26 | } 27 | 28 | declare module '*.wgsl' { 29 | const value: string 30 | export default value 31 | } 32 | -------------------------------------------------------------------------------- /src/types/vite-env.d.ts: -------------------------------------------------------------------------------- 1 | /// 2 | -------------------------------------------------------------------------------- /src/webgl/Particles.ts: -------------------------------------------------------------------------------- 1 | import * as THREE from 'three' 2 | import { acceleratedRaycast, computeBoundsTree, disposeBoundsTree } from 'three-mesh-bvh' 3 | import { MeshSurfaceSampler } from 'three/examples/jsm/math/MeshSurfaceSampler' 4 | import { gl } from './core/WebGL' 5 | 6 | THREE.BufferGeometry.prototype.computeBoundsTree = computeBoundsTree 7 | THREE.BufferGeometry.prototype.disposeBoundsTree = disposeBoundsTree 8 | THREE.Mesh.prototype.raycast = acceleratedRaycast 9 | 10 | export class Particles { 11 | private scene = new THREE.Scene() 12 | private renderTarget = new THREE.WebGLRenderTarget(gl.size.width, gl.size.height, { encoding: THREE.sRGBEncoding }) 13 | 14 | private readonly AMOUNT = 200 15 | 16 | private isAnimation = true 17 | private raycaster = new THREE.Raycaster() 18 | 19 | private particlesData: { velocity: THREE.Vector3 }[] = [] 20 | private particlePositions = new Float32Array(this.AMOUNT * 3) 21 | private particleColors = new Float32Array(this.AMOUNT * 3) 22 | 23 | private linePositions = new Float32Array(this.AMOUNT * this.AMOUNT * 3) 24 | private lineColors = new Float32Array(this.AMOUNT * this.AMOUNT * 3) 25 | 26 | private lineColorMax = 0 27 | 28 | constructor(private constraintMesh: THREE.Mesh) { 29 | this.init() 30 | this.createParticles() 31 | this.addEvents() 32 | } 33 | 34 | private init() { 35 | this.scene.background = new THREE.Color('#000') 36 | this.raycaster.firstHitOnly = true 37 | } 38 | 39 | private createParticles() { 40 | // particles 41 | const randVelo = (scale = 1) => (Math.random() * 2 - 1) * scale 42 | 43 | const sampler = new MeshSurfaceSampler(this.constraintMesh).build() 44 | const targetPosition = new THREE.Vector3() 45 | const targetNormal = new THREE.Vector3() 46 | 47 | for (let i = 0; i < this.AMOUNT; i++) { 48 | const [ix, iy, iz] = [3 * i, 3 * i + 1, 3 * i + 2] 49 | sampler.sample(targetPosition, targetNormal) 50 | targetPosition.add(targetNormal.multiplyScalar(-1 * 0.05)) 51 | 52 | this.particlePositions[ix] = targetPosition.x 53 | this.particlePositions[iy] = targetPosition.y 54 | this.particlePositions[iz] = targetPosition.z 55 | 56 | this.particlesData.push({ 57 | velocity: targetNormal 58 | .clone() 59 | .normalize() 60 | .add(new THREE.Vector3(randVelo(), randVelo(), randVelo())) 61 | .normalize(), 62 | }) 63 | } 64 | 65 | const geometry = new THREE.BufferGeometry() 66 | geometry.setAttribute( 67 | 'position', 68 | new THREE.BufferAttribute(this.particlePositions, 3).setUsage(THREE.DynamicDrawUsage), 69 | ) 70 | geometry.setAttribute('color', new THREE.BufferAttribute(this.particleColors, 3).setUsage(THREE.DynamicDrawUsage)) 71 | 72 | const material = new THREE.PointsMaterial({ 73 | vertexColors: true, 74 | size: 3, 75 | // blending: THREE.AdditiveBlending, 76 | transparent: true, 77 | sizeAttenuation: false, 78 | }) 79 | 80 | const points = new THREE.Points(geometry, material) 81 | points.name = 'points' 82 | this.scene.add(points) 83 | 84 | // line 85 | const lineGeometry = new THREE.BufferGeometry() 86 | lineGeometry.setAttribute( 87 | 'position', 88 | new THREE.BufferAttribute(this.linePositions, 3).setUsage(THREE.DynamicDrawUsage), 89 | ) 90 | lineGeometry.setAttribute('color', new THREE.BufferAttribute(this.lineColors, 3).setUsage(THREE.DynamicDrawUsage)) 91 | 92 | const lineMaterial = new THREE.LineBasicMaterial({ 93 | vertexColors: true, 94 | blending: THREE.AdditiveBlending, 95 | transparent: true, 96 | }) 97 | 98 | const lines = new THREE.LineSegments(lineGeometry, lineMaterial) 99 | lines.name = 'lines' 100 | this.scene.add(lines) 101 | } 102 | 103 | // ----------------------------------------------------------- 104 | // events 105 | private addEvents() { 106 | window.addEventListener('focus', this.handleFocus) 107 | window.addEventListener('blur', this.handleBlur) 108 | } 109 | 110 | private handleFocus = () => (this.isAnimation = true) 111 | 112 | private handleBlur = () => (this.isAnimation = false) 113 | 114 | // ----------------------------------------------------------- 115 | // animation 116 | private getColor(src: THREE.Vector3) { 117 | const r = THREE.MathUtils.smoothstep(src.x, -1, 1) 118 | const g = THREE.MathUtils.smoothstep(src.y, -1, 1) 119 | const b = THREE.MathUtils.smoothstep(src.z, -1, 1) 120 | return [r, g, b] 121 | } 122 | 123 | private update(dt: number) { 124 | if (!this.isAnimation) return 125 | 126 | let linePosCounter = 0 127 | let lineColCounter = 0 128 | 129 | for (let i = 0; i < this.lineColorMax / 3; i++) { 130 | // clear lines 131 | this.lineColors[3 * i] = 0 132 | this.lineColors[3 * i + 1] = 0 133 | this.lineColors[3 * i + 2] = 0 134 | } 135 | 136 | for (let i = 0; i < this.AMOUNT; i++) { 137 | // draw particles 138 | const [ix, iy, iz] = [3 * i, 3 * i + 1, 3 * i + 2] 139 | 140 | const velocity = this.particlesData[i].velocity 141 | 142 | this.particlePositions[ix] += velocity.x * dt 143 | this.particlePositions[iy] += velocity.y * dt 144 | this.particlePositions[iz] += velocity.z * dt 145 | 146 | const color = this.getColor(velocity) 147 | this.particleColors[ix] = color[0] 148 | this.particleColors[iy] = color[1] 149 | this.particleColors[iz] = color[2] 150 | 151 | this.raycaster.ray.origin.set(this.particlePositions[ix], this.particlePositions[iy], this.particlePositions[iz]) 152 | this.raycaster.ray.direction.copy(velocity) 153 | const hits = this.raycaster.intersectObject(this.constraintMesh) 154 | if (0 < hits.length) { 155 | if (hits[0].distance < dt * 2) velocity.reflect(hits[0].face!.normal) 156 | } else { 157 | this.particleColors[ix] = 0 158 | this.particleColors[iy] = 0 159 | this.particleColors[iz] = 0 160 | } 161 | 162 | // draw lines 163 | const posX = this.particlePositions[ix] 164 | const posY = this.particlePositions[iy] 165 | const posZ = this.particlePositions[iz] 166 | 167 | let lineCounter = 0 168 | 169 | for (let j = i + 1; j < this.AMOUNT; j++) { 170 | const [ixx, iyy, izz] = [3 * j, 3 * j + 1, 3 * j + 2] 171 | 172 | const nextVelocity = this.particlesData[j].velocity 173 | const nextPosX = this.particlePositions[ixx] + nextVelocity.x * dt 174 | const nextPosY = this.particlePositions[iyy] + nextVelocity.y * dt 175 | const nextPosZ = this.particlePositions[izz] + nextVelocity.z * dt 176 | 177 | const nextColor = this.getColor(nextVelocity) 178 | 179 | const distance = Math.sqrt((nextPosX - posX) ** 2 + (nextPosY - posY) ** 2 + (nextPosZ - posZ) ** 2) 180 | 181 | let alpha = 1 - THREE.MathUtils.smoothstep(distance, 0.1, 0.3) 182 | alpha *= 0.5 183 | 184 | this.linePositions[linePosCounter++] = posX 185 | this.linePositions[linePosCounter++] = posY 186 | this.linePositions[linePosCounter++] = posZ 187 | 188 | this.linePositions[linePosCounter++] = nextPosX 189 | this.linePositions[linePosCounter++] = nextPosY 190 | this.linePositions[linePosCounter++] = nextPosZ 191 | 192 | this.lineColors[lineColCounter++] = alpha * color[0] 193 | this.lineColors[lineColCounter++] = alpha * color[1] 194 | this.lineColors[lineColCounter++] = alpha * color[2] 195 | 196 | this.lineColors[lineColCounter++] = alpha * nextColor[0] 197 | this.lineColors[lineColCounter++] = alpha * nextColor[1] 198 | this.lineColors[lineColCounter++] = alpha * nextColor[2] 199 | 200 | if (0 < alpha && 20 <= ++lineCounter) { 201 | break 202 | } 203 | } 204 | } 205 | 206 | const points = this.scene.getObjectByName('points') as THREE.Mesh 207 | points.geometry.attributes.position.needsUpdate = true 208 | points.geometry.attributes.color.needsUpdate = true 209 | 210 | const lines = this.scene.getObjectByName('lines') as THREE.Mesh 211 | lines.geometry.attributes.position.needsUpdate = true 212 | lines.geometry.attributes.color.needsUpdate = true 213 | 214 | this.lineColorMax = lineColCounter 215 | } 216 | 217 | get texture() { 218 | return this.renderTarget.texture 219 | } 220 | 221 | render(dt: number) { 222 | this.update(dt) 223 | 224 | gl.renderer.setRenderTarget(this.renderTarget) 225 | gl.renderer.render(this.scene, gl.camera) 226 | gl.renderer.setRenderTarget(null) 227 | } 228 | 229 | dispose() { 230 | window.removeEventListener('focus', this.handleFocus) 231 | window.removeEventListener('blur', this.handleBlur) 232 | } 233 | } 234 | -------------------------------------------------------------------------------- /src/webgl/TCanvas.ts: -------------------------------------------------------------------------------- 1 | import * as THREE from 'three' 2 | import { resolvePath } from '../scripts/utils' 3 | import { gl } from './core/WebGL' 4 | import { Particles } from './Particles' 5 | import { Assets, loadAssets } from './utils/assetLoader' 6 | import { controls } from './utils/OrbitControls' 7 | import vertexShader from './shaders/vertex.glsl' 8 | import fragmentShader from './shaders/fragment.glsl' 9 | 10 | export class TCanvas { 11 | private animeID?: number 12 | private particles!: Particles 13 | 14 | private assets: Assets = { 15 | bunny: { path: resolvePath('/resources/bunny.glb') }, 16 | matcap: { path: resolvePath('/resources/matcap_1k.jpg') }, 17 | } 18 | 19 | constructor(private parentNode: ParentNode) { 20 | loadAssets(this.assets).then(() => { 21 | this.init() 22 | this.createObjects() 23 | this.setAnimationFrame() 24 | }) 25 | } 26 | 27 | private init() { 28 | gl.setup(this.parentNode.querySelector('.three-container')!) 29 | gl.scene.background = new THREE.Color('#fff') 30 | gl.camera.position.set(0, 0.5, 2.2) 31 | } 32 | 33 | private createObjects() { 34 | const bunny = (this.assets.bunny.data as any).scene.children[0] as THREE.Mesh 35 | 36 | // const geometry = new THREE.BoxGeometry() 37 | // const geometry = new THREE.TorusKnotGeometry(0.5, 0.2, 128, 32, 2, 1) 38 | const geometry = bunny.geometry 39 | const matrix4 = new THREE.Matrix4() 40 | geometry.applyMatrix4(matrix4.makeScale(1.5, 1.5, 1.5)) 41 | geometry.applyMatrix4(matrix4.makeTranslation(0.1, -0.1, 0)) 42 | geometry.computeBoundsTree() 43 | const material = new THREE.ShaderMaterial({ 44 | uniforms: { 45 | tParticles: { value: null }, 46 | tMatcap: { value: this.assets.matcap.data }, 47 | }, 48 | vertexShader, 49 | fragmentShader, 50 | side: THREE.DoubleSide, 51 | }) 52 | const mesh = new THREE.Mesh(geometry, material) 53 | gl.scene.add(mesh) 54 | 55 | this.particles = new Particles(mesh) 56 | material.uniforms.tParticles.value = this.particles.texture 57 | } 58 | 59 | // ---------------------------------- 60 | // animation frame 61 | private setAnimationFrame() { 62 | const anime = () => { 63 | const dt = gl.time.getDelta() * 0.3 64 | this.particles.render(dt) 65 | 66 | controls.update() 67 | gl.render() 68 | 69 | requestAnimationFrame(anime) 70 | } 71 | this.animeID = requestAnimationFrame(anime) 72 | } 73 | 74 | // ---------------------------------- 75 | // dispose 76 | dispose() { 77 | this.animeID && cancelAnimationFrame(this.animeID) 78 | gl.dispose() 79 | } 80 | } 81 | -------------------------------------------------------------------------------- /src/webgl/core/WebGL.ts: -------------------------------------------------------------------------------- 1 | import * as THREE from 'three' 2 | 3 | class WebGL { 4 | public renderer: THREE.WebGLRenderer 5 | public scene: THREE.Scene 6 | public camera: THREE.PerspectiveCamera 7 | public readonly time = new THREE.Clock() 8 | 9 | private resizeCallback?: () => void 10 | 11 | constructor() { 12 | const { width, height, aspect } = this.size 13 | 14 | this.renderer = new THREE.WebGLRenderer({ antialias: true, alpha: true }) 15 | this.renderer.setPixelRatio(window.devicePixelRatio) 16 | this.renderer.setSize(width, height) 17 | this.renderer.shadowMap.enabled = true 18 | this.renderer.outputEncoding = THREE.sRGBEncoding 19 | 20 | this.scene = new THREE.Scene() 21 | this.camera = new THREE.PerspectiveCamera(50, aspect, 0.01, 100) 22 | 23 | window.addEventListener('resize', this.handleResize) 24 | } 25 | 26 | private handleResize = () => { 27 | this.resizeCallback && this.resizeCallback() 28 | 29 | const { width, height, aspect } = this.size 30 | this.camera.aspect = aspect 31 | this.camera.updateProjectionMatrix() 32 | this.renderer.setSize(width, height) 33 | } 34 | 35 | get size() { 36 | const { innerWidth: width, innerHeight: height } = window 37 | return { width, height, aspect: width / height } 38 | } 39 | 40 | setup(container: HTMLElement) { 41 | container.appendChild(this.renderer.domElement) 42 | } 43 | 44 | setResizeCallback(callback: () => void) { 45 | this.resizeCallback = callback 46 | } 47 | 48 | getMesh(name: string) { 49 | return this.scene.getObjectByName(name) as THREE.Mesh 50 | } 51 | 52 | render() { 53 | this.renderer.render(this.scene, this.camera) 54 | } 55 | 56 | dispose() { 57 | gl.scene.traverse((child) => { 58 | if (child.type !== 'Scene') gl.scene.remove(child) 59 | }) 60 | } 61 | } 62 | 63 | export const gl = new WebGL() 64 | -------------------------------------------------------------------------------- /src/webgl/glsl/matcap.glsl: -------------------------------------------------------------------------------- 1 | vec2 matcap(vec3 eye, vec3 normal) { 2 | vec3 reflected = reflect(eye, normal); 3 | float m = 2.8284271247461903 * sqrt( reflected.z+1.0 ); 4 | return reflected.xy / m + 0.5; 5 | } -------------------------------------------------------------------------------- /src/webgl/shaders/fragment.glsl: -------------------------------------------------------------------------------- 1 | uniform sampler2D tParticles; 2 | uniform sampler2D tMatcap; 3 | varying vec2 v_screenUv; 4 | varying vec3 v_normal; 5 | varying vec3 v_eye; 6 | 7 | #include '../glsl/matcap.glsl' 8 | 9 | float rand(vec2 n) { 10 | return fract(sin(dot(n, vec2(12.9898, 4.1414))) * 43758.5453); 11 | } 12 | 13 | vec3 randVector(in vec3 p) { 14 | return normalize(vec3(rand(p.xy), rand(p.yz), rand(p.xz))); 15 | } 16 | 17 | void main() { 18 | vec3 roughness = randVector(v_normal) * 0.03; 19 | 20 | vec2 screenUv = v_screenUv + v_normal.xy * 0.01 + roughness.xy * 0.05; 21 | vec4 particlesTexture = texture2D(tParticles, screenUv); 22 | vec2 matcapUv = matcap(v_eye, v_normal + roughness); 23 | vec4 m = texture2D(tMatcap, matcapUv); 24 | 25 | vec3 color = particlesTexture.rgb; 26 | color += m.rgb; 27 | 28 | gl_FragColor = particlesTexture; 29 | gl_FragColor = vec4(color, 1.0); 30 | } -------------------------------------------------------------------------------- /src/webgl/shaders/vertex.glsl: -------------------------------------------------------------------------------- 1 | varying vec2 v_screenUv; 2 | varying vec3 v_normal; 3 | varying vec3 v_eye; 4 | 5 | void main() { 6 | v_normal = normalize(normalMatrix * normal); 7 | v_eye = normalize((modelViewMatrix * vec4( position, 1.0 )).xyz); 8 | 9 | gl_Position = projectionMatrix * modelViewMatrix * vec4( position, 1.0 ); 10 | 11 | v_screenUv = gl_Position.xy / gl_Position.w * 0.5 + 0.5; 12 | } -------------------------------------------------------------------------------- /src/webgl/utils/Mouse2D.ts: -------------------------------------------------------------------------------- 1 | class Mouse2D { 2 | position: [number, number] = [0, 0] 3 | 4 | constructor() { 5 | window.addEventListener('mousemove', this.handleMouseMove) 6 | window.addEventListener('touchmove', this.handleTouchMove) 7 | } 8 | 9 | private handleMouseMove = (e: MouseEvent) => { 10 | const x = (e.clientX / window.innerWidth) * 2 - 1 11 | const y = -1 * ((e.clientY / window.innerHeight) * 2 - 1) 12 | this.position = [x, y] 13 | } 14 | 15 | private handleTouchMove = (e: TouchEvent) => { 16 | const { pageX, pageY } = e.touches[0] 17 | const x = (pageX / window.innerWidth) * 2 - 1 18 | const y = -1 * ((pageY / window.innerHeight) * 2 - 1) 19 | this.position = [x, y] 20 | } 21 | 22 | dispose() { 23 | window.removeEventListener('mousemove', this.handleMouseMove) 24 | window.removeEventListener('touchmove', this.handleTouchMove) 25 | } 26 | } 27 | 28 | export const mouse2d = new Mouse2D() 29 | -------------------------------------------------------------------------------- /src/webgl/utils/OrbitControls.ts: -------------------------------------------------------------------------------- 1 | import { OrbitControls as OC } from 'three/examples/jsm/controls/OrbitControls' 2 | import { gl } from '../core/WebGL' 3 | 4 | class OrbitControls { 5 | private orbitControls: OC 6 | 7 | constructor() { 8 | this.orbitControls = new OC(gl.camera, gl.renderer.domElement) 9 | this.orbitControls.enableDamping = true 10 | this.orbitControls.dampingFactor = 0.1 11 | } 12 | 13 | disableDamping() { 14 | this.orbitControls.enableDamping = false 15 | } 16 | 17 | update() { 18 | this.orbitControls.update() 19 | } 20 | } 21 | 22 | export const controls = new OrbitControls() 23 | -------------------------------------------------------------------------------- /src/webgl/utils/assetLoader.ts: -------------------------------------------------------------------------------- 1 | import * as THREE from 'three' 2 | import { GLTF, GLTFLoader } from 'three/examples/jsm/loaders/GLTFLoader' 3 | import { RGBELoader } from 'three/examples/jsm/loaders/RGBELoader' 4 | 5 | export type Assets = { 6 | [key in string]: { 7 | data?: THREE.Texture | THREE.VideoTexture | GLTF 8 | path: string 9 | encoding?: boolean 10 | flipY?: boolean 11 | } 12 | } 13 | 14 | export async function loadAssets(assets: Assets) { 15 | const textureLoader = new THREE.TextureLoader() 16 | const gltfLoader = new GLTFLoader() 17 | const rgbeLoader = new RGBELoader() 18 | 19 | const getExtension = (path: string) => { 20 | const s = path.split('.') 21 | return s[s.length - 1] 22 | } 23 | 24 | await Promise.all( 25 | Object.values(assets).map(async (v) => { 26 | const extension = getExtension(v.path) 27 | 28 | if (['jpg', 'png', 'webp'].includes(extension)) { 29 | const texture = await textureLoader.loadAsync(v.path) 30 | texture.userData.aspect = texture.image.width / texture.image.height 31 | v.encoding && (texture.encoding = THREE.sRGBEncoding) 32 | v.flipY !== undefined && (texture.flipY = v.flipY) 33 | v.data = texture 34 | } else if (['glb'].includes(extension)) { 35 | const gltf = await gltfLoader.loadAsync(v.path) 36 | v.data = gltf 37 | } else if (['webm', 'mp4'].includes(extension)) { 38 | const video = document.createElement('video') 39 | video.src = v.path 40 | video.muted = true 41 | video.loop = true 42 | video.autoplay = true 43 | video.preload = 'metadata' 44 | video.playsInline = true 45 | // await video.play() 46 | const texture = new THREE.VideoTexture(video) 47 | texture.userData.aspect = video.videoWidth / video.videoHeight 48 | v.encoding && (texture.encoding = THREE.sRGBEncoding) 49 | v.data = texture 50 | } else if (['hdr'].includes(extension)) { 51 | const texture = await rgbeLoader.loadAsync(v.path) 52 | texture.mapping = THREE.EquirectangularReflectionMapping 53 | v.data = texture 54 | } 55 | }), 56 | ) 57 | } 58 | -------------------------------------------------------------------------------- /src/webgl/utils/coveredTexture.ts: -------------------------------------------------------------------------------- 1 | export function calcCoveredTextureScale(texture: THREE.Texture, aspect: number, target?: THREE.Vector2) { 2 | const imageAspect = texture.image.width / texture.image.height 3 | 4 | let result: [number, number] = [1, 1] 5 | if (aspect < imageAspect) result = [aspect / imageAspect, 1] 6 | else result = [1, imageAspect / aspect] 7 | 8 | target?.set(result[0], result[1]) 9 | 10 | return result 11 | } 12 | 13 | export function coveredTexture(texture: THREE.Texture, screenAspect: number) { 14 | texture.matrixAutoUpdate = false 15 | const scale = calcCoveredTextureScale(texture, screenAspect) 16 | texture.matrix.setUvTransform(0, 0, scale[0], scale[1], 0, 0.5, 0.5) 17 | 18 | return texture 19 | } 20 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "ESNext", 4 | "useDefineForClassFields": true, 5 | "module": "ESNext", 6 | "lib": ["ESNext", "DOM"], 7 | "moduleResolution": "Node", 8 | "strict": true, 9 | "sourceMap": true, 10 | "resolveJsonModule": true, 11 | "esModuleInterop": true, 12 | "noEmit": true, 13 | "noUnusedLocals": true, 14 | "noUnusedParameters": true, 15 | "noImplicitReturns": true, 16 | "skipLibCheck": true 17 | }, 18 | "include": ["src"] 19 | } 20 | -------------------------------------------------------------------------------- /vite.config.ts: -------------------------------------------------------------------------------- 1 | import path from 'path' 2 | import { defineConfig } from 'vite' 3 | import glsl from 'vite-plugin-glsl' 4 | 5 | export default defineConfig(({ mode }) => { 6 | console.log('⚓ ' + mode) 7 | return { 8 | root: './src', 9 | publicDir: '../public', 10 | base: mode === 'development' ? '/' : '/reflect-particles/', 11 | plugins: [glsl()], 12 | build: { 13 | rollupOptions: { 14 | input: { 15 | home: path.resolve(__dirname, './src/index.html'), 16 | }, 17 | }, 18 | outDir: '../dist', 19 | emptyOutDir: true, 20 | }, 21 | server: { 22 | host: true, 23 | }, 24 | } 25 | }) 26 | --------------------------------------------------------------------------------