monorepo/shaders/star-wars.glsl
2024-07-27 19:16:49 -04:00

252 lines
6.4 KiB
GLSL

// STAR WARS cellular automata
// ===========================
//
// an implementation of star wars (https://quuxplusone.github.io/blog/2020/06/29/star-wars-ca/)
//
// based on code for http://www.vexorian.com/2015/05/cloudy-conway.html?m=1
//
// built for use in https://github.com/markusfisch/ShaderEditor
//
// controls: tap top left to switch tool, top right to switch brush
#ifdef GL_FRAGMENT_PRECISION_HIGH
precision highp float;
#else
precision mediump float;
#endif
#define TOOLZONE > 0.8
// S_ entries mark data addresses
// x, chan, width, dec
// see rstat for more info
#define S_TOOL 0., 0, 1., 10.
#define T_FILL 0.
#define T_DEL 1.
#define T_ROTOR 2.
#define T_LATTICE 3.
#define T_WALLH 4.
#define S_BRUSH 0., 2, 1., 10.
#define BR_FINE 0.
#define BR_MED 1.
#define BR_LARGE 2.
#define BR_HUGE 3.
// ui fade out timer
#define S_UI 0., 1, 10., 100.
uniform vec2 resolution;
uniform int pointerCount;
uniform vec3 pointers[10];
uniform sampler2D backbuffer;
uniform int second;
uniform int frame;
float oneIfZero(float value) {
return step(abs(value), 0.05);
}
float oiz(float value) {
// one if zero alias
return step(abs(value), 0.05);
}
vec4 get4(float x, float y) {
return texture2D(backbuffer,
mod(gl_FragCoord.xy + vec2(x, y), resolution) / resolution);
}
vec4 get4abs(float x, float y) {
return texture2D(backbuffer, vec2(x, y) / resolution);
}
vec4 get4c(vec4 c) {
return texture2D(backbuffer, c.xy / resolution);
}
vec4 evaluate(float sum, float prev) {
vec4 cell = get4(0.0, 0.0);
float wasAlive = 1. - step(prev, 0.9);
float has2 = oneIfZero(sum - 2.0);
float has3 = oneIfZero(sum - 3.0);
float has4 = oneIfZero(sum - 4.0);
float has5 = oneIfZero(sum - 5.0);
// In this rule, a state-0 cell will become state 1
// iff it has two state-1 neighbours.
// A state-1 cell does not change if it has 3, 4 or 5
// state-1 neighbours, otherwise it will enter state 2
// next tick and then state 3 before dying.
// use the a channel for representing: states are 1.0, 0.2, 0.1, 0.0.
// 1 -> 1, 1 -> 2
float stay = wasAlive * max(min(has3 + has4 + has5, 1.), 0.2);
// 0 -> 1
float become = oneIfZero(prev) * has2;
// 2 -> 3, 3 -> 0
float decay = (1. - wasAlive) * max(prev - .1, 0.);
float state = max(stay, max(become, decay));
return vec4(
cell.g * 0.7,
oneIfZero(state - 1.) * 0.03 + cell.g * 0.95 + oneIfZero(state - .1) * 0.1,
oneIfZero(state - 0.2) * 0.2 + cell.g,
state
);
}
float get(float x, float y) {
return 1. - step(get4(x, y).a, 0.9);
}
// store state in (x, 0)'s rgb channels.
// different decimals for different state
// avoid passing data like .99 because floating point imprecision
// pass in dec 10, 100, 1000, and so on
// chan 0, 1, 2 for rgb
// width 1., 10., 100. to control how many decimals for one datum
float rstat(float x, int chan, float width, float dec) {
return floor(mod(get4abs(0.5 + x, 0.5)[chan] * dec, 10. * width));
}
// return delta to modified state, set to v
vec4 wstat(float x, int chan, float width, float dec, float v) {
vec4 ret;
ret[chan] = (v - rstat(x, chan, width, dec)) / dec;
return ret;
}
// return a color for a state
// prev is the a in the vec4 which we keep same
vec4 colstat(float s, float prev) {
return
oiz(s - 0.) * vec4(0.5, 0., 0., prev) +
oiz(s - 1.) * vec4(0.8, 0.5, 0., prev) +
oiz(s - 2.) * vec4(0.8, 0.8, 0., prev) +
oiz(s - 3.) * vec4(0., 0.5, 0., prev) +
oiz(s - 4.) * vec4(0.0, 0., 0.5, prev) +
oiz(s - 5.) * vec4(0.1, 0., 0.3, prev) +
oiz(s - 6.) * vec4(0.5, 0., 0.5, prev) +
oiz(s - 7.) * vec4(0.5, 0.5, 0.5, prev) +
oiz(s - 8.) * vec4(0., 0.5, 0.5, prev) +
oiz(s - 9.) * vec4(0., 0., 0., prev);
}
// get brush size
float getbrush() {
float id = rstat(S_BRUSH);
float side = max(resolution.x, resolution.y);
return side * (
oiz(id - 0.) * .02 +
oiz(id - 1.) * .07 +
oiz(id - 2.) * .15 +
oiz(id - 3.) * .4
);
}
void main() {
float sum =
get(-1.0, -1.0) +
get(-1.0, 0.0) +
get(-1.0, 1.0) +
get(0.0, -1.0) +
get(0.0, 1.0) +
get(1.0, -1.0) +
get(1.0, 0.0) +
get(1.0, 1.0);
vec2 center = resolution /2.;
float prev = get4(0., 0.).a;
bool switchTool = false;
vec2 puv = vec2(1.);
if (pointerCount > 0) {
puv = pointers[0].xy / resolution;
switchTool = puv.y TOOLZONE;
}
float tapSize = getbrush();
float tool = rstat(S_TOOL);
for (int n = 0; n < pointerCount; ++n) {
if (!switchTool && abs(pointers[n].y - gl_FragCoord.y) < tapSize && tool == T_WALLH) {
prev = 0.;
sum = 0.;
float c = mod(gl_FragCoord.y - .5, 6.);
if (
c < 4.
&& (mod(c, 3.) == 0. || mod(c + gl_FragCoord.x - .5, 2.) == 0.)
)
sum = 2.;
break;
} else if (!switchTool && distance(pointers[n].xy, gl_FragCoord.xy) < tapSize) {
prev = 0.;
sum = 0.;
if (tool == T_ROTOR) {
vec2 modc = mod(gl_FragCoord.xy, 9.) - 0.5;
if (abs(modc.y - 2.) + abs(modc.x - 2.) <= 1.)
sum = 2.0;
} else if (tool == T_LATTICE) {
// lattice: 01101101
if (mod(mod(gl_FragCoord.x + gl_FragCoord.y, 8.), 3.) >= 1.)
sum = 2.;
} else if (tool == T_DEL)
sum = 0.;
else if (tool == T_FILL)
sum = 2.;
break;
}
}
if (frame == 0)
gl_FragColor = vec4(0.);
else
gl_FragColor = evaluate(sum, prev);
vec2 uv = gl_FragCoord.xy / resolution.xy;
float uiTimer = rstat(S_UI);
if (gl_FragCoord.xy == vec2(0.5, 0.5)) {
gl_FragColor = get4c(gl_FragCoord);
if (switchTool && puv.x < 0.5) {
if (uiTimer < 94.) {
// change tool
float new = mod(rstat(S_TOOL) + 1., 5.);
gl_FragColor += wstat(S_TOOL, new);
// show ui
gl_FragColor += wstat(S_UI, 96.);
}
} else if (switchTool && puv.x > 0.5) {
if (uiTimer < 94.) {
// change brush size
float new = mod(rstat(S_BRUSH) + 1., 4.);
gl_FragColor += wstat(S_BRUSH, new);
// show ui
gl_FragColor += wstat(S_UI, 96.);
}
} else if (uiTimer > 50.)
gl_FragColor += wstat(S_UI, uiTimer - 1.);
else if (uiTimer <= 50.)
gl_FragColor += wstat(S_UI, max(0., uiTimer - 15.));
} else if (uv.x < 0.5 && uv.y TOOLZONE) {
gl_FragColor.rgb *= 1. - uiTimer / 100.;
vec4 statCol = colstat(rstat(S_TOOL), gl_FragColor.a);
gl_FragColor.a = 0.;
statCol.rgb *= uiTimer / 100.;
gl_FragColor += statCol;
} else if (uv.x > 0.5 && uv.y TOOLZONE) {
gl_FragColor.rgb *= 1. - uiTimer / 100.;
vec4 statCol = colstat(rstat(S_BRUSH), gl_FragColor.a);
gl_FragColor.a = 0.;
statCol.rgb *= uiTimer / 100.;
gl_FragColor += statCol;
}
}