Skip to content

Instantly share code, notes, and snippets.

Created January 1, 2016 06:37
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save anonymous/a0d0d6cad8658e4ef836 to your computer and use it in GitHub Desktop.
Save anonymous/a0d0d6cad8658e4ef836 to your computer and use it in GitHub Desktop.
/a/ EoE crosses script
var xImgB64 = '';
var WinterFlake = {
opts: null,
canvas: null,
gl: null,
particles: null,
buffer: null,
run: function(opts) {
var canvas, gl, buffer, key;
this.opts = {
src: window.xImgB64,
minCount: 40,
maxCount: 40,
minSize: 35,
maxSize: 165,
speed: 0.0015
};
if (opts) {
for (key in opts) {
this.opts[key] = opts[key];
}
}
canvas = document.createElement('canvas');
canvas.id = 'wf-canvas';
canvas.style.pointerEvents = 'none';
canvas.style.position = 'fixed';
canvas.style.left = canvas.style.top = '0';
canvas.style.width = '100%';
canvas.style.height = '100%';
canvas.width = document.documentElement.clientWidth;
canvas.height = document.documentElement.clientHeight;
gl = canvas.getContext('webgl') || canvas.getContext('experimental-webgl');
this.canvas = canvas;
this.gl = gl;
gl.viewport(0.0, 0.0, canvas.width, canvas.height);
gl.disable(gl.DEPTH_TEST);
gl.enable(gl.BLEND);
gl.blendFunc(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA);
buffer = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, buffer);
if (!this.setShaders()) {
this.destroy();
return false;
}
this.buildParticles();
this.setTexture();
document.body.appendChild(canvas);
this.mouseX = 0.0;
window.addEventListener('resize', this.onWindowResize, false);
},
buildParticles: function() {
var count, i, ii, buffer, particles;
count = (0 | Math.random() * (this.opts.maxCount - this.opts.minCount))
+ this.opts.minCount;
buffer = new Float32Array(count * 4);
particles = new Array(count);
for (i = 0; i < count; ++i) {
ii = i * 4;
particles[i] = {
x: buffer[ii] = this.getRandomPos(),
y: buffer[ii + 1] = this.getRandomPos() - 2.25,
size: buffer[ii + 2] = this.getRandomSize(),
dir: this.getRandomDir()
};
buffer[ii + 3] = this.getRandomSpt();
}
this.particles = particles;
this.buffer = buffer;
},
normalize: function(vec) {
var x, y, length, len;
x = vec[0];
y = vec[1];
length = x * x + y * y;
if (length > 0) {
len = 1 / Math.sqrt(length);
vec[0] = vec[0] * length;
vec[1] = vec[1] * length;
}
},
onWindowResize: function() {
var self = WinterFlake;
self.canvas.width = document.documentElement.clientWidth;
self.canvas.height = document.documentElement.clientHeight;
self.gl.viewport(0.0, 0.0, self.canvas.width, self.canvas.height);
},
getRandomPos: function() {
return Math.random() * 2.2 - 1.1;
},
getRandomSize: function() {
return Math.random() * (this.opts.maxSize - this.opts.minSize) + this.opts.minSize;
},
getRandomDir: function() {
return this.opts.speed + Math.random() * this.opts.speed;
},
getRandomSpt: function() {
return Math.floor((Math.random() * 4));
},
destroy: function() {
window.removeEventListener('resize', this.onWindowResize, false);
if (this.canvas && this.canvas.parentNode) {
this.canvas.parentNode.removeChild(this.canvas);
}
this.canvas = this.gl = this.particles = this.buffer = this.opts
= this.mouseX = null;
},
updateBuffer: function() {
var i, ii, p, count;
count = this.particles.length;
for (i = 0; i < count; ++i) {
ii = i * 4;
p = this.particles[i];
if (p.y > 1.25) {
p.x = this.buffer[ii] = this.getRandomPos();
p.y = this.buffer[ii + 1] = -1.25,
p.size = this.buffer[ii + 2] = this.getRandomSize();
this.buffer[ii + 3] = this.getRandomSpt();
p.dir = this.getRandomDir();
}
else {
this.buffer[ii + 1] = p.y = p.y + p.dir;
}
}
},
draw: function() {
var gl;
gl = this.gl;
gl.bufferData(gl.ARRAY_BUFFER, WinterFlake.buffer, gl.DYNAMIC_DRAW);
gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);
gl.drawArrays(gl.POINTS, 0, WinterFlake.particles.length);
},
setTexture: function() {
var img;
img = new Image();
img.onload = this.onTextureLoaded;
img.src = this.opts.src;
},
onTextureLoaded: function() {
var tex, gl;
gl = WinterFlake.gl;
tex = gl.createTexture();
gl.activeTexture(gl.TEXTURE0);
gl.bindTexture(gl.TEXTURE_2D, tex);
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, this);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR_MIPMAP_NEAREST);
gl.generateMipmap(gl.TEXTURE_2D);
WinterFlake.onAnimationFrame();
},
onAnimationFrame: function() {
if (!WinterFlake.canvas) {
return;
}
requestAnimationFrame(WinterFlake.onAnimationFrame);
WinterFlake.updateBuffer();
WinterFlake.draw();
},
setShaders: function() {
var gl, src, shader, prog, attr;
gl = WinterFlake.gl;
prog = gl.createProgram();
// Vertex
src = '\
attribute vec2 pos;\
attribute float size;\
attribute float spti;\
varying float spto;\
void main() {\
gl_PointSize = size;\
gl_Position = vec4(pos, -1.0, 1.0);\
spto = spti;\
}';
shader = gl.createShader(gl.VERTEX_SHADER);
gl.shaderSource(shader, src);
gl.compileShader(shader);
if (!gl.getShaderParameter(shader, gl.COMPILE_STATUS)) {
console.log("Couldn't compile vertex shader: " + gl.getShaderInfoLog(shader));
return false;
}
gl.attachShader(prog, shader);
// Fragment
src = '\
precision mediump float;\
uniform sampler2D sampler;\
varying float spto;\
void main() {\
vec2 pos = gl_PointCoord * 0.5;\
if (spto >= 3.0) { pos.x = pos.x + 0.5; pos.y = pos.y + 0.5; }\
else if (spto >= 2.0) { pos.y = pos.y + 0.5; }\
else if (spto >= 1.0) { pos.x = pos.x + 0.5; }\
gl_FragColor = texture2D(sampler, pos);\
}';
shader = gl.createShader(gl.FRAGMENT_SHADER);
gl.shaderSource(shader, src);
gl.compileShader(shader);
if (!gl.getShaderParameter(shader, gl.COMPILE_STATUS)) {
console.log("Couldn't compile fragment shader: " + gl.getShaderInfoLog(shader));
return false;
}
gl.attachShader(prog, shader);
gl.linkProgram(prog);
if (!gl.getProgramParameter(prog, gl.LINK_STATUS)) {
console.log("Couldn't link program");
return false;
}
gl.useProgram(prog);
this.shader = prog;
attr = gl.getAttribLocation(prog, 'pos');
gl.enableVertexAttribArray(attr);
gl.vertexAttribPointer(attr, 2, gl.FLOAT, false, 16, 0);
attr = gl.getAttribLocation(prog, 'size');
gl.enableVertexAttribArray(attr);
gl.vertexAttribPointer(attr, 1, gl.FLOAT, false, 16, 8);
attr = gl.getAttribLocation(prog, 'spti');
gl.enableVertexAttribArray(attr);
gl.vertexAttribPointer(attr, 1, gl.FLOAT, false, 16, 12);
attr = gl.getUniformLocation(prog, 'sampler');
gl.uniform1i(attr, 0);
return true;
},
};
function checkCapabilities() {
var el;
if (window.matchMedia) {
if (window.matchMedia('(max-width: 480px)').matches
|| window.matchMedia('(max-device-width: 480px)').matches) {
return false;
}
}
if (!window.WebGLRenderingContext) {
console.log('No WebGL.');
return false;
}
el = document.createElement('x');
el.style.cssText = 'pointer-events:auto';
if (el.style.pointerEvents !== 'auto') {
console.log('No Pointer Events.');
return false;
}
return true;
}
function toggleCrosses(e) {
e && e.preventDefault();
try {
if (!WinterFlake.canvas) {
WinterFlake.run();
this.innerHTML = ':stopcross:';
localStorage.removeItem('4chan-tmp-toggle');
}
else {
this.innerHTML = ':startcross:';
WinterFlake.destroy();
localStorage.setItem('4chan-tmp-toggle', 'stopcross');
}
}
catch (ex) { console.log(ex) }
}
function initWF() {
var el, o, frag, flag;
if (!checkCapabilities()) {
return;
}
o = document.getElementById('navtopright');
frag = document.createDocumentFragment();
frag.appendChild(document.createTextNode('['));
el = document.createElement('a');
el.href = '#';
el.innerHTML = ':startcross:';
el.addEventListener('click', toggleCrosses, false);
frag.appendChild(el);
frag.appendChild(document.createTextNode('] '));
o.insertBefore(frag, o.firstChild);
try {
flag = localStorage.getItem('4chan-tmp-toggle');
if (!flag || flag !== 'stopcross') {
toggleCrosses.call(el);
}
}
catch (e) {}
}
document.addEventListener('DOMContentLoaded', initWF, false);
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment