#snow
View Compiled
#snow
display: block
position: fixed
left: 0
top: 0
right: 0
bottom: 0
background-color: #000000
background-image: linear-gradient(to bottom, #000000, #050505)
View Compiled
// https://github.com/bsehovac/shader-program
const snowflake = ''
const holder = document.querySelector( '#snow' )
const count = 7000
let wind = {
current: 0,
force: 0.1,
target: 0.1,
min: 0.1,
max: 0.25,
easing: 0.005
}
const snow = new ShaderProgram( holder, {
depthTest: false,
texture: snowflake,
uniforms: {
worldSize: { type: 'vec3', value: [ 0, 0, 0 ] },
gravity: { type: 'float', value: 100 },
wind:{ type: 'float', value: 0 },
},
buffers: {
size: { size: 1, data: [] },
rotation: { size: 3, data: [] },
speed: { size: 3, data: [] },
},
vertex: `
precision highp float;
attribute vec4 a_position;
attribute vec4 a_color;
attribute vec3 a_rotation;
attribute vec3 a_speed;
attribute float a_size;
uniform float u_time;
uniform vec2 u_mousemove;
uniform vec2 u_resolution;
uniform mat4 u_projection;
uniform vec3 u_worldSize;
uniform float u_gravity;
uniform float u_wind;
varying vec4 v_color;
varying float v_rotation;
void main() {
v_color = a_color;
v_rotation = a_rotation.x + u_time * a_rotation.y;
vec3 pos = a_position.xyz;
pos.x = mod(pos.x + u_time + u_wind * a_speed.x, u_worldSize.x * 2.0) - u_worldSize.x;
pos.y = mod(pos.y - u_time * a_speed.y * u_gravity, u_worldSize.y * 2.0) - u_worldSize.y;
pos.x += sin(u_time * a_speed.z) * a_rotation.z;
pos.z += cos(u_time * a_speed.z) * a_rotation.z;
gl_Position = u_projection * vec4( pos.xyz, a_position.w );
gl_PointSize = ( a_size / gl_Position.w ) * 100.0;
}`,
fragment: `
precision highp float;
uniform sampler2D u_texture;
varying vec4 v_color;
varying float v_rotation;
void main() {
vec2 rotated = vec2(
cos(v_rotation) * (gl_PointCoord.x - 0.5) + sin(v_rotation) * (gl_PointCoord.y - 0.5) + 0.5,
cos(v_rotation) * (gl_PointCoord.y - 0.5) - sin(v_rotation) * (gl_PointCoord.x - 0.5) + 0.5
);
vec4 snowflake = texture2D(u_texture, rotated);
gl_FragColor = vec4(snowflake.rgb, snowflake.a * v_color.a);
}`,
onResize( w, h, dpi ) {
const position = [], color = [], size = [], rotation = [], speed = []
// z in range from -80 to 80, camera distance is 100
// max height at z of -80 is 110
const height = 110
const width = w / h * height
const depth = 80
Array.from( { length: w / h * count }, snowflake => {
position.push(
-width + Math.random() * width * 2,
-height + Math.random() * height * 2,
Math.random() * depth * 2
)
speed.push(// 0, 0, 0 )
1 + Math.random(),
1 + Math.random(),
Math.random() * 10
) // x, y, sinusoid
rotation.push(
Math.random() * 2 * Math.PI,
Math.random() * 20,
Math.random() * 10
) // angle, speed, sinusoid
color.push(
1,
1,
1,
0.1 + Math.random() * 0.2
)
size.push(
5 * Math.random() * 5 * ( h * dpi / 1000 )
)
} )
this.uniforms.worldSize = [ width, height, depth ]
this.buffers.position = position
this.buffers.color = color
this.buffers.rotation = rotation
this.buffers.size = size
this.buffers.speed = speed
},
onUpdate( delta ) {
wind.force += ( wind.target - wind.force ) * wind.easing
wind.current += wind.force * ( delta * 0.2 )
this.uniforms.wind = wind.current
if ( Math.random() > 0.995 ) {
wind.target = ( wind.min + Math.random() * ( wind.max - wind.min ) ) * ( Math.random() > 0.5 ? -1 : 1 )
}
// stats.update()
},
} )
This Pen doesn't use any external CSS resources.