├── src └── app │ ├── favicon.ico │ ├── rain │ ├── img │ │ ├── bg.jpg │ │ ├── drop-alpha.png │ │ ├── drop-color.png │ │ ├── drop-shine.png │ │ ├── drop-shine2.png │ │ ├── water │ │ │ ├── texture-bg.png │ │ │ └── texture-fg.png │ │ └── weather │ │ │ ├── texture-rain-bg.png │ │ │ ├── texture-rain-fg.png │ │ │ ├── texture-sun-bg.png │ │ │ ├── texture-sun-fg.png │ │ │ ├── texture-drizzle-bg.png │ │ │ ├── texture-drizzle-fg.png │ │ │ ├── texture-fallout-bg.png │ │ │ ├── texture-fallout-fg.png │ │ │ ├── texture-storm-lightning-bg.png │ │ │ └── texture-storm-lightning-fg.png │ ├── times.jsx │ ├── shaders │ │ ├── simple.vert │ │ └── water.frag │ ├── create-canvas.jsx │ ├── random.jsx │ ├── rain-utils.jsx │ ├── image-loader.jsx │ ├── gl-obj.jsx │ ├── page.jsx │ ├── webgl.jsx │ ├── RainEffect.jsx │ ├── rain-renderer.jsx │ ├── weather-utils.jsx │ └── raindrops.jsx │ ├── snow │ ├── snowflake.png │ ├── page.jsx │ ├── ShaderProgram.jsx │ └── SnowEffect.jsx │ ├── fog │ ├── fog-element.png │ ├── dense-fog-element.png │ ├── page.jsx │ └── FogEffect.jsx │ ├── page.js │ ├── globals.css │ ├── layout.js │ └── components │ └── Navbar.jsx ├── jsconfig.json ├── postcss.config.mjs ├── public └── assets │ ├── budapest-rain.png │ ├── new-york-snow.png │ └── san-francisco-fog.png ├── next.config.mjs ├── eslint.config.mjs ├── .gitignore ├── package.json ├── LICENSE └── README.md /src/app/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/favicon.ico -------------------------------------------------------------------------------- /jsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "paths": { 4 | "@/*": ["./src/*"] 5 | } 6 | } 7 | } 8 | -------------------------------------------------------------------------------- /src/app/rain/img/bg.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/bg.jpg -------------------------------------------------------------------------------- /postcss.config.mjs: -------------------------------------------------------------------------------- 1 | const config = { 2 | plugins: ["@tailwindcss/postcss"], 3 | }; 4 | 5 | export default config; 6 | -------------------------------------------------------------------------------- /src/app/snow/snowflake.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/snow/snowflake.png -------------------------------------------------------------------------------- /src/app/fog/fog-element.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/fog/fog-element.png -------------------------------------------------------------------------------- /public/assets/budapest-rain.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/public/assets/budapest-rain.png -------------------------------------------------------------------------------- /public/assets/new-york-snow.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/public/assets/new-york-snow.png -------------------------------------------------------------------------------- /src/app/rain/img/drop-alpha.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/drop-alpha.png -------------------------------------------------------------------------------- /src/app/rain/img/drop-color.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/drop-color.png -------------------------------------------------------------------------------- /src/app/rain/img/drop-shine.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/drop-shine.png -------------------------------------------------------------------------------- /src/app/fog/dense-fog-element.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/fog/dense-fog-element.png -------------------------------------------------------------------------------- /src/app/page.js: -------------------------------------------------------------------------------- 1 | import { redirect } from 'next/navigation'; 2 | 3 | export default function Home() { 4 | redirect('/snow'); 5 | } 6 | -------------------------------------------------------------------------------- /src/app/rain/img/drop-shine2.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/drop-shine2.png -------------------------------------------------------------------------------- /public/assets/san-francisco-fog.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/public/assets/san-francisco-fog.png -------------------------------------------------------------------------------- /src/app/rain/times.jsx: -------------------------------------------------------------------------------- 1 | export default function times(n,f){ 2 | for (let i = 0; i < n; i++) { 3 | f.call(this,i); 4 | } 5 | } -------------------------------------------------------------------------------- /src/app/rain/img/water/texture-bg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/water/texture-bg.png -------------------------------------------------------------------------------- /src/app/rain/img/water/texture-fg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/water/texture-fg.png -------------------------------------------------------------------------------- /src/app/rain/img/weather/texture-rain-bg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/weather/texture-rain-bg.png -------------------------------------------------------------------------------- /src/app/rain/img/weather/texture-rain-fg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/weather/texture-rain-fg.png -------------------------------------------------------------------------------- /src/app/rain/img/weather/texture-sun-bg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/weather/texture-sun-bg.png -------------------------------------------------------------------------------- /src/app/rain/img/weather/texture-sun-fg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/weather/texture-sun-fg.png -------------------------------------------------------------------------------- /src/app/rain/img/weather/texture-drizzle-bg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/weather/texture-drizzle-bg.png -------------------------------------------------------------------------------- /src/app/rain/img/weather/texture-drizzle-fg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/weather/texture-drizzle-fg.png -------------------------------------------------------------------------------- /src/app/rain/img/weather/texture-fallout-bg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/weather/texture-fallout-bg.png -------------------------------------------------------------------------------- /src/app/rain/img/weather/texture-fallout-fg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/weather/texture-fallout-fg.png -------------------------------------------------------------------------------- /src/app/rain/shaders/simple.vert: -------------------------------------------------------------------------------- 1 | precision mediump float; 2 | 3 | attribute vec2 a_position; 4 | 5 | void main() { 6 | gl_Position = vec4(a_position,0.0,1.0); 7 | } -------------------------------------------------------------------------------- /src/app/rain/img/weather/texture-storm-lightning-bg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/weather/texture-storm-lightning-bg.png -------------------------------------------------------------------------------- /src/app/rain/img/weather/texture-storm-lightning-fg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rauschermate/react-weather-effects/HEAD/src/app/rain/img/weather/texture-storm-lightning-fg.png -------------------------------------------------------------------------------- /src/app/rain/create-canvas.jsx: -------------------------------------------------------------------------------- 1 | export default function createCanvas(width, height) { 2 | if (typeof document === "undefined") { 3 | return null; 4 | } 5 | let canvas = document.createElement("canvas"); 6 | canvas.width = width; 7 | canvas.height = height; 8 | return canvas; 9 | } -------------------------------------------------------------------------------- /next.config.mjs: -------------------------------------------------------------------------------- 1 | /** @type {import('next').NextConfig} */ 2 | const nextConfig = { 3 | webpack(config) { 4 | config.module.rules.push({ 5 | test: /\.(glsl|vert|frag)$/, 6 | use: 'raw-loader', 7 | type: 'javascript/auto', 8 | }); 9 | return config; 10 | }, 11 | }; 12 | 13 | export default nextConfig; 14 | -------------------------------------------------------------------------------- /eslint.config.mjs: -------------------------------------------------------------------------------- 1 | import { dirname } from "path"; 2 | import { fileURLToPath } from "url"; 3 | import { FlatCompat } from "@eslint/eslintrc"; 4 | 5 | const __filename = fileURLToPath(import.meta.url); 6 | const __dirname = dirname(__filename); 7 | 8 | const compat = new FlatCompat({ 9 | baseDirectory: __dirname, 10 | }); 11 | 12 | const eslintConfig = [...compat.extends("next/core-web-vitals")]; 13 | 14 | export default eslintConfig; 15 | -------------------------------------------------------------------------------- /src/app/rain/random.jsx: -------------------------------------------------------------------------------- 1 | export function random(from=null,to=null,interpolation=null){ 2 | if(from==null){ 3 | from=0; 4 | to=1; 5 | }else if(from!=null && to==null){ 6 | to=from; 7 | from=0; 8 | } 9 | const delta=to-from; 10 | 11 | if(interpolation==null){ 12 | interpolation=(n)=>{ 13 | return n; 14 | } 15 | } 16 | return from+(interpolation(Math.random())*delta); 17 | } 18 | export function chance(c){ 19 | return random()<=c; 20 | } -------------------------------------------------------------------------------- /src/app/globals.css: -------------------------------------------------------------------------------- 1 | @import "tailwindcss"; 2 | 3 | :root { 4 | --background: #ffffff; 5 | --foreground: #171717; 6 | } 7 | 8 | @theme inline { 9 | --color-background: var(--background); 10 | --color-foreground: var(--foreground); 11 | --font-sans: var(--font-geist-sans); 12 | --font-mono: var(--font-geist-mono); 13 | } 14 | 15 | @media (prefers-color-scheme: dark) { 16 | :root { 17 | --background: #0a0a0a; 18 | --foreground: #ededed; 19 | } 20 | } 21 | 22 | body { 23 | background: var(--background); 24 | color: var(--foreground); 25 | font-family: Arial, Helvetica, sans-serif; 26 | } 27 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. 2 | 3 | # dependencies 4 | /node_modules 5 | /.pnp 6 | .pnp.* 7 | .yarn/* 8 | !.yarn/patches 9 | !.yarn/plugins 10 | !.yarn/releases 11 | !.yarn/versions 12 | 13 | # testing 14 | /coverage 15 | 16 | # next.js 17 | /.next/ 18 | /out/ 19 | 20 | # production 21 | /build 22 | 23 | # misc 24 | .DS_Store 25 | *.pem 26 | 27 | # debug 28 | npm-debug.log* 29 | yarn-debug.log* 30 | yarn-error.log* 31 | .pnpm-debug.log* 32 | 33 | # env files (can opt-in for committing if needed) 34 | .env* 35 | 36 | # vercel 37 | .vercel 38 | 39 | # typescript 40 | *.tsbuildinfo 41 | next-env.d.ts 42 | -------------------------------------------------------------------------------- /src/app/layout.js: -------------------------------------------------------------------------------- 1 | import { Geist, Geist_Mono } from "next/font/google"; 2 | import "./globals.css"; 3 | import Navbar from "./components/Navbar"; 4 | 5 | const geistSans = Geist({ 6 | variable: "--font-geist-sans", 7 | subsets: ["latin"], 8 | }); 9 | 10 | const geistMono = Geist_Mono({ 11 | variable: "--font-geist-mono", 12 | subsets: ["latin"], 13 | }); 14 | 15 | export const metadata = { 16 | title: "3D demo", 17 | description: "by Mate Rauscher", 18 | }; 19 | 20 | export default function RootLayout({ children }) { 21 | return ( 22 | 23 | 26 | 27 | {children} 28 | 29 | 30 | ); 31 | } 32 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "threejs", 3 | "version": "0.1.0", 4 | "private": true, 5 | "scripts": { 6 | "dev": "next dev --turbopack", 7 | "build": "next build", 8 | "start": "next start", 9 | "lint": "next lint" 10 | }, 11 | "dependencies": { 12 | "@gsap/react": "^2.1.2", 13 | "@react-three/drei": "^10.3.0", 14 | "@react-three/fiber": "^9.1.2", 15 | "gsap": "^3.13.0", 16 | "maath": "^0.10.8", 17 | "next": "15.3.4", 18 | "raw-loader": "^4.0.2", 19 | "react": "^19.0.0", 20 | "react-dom": "^19.0.0", 21 | "three": "^0.177.0" 22 | }, 23 | "devDependencies": { 24 | "@eslint/eslintrc": "^3", 25 | "@tailwindcss/postcss": "^4", 26 | "eslint": "^9", 27 | "eslint-config-next": "15.3.4", 28 | "tailwindcss": "^4" 29 | } 30 | } 31 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2024 Mate Rauscher 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /src/app/rain/rain-utils.jsx: -------------------------------------------------------------------------------- 1 | // Weather data for different types 2 | export const weatherData = { 3 | rain: { 4 | minR: 10, 5 | maxR: 40, 6 | rainChance: 0.35, 7 | rainLimit: 6, 8 | drizzle: 50, 9 | drizzleSize: [2, 4.5], 10 | raining: true, 11 | trailRate: 1, 12 | trailScaleRange: [0.2, 0.35], 13 | flashChance: 0 14 | }, 15 | storm: { 16 | minR: 15, 17 | maxR: 45, 18 | rainChance: 0.55, 19 | rainLimit: 6, 20 | drizzle: 80, 21 | drizzleSize: [2, 6], 22 | trailRate: 1, 23 | trailScaleRange: [0.15, 0.3], 24 | flashChance: 0.1 25 | }, 26 | fallout: { 27 | minR: 15, 28 | maxR: 45, 29 | rainChance: 0.45, 30 | rainLimit: 6, 31 | drizzle: 20, 32 | drizzleSize: [2, 4.5], 33 | raining: true, 34 | trailRate: 4, 35 | trailScaleRange: [0.2, 0.35], 36 | flashChance: 0.6 37 | }, 38 | drizzle: { 39 | minR: 10, 40 | maxR: 40, 41 | rainChance: 0.15, 42 | rainLimit: 2, 43 | drizzle: 10, 44 | drizzleSize: [2, 4.5], 45 | raining: true, 46 | trailRate: 1, 47 | trailScaleRange: [0.2, 0.35], 48 | flashChance: 0 49 | } 50 | }; 51 | -------------------------------------------------------------------------------- /src/app/rain/image-loader.jsx: -------------------------------------------------------------------------------- 1 | function loadImage(src,i,onLoad){ 2 | return new Promise((resolve,reject)=>{ 3 | if(typeof src=="string"){ 4 | src={ 5 | name:"image"+i, 6 | src, 7 | }; 8 | } 9 | 10 | let img=new Image(); 11 | src.img=img; 12 | 13 | img.addEventListener("load",(event)=>{ 14 | if(typeof onLoad=="function"){ 15 | onLoad.call(null,img,i); 16 | } 17 | 18 | resolve(src); 19 | }); 20 | 21 | img.src=src.src.src; 22 | }) 23 | } 24 | 25 | function loadImages(images,onLoad){ 26 | return Promise.all(images.map((src,i)=>{ 27 | return loadImage(src,i,onLoad); 28 | })); 29 | } 30 | 31 | export default function ImageLoader(images,onLoad){ 32 | return new Promise((resolve,reject)=>{ 33 | loadImages(images,onLoad).then((loadedImages)=>{ 34 | let r={}; 35 | loadedImages.forEach((curImage)=>{ 36 | r[curImage.name]={ 37 | img:curImage.img, 38 | src:curImage.src, 39 | }; 40 | }); 41 | resolve(r); 42 | }); 43 | }) 44 | } -------------------------------------------------------------------------------- /src/app/rain/gl-obj.jsx: -------------------------------------------------------------------------------- 1 | import * as WebGL from "./webgl"; 2 | 3 | function GL(canvas,options,vert,frag){ 4 | this.init(canvas,options,vert,frag); 5 | } 6 | GL.prototype={ 7 | canvas:null, 8 | gl:null, 9 | program:null, 10 | width:0, 11 | height:0, 12 | init(canvas,options,vert,frag){ 13 | this.canvas=canvas; 14 | this.width=canvas.width; 15 | this.height=canvas.height; 16 | this.gl=WebGL.getContext(canvas,options); 17 | this.program=this.createProgram(vert,frag); 18 | this.useProgram(this.program); 19 | }, 20 | createProgram(vert,frag){ 21 | let program=WebGL.createProgram(this.gl,vert,frag); 22 | return program; 23 | }, 24 | useProgram(program){ 25 | this.program=program; 26 | this.gl.useProgram(program); 27 | }, 28 | createTexture(source,i){ 29 | return WebGL.createTexture(this.gl,source,i); 30 | }, 31 | createUniform(type,name,...v){ 32 | WebGL.createUniform(this.gl,this.program,type,name,...v); 33 | }, 34 | activeTexture(i){ 35 | WebGL.activeTexture(this.gl,i); 36 | }, 37 | updateTexture(source){ 38 | WebGL.updateTexture(this.gl,source); 39 | }, 40 | draw(){ 41 | WebGL.setRectangle(this.gl, -1, -1, 2, 2); 42 | this.gl.drawArrays(this.gl.TRIANGLES, 0, 6); 43 | } 44 | } 45 | 46 | export default GL; -------------------------------------------------------------------------------- /src/app/snow/page.jsx: -------------------------------------------------------------------------------- 1 | "use client" 2 | 3 | import SnowEffect from "./SnowEffect"; 4 | import React, { useState } from "react"; 5 | 6 | const SNOW_TYPES = [ 7 | { type: "gentle", label: "Gentle" }, 8 | { type: "storm", label: "Storm" }, 9 | ]; 10 | 11 | const BG_IMAGE = "https://images.unsplash.com/photo-1518391846015-55a9cc003b25?q=80&w=3540&auto=format&fit=crop&ixlib=rb-4.1.0&ixid=M3wxMjA3fDB8MHxwaG90by1wYWdlfHx8fGVufDB8fHx8fA%3D%3D"; 12 | 13 | export default function SnowPage() { 14 | const [snowType, setSnowType] = useState("gentle"); 15 | 16 | return ( 17 | <> 18 | 19 |
28 | {SNOW_TYPES.map(({ type, label }) => ( 29 | 42 | ))} 43 |
44 | 45 | ); 46 | } -------------------------------------------------------------------------------- /src/app/fog/page.jsx: -------------------------------------------------------------------------------- 1 | "use client" 2 | 3 | import dynamic from 'next/dynamic'; 4 | import React, { useState } from "react"; 5 | 6 | const FogEffect = dynamic(() => import('./FogEffect'), { ssr: false }); 7 | 8 | const FOG_TYPES = [ 9 | { type: "light", label: "Light" }, 10 | { type: "dense", label: "Dense" }, 11 | ]; 12 | 13 | const BG_IMAGE = "https://images.unsplash.com/photo-1719858403455-9a2582eca805?q=80&w=1599&auto=format&fit=crop&ixlib=rb-4.1.0&ixid=M3wxMjA3fDB8MHxwaG90by1wYWdlfHx8fGVufDB8fHx8fA%3D%3D"; 14 | 15 | export default function FogPage() { 16 | const [fogType, setFogType] = useState("light"); 17 | 18 | return ( 19 | <> 20 | 21 |
30 | {FOG_TYPES.map(({ type, label }) => ( 31 | 44 | ))} 45 |
46 | 47 | ); 48 | } -------------------------------------------------------------------------------- /src/app/rain/page.jsx: -------------------------------------------------------------------------------- 1 | "use client" 2 | 3 | import RainEffect from "./RainEffect"; 4 | import React, { useState } from "react"; 5 | 6 | const RAIN_TYPES = [ 7 | { type: "rain", label: "Rain" }, 8 | { type: "storm", label: "Storm" }, 9 | { type: "drizzle", label: "Drizzle" }, 10 | { type: "fallout", label: "Fallout" }, 11 | ]; 12 | 13 | const BG_IMAGE = "https://images.unsplash.com/photo-1541343672885-9be56236302a?q=80&w=1587&auto=format&fit=crop&ixlib=rb-4.1.0&ixid=M3wxMjA3fDB8MHxwaG90by1wYWdlfHx8fGVufDB8fHx8fA%3D%3D"; 14 | 15 | export default function RainPage() { 16 | const [rainType, setRainType] = useState("rain"); 17 | 18 | return ( 19 | <> 20 | 21 |
30 | {RAIN_TYPES.map(({ type, label }) => ( 31 | 44 | ))} 45 |
46 | 47 | ); 48 | } -------------------------------------------------------------------------------- /src/app/components/Navbar.jsx: -------------------------------------------------------------------------------- 1 | "use client"; 2 | 3 | import Link from "next/link"; 4 | import { usePathname } from 'next/navigation'; 5 | 6 | import Image from 'next/image'; 7 | 8 | const NAV_ITEMS = [ 9 | { 10 | href: '/rain', 11 | label: 'Rain', 12 | icon: '/assets/budapest-rain.png', 13 | }, 14 | { 15 | href: '/snow', 16 | label: 'Snow', 17 | icon: '/assets/new-york-snow.png', 18 | }, 19 | { 20 | href: '/fog', 21 | label: 'Fog', 22 | icon: '/assets/san-francisco-fog.png', 23 | }, 24 | ]; 25 | 26 | export default function Navbar() { 27 | const pathname = usePathname(); 28 | return ( 29 | 68 | ); 69 | } -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # 🌦️ Animated React Weather Effects 2 | 3 | A beautiful, interactive weather effects demo built with React, Next.js, WebGL, and Three.js. Experience realistic **rain**, **snow**, and **fog** scenes, each with unique visual effects and controls. 4 | 5 | ## 🚀 Demo 6 | 7 | [Live demo](https://react-weather-effects.vercel.app/) 8 | 9 | ## ✨ Features 10 | 11 | - **Rain:** Realistic WebGL rain with custom shaders, and lightning effects. 12 | - **Snow:** Gentle and stormy snow scenes powered by Three.js particle systems. 13 | - **Fog:** Light and dense fog overlays using Three.js and custom blending. 14 | - **Interactive:** Switch between weather types and subtypes (e.g., storm, drizzle) with a modern UI. 15 | - **Responsive:** Works on desktop and mobile browsers. 16 | 17 | ## 🌈 Weather Types 18 | 19 | - **Rain** 20 | - Rain 21 | - Storm (with lightning) 22 | - Drizzle 23 | - Fallout 24 | - **Snow** 25 | - Gentle 26 | - Storm 27 | - **Fog** 28 | - Light 29 | - Dense 30 | 31 | ## 🛠️ Technology 32 | 33 | - **React** & **Next.js** (App Router) 34 | - **WebGL** (custom shaders for rain) 35 | - **Three.js** (snow and fog effects) 36 | - **GSAP** (for smooth lightning and fog animations) 37 | - **Tailwind CSS** (for modern UI) 38 | 39 | ## 🖥️ Local Development 40 | 41 | ```bash 42 | npm install 43 | npm run dev 44 | ``` 45 | 46 | Open [http://localhost:3000](http://localhost:3000) in your browser. 47 | 48 | ## 🗂️ Project Structure 49 | 50 | - `/src/app/rain/` – Rain effect (WebGL, shaders, rain types) 51 | - `/src/app/snow/` – Snow effect (Three.js, snow types) 52 | - `/src/app/fog/` – Fog effect (Three.js, fog types) 53 | - `/src/app/components/Navbar.jsx` – Navigation bar for switching weather types 54 | 55 | --- 56 | 57 | ## 🙏 Credits 58 | 59 | - **Rain shaders & inspiration:** 60 | [Lucas Bebber – RainEffect](https://github.com/codrops/RainEffect) 61 | [Shadertoy rain shader](https://www.shadertoy.com/view/ltffzl) 62 | - **Snow & fog inspiration:** 63 | [React three Fiber docs](https://r3f.docs.pmnd.rs/getting-started/examples) 64 | [Freezing cube](https://boytchev.github.io/etudes/webgl/freezing-cube.html) 65 | [Images from Unsplash](https://unsplash.com) 66 | [Snow effect inspiration #1](https://codepen.io/bsehovac/pen/GPwXxq) 67 | [Snow effect inspiration #2](https://codepen.io/bsehovac/full/GPwXxq) 68 | [Smoke effect](https://codepen.io/daniel3toma/pen/dybjNbZ) 69 | 70 | --- 71 | 72 | ## 📄 License 73 | 74 | MIT 75 | -------------------------------------------------------------------------------- /src/app/rain/shaders/water.frag: -------------------------------------------------------------------------------- 1 | 2 | precision mediump float; 3 | 4 | // textures 5 | uniform sampler2D u_waterMap; 6 | uniform sampler2D u_textureShine; 7 | uniform sampler2D u_textureFg; 8 | uniform sampler2D u_textureBg; 9 | 10 | // the texCoords passed in from the vertex shader. 11 | varying vec2 v_texCoord; 12 | uniform vec2 u_resolution; 13 | uniform vec2 u_parallax; 14 | uniform float u_parallaxFg; 15 | uniform float u_parallaxBg; 16 | uniform float u_textureRatio; 17 | uniform bool u_renderShine; 18 | uniform bool u_renderShadow; 19 | uniform float u_minRefraction; 20 | uniform float u_refractionDelta; 21 | uniform float u_brightness; 22 | uniform float u_alphaMultiply; 23 | uniform float u_alphaSubtract; 24 | 25 | // alpha-blends two colors 26 | vec4 blend(vec4 bg,vec4 fg){ 27 | vec3 bgm=bg.rgb*bg.a; 28 | vec3 fgm=fg.rgb*fg.a; 29 | float ia=1.0-fg.a; 30 | float a=(fg.a + bg.a * ia); 31 | vec3 rgb; 32 | if(a!=0.0){ 33 | rgb=(fgm + bgm * ia) / a; 34 | }else{ 35 | rgb=vec3(0.0,0.0,0.0); 36 | } 37 | return vec4(rgb,a); 38 | } 39 | 40 | vec2 pixel(){ 41 | return vec2(1.0,1.0)/u_resolution; 42 | } 43 | 44 | vec2 parallax(float v){ 45 | return u_parallax*pixel()*v; 46 | } 47 | 48 | vec2 texCoord(){ 49 | return vec2(gl_FragCoord.x, u_resolution.y-gl_FragCoord.y)/u_resolution; 50 | } 51 | 52 | // scales the bg up and proportionally to fill the container 53 | vec2 scaledTexCoord(){ 54 | float ratio=u_resolution.x/u_resolution.y; 55 | vec2 scale=vec2(1.0,1.0); 56 | vec2 offset=vec2(0.0,0.0); 57 | float ratioDelta=ratio-u_textureRatio; 58 | if(ratioDelta>=0.0){ 59 | scale.y=(1.0+ratioDelta); 60 | offset.y=ratioDelta/2.0; 61 | }else{ 62 | scale.x=(1.0-ratioDelta); 63 | offset.x=-ratioDelta/2.0; 64 | } 65 | return (texCoord()+offset)/scale; 66 | } 67 | 68 | // get color from fg 69 | vec4 fgColor(float x, float y){ 70 | float p2=u_parallaxFg*2.0; 71 | vec2 scale=vec2( 72 | (u_resolution.x+p2)/u_resolution.x, 73 | (u_resolution.y+p2)/u_resolution.y 74 | ); 75 | 76 | vec2 scaledTexCoord=texCoord()/scale; 77 | vec2 offset=vec2( 78 | (1.0-(1.0/scale.x))/2.0, 79 | (1.0-(1.0/scale.y))/2.0 80 | ); 81 | 82 | return texture2D(u_waterMap, 83 | (scaledTexCoord+offset)+(pixel()*vec2(x,y))+parallax(u_parallaxFg) 84 | ); 85 | } 86 | 87 | void main() { 88 | vec4 bg=texture2D(u_textureBg,scaledTexCoord()+parallax(u_parallaxBg)); 89 | 90 | vec4 cur = fgColor(0.0,0.0); 91 | 92 | float d=cur.b; // "thickness" 93 | float x=cur.g; 94 | float y=cur.r; 95 | 96 | float a=clamp(cur.a*u_alphaMultiply-u_alphaSubtract, 0.0,1.0); 97 | 98 | vec2 refraction = (vec2(x,y)-0.5)*2.0; 99 | vec2 refractionParallax=parallax(u_parallaxBg-u_parallaxFg); 100 | vec2 refractionPos = scaledTexCoord() 101 | + (pixel()*refraction*(u_minRefraction+(d*u_refractionDelta))) 102 | + refractionParallax; 103 | 104 | vec4 tex=texture2D(u_textureFg,refractionPos); 105 | 106 | if(u_renderShine){ 107 | float maxShine=490.0; 108 | float minShine=maxShine*0.18; 109 | vec2 shinePos=vec2(0.5,0.5) + ((1.0/512.0)*refraction)* -(minShine+((maxShine-minShine)*d)); 110 | vec4 shine=texture2D(u_textureShine,shinePos); 111 | tex=blend(tex,shine); 112 | } 113 | 114 | vec4 fg=vec4(tex.rgb*u_brightness,a); 115 | 116 | if(u_renderShadow){ 117 | float borderAlpha = fgColor(0.,0.-(d*6.0)).a; 118 | borderAlpha=borderAlpha*u_alphaMultiply-(u_alphaSubtract+0.5); 119 | borderAlpha=clamp(borderAlpha,0.,1.); 120 | borderAlpha*=0.2; 121 | vec4 border=vec4(0.,0.,0.,borderAlpha); 122 | fg=blend(border,fg); 123 | } 124 | 125 | gl_FragColor = blend(bg,fg); 126 | } 127 | -------------------------------------------------------------------------------- /src/app/rain/webgl.jsx: -------------------------------------------------------------------------------- 1 | export function getContext(canvas, options={}) { 2 | let contexts = ["webgl", "experimental-webgl"]; 3 | let context = null; 4 | 5 | contexts.some(name=>{ 6 | try{ 7 | context = canvas.getContext(name,options); 8 | }catch(e){}; 9 | return context!=null; 10 | }); 11 | 12 | if(context==null){ 13 | document.body.classList.add("no-webgl"); 14 | } 15 | 16 | return context; 17 | } 18 | 19 | export function createProgram(gl,vertexScript,fragScript){ 20 | let vertexShader = createShader(gl, vertexScript, gl.VERTEX_SHADER); 21 | let fragShader = createShader(gl, fragScript, gl.FRAGMENT_SHADER); 22 | 23 | let program = gl.createProgram(); 24 | gl.attachShader(program, vertexShader); 25 | gl.attachShader(program, fragShader); 26 | 27 | gl.linkProgram(program); 28 | 29 | let linked = gl.getProgramParameter(program, gl.LINK_STATUS); 30 | if (!linked) { 31 | var lastError = gl.getProgramInfoLog(program); 32 | error("Error in program linking: " + lastError); 33 | gl.deleteProgram(program); 34 | return null; 35 | } 36 | 37 | var positionLocation = gl.getAttribLocation(program, "a_position"); 38 | var texCoordLocation = gl.getAttribLocation(program, "a_texCoord"); 39 | 40 | var texCoordBuffer = gl.createBuffer(); 41 | gl.bindBuffer(gl.ARRAY_BUFFER, texCoordBuffer); 42 | gl.bufferData(gl.ARRAY_BUFFER, new Float32Array([ 43 | -1.0, -1.0, 44 | 1.0, -1.0, 45 | -1.0, 1.0, 46 | -1.0, 1.0, 47 | 1.0, -1.0, 48 | 1.0, 1.0 49 | ]), gl.STATIC_DRAW); 50 | gl.enableVertexAttribArray(texCoordLocation); 51 | gl.vertexAttribPointer(texCoordLocation, 2, gl.FLOAT, false, 0, 0); 52 | 53 | // Create a buffer for the position of the rectangle corners. 54 | var buffer = gl.createBuffer(); 55 | gl.bindBuffer(gl.ARRAY_BUFFER, buffer); 56 | gl.enableVertexAttribArray(positionLocation); 57 | gl.vertexAttribPointer(positionLocation, 2, gl.FLOAT, false, 0, 0); 58 | 59 | return program; 60 | } 61 | 62 | export function createShader(gl,script,type){ 63 | let shader = gl.createShader(type); 64 | gl.shaderSource(shader,script); 65 | gl.compileShader(shader); 66 | 67 | let compiled = gl.getShaderParameter(shader, gl.COMPILE_STATUS); 68 | 69 | if (!compiled) { 70 | let lastError = gl.getShaderInfoLog(shader); 71 | error("Error compiling shader '" + shader + "':" + lastError); 72 | gl.deleteShader(shader); 73 | return null; 74 | } 75 | return shader; 76 | } 77 | export function createTexture(gl,source,i){ 78 | var texture = gl.createTexture(); 79 | activeTexture(gl,i); 80 | gl.bindTexture(gl.TEXTURE_2D, texture); 81 | 82 | // Set the parameters so we can render any size image. 83 | gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); 84 | gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); 85 | gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR); 86 | gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR); 87 | 88 | if ( source == null ) { 89 | return texture; 90 | } else { 91 | updateTexture(gl,source); 92 | } 93 | 94 | return texture; 95 | } 96 | export function createUniform(gl,program,type,name,...args){ 97 | let location=gl.getUniformLocation(program,"u_"+name); 98 | gl["uniform"+type](location,...args); 99 | } 100 | export function activeTexture(gl,i){ 101 | gl.activeTexture(gl["TEXTURE"+i]); 102 | } 103 | export function updateTexture(gl, source) { 104 | const isCanvas = typeof HTMLCanvasElement !== "undefined" && source instanceof HTMLCanvasElement; 105 | const isImage = typeof HTMLImageElement !== "undefined" && source instanceof HTMLImageElement; 106 | const isVideo = typeof HTMLVideoElement !== "undefined" && source instanceof HTMLVideoElement; 107 | if (!source || (!isCanvas && !isImage && !isVideo)) { 108 | console.warn("updateTexture: source is not a valid canvas/image/video", source, new Error().stack); 109 | return; 110 | } 111 | gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, source); 112 | } 113 | export function setRectangle(gl, x, y, width, height) { 114 | var x1 = x; 115 | var x2 = x + width; 116 | var y1 = y; 117 | var y2 = y + height; 118 | gl.bufferData(gl.ARRAY_BUFFER, new Float32Array([ 119 | x1, y1, 120 | x2, y1, 121 | x1, y2, 122 | x1, y2, 123 | x2, y1, 124 | x2, y2]), gl.STATIC_DRAW); 125 | } 126 | 127 | function error(msg){ 128 | console.error(msg); 129 | } -------------------------------------------------------------------------------- /src/app/rain/RainEffect.jsx: -------------------------------------------------------------------------------- 1 | import React, { useEffect } from 'react'; 2 | 3 | import RainRenderer from "./rain-renderer"; 4 | import TweenLite from 'gsap'; 5 | import times from './times'; 6 | import {random,chance} from './random'; 7 | import Raindrops from "./raindrops"; 8 | import loadImages from "./image-loader"; 9 | import createCanvas from "./create-canvas"; 10 | import { weatherData } from './rain-utils'; 11 | 12 | import DropColor from './img/drop-color.png'; 13 | import DropAlpha from './img/drop-alpha.png'; 14 | 15 | 16 | const RainEffect = (props) => { 17 | const {type, backgroundImageUrl } = props; 18 | 19 | let canvas, dropAlpha, dropColor, raindrops, textureFg, textureFgCtx, textureBg, textureBgCtx, renderer, curWeatherData; 20 | let backgroundImage = null; 21 | let intervalId = undefined; 22 | let blend = {v:0}; 23 | 24 | let textureFgSize = { 25 | width: 100, 26 | height: 100 27 | } 28 | let textureBgSize = { 29 | width: 100, 30 | height: 100 31 | } 32 | 33 | 34 | useEffect(() => { 35 | setBackgroundImage(backgroundImageUrl); 36 | },[backgroundImageUrl, type]); 37 | 38 | // Set the background image and initialize rain effect after image loads 39 | const setBackgroundImage = (backgroundImageUrl) => { 40 | if (typeof window === 'undefined') return; // SSR safety 41 | backgroundImage = new window.Image(); 42 | backgroundImage.crossOrigin = 'anonymous'; 43 | backgroundImage.src = backgroundImageUrl; 44 | 45 | backgroundImage.onload = () => { 46 | textureFgSize = { 47 | width: backgroundImage.naturalWidth, 48 | height: backgroundImage.naturalHeight 49 | } 50 | textureBgSize = { 51 | width: backgroundImage.naturalWidth, 52 | height: backgroundImage.naturalHeight 53 | } 54 | } 55 | 56 | 57 | 58 | loadTextures().then(() => { 59 | init(type); 60 | }); 61 | } 62 | 63 | // Load drop textures 64 | const loadTextures = () => { 65 | return loadImages([ 66 | { name:"dropAlpha", src: DropAlpha }, 67 | { name:"dropColor", src: DropColor }, 68 | ]).then(function (images){ 69 | dropColor = images.dropColor.img; 70 | dropAlpha = images.dropAlpha.img; 71 | }); 72 | } 73 | 74 | const init = (type = 'rain') => { 75 | canvas = document.querySelector('#container-weather'); 76 | const container = canvas.parentElement; // or a specific container element 77 | const dpi = window.devicePixelRatio || 1; 78 | 79 | const rect = container.getBoundingClientRect(); 80 | 81 | canvas.width = rect.width * dpi; 82 | canvas.height = rect.height * dpi; 83 | 84 | canvas.style.width = `${rect.width}px`; 85 | canvas.style.height = `${rect.height}px`; 86 | 87 | 88 | raindrops=new Raindrops( 89 | canvas.width, 90 | canvas.height, 91 | dpi, 92 | dropAlpha, 93 | dropColor,{ 94 | trailRate:1, 95 | trailScaleRange:[0.2,0.45], 96 | collisionRadius : 0.45, 97 | dropletsCleaningRadiusMultiplier : 0.28, 98 | } 99 | ); 100 | 101 | 102 | textureFg = createCanvas(textureFgSize.width,textureFgSize.height); 103 | textureFgCtx = textureFg.getContext('2d'); 104 | textureBg = createCanvas(textureBgSize.width,textureBgSize.height); 105 | 106 | textureBgCtx = textureBg.getContext('2d'); 107 | 108 | 109 | generateTextures(backgroundImage, backgroundImage); 110 | renderer = new RainRenderer(canvas, raindrops.canvas, textureFg, textureBg, null,{ 111 | brightness:1.04, 112 | alphaMultiply:16, 113 | alphaSubtract:4, 114 | minRefraction: 128 115 | // minRefraction:256, 116 | // maxRefraction:512 117 | }); 118 | 119 | curWeatherData = { ...weatherData[type], fg: backgroundImage, bg: backgroundImage }; 120 | if (type === 'storm' || type === 'fallout') { 121 | setupLightningFlicker(); 122 | } 123 | if (raindrops && curWeatherData) { 124 | raindrops.options = Object.assign(raindrops.options, curWeatherData); 125 | raindrops.clearDrops(); 126 | } 127 | } 128 | 129 | // Generate foreground/background textures for the rain renderer 130 | const generateTextures = (fg, bg, x=0, y=0, alpha=1) => { 131 | if (!fg || !bg || (fg instanceof HTMLImageElement && !fg.complete) || (bg instanceof HTMLImageElement && !bg.complete)) { return; } 132 | textureFgCtx.globalAlpha = alpha; 133 | textureFgCtx.drawImage(fg, x, y, textureFgSize.width, textureFgSize.height); 134 | textureBgCtx.globalAlpha = alpha; 135 | textureBgCtx.drawImage(bg, x, y, textureBgSize.width, textureBgSize.height); 136 | } 137 | 138 | // Lightning flicker effect for storm weather 139 | const setupLightningFlicker = () => { 140 | const minInterval = 1000; // minimum time between flickers 141 | const maxInterval = 5000; // maximum time between flickers 142 | const flashChance = curWeatherData && typeof curWeatherData.flashChance === 'number' ? curWeatherData.flashChance : 0; 143 | const interval = minInterval + (maxInterval - minInterval) * (1 - flashChance); 144 | 145 | intervalId = setInterval(() => { 146 | const flicker = Math.random() * 2.0; 147 | if (renderer && renderer.gl) { 148 | renderer.gl.useProgram(renderer.programWater); 149 | renderer.gl.createUniform("1f", "lightningFlash", flicker); 150 | setTimeout(() => { 151 | renderer.gl.useProgram(renderer.programWater); 152 | renderer.gl.createUniform("1f", "lightningFlash", 0.0); 153 | }, 100 + Math.random() * 200); // Flicker lasts 100-300ms 154 | } 155 | }, interval); 156 | } 157 | 158 | 159 | return ( 160 |
161 | 162 |
163 | ) 164 | 165 | } 166 | 167 | export default RainEffect; -------------------------------------------------------------------------------- /src/app/fog/FogEffect.jsx: -------------------------------------------------------------------------------- 1 | "use client"; 2 | 3 | import React, { useRef, useMemo } from 'react'; 4 | import { Canvas, useFrame, useLoader } from '@react-three/fiber'; 5 | import * as THREE from 'three'; 6 | import fogElementImg from './fog-element.png'; 7 | import denseFogElementImg from './dense-fog-element.png'; 8 | import gsap from 'gsap'; 9 | 10 | // Settings for light and dense fog 11 | const LIGHT_FOG_SETTINGS = { 12 | count: 18, 13 | fogElementRatio: 0.8, // 80% fog-element, 20% dense-fog-element 14 | alphaMin: 0.18, 15 | alphaMax: 0.32, 16 | scaleMin: 0.7, 17 | scaleMax: 1.7, 18 | moveSpeed: 0.02, 19 | }; 20 | const DENSE_FOG_SETTINGS = { 21 | count: 38, 22 | fogElementRatio: 0.35, // 35% fog-element, 65% dense-fog-element 23 | alphaMin: 0.10, 24 | alphaMax: 0.22, 25 | scaleMin: 1.0, 26 | scaleMax: 2.2, 27 | moveSpeed: 0.04, 28 | }; 29 | 30 | function FogSprite({ texture, initial, moveSpeed, windOffset }) { 31 | const mesh = useRef(); 32 | const visibleRef = useRef(false); // Track if currently visible 33 | useFrame((state) => { 34 | if (mesh.current) { 35 | const t = state.clock.getElapsedTime(); 36 | // Add wind offset to base position 37 | let x = initial.x + (windOffset?.current?.x || 0) + Math.sin(t * initial.driftSpeedX + initial.driftPhaseX) * initial.driftAmountX; 38 | let y = initial.y + (windOffset?.current?.y || 0) + Math.cos(t * initial.driftSpeedY + initial.driftPhaseY) * initial.driftAmountY; 39 | // Loop position in [-1.1, 1.1] for seamless looping 40 | if (x > 1.1) x -= 2.2; 41 | if (x < -1.1) x += 2.2; 42 | if (y > 1.1) y -= 2.2; 43 | if (y < -1.1) y += 2.2; 44 | mesh.current.position.x = x; 45 | mesh.current.position.y = y; 46 | // Determine if inside visible area 47 | const isVisible = x > -1 && x < 1 && y > -1 && y < 1; 48 | // Animate opacity with GSAP only on enter/exit 49 | if (mesh.current.material) { 50 | if (isVisible && !visibleRef.current) { 51 | // Entering: fade in 52 | gsap.to(mesh.current.material, { opacity: initial.alpha, duration: .5, overwrite: true }); 53 | visibleRef.current = true; 54 | } else if (!isVisible && visibleRef.current) { 55 | // Exiting: fade out 56 | gsap.to(mesh.current.material, { opacity: 0, duration: .5, overwrite: true }); 57 | visibleRef.current = false; 58 | } 59 | } 60 | } 61 | }); 62 | return ( 63 | 64 | 65 | 71 | 72 | ); 73 | } 74 | 75 | // Add a helper component for global wind animation 76 | function FogWindController({ windAngle, windOffset }) { 77 | useFrame(() => { 78 | // Occasionally nudge wind angle 79 | if (Math.random() > 0.995) { 80 | windAngle.current += (Math.random() - 0.5) * 0.2; // Small nudge 81 | // Clamp angle to [-PI, PI] 82 | if (windAngle.current > Math.PI) windAngle.current -= 2 * Math.PI; 83 | if (windAngle.current < -Math.PI) windAngle.current += 2 * Math.PI; 84 | } 85 | // Much slower wind movement 86 | const windSpeed = 0.0007; // Much slower 87 | windOffset.current.x += Math.cos(windAngle.current) * windSpeed; 88 | windOffset.current.y += Math.sin(windAngle.current) * windSpeed; 89 | // Loop wind offset in [-1, 1] for seamless looping 90 | if (windOffset.current.x > 1) windOffset.current.x -= 2; 91 | if (windOffset.current.x < -1) windOffset.current.x += 2; 92 | if (windOffset.current.y > 1) windOffset.current.y -= 2; 93 | if (windOffset.current.y < -1) windOffset.current.y += 2; 94 | }); 95 | return null; 96 | } 97 | 98 | export default function FogEffect({ backgroundImageUrl, type = 'light' }) { 99 | // Load both fog textures 100 | const [fogElement, denseFogElement] = useLoader(THREE.TextureLoader, [ 101 | fogElementImg.src, 102 | denseFogElementImg.src, 103 | ]); 104 | const settings = type === 'dense' ? DENSE_FOG_SETTINGS : LIGHT_FOG_SETTINGS; 105 | 106 | // Wind angle (in radians) and wind offset 107 | const windAngle = React.useRef(0); // 0 = right 108 | const windOffset = React.useRef({ x: 0, y: 0 }); 109 | 110 | // Precompute sprite initial states 111 | const sprites = useMemo(() => { 112 | return Array.from({ length: settings.count }).map(() => { 113 | // Randomly choose which texture to use 114 | const useFogElement = Math.random() < settings.fogElementRatio; 115 | const texture = useFogElement ? fogElement : denseFogElement; 116 | // Random position in [-1, 1] (covering the screen) 117 | const x = (Math.random() - 0.5) * 2.2; 118 | const y = (Math.random() - 0.5) * 2.2; 119 | // Random scale 120 | const scale = settings.scaleMin + Math.random() * (settings.scaleMax - settings.scaleMin); 121 | // Random alpha 122 | const alpha = settings.alphaMin + Math.random() * (settings.alphaMax - settings.alphaMin); 123 | // Random drift parameters 124 | const driftAmountX = 0.08 + Math.random() * 0.18; 125 | const driftAmountY = 0.08 + Math.random() * 0.18; 126 | const driftSpeedX = settings.moveSpeed * (0.7 + Math.random() * 0.6); 127 | const driftSpeedY = settings.moveSpeed * (0.7 + Math.random() * 0.6); 128 | const driftPhaseX = Math.random() * Math.PI * 2; 129 | const driftPhaseY = Math.random() * Math.PI * 2; 130 | return { 131 | texture, 132 | x, 133 | y, 134 | scale, 135 | alpha, 136 | driftAmountX, 137 | driftAmountY, 138 | driftSpeedX, 139 | driftSpeedY, 140 | driftPhaseX, 141 | driftPhaseY, 142 | }; 143 | }); 144 | }, [fogElement, denseFogElement, settings]); 145 | 146 | return ( 147 |
157 | 161 | 162 | {sprites.map((init, i) => ( 163 | 170 | ))} 171 | 172 |
173 | ); 174 | } -------------------------------------------------------------------------------- /src/app/rain/rain-renderer.jsx: -------------------------------------------------------------------------------- 1 | // import * as WebGL from "./webgl.jsx"; 2 | import GL from "./gl-obj.jsx"; 3 | // import loadImages from "./image-loader.jsx"; 4 | import createCanvas from "./create-canvas.jsx"; 5 | // let requireShaderScript = require("glslify"); 6 | 7 | // let vertShader = requireShaderScript('./shaders/simple.vert'); 8 | // let fragShader = requireShaderScript('./shaders/water.frag'); 9 | 10 | let vertShader = 11 | 'precision mediump float;\n' + 12 | 'attribute vec2 a_position;\n' + 13 | 'void main() {\n' + 14 | ' gl_Position = vec4(a_position,0.0,1.0);\n' + 15 | '}\n' ; 16 | 17 | let fragShader = 18 | ' precision mediump float;\n ' + 19 | ' \n' + 20 | ' // textures \n' + 21 | ' uniform sampler2D u_waterMap;\n ' + 22 | ' uniform sampler2D u_textureShine;\n ' + 23 | ' uniform sampler2D u_textureFg;\n ' + 24 | ' uniform sampler2D u_textureBg;\n ' + 25 | ' ' + 26 | ' // the texCoords passed in from the vertex shader. \n' + 27 | ' varying vec2 v_texCoord;\n ' + 28 | ' uniform vec2 u_resolution;\n ' + 29 | ' uniform vec2 u_parallax;\n ' + 30 | ' uniform float u_parallaxFg;\n ' + 31 | ' uniform float u_parallaxBg;\n ' + 32 | ' uniform float u_textureRatio;\n ' + 33 | ' uniform bool u_renderShine;\n ' + 34 | ' uniform bool u_renderShadow;\n ' + 35 | ' uniform float u_minRefraction;\n ' + 36 | ' uniform float u_refractionDelta;\n ' + 37 | ' uniform float u_brightness;\n ' + 38 | ' uniform float u_alphaMultiply;\n ' + 39 | ' uniform float u_alphaSubtract;\n ' + 40 | ' uniform float u_lightningFlash;\n ' + 41 | ' ' + 42 | ' // alpha-blends two colors \n' + 43 | ' vec4 blend(vec4 bg,vec4 fg){ \n' + 44 | ' vec3 bgm=bg.rgb*bg.a;\n ' + 45 | ' vec3 fgm=fg.rgb*fg.a;\n ' + 46 | ' float ia=1.0-fg.a;\n ' + 47 | ' float a=(fg.a + bg.a * ia);\n ' + 48 | ' vec3 rgb;\n ' + 49 | ' if(a!=0.0){ \n' + 50 | ' rgb=(fgm + bgm * ia) / a;\n ' + 51 | ' }else{ \n' + 52 | ' rgb=vec3(0.0,0.0,0.0);\n ' + 53 | ' } \n' + 54 | ' return vec4(rgb,a);\n ' + 55 | ' } \n' + 56 | ' \n' + 57 | ' vec2 pixel(){ \n' + 58 | ' return vec2(1.0,1.0)/u_resolution;\n ' + 59 | ' } \n' + 60 | ' \n' + 61 | ' vec2 parallax(float v){ \n' + 62 | ' return u_parallax*pixel()*v;\n ' + 63 | ' } \n' + 64 | ' \n' + 65 | ' vec2 texCoord(){ \n' + 66 | ' return vec2(gl_FragCoord.x, u_resolution.y-gl_FragCoord.y)/u_resolution;\n ' + 67 | ' } \n' + 68 | ' \n' + 69 | ' // scales the bg up and proportionally to fill the container \n' + 70 | ' vec2 scaledTexCoord(){ \n' + 71 | ' float ratio=u_resolution.x/u_resolution.y;\n ' + 72 | ' vec2 scale=vec2(1.0,1.0);\n ' + 73 | ' vec2 offset=vec2(0.0,0.0);\n ' + 74 | ' float ratioDelta=ratio-u_textureRatio;\n ' + 75 | ' if(ratioDelta>=0.0){ \n' + 76 | ' scale.y=(1.0+ratioDelta);\n ' + 77 | ' offset.y=ratioDelta/2.0;\n ' + 78 | ' }else{ \n' + 79 | ' scale.x=(1.0-ratioDelta);\n ' + 80 | ' offset.x=-ratioDelta/2.0;\n ' + 81 | ' } \n' + 82 | ' return (texCoord()+offset)/scale;\n ' + 83 | ' } \n' + 84 | ' \n' + 85 | ' // get color from fg \n' + 86 | ' vec4 fgColor(float x, float y){ \n' + 87 | ' float p2=u_parallaxFg*2.0;\n ' + 88 | ' vec2 scale=vec2( \n' + 89 | ' (u_resolution.x+p2)/u_resolution.x, \n' + 90 | ' (u_resolution.y+p2)/u_resolution.y \n' + 91 | ' );\n ' + 92 | ' \n' + 93 | ' vec2 scaledTexCoord=texCoord()/scale;\n ' + 94 | ' vec2 offset=vec2( \n' + 95 | ' (1.0-(1.0/scale.x))/2.0, \n' + 96 | ' (1.0-(1.0/scale.y))/2.0 \n' + 97 | ' );\n ' + 98 | ' \n' + 99 | ' return texture2D(u_waterMap, \n' + 100 | ' (scaledTexCoord+offset)+(pixel()*vec2(x,y))+parallax(u_parallaxFg) \n' + 101 | ' );\n ' + 102 | ' } \n' + 103 | ' \n' + 104 | ' void main() { \n' + 105 | ' vec4 bg=texture2D(u_textureBg,scaledTexCoord()+parallax(u_parallaxBg));\n ' + 106 | ' bg.rgb *= (1.0 + 0.5 * u_lightningFlash);\n ' + 107 | ' \n' + 108 | ' vec4 cur = fgColor(0.0,0.0);\n ' + 109 | ' ' + 110 | ' float d=cur.b;\n // "thickness" \n' + 111 | ' float x=cur.g;\n ' + 112 | ' float y=cur.r;\n ' + 113 | ' \n' + 114 | ' float a=clamp(cur.a*u_alphaMultiply-u_alphaSubtract, 0.0,1.0);\n ' + 115 | ' \n' + 116 | ' vec2 refraction = (vec2(x,y)-0.5)*2.0;\n ' + 117 | ' vec2 refractionParallax=parallax(u_parallaxBg-u_parallaxFg);\n ' + 118 | ' vec2 refractionPos = scaledTexCoord() \n' + 119 | ' + (pixel()*refraction*(u_minRefraction+(d*u_refractionDelta))) \n' + 120 | ' + refractionParallax;\n ' + 121 | ' \n' + 122 | ' vec4 tex=texture2D(u_textureFg,refractionPos);\n ' + 123 | ' \n' + 124 | ' if(u_renderShine){ \n' + 125 | ' float maxShine=490.0;\n ' + 126 | ' float minShine=maxShine*0.18;\n ' + 127 | ' vec2 shinePos=vec2(0.5,0.5) + ((1.0/512.0)*refraction)* -(minShine+((maxShine-minShine)*d));\n ' + 128 | ' vec4 shine=texture2D(u_textureShine,shinePos);\n ' + 129 | ' tex=blend(tex,shine);\n ' + 130 | ' } \n' + 131 | ' \n' + 132 | ' vec4 fg=vec4(tex.rgb*u_brightness*a*(1.0 + 0.5 * u_lightningFlash),a);\n ' + 133 | ' \n' + 134 | ' if(u_renderShadow){ \n' + 135 | ' float borderAlpha = fgColor(0.,0.-(d*6.0)).a;\n ' + 136 | ' borderAlpha=borderAlpha*u_alphaMultiply-(u_alphaSubtract+0.5);\n ' + 137 | ' borderAlpha=clamp(borderAlpha,0.,1.);\n ' + 138 | ' borderAlpha*=0.2;\n ' + 139 | ' vec4 border=vec4(0.,0.,0.,borderAlpha);\n ' + 140 | ' fg=blend(border,fg);\n ' + 141 | ' } \n' + 142 | ' \n' + 143 | ' gl_FragColor = blend(bg,fg);\n ' + 144 | ' } \n'; 145 | 146 | const defaultOptions={ 147 | renderShadow:false, 148 | minRefraction:256, 149 | maxRefraction:512, 150 | brightness:1, 151 | alphaMultiply:20, 152 | alphaSubtract:5, 153 | parallaxBg:5, 154 | parallaxFg:20 155 | } 156 | function RainRenderer(canvas,canvasLiquid, imageFg, imageBg, imageShine=null,options={}){ 157 | this.canvas=canvas; 158 | this.canvasLiquid=canvasLiquid; 159 | this.imageShine=imageShine; 160 | this.imageFg=imageFg; 161 | this.imageBg=imageBg; 162 | this.options=Object.assign({},defaultOptions, options); 163 | this.init(); 164 | } 165 | 166 | RainRenderer.prototype={ 167 | canvas:null, 168 | gl:null, 169 | canvasLiquid:null, 170 | width:0, 171 | height:0, 172 | imageShine:"", 173 | imageFg:"", 174 | imageBg:"", 175 | textures:null, 176 | programWater:null, 177 | programBlurX:null, 178 | programBlurY:null, 179 | parallaxX:0, 180 | parallaxY:0, 181 | renderShadow:false, 182 | options:null, 183 | init(){ 184 | this.width=this.canvas.width; 185 | this.height=this.canvas.height; 186 | this.gl=new GL(this.canvas, {alpha:false},vertShader,fragShader); 187 | let gl=this.gl; 188 | this.programWater=gl.program; 189 | 190 | gl.createUniform("2f","resolution",this.width,this.height); 191 | gl.createUniform("1f","textureRatio",this.imageBg.width/this.imageBg.height); 192 | gl.createUniform("1i","renderShine",this.imageShine==null?false:true); 193 | gl.createUniform("1i","renderShadow",this.options.renderShadow); 194 | gl.createUniform("1f","minRefraction",this.options.minRefraction); 195 | gl.createUniform("1f","refractionDelta",this.options.maxRefraction-this.options.minRefraction); 196 | gl.createUniform("1f","brightness",this.options.brightness); 197 | gl.createUniform("1f","alphaMultiply",this.options.alphaMultiply); 198 | gl.createUniform("1f","alphaSubtract",this.options.alphaSubtract); 199 | gl.createUniform("1f","parallaxBg",this.options.parallaxBg); 200 | gl.createUniform("1f","parallaxFg",this.options.parallaxFg); 201 | gl.createUniform("1f","lightningFlash",0.0); 202 | 203 | 204 | gl.createTexture(null,0); 205 | 206 | this.textures=[ 207 | {name:'textureShine', img:this.imageShine==null?createCanvas(2,2):this.imageShine}, 208 | {name:'textureFg', img:this.imageFg}, 209 | {name:'textureBg', img:this.imageBg} 210 | ]; 211 | 212 | this.textures.forEach((texture,i)=>{ 213 | gl.createTexture(texture.img,i+1); 214 | gl.createUniform("1i",texture.name,i+1); 215 | }); 216 | 217 | this.draw(); 218 | }, 219 | draw(){ 220 | this.gl.useProgram(this.programWater); 221 | this.gl.createUniform("2f", "parallax", this.parallaxX,this.parallaxY); 222 | this.updateTexture(); 223 | this.gl.draw(); 224 | 225 | requestAnimationFrame(this.draw.bind(this)); 226 | }, 227 | updateTextures(){ 228 | this.textures.forEach((texture,i)=>{ 229 | this.gl.activeTexture(i+1); 230 | this.gl.updateTexture(texture.img); 231 | }) 232 | }, 233 | updateTexture(){ 234 | this.gl.activeTexture(0); 235 | this.gl.updateTexture(this.canvasLiquid); 236 | }, 237 | resize(){ 238 | 239 | }, 240 | // get overlayTexture(){ 241 | 242 | // }, 243 | // set overlayTexture(v){ 244 | 245 | // } 246 | } 247 | 248 | export default RainRenderer; -------------------------------------------------------------------------------- /src/app/rain/weather-utils.jsx: -------------------------------------------------------------------------------- 1 | import TweenLite from 'gsap'; 2 | import RainRenderer from "./rain-renderer"; 3 | import Raindrops from "./raindrops"; 4 | import loadImages from "./image-loader"; 5 | import createCanvas from "./create-canvas"; 6 | import times from './times'; 7 | import {random,chance} from './random'; 8 | 9 | import DropColor from './img/drop-color.png'; 10 | import DropAlpha from './img/drop-alpha.png'; 11 | 12 | let textureStormLightningFg, textureStormLightningBg, dropColor, dropAlpha; 13 | 14 | let textureFg, 15 | textureFgCtx, 16 | textureBg, 17 | textureBgCtx; 18 | 19 | let textureBgSize = { 20 | width: window.innerWidth, 21 | height: window.innerHeight 22 | } 23 | let textureFgSize = { 24 | width:96, 25 | height:64 26 | } 27 | 28 | // const blankFg = createCanvas(textureFgSize.width, textureFgSize.height); 29 | 30 | let raindrops, 31 | renderer, 32 | canvas; 33 | 34 | let weatherData = null; 35 | let curWeatherData = null; 36 | let blend = {v:0}; 37 | let intervalId = undefined; 38 | 39 | let backgroundImage = null; 40 | let lastWeatherType = 'rain'; // Track the last used weather type 41 | let resizeTimeout = null; 42 | 43 | // Set the background image and initialize rain effect after image loads 44 | export function setBackgroundImage(url, type = 'rain') { 45 | if (typeof window === 'undefined') return; // SSR safety 46 | backgroundImage = new window.Image(); 47 | backgroundImage.crossOrigin = 'anonymous'; 48 | 49 | console.log('textureBgSize', textureBgSize); 50 | backgroundImage.onload = () => { 51 | // Calculate scaled dimensions while preserving aspect ratio 52 | console.log('backgroundImage natural w h', backgroundImage.naturalWidth, backgroundImage.naturalHeight); 53 | const imageRatio = backgroundImage.naturalWidth / backgroundImage.naturalHeight; 54 | const containerRatio = textureBgSize.width / textureBgSize.height; 55 | console.log('imageRatio', imageRatio); 56 | console.log('containerRatio', containerRatio); 57 | if (imageRatio > containerRatio) { 58 | // Image is wider than target - scale by width 59 | backgroundImage.height = textureBgSize.height; 60 | backgroundImage.width = textureBgSize.height * imageRatio; 61 | } else { 62 | // Image is taller than target - scale by height 63 | backgroundImage.width = textureBgSize.width; 64 | backgroundImage.height = textureBgSize.width / imageRatio; 65 | } 66 | console.log('backgroundImage w h', backgroundImage.width, backgroundImage.height); 67 | loadTextures().then(() => { 68 | init(type, backgroundImage); 69 | }); 70 | }; 71 | backgroundImage.src = url; 72 | console.log('backgroundImage at onload', backgroundImage); 73 | } 74 | 75 | // Load drop textures 76 | export function loadTextures() { 77 | 78 | return loadImages([ 79 | { name:"dropAlpha", src: DropAlpha }, 80 | { name:"dropColor", src: DropColor }, 81 | ]).then(function (images){ 82 | dropColor = images.dropColor.img; 83 | dropAlpha = images.dropAlpha.img; 84 | }); 85 | } 86 | 87 | // Set up weather data and current weather 88 | function setupWeather(type) { 89 | 90 | setupWeatherData(); 91 | curWeatherData = weatherData[type]; 92 | 93 | if (raindrops && curWeatherData) { 94 | raindrops.options = Object.assign(raindrops.options, curWeatherData); 95 | raindrops.clearDrops(); 96 | } 97 | } 98 | 99 | // Handle window resize for responsive canvas 100 | function handleResize() { 101 | // resizeCanvas(); 102 | if (typeof window === 'undefined') return; 103 | if (backgroundImage && backgroundImage.complete) { 104 | setBackgroundImage(backgroundImage.src, lastWeatherType); 105 | } 106 | } 107 | 108 | // Make canvas match its CSS size 109 | // function resizeCanvas() { 110 | // canvas.width = canvas.clientWidth; 111 | // canvas.height = canvas.clientHeight; 112 | // } 113 | 114 | // Enable responsive canvas resizing 115 | export function enableResponsiveCanvas() { 116 | if (typeof window === 'undefined') return; 117 | window.addEventListener('resize', handleResize); 118 | } 119 | 120 | // Initialize rain effect and renderer 121 | function init(type = 'rain', backgroundImage) { 122 | if (typeof window === 'undefined' || typeof document === 'undefined') return; 123 | lastWeatherType = type; // Save the last used weather type 124 | canvas = document.querySelector('#container-weather'); 125 | // var dpi = window.devicePixelRatio; 126 | // canvas.width = window.innerWidth * dpi; 127 | // canvas.height = window.innerHeight * dpi; 128 | // canvas.style.width = window.innerWidth + "px"; 129 | // canvas.style.height = window.innerHeight + "px"; 130 | const container = canvas.parentElement; // or a specific container element 131 | console.log('container', container); 132 | const dpi = window.devicePixelRatio || 1; 133 | 134 | const rect = container.getBoundingClientRect(); 135 | console.log('rect', rect); 136 | canvas.width = rect.width * dpi; 137 | canvas.height = rect.height * dpi; 138 | 139 | canvas.style.width = `${rect.width}px`; 140 | canvas.style.height = `${rect.height}px`; 141 | 142 | // // Center the image in the canvas 143 | // const x = (canvas.width - backgroundImage.width) / 2; 144 | // const y = (canvas.height - backgroundImage.height) / 2; 145 | 146 | 147 | console.log('window.innerWidth', window.innerWidth); 148 | console.log('dpi', dpi); 149 | console.log('canvas.clientWidth', canvas.clientWidth); 150 | console.log('canvas.clientHeight', canvas.clientHeight); 151 | console.log('canvas.width', canvas.width); 152 | console.log('canvas.height', canvas.height); 153 | console.log('canvas.style.width', canvas.style.width); 154 | console.log('canvas.style.height', canvas.style.height); 155 | 156 | textureBgSize = { width: rect.width * dpi, height: rect.height * dpi }; 157 | 158 | raindrops=new Raindrops( 159 | canvas.width, 160 | canvas.height, 161 | dpi, 162 | dropAlpha, 163 | dropColor,{ 164 | trailRate:1, 165 | trailScaleRange:[0.2,0.45], 166 | collisionRadius : 0.45, 167 | dropletsCleaningRadiusMultiplier : 0.28, 168 | } 169 | ); 170 | 171 | textureFg = createCanvas(textureFgSize.width,textureFgSize.height); 172 | textureFgCtx = textureFg.getContext('2d'); 173 | textureBg = createCanvas(textureBgSize.width,textureBgSize.height); 174 | textureBgCtx = textureBg.getContext('2d'); 175 | 176 | 177 | generateTextures(backgroundImage, backgroundImage); 178 | renderer = new RainRenderer(canvas, raindrops.canvas, textureFg, textureBg, null,{ 179 | brightness:1.04, 180 | alphaMultiply:6, 181 | alphaSubtract:3, 182 | minRefraction: 128 183 | // minRefraction:256, 184 | // maxRefraction:512 185 | }); 186 | 187 | setupWeather(type); 188 | if (curWeatherData && curWeatherData.flashChance) { 189 | setupFlash(); 190 | } 191 | } 192 | 193 | // Set up lightning flash effect for storm weather 194 | function setupFlash() { 195 | intervalId = setInterval(()=>{ 196 | if(chance(curWeatherData.flashChance)){ 197 | flash(curWeatherData.bg,curWeatherData.fg,curWeatherData.flashBg,curWeatherData.flashFg); 198 | } 199 | },500); 200 | } 201 | 202 | // Set up weather data for different types 203 | function setupWeatherData() { 204 | var defaultWeather = { 205 | minR: 10, 206 | maxR: 40, 207 | rainChance: 0.35, 208 | rainLimit: 6, 209 | drizzle: 50, 210 | drizzleSize: [2, 4.5], 211 | raining: true, 212 | trailRate: 1, 213 | trailScaleRange: [0.2, 0.35], 214 | fg: backgroundImage, 215 | bg: backgroundImage, 216 | flashFg: null, 217 | flashBg: null, 218 | flashChance: 0 219 | }; 220 | 221 | function weather(data) { 222 | return Object.assign({}, defaultWeather, data); 223 | }; 224 | 225 | weatherData = { 226 | rain: weather({ 227 | rainChance: 0.35, 228 | rainLimit: 6, 229 | drizzle: 50, 230 | raining: true, 231 | // trailRate:2.5, 232 | fg: backgroundImage, 233 | bg: backgroundImage 234 | }), 235 | storm: weather({ 236 | minR: 20, 237 | maxR: 45, 238 | rainChance: 0.55, 239 | rainLimit: 6, 240 | drizzle: 80, 241 | drizzleSize: [2, 6], 242 | trailRate: 1, 243 | trailScaleRange: [0.15, 0.3], 244 | fg: backgroundImage, 245 | bg: backgroundImage, 246 | flashFg: textureStormLightningFg, 247 | flashBg: textureStormLightningBg, 248 | flashChance: 0.1 249 | }), 250 | fallout: weather({ 251 | rainChance: 0.35, 252 | rainLimit: 6, 253 | drizzle: 20, 254 | trailRate: 4, 255 | fg: backgroundImage, 256 | bg: backgroundImage 257 | }), 258 | drizzle: weather({ 259 | rainChance: 0.15, 260 | rainLimit: 2, 261 | drizzle: 10, 262 | fg: backgroundImage, 263 | bg: backgroundImage 264 | }), 265 | sunny: weather({ 266 | rainChance: 0, 267 | rainLimit: 0, 268 | drizzle: 0, 269 | raining: false, 270 | fg: backgroundImage, 271 | bg: backgroundImage 272 | }) 273 | }; 274 | } 275 | 276 | // Lightning flash animation 277 | function flash(baseBg, baseFg, flashBg, flashFg) { 278 | let flashValue={v:0}; 279 | function transitionFlash(to,t=0.025){ 280 | return new Promise((resolve,reject)=>{ 281 | TweenLite.to(flashValue,t,{ 282 | v:to, 283 | // ease:Quint.easeOut, 284 | onUpdate:()=>{ 285 | generateTextures(baseFg,baseBg); 286 | generateTextures(flashFg,flashBg,flashValue.v); 287 | renderer.updateTextures(); 288 | }, 289 | onComplete:()=>{ 290 | resolve(); 291 | } 292 | }); 293 | }); 294 | } 295 | 296 | let lastFlash=transitionFlash(1); 297 | times(random(2,7),(i)=>{ 298 | lastFlash=lastFlash.then(()=>{ 299 | return transitionFlash(random(0.1,1)) 300 | }) 301 | }) 302 | lastFlash=lastFlash.then(()=>{ 303 | return transitionFlash(1,0.1); 304 | }).then(()=>{ 305 | transitionFlash(0,0.25); 306 | }); 307 | } 308 | 309 | // Generate foreground/background textures for the rain renderer 310 | function generateTextures(fg, bg, x=0, y=0, alpha=1) { 311 | if ( 312 | !fg || 313 | !bg || 314 | (fg instanceof HTMLImageElement && !fg.complete) || 315 | (bg instanceof HTMLImageElement && !bg.complete) 316 | ) { 317 | // Image not ready, skip drawing 318 | return; 319 | } 320 | textureFgCtx.globalAlpha = alpha; 321 | textureFgCtx.drawImage(fg, x, y, textureFgSize.width, textureFgSize.height); 322 | console.log('x and y in generateTextures', x, y); 323 | 324 | textureBgCtx.globalAlpha = alpha; 325 | textureBgCtx.drawImage(bg, x, y, textureBgSize.width, textureBgSize.height); 326 | } 327 | 328 | // Clean up weather effect and clear intervals 329 | export function cleanWeather(){ 330 | //TODO enhance the cleanup function 331 | // raindrops.clean(); 332 | clearInterval(intervalId); 333 | // textureRainFg = null; 334 | // textureRainBg = null; 335 | // textureStormLightningFg = null; 336 | // textureStormLightningBg = null; 337 | // textureFalloutFg = null; 338 | // textureFalloutBg = null; 339 | // textureSunFg = null; 340 | // textureSunBg = null; 341 | // textureDrizzleFg = null; 342 | // textureDrizzleBg = null; 343 | // dropColor = null; 344 | // dropAlpha = null; 345 | // textureFg = null; 346 | // textureFgCtx = null; 347 | // textureBg = null; 348 | // textureBgCtx = null; 349 | // textureBgSize = null; 350 | // textureFgSize = null; 351 | // raindrops = null; 352 | // renderer = null; 353 | // canvas = null; 354 | // parallax = null; 355 | weatherData = null; 356 | curWeatherData = null; 357 | // blend = null; 358 | intervalId = null; 359 | } -------------------------------------------------------------------------------- /src/app/rain/raindrops.jsx: -------------------------------------------------------------------------------- 1 | // import loadImages from "./image-loader.jsx"; 2 | import times from "./times.jsx"; 3 | import createCanvas from "./create-canvas.jsx"; 4 | import {random, chance} from "./random.jsx"; 5 | 6 | let dropSize=64; 7 | const Drop={ 8 | x:0, 9 | y:0, 10 | r:0, 11 | spreadX:0, 12 | spreadY:0, 13 | momentum:0, 14 | momentumX:0, 15 | lastSpawn:0, 16 | nextSpawn:0, 17 | parent:null, 18 | isNew:true, 19 | killed:false, 20 | shrink:0, 21 | } 22 | const defaultOptions={ 23 | minR:10, 24 | maxR:40, 25 | maxDrops:900, 26 | rainChance:0.3, 27 | rainLimit:3, 28 | dropletsRate:50, 29 | dropletsSize:[2,4], 30 | dropletsCleaningRadiusMultiplier:0.43, 31 | raining:true, 32 | globalTimeScale:1, 33 | trailRate:1, 34 | autoShrink:true, 35 | spawnArea:[-0.1,0.95], 36 | trailScaleRange:[0.2,0.5], 37 | collisionRadius:0.65, 38 | collisionRadiusIncrease:0.01, 39 | dropFallMultiplier:1, 40 | collisionBoostMultiplier:0.05, 41 | collisionBoost:1, 42 | } 43 | 44 | function Raindrops(width,height,scale,dropAlpha,dropColor,options={}){ 45 | this.width=width; 46 | this.height=height; 47 | this.scale=scale; 48 | this.dropAlpha=dropAlpha; 49 | this.dropColor=dropColor; 50 | this.options=Object.assign({},defaultOptions,options); 51 | this.init(); 52 | } 53 | Raindrops.prototype={ 54 | dropColor:null, 55 | dropAlpha:null, 56 | canvas:null, 57 | ctx:null, 58 | width:0, 59 | height:0, 60 | scale:0, 61 | dropletsPixelDensity:1, 62 | droplets:null, 63 | dropletsCtx:null, 64 | dropletsCounter:0, 65 | drops:null, 66 | dropsGfx:null, 67 | clearDropletsGfx:null, 68 | textureCleaningIterations:0, 69 | lastRender:null, 70 | 71 | options:null, 72 | 73 | init(){ 74 | this.timeoutID = undefined; 75 | this.canvas = createCanvas(this.width,this.height); 76 | this.ctx = this.canvas.getContext('2d'); 77 | 78 | this.droplets = createCanvas(this.width*this.dropletsPixelDensity,this.height*this.dropletsPixelDensity); 79 | this.dropletsCtx = this.droplets.getContext('2d'); 80 | 81 | this.drops=[]; 82 | this.dropsGfx=[]; 83 | 84 | this.renderDropsGfx(); 85 | 86 | this.update(); 87 | }, 88 | get deltaR(){ 89 | return this.options.maxR-this.options.minR; 90 | }, 91 | get area(){ 92 | return (this.width*this.height)/this.scale; 93 | }, 94 | get areaMultiplier(){ 95 | return Math.sqrt(this.area/(1024*768)); 96 | }, 97 | drawDroplet(x,y,r){ 98 | this.drawDrop(this.dropletsCtx,Object.assign(Object.create(Drop),{ 99 | x:x*this.dropletsPixelDensity, 100 | y:y*this.dropletsPixelDensity, 101 | r:r*this.dropletsPixelDensity 102 | })); 103 | }, 104 | 105 | renderDropsGfx(){ 106 | let dropBuffer=createCanvas(dropSize,dropSize); 107 | let dropBufferCtx=dropBuffer.getContext('2d'); 108 | this.dropsGfx=Array.apply(null,Array(255)) 109 | .map((cur,i)=>{ 110 | let drop=createCanvas(dropSize,dropSize); 111 | let dropCtx=drop.getContext('2d'); 112 | 113 | dropBufferCtx.clearRect(0,0,dropSize,dropSize); 114 | 115 | // color 116 | dropBufferCtx.globalCompositeOperation="source-over"; 117 | dropBufferCtx.drawImage(this.dropColor,0,0,dropSize,dropSize); 118 | 119 | // blue overlay, for depth 120 | dropBufferCtx.globalCompositeOperation="screen"; 121 | dropBufferCtx.fillStyle="rgba(0,0,"+i+",1)"; 122 | dropBufferCtx.fillRect(0,0,dropSize,dropSize); 123 | 124 | // alpha 125 | dropCtx.globalCompositeOperation="source-over"; 126 | dropCtx.drawImage(this.dropAlpha,0,0,dropSize,dropSize); 127 | 128 | dropCtx.globalCompositeOperation="source-in"; 129 | dropCtx.drawImage(dropBuffer,0,0,dropSize,dropSize); 130 | return drop; 131 | }); 132 | 133 | // create circle that will be used as a brush to remove droplets 134 | this.clearDropletsGfx=createCanvas(128,128); 135 | let clearDropletsCtx=this.clearDropletsGfx.getContext("2d"); 136 | clearDropletsCtx.fillStyle="#000"; 137 | clearDropletsCtx.beginPath(); 138 | clearDropletsCtx.arc(64,64,64,0,Math.PI*2); 139 | clearDropletsCtx.fill(); 140 | }, 141 | drawDrop(ctx,drop){ 142 | if(this.dropsGfx.length>0){ 143 | let x=drop.x; 144 | let y=drop.y; 145 | let r=drop.r; 146 | let spreadX=drop.spreadX; 147 | let spreadY=drop.spreadY; 148 | 149 | let scaleX=1; 150 | let scaleY=1.5; 151 | 152 | let d=Math.max(0,Math.min(1,((r-this.options.minR)/(this.deltaR))*0.9)); 153 | d*=1/(((drop.spreadX+drop.spreadY)*0.5)+1); 154 | 155 | ctx.globalAlpha=1; 156 | ctx.globalCompositeOperation="source-over"; 157 | 158 | d=Math.floor(d*(this.dropsGfx.length-1)); 159 | ctx.drawImage( 160 | this.dropsGfx[d], 161 | (x-(r*scaleX*(spreadX+1)))*this.scale, 162 | (y-(r*scaleY*(spreadY+1)))*this.scale, 163 | (r*2*scaleX*(spreadX+1))*this.scale, 164 | (r*2*scaleY*(spreadY+1))*this.scale 165 | ); 166 | } 167 | }, 168 | clearDroplets(x,y,r=30){ 169 | let ctx=this.dropletsCtx; 170 | ctx.globalCompositeOperation="destination-out"; 171 | ctx.drawImage( 172 | this.clearDropletsGfx, 173 | (x-r)*this.dropletsPixelDensity*this.scale, 174 | (y-r)*this.dropletsPixelDensity*this.scale, 175 | (r*2)*this.dropletsPixelDensity*this.scale, 176 | (r*2)*this.dropletsPixelDensity*this.scale*1.5 177 | ) 178 | }, 179 | clearCanvas(){ 180 | this.ctx.clearRect(0,0,this.width,this.height); 181 | }, 182 | createDrop(options){ 183 | if(this.drops.length >= this.options.maxDrops*this.areaMultiplier) return null; 184 | 185 | return Object.assign(Object.create(Drop),options); 186 | }, 187 | addDrop(drop){ 188 | if(this.drops.length >= this.options.maxDrops*this.areaMultiplier || drop==null) return false; 189 | 190 | this.drops.push(drop); 191 | return true; 192 | }, 193 | updateRain(timeScale){ 194 | let rainDrops=[]; 195 | if(this.options.raining){ 196 | let limit=this.options.rainLimit*timeScale*this.areaMultiplier; 197 | let count=0; 198 | while(chance(this.options.rainChance*timeScale*this.areaMultiplier) && count{ 201 | return Math.pow(n,3); 202 | }); 203 | let rainDrop=this.createDrop({ 204 | x:random(this.width/this.scale), 205 | y:random((this.height/this.scale)*this.options.spawnArea[0],(this.height/this.scale)*this.options.spawnArea[1]), 206 | r:r, 207 | momentum:1+((r-this.options.minR)*0.1)+random(2), 208 | spreadX:1.5, 209 | spreadY:1.5, 210 | }); 211 | if(rainDrop!=null){ 212 | rainDrops.push(rainDrop); 213 | } 214 | } 215 | } 216 | return rainDrops; 217 | }, 218 | clearDrops(){ 219 | this.drops.forEach((drop)=>{ 220 | // TODO this has to be an array of timeouts 221 | this.timeoutID = setTimeout(()=>{ 222 | drop.shrink=0.1+(random(0.5)); 223 | },random(1200)) 224 | }) 225 | this.clearTexture(); 226 | }, 227 | clearTexture(){ 228 | this.textureCleaningIterations=50; 229 | // clearTimeout(this.timeoutID); 230 | }, 231 | clean(){ 232 | // TODO clear array of timeouts 233 | // clearTimeout(this.timeoutID); 234 | }, 235 | updateDroplets(timeScale){ 236 | if(this.textureCleaningIterations>0){ 237 | this.textureCleaningIterations-=1*timeScale; 238 | this.dropletsCtx.globalCompositeOperation="destination-out"; 239 | this.dropletsCtx.fillStyle="rgba(0,0,0,"+(0.05*timeScale)+")"; 240 | this.dropletsCtx.fillRect(0,0, 241 | this.width*this.dropletsPixelDensity,this.height*this.dropletsPixelDensity); 242 | } 243 | if(this.options.raining){ 244 | this.dropletsCounter+=this.options.dropletsRate*timeScale*this.areaMultiplier; 245 | times(this.dropletsCounter,(i)=>{ 246 | this.dropletsCounter--; 247 | this.drawDroplet( 248 | random(this.width/this.scale), 249 | random(this.height/this.scale), 250 | random(...this.options.dropletsSize,(n)=>{ 251 | return n*n; 252 | }) 253 | ) 254 | }); 255 | } 256 | this.ctx.drawImage(this.droplets,0,0,this.width,this.height); 257 | }, 258 | updateDrops(timeScale){ 259 | let newDrops=[]; 260 | 261 | this.updateDroplets(timeScale); 262 | let rainDrops=this.updateRain(timeScale); 263 | newDrops=newDrops.concat(rainDrops); 264 | 265 | this.drops.sort((a,b)=>{ 266 | let va=(a.y*(this.width/this.scale))+a.x; 267 | let vb=(b.y*(this.width/this.scale))+b.x; 268 | return va>vb?1:va===vb?0:-1; 269 | }); 270 | 271 | this.drops.forEach(function(drop,i){ 272 | if(!drop.killed){ 273 | // update gravity 274 | // (chance of drops "creeping down") 275 | if(chance((drop.r-(this.options.minR*this.options.dropFallMultiplier)) * (0.1/this.deltaR) * timeScale)){ 276 | drop.momentum += random((drop.r/this.options.maxR)*4); 277 | } 278 | // clean small drops 279 | if(this.options.autoShrink && drop.r<=this.options.minR && chance(0.05*timeScale)){ 280 | drop.shrink+=0.01; 281 | } 282 | //update shrinkage 283 | drop.r -= drop.shrink*timeScale; 284 | if(drop.r<=0) drop.killed=true; 285 | 286 | // update trails 287 | if(this.options.raining){ 288 | drop.lastSpawn+=drop.momentum*timeScale*this.options.trailRate; 289 | if(drop.lastSpawn>drop.nextSpawn){ 290 | let trailDrop=this.createDrop({ 291 | x:drop.x+(random(-drop.r,drop.r)*0.1), 292 | y:drop.y-(drop.r*0.01), 293 | r:drop.r*random(...this.options.trailScaleRange), 294 | spreadY:drop.momentum*0.1, 295 | parent:drop, 296 | }); 297 | 298 | if(trailDrop!=null){ 299 | newDrops.push(trailDrop); 300 | 301 | drop.r*=Math.pow(0.97,timeScale); 302 | drop.lastSpawn=0; 303 | drop.nextSpawn=random(this.options.minR,this.options.maxR)-(drop.momentum*2*this.options.trailRate)+(this.options.maxR-drop.r); 304 | } 305 | } 306 | } 307 | 308 | //normalize spread 309 | drop.spreadX*=Math.pow(0.4,timeScale); 310 | drop.spreadY*=Math.pow(0.7,timeScale); 311 | 312 | //update position 313 | let moved=drop.momentum>0; 314 | if(moved && !drop.killed){ 315 | drop.y+=drop.momentum*this.options.globalTimeScale; 316 | drop.x+=drop.momentumX*this.options.globalTimeScale; 317 | if(drop.y>(this.height/this.scale)+drop.r){ 318 | drop.killed=true; 319 | } 320 | } 321 | 322 | // collision 323 | let checkCollision=(moved || drop.isNew) && !drop.killed; 324 | drop.isNew=false; 325 | 326 | if(checkCollision){ 327 | this.drops.slice(i+1,i+70).forEach((drop2)=>{ 328 | //basic check 329 | if( 330 | drop !== drop2 && 331 | drop.r > drop2.r && 332 | drop.parent !== drop2 && 333 | drop2.parent !== drop && 334 | !drop2.killed 335 | ){ 336 | let dx=drop2.x-drop.x; 337 | let dy=drop2.y-drop.y; 338 | var d=Math.sqrt((dx*dx)+(dy*dy)); 339 | //if it's within acceptable distance 340 | if(d<(drop.r+drop2.r)*(this.options.collisionRadius+(drop.momentum*this.options.collisionRadiusIncrease*timeScale))){ 341 | let pi=Math.PI; 342 | let r1=drop.r; 343 | let r2=drop2.r; 344 | let a1=pi*(r1*r1); 345 | let a2=pi*(r2*r2); 346 | let targetR=Math.sqrt((a1+(a2*0.8))/pi); 347 | if(targetR>this.maxR){ 348 | targetR=this.maxR; 349 | } 350 | drop.r=targetR; 351 | drop.momentumX+=dx*0.1; 352 | drop.spreadX=0; 353 | drop.spreadY=0; 354 | drop2.killed=true; 355 | drop.momentum=Math.max(drop2.momentum,Math.min(40,drop.momentum+(targetR*this.options.collisionBoostMultiplier)+this.options.collisionBoost)); 356 | } 357 | } 358 | }); 359 | } 360 | 361 | //slowdown momentum 362 | drop.momentum-=Math.max(1,(this.options.minR*0.5)-drop.momentum)*0.1*timeScale; 363 | if(drop.momentum<0) drop.momentum=0; 364 | drop.momentumX*=Math.pow(0.7,timeScale); 365 | 366 | 367 | if(!drop.killed){ 368 | newDrops.push(drop); 369 | if(moved && this.options.dropletsRate>0) this.clearDroplets(drop.x,drop.y,drop.r*this.options.dropletsCleaningRadiusMultiplier); 370 | this.drawDrop(this.ctx, drop); 371 | } 372 | 373 | } 374 | },this); 375 | 376 | this.drops = newDrops; 377 | }, 378 | update(){ 379 | this.clearCanvas(); 380 | 381 | let now=Date.now(); 382 | if(this.lastRender==null) this.lastRender=now; 383 | let deltaT=now-this.lastRender; 384 | let timeScale=deltaT/((1/60)*1000); 385 | if(timeScale>1.1) timeScale=1.1; 386 | timeScale*=this.options.globalTimeScale; 387 | this.lastRender=now; 388 | 389 | this.updateDrops(timeScale); 390 | 391 | requestAnimationFrame(this.update.bind(this)); 392 | } 393 | } 394 | 395 | export default Raindrops; -------------------------------------------------------------------------------- /src/app/snow/ShaderProgram.jsx: -------------------------------------------------------------------------------- 1 | export default class ShaderProgram { 2 | 3 | constructor( holder, options = {} ) { 4 | 5 | options = Object.assign( { 6 | antialias: false, 7 | depthTest: false, 8 | mousemove: false, 9 | autosize: true, 10 | msaa: 0, 11 | vertex: ` 12 | precision highp float; 13 | 14 | attribute vec4 a_position; 15 | attribute vec4 a_color; 16 | 17 | uniform float u_time; 18 | uniform vec2 u_resolution; 19 | uniform vec2 u_mousemove; 20 | uniform mat4 u_projection; 21 | 22 | varying vec4 v_color; 23 | 24 | void main() { 25 | 26 | gl_Position = u_projection * a_position; 27 | gl_PointSize = (10.0 / gl_Position.w) * 100.0; 28 | 29 | v_color = a_color; 30 | 31 | }`, 32 | fragment: ` 33 | precision highp float; 34 | 35 | uniform sampler2D u_texture; 36 | uniform int u_hasTexture; 37 | 38 | varying vec4 v_color; 39 | 40 | void main() { 41 | 42 | if ( u_hasTexture == 1 ) { 43 | 44 | gl_FragColor = v_color * texture2D(u_texture, gl_PointCoord); 45 | 46 | } else { 47 | 48 | gl_FragColor = v_color; 49 | 50 | } 51 | 52 | }`, 53 | uniforms: {}, 54 | buffers: {}, 55 | camera: {}, 56 | texture: null, 57 | onUpdate: ( () => {} ), 58 | onResize: ( () => {} ), 59 | }, options ) 60 | 61 | const uniforms = Object.assign( { 62 | time: { type: 'float', value: 0 }, 63 | hasTexture: { type: 'int', value: 0 }, 64 | resolution: { type: 'vec2', value: [ 0, 0 ] }, 65 | mousemove: { type: 'vec2', value: [ 0, 0 ] }, 66 | projection: { type: 'mat4', value: [ 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1 ] }, 67 | }, options.uniforms ) 68 | 69 | const buffers = Object.assign( { 70 | position: { size: 3, data: [] }, 71 | color: { size: 4, data: [] }, 72 | }, options.buffers ) 73 | 74 | const camera = Object.assign( { 75 | fov: 60, 76 | near: 1, 77 | far: 10000, 78 | aspect: 1, 79 | z: 100, 80 | perspective: true, 81 | }, options.camera ) 82 | 83 | const canvas = document.createElement( 'canvas' ) 84 | const gl = canvas.getContext( 'webgl', { antialias: options.antialias } ) 85 | 86 | if ( ! gl ) return false 87 | 88 | this.count = 0 89 | this.gl = gl 90 | this.canvas = canvas 91 | this.camera = camera 92 | this.holder = holder 93 | this.msaa = options.msaa 94 | this.onUpdate = options.onUpdate 95 | this.onResize = options.onResize 96 | this.data = {} 97 | 98 | holder.appendChild( canvas ) 99 | 100 | this.createProgram( options.vertex, options.fragment ) 101 | 102 | this.createBuffers( buffers ) 103 | this.createUniforms( uniforms ) 104 | 105 | this.updateBuffers() 106 | this.updateUniforms() 107 | 108 | this.createTexture( options.texture ) 109 | 110 | gl.enable( gl.BLEND ) 111 | gl.enable( gl.CULL_FACE ) 112 | gl.blendFunc( gl.SRC_ALPHA, gl.ONE ) 113 | gl[ options.depthTest ? 'enable' : 'disable' ]( gl.DEPTH_TEST ) 114 | 115 | if ( options.autosize ) 116 | window.addEventListener( 'resize', e => this.resize( e ), false ) 117 | if ( options.mousemove ) 118 | window.addEventListener( 'mousemove', e => this.mousemove( e ), false ) 119 | 120 | this.resize() 121 | 122 | this.update = this.update.bind( this ) 123 | this.time = { start: performance.now(), old: performance.now() } 124 | this.update() 125 | 126 | } 127 | 128 | mousemove( e ) { 129 | 130 | let x = e.pageX / this.width * 2 - 1 131 | let y = e.pageY / this.height * 2 - 1 132 | 133 | this.uniforms.mousemove = [ x, y ] 134 | 135 | } 136 | 137 | resize( e ) { 138 | 139 | const holder = this.holder 140 | const canvas = this.canvas 141 | const gl = this.gl 142 | 143 | const width = this.width = holder.offsetWidth 144 | const height = this.height = holder.offsetHeight 145 | const aspect = this.aspect = width / height 146 | const dpi = this.dpi = Math.max( this.msaa ? 2 : 1, devicePixelRatio ) 147 | 148 | canvas.width = width * dpi 149 | canvas.height = height * dpi 150 | canvas.style.width = width + 'px' 151 | canvas.style.height = height + 'px' 152 | 153 | gl.viewport( 0, 0, width * dpi, height * dpi ) 154 | gl.clearColor( 0, 0, 0, 0 ) 155 | 156 | this.uniforms.resolution = [ width, height ] 157 | this.uniforms.projection = this.setProjection( aspect ) 158 | 159 | this.onResize( width, height, dpi ) 160 | 161 | } 162 | 163 | setProjection( aspect ) { 164 | 165 | const camera = this.camera 166 | 167 | if ( camera.perspective ) { 168 | 169 | camera.aspect = aspect 170 | 171 | const fovRad = camera.fov * ( Math.PI / 180 ) 172 | const f = Math.tan( Math.PI * 0.5 - 0.5 * fovRad ) 173 | const rangeInv = 1.0 / ( camera.near - camera.far ) 174 | 175 | const matrix = [ 176 | f / camera.aspect, 0, 0, 0, 177 | 0, f, 0, 0, 178 | 0, 0, (camera.near + camera.far) * rangeInv, -1, 179 | 0, 0, camera.near * camera.far * rangeInv * 2, 0 180 | ] 181 | 182 | matrix[ 14 ] += camera.z 183 | matrix[ 15 ] += camera.z 184 | 185 | return matrix 186 | 187 | } else { 188 | 189 | return [ 190 | 2 / this.width, 0, 0, 0, 191 | 0, -2 / this.height, 0, 0, 192 | 0, 0, 1, 0, 193 | -1, 1, 0, 1, 194 | ] 195 | 196 | } 197 | 198 | } 199 | 200 | createShader( type, source ) { 201 | 202 | const gl = this.gl 203 | const shader = gl.createShader( type ) 204 | 205 | gl.shaderSource( shader, source ) 206 | gl.compileShader( shader ) 207 | 208 | if ( gl.getShaderParameter (shader, gl.COMPILE_STATUS ) ) { 209 | 210 | return shader 211 | 212 | } else { 213 | 214 | console.log( gl.getShaderInfoLog( shader ) ) 215 | gl.deleteShader( shader ) 216 | 217 | } 218 | 219 | } 220 | 221 | createProgram( vertex, fragment ) { 222 | 223 | const gl = this.gl 224 | 225 | const vertexShader = this.createShader( gl.VERTEX_SHADER, vertex ) 226 | const fragmentShader = this.createShader( gl.FRAGMENT_SHADER, fragment ) 227 | 228 | const program = gl.createProgram() 229 | 230 | gl.attachShader( program, vertexShader ) 231 | gl.attachShader( program, fragmentShader ) 232 | gl.linkProgram( program ) 233 | 234 | if ( gl.getProgramParameter( program, gl.LINK_STATUS ) ) { 235 | 236 | gl.useProgram( program ) 237 | this.program = program 238 | 239 | } else { 240 | 241 | console.log( gl.getProgramInfoLog( program ) ) 242 | gl.deleteProgram( program ) 243 | 244 | } 245 | 246 | } 247 | 248 | createUniforms( data ) { 249 | 250 | const gl = this.gl 251 | const uniforms = this.data.uniforms = data 252 | const values = this.uniforms = {} 253 | 254 | Object.keys( uniforms ).forEach( name => { 255 | 256 | const uniform = uniforms[ name ] 257 | 258 | uniform.location = gl.getUniformLocation( this.program, 'u_' + name ) 259 | 260 | Object.defineProperty( values, name, { 261 | set: value => { 262 | 263 | uniforms[ name ].value = value 264 | this.setUniform( name, value ) 265 | 266 | }, 267 | get: () => uniforms[ name ].value 268 | } ) 269 | 270 | } ) 271 | 272 | } 273 | 274 | setUniform( name, value ) { 275 | 276 | const gl = this.gl 277 | const uniform = this.data.uniforms[ name ] 278 | 279 | uniform.value = value 280 | 281 | switch ( uniform.type ) { 282 | case 'int': { 283 | gl.uniform1i( uniform.location, value ) 284 | break 285 | } 286 | case 'float': { 287 | gl.uniform1f( uniform.location, value ) 288 | break 289 | } 290 | case 'vec2': { 291 | gl.uniform2f( uniform.location, ...value ) 292 | break 293 | } 294 | case 'vec3': { 295 | gl.uniform3f( uniform.location, ...value ) 296 | break 297 | } 298 | case 'vec4': { 299 | gl.uniform4f( uniform.location, ...value ) 300 | break 301 | } 302 | case 'mat2': { 303 | gl.uniformMatrix2fv( uniform.location, false, value ) 304 | break 305 | } 306 | case 'mat3': { 307 | gl.uniformMatrix3fv( uniform.location, false, value ) 308 | break 309 | } 310 | case 'mat4': { 311 | gl.uniformMatrix4fv( uniform.location, false, value ) 312 | break 313 | } 314 | } 315 | 316 | // ivec2 : uniform2i, 317 | // ivec3 : uniform3i, 318 | // ivec4 : uniform4i, 319 | // sampler2D : uniform1i, 320 | // samplerCube : uniform1i, 321 | // bool : uniform1i, 322 | // bvec2 : uniform2i, 323 | // bvec3 : uniform3i, 324 | // bvec4 : uniform4i, 325 | 326 | } 327 | 328 | updateUniforms() { 329 | 330 | const gl = this.gl 331 | const uniforms = this.data.uniforms 332 | 333 | Object.keys( uniforms ).forEach( name => { 334 | 335 | const uniform = uniforms[ name ] 336 | 337 | this.uniforms[ name ] = uniform.value 338 | 339 | } ) 340 | 341 | } 342 | 343 | createBuffers( data ) { 344 | 345 | const gl = this.gl 346 | const buffers = this.data.buffers = data 347 | const values = this.buffers = {} 348 | 349 | Object.keys( buffers ).forEach( name => { 350 | 351 | const buffer = buffers[ name ] 352 | 353 | buffer.buffer = this.createBuffer( 'a_' + name, buffer.size ) 354 | 355 | Object.defineProperty( values, name, { 356 | set: data => { 357 | 358 | buffers[ name ].data = data 359 | this.setBuffer( name, data ) 360 | 361 | if ( name == 'position' ) 362 | this.count = buffers.position.data.length / 3 363 | 364 | }, 365 | get: () => buffers[ name ].data 366 | } ) 367 | 368 | } ) 369 | 370 | } 371 | 372 | createBuffer( name, size ) { 373 | 374 | const gl = this.gl 375 | const program = this.program 376 | 377 | const index = gl.getAttribLocation( program, name ) 378 | const buffer = gl.createBuffer() 379 | 380 | gl.bindBuffer( gl.ARRAY_BUFFER, buffer ) 381 | gl.enableVertexAttribArray( index ) 382 | gl.vertexAttribPointer( index, size, gl.FLOAT, false, 0, 0 ) 383 | 384 | return buffer 385 | 386 | } 387 | 388 | setBuffer( name, data ) { 389 | 390 | const gl = this.gl 391 | const buffers = this.data.buffers 392 | 393 | if ( name == null && ! gl.bindBuffer( gl.ARRAY_BUFFER, null ) ) return 394 | 395 | gl.bindBuffer( gl.ARRAY_BUFFER, buffers[ name ].buffer ) 396 | gl.bufferData( gl.ARRAY_BUFFER, new Float32Array( data ), gl.STATIC_DRAW ) 397 | 398 | } 399 | 400 | updateBuffers() { 401 | 402 | const gl = this.gl 403 | const buffers = this.buffers 404 | 405 | Object.keys( buffers ).forEach( name => 406 | buffers[ name ] = buffer.data 407 | ) 408 | 409 | this.setBuffer( null ) 410 | 411 | } 412 | 413 | createTexture( src ) { 414 | 415 | const gl = this.gl 416 | const texture = gl.createTexture() 417 | 418 | gl.bindTexture( gl.TEXTURE_2D, texture ) 419 | gl.texImage2D( gl.TEXTURE_2D, 0, gl.RGBA, 1, 1, 0, gl.RGBA, gl.UNSIGNED_BYTE, new Uint8Array( [ 0, 0, 0, 0 ] ) ) 420 | 421 | this.texture = texture 422 | 423 | if ( src ) { 424 | 425 | this.uniforms.hasTexture = 1 426 | this.loadTexture( src ) 427 | 428 | } 429 | 430 | } 431 | 432 | loadTexture( src ) { 433 | 434 | const gl = this.gl 435 | const texture = this.texture 436 | 437 | const textureImage = new Image() 438 | 439 | textureImage.onload = () => { 440 | 441 | gl.bindTexture( gl.TEXTURE_2D, texture ) 442 | 443 | gl.texImage2D( gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, textureImage ) 444 | 445 | gl.texParameteri( gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR ) 446 | gl.texParameteri( gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR ) 447 | 448 | gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE) 449 | gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE) 450 | 451 | // gl.generateMipmap( gl.TEXTURE_2D ) 452 | 453 | } 454 | 455 | textureImage.src = src 456 | 457 | } 458 | 459 | update() { 460 | 461 | const gl = this.gl 462 | 463 | const now = performance.now() 464 | const elapsed = ( now - this.time.start ) / 5000 465 | const delta = now - this.time.old 466 | this.time.old = now 467 | 468 | this.uniforms.time = elapsed 469 | 470 | if ( this.count > 0 ) { 471 | gl.clear( gl.COLORBUFFERBIT ) 472 | gl.drawArrays( gl.POINTS, 0, this.count ) 473 | } 474 | 475 | this.onUpdate( delta ) 476 | 477 | requestAnimationFrame( this.update ) 478 | 479 | } 480 | 481 | } -------------------------------------------------------------------------------- /src/app/snow/SnowEffect.jsx: -------------------------------------------------------------------------------- 1 | import React, { useRef, useMemo } from "react"; 2 | import { Canvas, useFrame, extend } from "@react-three/fiber"; 3 | import * as THREE from "three"; 4 | import snowflake from "./snowflake.png"; 5 | 6 | 7 | // https://github.com/bsehovac/shader-program 8 | 9 | // const snowflake = ''; 10 | 11 | const count = 7000; 12 | 13 | let wind = { 14 | current: 0, 15 | force: 0.1, 16 | target: 0.1, 17 | min: 0.1, 18 | max: 0.25, 19 | easing: 0.005 20 | }; 21 | 22 | // Shaders from your original code 23 | const vertexShader = ` 24 | precision highp float; 25 | attribute float size; 26 | attribute vec3 rotation; 27 | attribute vec3 speed; 28 | attribute vec4 a_color; 29 | attribute float scale; 30 | attribute float distortion; 31 | attribute float brightness; 32 | attribute float contrast; 33 | attribute float rotationOffset; 34 | attribute float flipX; 35 | attribute float flipY; 36 | attribute float warp; 37 | varying vec4 v_color; 38 | varying float v_rotation; 39 | varying float v_scale; 40 | varying float v_distortion; 41 | varying float v_brightness; 42 | varying float v_contrast; 43 | varying float v_rotationOffset; 44 | varying float v_flipX; 45 | varying float v_flipY; 46 | varying float v_warp; 47 | uniform float u_time; 48 | uniform vec3 u_worldSize; 49 | uniform float u_gravity; 50 | uniform float u_wind; 51 | void main() { 52 | v_color = a_color; 53 | v_rotation = rotation.x + u_time * rotation.y; 54 | v_scale = scale; 55 | v_distortion = distortion; 56 | v_brightness = brightness; 57 | v_contrast = contrast; 58 | v_rotationOffset = rotationOffset; 59 | v_flipX = flipX; 60 | v_flipY = flipY; 61 | v_warp = warp; 62 | vec3 pos = position; 63 | pos.x = mod(pos.x + u_time + u_wind * speed.x, u_worldSize.x * 2.0) - u_worldSize.x; 64 | pos.y = mod(pos.y - u_time * speed.y * u_gravity, u_worldSize.y * 2.0) - u_worldSize.y; 65 | pos.x += sin(u_time * speed.z) * rotation.z; 66 | pos.z += cos(u_time * speed.z) * rotation.z; 67 | gl_Position = projectionMatrix * modelViewMatrix * vec4(pos, 1.0); 68 | gl_PointSize = (size * v_scale / gl_Position.w) * 100.0; 69 | } 70 | `; 71 | 72 | const fragmentShader = ` 73 | precision highp float; 74 | uniform sampler2D u_texture; 75 | uniform float u_time; 76 | varying vec4 v_color; 77 | varying float v_rotation; 78 | varying float v_scale; 79 | varying float v_distortion; 80 | varying float v_brightness; 81 | varying float v_contrast; 82 | varying float v_rotationOffset; 83 | varying float v_flipX; 84 | varying float v_flipY; 85 | varying float v_warp; 86 | 87 | // Simple noise function 88 | float noise(vec2 p) { 89 | return fract(sin(dot(p, vec2(12.9898, 78.233))) * 43758.5453); 90 | } 91 | 92 | void main() { 93 | vec2 coord = gl_PointCoord - 0.5; 94 | // Apply random scale 95 | coord *= v_scale; 96 | // Add random distortion based on noise and time 97 | float distortion_amount = v_distortion * 0.15; 98 | coord += vec2( 99 | noise(coord * 10.0 + u_time * 0.1) * distortion_amount, 100 | noise(coord * 10.0 + u_time * 0.1 + 1.0) * distortion_amount 101 | ); 102 | // Apply warp (skew) 103 | coord.x += v_warp * coord.y; 104 | coord.y += v_warp * coord.x; 105 | // Apply flip 106 | if (v_flipX > 0.5) coord.x = -coord.x; 107 | if (v_flipY > 0.5) coord.y = -coord.y; 108 | // Apply random rotation offset 109 | float angle = v_rotation + v_rotationOffset; 110 | vec2 rotated = vec2( 111 | cos(angle) * coord.x + sin(angle) * coord.y, 112 | cos(angle) * coord.y - sin(angle) * coord.x 113 | ) + 0.5; 114 | vec4 snowflake = texture2D(u_texture, rotated); 115 | // Apply brightness and contrast 116 | vec3 color = snowflake.rgb; 117 | color = (color - 0.5) * v_contrast + 0.5 + v_brightness; 118 | color = clamp(color, 0.0, 1.0); 119 | gl_FragColor = vec4(color, snowflake.a * v_color.a); 120 | } 121 | `; 122 | 123 | // Settings for gentle and storm snow 124 | const GENTLE_SETTINGS = { 125 | count: 3000, 126 | gravity: 20, 127 | colorAlphaMin: 0.2, 128 | colorAlphaMax: 0.6, 129 | sizeMin: 5, 130 | sizeMax: 15, 131 | scaleMin: 0.5, 132 | scaleMax: 1.5, 133 | distortionMin: 0.1, 134 | distortionMax: 0.5, 135 | brightnessMin: -0.1, 136 | brightnessMax: 0.2, 137 | contrastMin: 0.8, 138 | contrastMax: 1.2, 139 | wind: { 140 | force: 0.05, 141 | target: 0.05, 142 | min: 0.02, 143 | max: 0.1, 144 | easing: 0.002, 145 | }, 146 | windDirectionChangeFreq: 0.97, 147 | windDirectionChangeAmount: 1.0, 148 | speedYMin: 0.5, 149 | speedYMax: 1.0, 150 | speedXMin: 0.2, 151 | speedXMax: 0.5, 152 | swayMax: 5, 153 | }; 154 | 155 | const STORM_SETTINGS = { 156 | count: 7000, 157 | gravity: 45, 158 | colorAlphaMin: 0.25, 159 | colorAlphaMax: 0.8, 160 | sizeMin: 7, 161 | sizeMax: 18, 162 | scaleMin: 0.7, 163 | scaleMax: 2.2, 164 | distortionMin: 0.2, 165 | distortionMax: 1.0, 166 | brightnessMin: -0.2, 167 | brightnessMax: 0.3, 168 | contrastMin: 0.7, 169 | contrastMax: 1.3, 170 | wind: { 171 | force: 0.15, 172 | target: 0.2, 173 | min: 0.08, 174 | max: 0.35, 175 | easing: 0.01, 176 | }, 177 | windDirectionChangeFreq: 0.995, 178 | windDirectionChangeAmount: 0.2, 179 | speedYMin: 1.2, 180 | speedYMax: 2.0, 181 | speedXMin: 0.4, 182 | speedXMax: 1.0, 183 | swayMax: 12, 184 | }; 185 | 186 | function SnowParticles({ settings }) { 187 | const mesh = useRef(); 188 | const worldSize = [110, 110, 80]; 189 | const gravity = settings.gravity; 190 | const wind = useRef({ 191 | current: 0, 192 | force: settings.wind.force, 193 | target: settings.wind.target, 194 | min: settings.wind.min, 195 | max: settings.wind.max, 196 | easing: settings.wind.easing, 197 | }); 198 | // Wind angle in radians 199 | const windAngle = React.useRef(0); // 0 = right, PI = left 200 | 201 | // Generate attributes 202 | const { positions, colors, sizes, rotations, speeds, scales, distortions, brightnesses, contrasts, rotationOffsets, flipXs, flipYs, warps } = useMemo(() => { 203 | const positions = []; 204 | const colors = []; 205 | const sizes = []; 206 | const rotations = []; 207 | const speeds = []; 208 | const scales = []; 209 | const distortions = []; 210 | const brightnesses = []; 211 | const contrasts = []; 212 | const rotationOffsets = []; 213 | const flipXs = []; 214 | const flipYs = []; 215 | const warps = []; 216 | for (let i = 0; i < settings.count; i++) { 217 | // Position 218 | positions.push( 219 | -worldSize[0] + Math.random() * worldSize[0] * 2, 220 | -worldSize[1] + Math.random() * worldSize[1] * 2, 221 | Math.random() * worldSize[2] * 2 222 | ); 223 | // Speed 224 | speeds.push( 225 | settings.speedXMin + Math.random() * (settings.speedXMax - settings.speedXMin), 226 | settings.speedYMin + Math.random() * (settings.speedYMax - settings.speedYMin), 227 | Math.random() * settings.swayMax 228 | ); 229 | // Rotation 230 | rotations.push( 231 | Math.random() * 2 * Math.PI, 232 | Math.random() * 20, 233 | Math.random() * 10 234 | ); 235 | // Color (RGBA) 236 | colors.push(1, 1, 1, settings.colorAlphaMin + Math.random() * (settings.colorAlphaMax - settings.colorAlphaMin)); 237 | // Size 238 | sizes.push(settings.sizeMin + Math.random() * (settings.sizeMax - settings.sizeMin)); 239 | // Random scale 240 | scales.push(settings.scaleMin + Math.random() * (settings.scaleMax - settings.scaleMin)); 241 | // Random distortion 242 | distortions.push(settings.distortionMin + Math.random() * (settings.distortionMax - settings.distortionMin)); 243 | // Random brightness 244 | brightnesses.push(settings.brightnessMin + Math.random() * (settings.brightnessMax - settings.brightnessMin)); 245 | // Random contrast 246 | contrasts.push(settings.contrastMin + Math.random() * (settings.contrastMax - settings.contrastMin)); 247 | // Random rotation offset 248 | rotationOffsets.push(Math.random() * Math.PI * 2); 249 | // Random flip 250 | flipXs.push(Math.random() > 0.5 ? 1 : 0); 251 | flipYs.push(Math.random() > 0.5 ? 1 : 0); 252 | // Random warp 253 | warps.push(-0.3 + Math.random() * 0.6); // -0.3 to 0.3 254 | } 255 | return { 256 | positions: new Float32Array(positions), 257 | colors: new Float32Array(colors), 258 | sizes: new Float32Array(sizes), 259 | rotations: new Float32Array(rotations), 260 | speeds: new Float32Array(speeds), 261 | scales: new Float32Array(scales), 262 | distortions: new Float32Array(distortions), 263 | brightnesses: new Float32Array(brightnesses), 264 | contrasts: new Float32Array(contrasts), 265 | rotationOffsets: new Float32Array(rotationOffsets), 266 | flipXs: new Float32Array(flipXs), 267 | flipYs: new Float32Array(flipYs), 268 | warps: new Float32Array(warps), 269 | }; 270 | }, [settings]); 271 | 272 | // Texture loading (async) 273 | const [texture, setTexture] = React.useState(); 274 | React.useEffect(() => { 275 | const loader = new THREE.TextureLoader(); 276 | loader.load(snowflake.src, (tex) => { 277 | setTexture(tex); 278 | }); 279 | }, []); 280 | 281 | // Uniforms 282 | const uniforms = useMemo( 283 | () => ({ 284 | u_time: { value: 0 }, 285 | u_texture: { value: texture }, 286 | u_worldSize: { value: worldSize }, 287 | u_gravity: { value: gravity }, 288 | u_wind: { value: 0 }, 289 | }), 290 | [texture, gravity] 291 | ); 292 | 293 | useFrame((state, delta) => { 294 | // Wind logic 295 | const w = wind.current; 296 | w.force += (w.target - w.force) * w.easing; 297 | w.current += w.force * (delta * 0.2); 298 | // Wind angle logic: nudge angle 299 | if (Math.random() > settings.windDirectionChangeFreq) { 300 | // Gentle: nudge more often/larger, Storm: less often/smaller 301 | const nudge = (Math.random() - 0.5) * settings.windDirectionChangeAmount; 302 | windAngle.current += nudge; 303 | // Clamp angle to [-PI, PI] for stability 304 | if (windAngle.current > Math.PI) windAngle.current -= 2 * Math.PI; 305 | if (windAngle.current < -Math.PI) windAngle.current += 2 * Math.PI; 306 | } 307 | // Compute wind X (horizontal) from angle 308 | const windX = Math.cos(windAngle.current); 309 | // Optionally, you could use windY = Math.sin(windAngle.current) for vertical modulation 310 | uniforms.u_wind.value = w.current * windX; 311 | // Subtle wind change (gentle drift) 312 | if (Math.random() > 0.98) { 313 | w.target += (Math.random() - 0.5) * 0.01; // Small nudge 314 | w.target = Math.max(w.min, Math.min(w.max, w.target)); 315 | } 316 | // Occasional strong/random wind change 317 | if (Math.random() > 0.995) { 318 | w.target = (w.min + Math.random() * (w.max - w.min)) * (Math.random() > 0.5 ? -1 : 1); 319 | } 320 | uniforms.u_time.value = state.clock.getElapsedTime(); 321 | }); 322 | 323 | if (!texture) return null; 324 | 325 | return ( 326 | 327 | 328 | 334 | 340 | 346 | 352 | 358 | 364 | 370 | 376 | 382 | 388 | 394 | 400 | 406 | 407 | 414 | 415 | ); 416 | } 417 | 418 | /** 419 | * SnowEffect component 420 | * Props: 421 | * type: 'gentle' | 'storm' (default: 'gentle') 422 | * - 'gentle': Gentle snowfall 423 | * - 'storm': Heavy snowstorm 424 | */ 425 | 426 | export default function SnowEffect({ type = 'gentle', backgroundImageUrl }) { 427 | const settings = type === 'storm' ? STORM_SETTINGS : GENTLE_SETTINGS; 428 | return ( 429 |
430 | 431 | 432 | 433 |
434 | ); 435 | } --------------------------------------------------------------------------------