{
const canvas = DOM.canvas(600, 600);
const glsl = SwissGL(canvas);
const worldExtent = 15;
function render(t) {
const K = 6;
const F = glsl(`
float(I.x==I.y) + 0.1*float(I.x==I.y+1)`,
{size:[K,K], format:'r16f'});
glsl({F}, `F(I/40).x*3.0`);
const points = glsl({size:[30,10], story:3, format:'rgba32f', tag:'points'});
for (let i=0; i<2; ++i) {
glsl({K, seed:123}, `
vec2 pos = (hash(ivec3(I, seed)).xy-0.5)*10.0;
float color = floor(UV.x*K);
out0 = vec4(pos, 0.0, color);`,
points);
}
function loop() {
const dt = 0.1;
glsl({F, worldExtent, repulsion, inertia, dt,
past:points[1]}, `
// this function wraps positions and velocities to
// [-worldExtent/2, worldExtent/2] range
vec3 wrap(vec3 p) {
return (fract(p/worldExtent+0.5)-0.5)*worldExtent;
}
void fragment() {
// read the current particle state
out0 = Src(I);
vec3 force=vec3(0); // force accumulator
// iterate over particles
for (int y=0; y<ViewSize.y; ++y)
for (int x=0; x<ViewSize.x; ++x) {
// reading the state of another particle
vec4 data1 = Src(ivec2(x,y));
vec3 dpos = wrap(data1.xyz-out0.xyz);
// calculate distance
float r = length(dpos);
if (r>3.0) continue;
dpos /= r+1e-8;
// calculate repulsion and interaction forces
float rep = max(1.0-r, 0.0)*repulsion;
float f = F(ivec2(out0.w, data1.w)).x;
float inter = f*max(1.0-abs(r-2.0), 0.0);
force += dpos*(inter-rep);
}
// fetch the past state to compute velocity
vec3 vel = wrap(out0.xyz-past(I).xyz)*pow(inertia, dt);
// update particle position
out0.xyz = wrap(out0.xyz+vel+0.5*force*(dt*dt));
}
`, points);
glsl({K, worldExtent,
points: points[0],
Grid: points[0].size,
Aspect:'fit',
Blend: 'd*(1-sa)+s*sa'}, `
// the code below is available in both
// vertex and fragment shaders
varying vec3 color;
//VERT start of vertex-only section
// vertex function is called
vec4 vertex() {
// get current particle data
vec4 d = points(ID);
// populate varyings to use in fragment shader
color = cos((d.w/K+vec3(0,0.33,0.66))*TAU)*0.5+0.5;
// emit normalized on-screen vertex position
// 'vec2 XY' is contains coordinates of the quad vertex in -1..1 range
return vec4(2.0*(d.xy+XY/8.0)/worldExtent, 0.0, 1.0);
}
//FRAG start of fragment-only section
void fragment() {
out0 = vec4(1.0);
// Compute the fragment transparency depending
// on the distance from the quad center.
// Interpolated XY is also available in the fragment shader.
float alpha = smoothstep(1.0, 0.6, length(XY));
// set the fragment color
out0 = vec4(color, alpha);
}`);
requestAnimationFrame(loop);
}
loop();
}
render();
return canvas;
}