// ==UserScript==
// @name youtubewall
// @description Fancy Youtube Fullscreen Graphic Demo
// @namespace youtubewall
// @include https://www.youtube.com/*
// @include http://www.youtube.com/*
// @version 1.1
// @grant none
// @run-at document-end
// ==/UserScript==
var SHADER_FRAGMENT = "precision mediump float;\n" +
"\n" +
"varying vec2 vTextureCoord;\n" +
"varying vec2 vTextureCoord2;\n" +
"varying vec2 vOther;\n" +
"varying float si1;\n" +
"varying float si4;\n" +
"\n" +
"uniform sampler2D diffuseTextureSampler;\n" +
"uniform sampler2D normalTextureSampler;\n" +
"\n" +
"vec3 texArc(float wi, float r) {\n" +
" vec2 c = vec2(0.5 * r * sin(wi) + 0.5, 0.5 * r * cos(wi) + 0.5);\n" +
" return vec3(texture2D(diffuseTextureSampler, c)) ;\n" +
" //return (vec3(c, 0.0));\n" +
"}\n" +
"\n" +
"void main(void) {\n" +
" vec3 bcol = vec3(texture2D(diffuseTextureSampler, vTextureCoord));\n" +
" vec3 wcol = vec3(texture2D(normalTextureSampler, vTextureCoord2));\n" +
"\n" +
" float ax = 0.0;\n" +
" float ay = 0.0;\n" +
" float dx = 0.0;\n" +
" float dy = 0.0;\n" +
" float ux = si4 + vTextureCoord.x * (1.0 - 2.0 * si4);\n" +
" float uy = si4 + vTextureCoord.y * (1.0 - 2.0 * si4);\n" +
" \n" +
" if (vTextureCoord.x > 1.0) { ax = vTextureCoord.x - 1.0; ux = 1.0 - si4; dx = -1.0; }\n" +
" if (vTextureCoord.y > 1.0) { ay = vTextureCoord.y - 1.0; uy = 1.0 - si4; dy = -1.0; }\n" +
" if (vTextureCoord.x < 0.0) { ax = vTextureCoord.x; ux = si4; dx = 1.0; }\n" +
" if (vTextureCoord.y < 0.0) { ay = vTextureCoord.y; uy = si4; dy = 1.0; }\n" +
"\n" +
" \n" +
" vec4 a1, a2, a3, a4, a5, a6, a7, a8, a9, a10, a11, a12, a13, a14, a15, a16, a17;\n" +
" \n" +
" float dst = sqrt(ax * ax + ay * ay) * 2.0;\n" +
" \n" +
" float myx = vTextureCoord.x - 0.5;\n" +
" float myy = vTextureCoord.y - 0.5;\n" +
" float s1 = si1 + dst * 0.02;\n" +
" float r = 0.95 + 0.05 * sqrt(myy * myy + myx * myx);\n" +
" float wi = atan(myx, myy);\n" +
" \n" +
" bcol = (texArc(wi, 1.0) + \n" +
" 0.5 * texArc(wi - 0.015, r) + \n" +
" 0.5 * texArc(wi + 0.015, r) + \n" +
" 0.25 * texArc(wi - 0.04, r) + \n" +
" 0.25 * texArc(wi + 0.04, r)) / 2.5;\n" +
" \n" +
" //a9 = texture2D(diffuseTextureSampler, vec2(ux , uy));\n" +
" \n" +
" gl_FragColor = vec4(0.2 * wcol + 0.8 * wcol * bcol * 1.0 / sqrt(1.0 + dst * dst), 1.0);\n" +
"// gl_FragColor = vec4(bcol * 1.0 / sqrt(1.0 + dst * dst), 1.0);\n" +
"}\n";
var SHADER_VERTEX = "precision mediump float;\n" +
"\n" +
"attribute vec3 vPosition;\n" +
"\n" +
"#define imagesize 0.5\n" +
"\n" +
"varying vec2 vTextureCoord2;\n" +
"varying vec2 vTextureCoord;\n" +
"varying vec2 vOther;\n" +
"\n" +
"#define d 0.005\n" +
"\n" +
"varying float si1;\n" +
"varying float si4;\n" +
"\n" +
"void main(void) {\n" +
" gl_Position = vec4(vPosition, 1.0);\n" +
" vTextureCoord2 = vec2((vPosition.x + 1.0), 2.0 - (vPosition.y + 1.0)) * 0.5;\n" +
" vTextureCoord = 0.5 + (vTextureCoord2 - 0.5) / imagesize;\n" +
" \n" +
" si1 = d / imagesize;\n" +
" si4 = 8.0 * si1;\n" +
"}\n";
/* RENDER.JS */
function Event() {
this.ctor = function() {
this.handlers = [];
};
this.add = function(handler) {
this.handlers.push(handler);
};
this.remove = function(handler) {
this.handlers.removeValue(handler);
};
this.invoke = function(a, b, c, d, e, f, g, h, i) {
for (var i = 0; i < this.handlers.length; i++) {
try {
this.handlers[i](a, b, c, d, e, f, g, h, i);
}
catch (e) {
// whatup?!
}
}
}
this.ctor();
}
function pglRender(gl) {
this.gl = gl;
this.drawable = null;
this.frames = 0;
this.beforeRender = new Event();
this.afterRender = new Event();
this.beforeInit = new Event();
this.myLoop = null;
this.init = function() {
this.beforeInit.invoke(this);
this.drawable.init(gl);
};
this.startLoop = function() {
var self = this;
this.myLoop = window.setInterval(function() { self.draw(); }, 25);
};
this.stop = function() {
window.clearInterval(this.myLoop);
};
this.draw = function() {
this.beforeRender.invoke(this.gl, this);
this.drawable.draw(this.gl, this);
this.afterRender.invoke(this.gl, this);
this.frames++;
};
}
/* VIDEO.JS */
function pglVideo(gl, videoParam) {
if (typeof videoParam === 'string') {
this.videoUrl = videoParam;
this.videoElement = null;
} else {
this.videoUrl = "";
this.videoElement = videoParam;
}
//@Override
this.texture = null;
this.gl = gl;
this.init = function() {
var self = this;
if (this.videoElement == null)
this.createVideoElement();
this.createTexture();
// this.render.beforeRender.add(function() { self.loadTexture(); });
};
this.createVideoElement = function() {
this.videoElement = document.createElement('video');
this.videoElement.src = this.videoUrl;
this.videoElement.autoplay = true;
this.videoElement.preload = 'auto';
this.videoElement.loop = true;
// this.videoElement.style.display = 'none';
// document.body.appendChild(this.videoElement);
};
this.createTexture = function() {
this.texture = this.gl.createTexture();
};
this.updateTexture = function() {
this.gl.bindTexture(this.gl.TEXTURE_2D, this.texture);
this.gl.texImage2D(this.gl.TEXTURE_2D, 0,this. gl.RGBA, this.gl.RGBA, this.gl.UNSIGNED_BYTE, this.videoElement);
// this.gl.texImage2D(this.gl.TEXTURE_2D, 0, this.imageElement, true);
this.gl.texParameteri(this.gl.TEXTURE_2D, this.gl.TEXTURE_MAG_FILTER, this.gl.LINEAR);
this.gl.texParameteri(this.gl.TEXTURE_2D, this.gl.TEXTURE_MIN_FILTER, this.gl.LINEAR);
this.gl.texParameteri(this.gl.TEXTURE_2D, this.gl.TEXTURE_WRAP_S, this.gl.CLAMP_TO_EDGE);
this.gl.texParameteri(this.gl.TEXTURE_2D, this.gl.TEXTURE_WRAP_T, this.gl.CLAMP_TO_EDGE);
// this.gl.generateMipmap(this.gl.TEXTURE_2D);
this.gl.bindTexture(this.gl.TEXTURE_2D, null);
};
this.init();
}
/* IMAGE.JS */
function pglImage(gl, imageParam) {
if (typeof imageParam === 'string') {
this.imageUrl = imageParam;
this.imageElement = null;
} else {
this.imageUrl = "";
this.imageElement = imageParam;
}
//@Override
this.texture = null;
this.gl = gl;
this.init = function() {
var self = this;
this.createImageElement();
this.createTexture();
};
this.createImageElement = function() {
var self = this;
if (this.imageElement == null) {
this.imageElement = new Image();
this.imageElement.src = this.imageUrl;
this.imageElement.addEventListener('load', function() { self.loadTexture(); }, false);
} else {
window.setTimeout(function() { self.loadTexture(); }, 500);
}
};
this.createTexture = function() {
console.log('ctex');
this.texture = this.gl.createTexture();
};
this.loadTexture = function() {
this.gl.bindTexture(this.gl.TEXTURE_2D, this.texture);
this.gl.texImage2D(this.gl.TEXTURE_2D, 0,this. gl.RGBA, this.gl.RGBA, this.gl.UNSIGNED_BYTE, this.imageElement);
// this.gl.texImage2D(this.gl.TEXTURE_2D, 0, this.imageElement, true);
this.gl.texParameteri(this.gl.TEXTURE_2D, this.gl.TEXTURE_MAG_FILTER, this.gl.LINEAR);
this.gl.texParameteri(this.gl.TEXTURE_2D, this.gl.TEXTURE_MIN_FILTER, this.gl.LINEAR);
this.gl.texParameteri(this.gl.TEXTURE_2D, this.gl.TEXTURE_WRAP_S, this.gl.CLAMP_TO_EDGE);
this.gl.texParameteri(this.gl.TEXTURE_2D, this.gl.TEXTURE_WRAP_T, this.gl.CLAMP_TO_EDGE);
// this.gl.generateMipmap(this.gl.TEXTURE_2D);
this.gl.bindTexture(this.gl.TEXTURE_2D, null);
};
this.init();
}
/* SCENE.JS */
function pglDemoScene() {
this.cube = new pglDemoCube();
this.shdr = new pglShader();
this.init = function(gl) {
this.cube.init(gl);
};
this.draw = function(gl) {
gl.clearColor(0, 0, 0, 1);
gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);
this.shdr.use(gl);
this.cube.draw(gl, this);
};
}
/* DEMOCUBE.JS */
var pglDemoTriangleVertices = [
-1.0, -1.0, 0.0,
1.0, -1.0, 0.0,
1.0, 1.0, 0.0,
-1.0, -1.0, 0.0,
-1.0, 1.0, 0.0,
1.0, 1.0, 0.0];
var pglDemoCubeVertices = [
// Front face
-1.0, -1.0, 1.0,
1.0, -1.0, 1.0,
1.0, 1.0, 1.0,
-1.0, 1.0, 1.0,
// Back face
-1.0, -1.0, -1.0,
-1.0, 1.0, -1.0,
1.0, 1.0, -1.0,
1.0, -1.0, -1.0,
// Top face
-1.0, 1.0, -1.0,
-1.0, 1.0, 1.0,
1.0, 1.0, 1.0,
1.0, 1.0, -1.0,
// Bottom face
-1.0, -1.0, -1.0,
1.0, -1.0, -1.0,
1.0, -1.0, 1.0,
-1.0, -1.0, 1.0,
// Right face
1.0, -1.0, -1.0,
1.0, 1.0, -1.0,
1.0, 1.0, 1.0,
1.0, -1.0, 1.0,
// Left face
-1.0, -1.0, -1.0,
-1.0, -1.0, 1.0,
-1.0, 1.0, 1.0,
-1.0, 1.0, -1.0
];
var pglDemoCubeVertexNormals = [
// Front
0.0, 0.0, 1.0,
0.0, 0.0, 1.0,
0.0, 0.0, 1.0,
0.0, 0.0, 1.0,
// Back
0.0, 0.0, -1.0,
0.0, 0.0, -1.0,
0.0, 0.0, -1.0,
0.0, 0.0, -1.0,
// Top
0.0, 1.0, 0.0,
0.0, 1.0, 0.0,
0.0, 1.0, 0.0,
0.0, 1.0, 0.0,
// Bottom
0.0, -1.0, 0.0,
0.0, -1.0, 0.0,
0.0, -1.0, 0.0,
0.0, -1.0, 0.0,
// Right
1.0, 0.0, 0.0,
1.0, 0.0, 0.0,
1.0, 0.0, 0.0,
1.0, 0.0, 0.0,
// Left
-1.0, 0.0, 0.0,
-1.0, 0.0, 0.0,
-1.0, 0.0, 0.0,
-1.0, 0.0, 0.0
];
var pglDemoCubeTextureCoordinates = [
// Front
0.0, 0.0,
1.0, 0.0,
1.0, 1.0,
0.0, 1.0,
// Back
0.0, 0.0,
1.0, 0.0,
1.0, 1.0,
0.0, 1.0,
// Top
0.0, 0.0,
1.0, 0.0,
1.0, 1.0,
0.0, 1.0,
// Bottom
0.0, 0.0,
1.0, 0.0,
1.0, 1.0,
0.0, 1.0,
// Right
0.0, 0.0,
1.0, 0.0,
1.0, 1.0,
0.0, 1.0,
// Left
0.0, 0.0,
1.0, 0.0,
1.0, 1.0,
0.0, 1.0
];
var pglDemoCubeVertexIndices = [
0, 1, 2, 0, 2, 3, // front
4, 5, 6, 4, 6, 7, // back
8, 9, 10, 8, 10, 11, // top
12, 13, 14, 12, 14, 15, // bottom
16, 17, 18, 16, 18, 19, // right
20, 21, 22, 20, 22, 23 // left
];
function pglDemoCube() {
this.tex = null;
this.vertices = null;
this.normals = null;
this.texcoords = null;
this.indices = null;
this.init = function(gl) {
this.tex2 = new pglImage(gl, GLOBAL_IMAGE_PARAM);
this.tex = new pglVideo(gl, GLOBAL_VIDEO_PARAM);
this.vertices = gl.createBuffer();
this.normals = gl.createBuffer();
this.texcoords = gl.createBuffer();
this.indices = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, this.vertices);
gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(pglDemoTriangleVertices), gl.STATIC_DRAW);
gl.bindBuffer(gl.ARRAY_BUFFER, this.normals);
gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(pglDemoCubeVertexNormals), gl.STATIC_DRAW);
gl.bindBuffer(gl.ARRAY_BUFFER, this.texcoords);
gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(pglDemoCubeTextureCoordinates), gl.STATIC_DRAW);
gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indices);
gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(pglDemoCubeVertexIndices), gl.STATIC_DRAW);
};
this.draw = function(gl, scene) {
this.tex.updateTexture(gl);
gl.activeTexture(gl.TEXTURE0);
gl.bindTexture(gl.TEXTURE_2D, this.tex.texture);
gl.uniform1i(scene.shdr.diffuseTexture, 0);
gl.activeTexture(gl.TEXTURE1);
gl.bindTexture(gl.TEXTURE_2D, this.tex2.texture);
gl.uniform1i(scene.shdr.normalTexture, 1);
gl.enableVertexAttribArray(0);
gl.bindBuffer(gl.ARRAY_BUFFER, this.vertices);
gl.vertexAttribPointer(
0, // attribute 0. No particular reason for 0, but must match the layout in the shader.
3, // size
gl.FLOAT, // type
gl.FALSE, // normalized?
0, // stride
0 // array buffer offset
);
// Draw the triangle !
gl.drawArrays(gl.TRIANGLES, 0, 6); // Starting from vertex 0; 3 vertices total -> 1 triangle
gl.disableVertexAttribArray(0);
};
}
/* SHADER.JS */
function pglShader() {
this.shaderProc = null;
this.vertexShader = null;
this.fragmentShader = null;
this.diffuseTexture = null;
this.normalTexture = null;
this.compileShaders = function(gl, vCode, fCode) {
this.shaderProc = gl.createProgram();
this.vertexShader = gl.createShader(gl.VERTEX_SHADER);
this.fragmentShader = gl.createShader(gl.FRAGMENT_SHADER);
gl.shaderSource(this.vertexShader, vCode);
gl.shaderSource(this.fragmentShader, fCode);
gl.compileShader(this.vertexShader);
gl.compileShader(this.fragmentShader);
gl.attachShader(this.shaderProc, this.vertexShader);
gl.attachShader(this.shaderProc, this.fragmentShader);
gl.bindAttribLocation(this.shaderProc, 0, "vPosition");
gl.linkProgram(this.shaderProc);
this.diffuseTexture = gl.getUniformLocation(this.shaderProc, "diffuseTextureSampler");
this.normalTexture = gl.getUniformLocation(this.shaderProc, "normalTextureSampler");
}
this.use = function(gl) {
gl.useProgram(this.shaderProc);
}
}
var GLOBAL_BUTTON_ID = 'sdf8sdoa7ubhsdv9o8h';
var GLOBAL_VIDEO_PARAM = 'big-buck-bunny_trailer.webm';
var GLOBAL_IMAGE_PARAM = '';
var GLOBAL_IMAGE_TILE = GLOBAL_IMAGE_PARAM;
var s;
var r;
var gl;
var c;
var video;
var video_parent;
var video_style;
var box;
function inject_btn() {
if (document.getElementById(GLOBAL_BUTTON_ID)) {
} else {
var header = document.getElementById('watch-header');
var btn = document.createElement('button');
btn.id = GLOBAL_BUTTON_ID;
btn.innerHTML = "CUSTOM FULLSCREEN";
header.insertBefore(btn, header.firstChild);
btn.onclick = main;
}
}
function exit() {
if (document.mozFullScreenElement == null) {
r.stop();
box.remove();
video_parent.insertBefore(video, video_parent.firstChild);
video.style = video_style;
video.controls = false;
video.play();
}
}
function genbg(w, h, cb) {
var si = new Image();
si.src = GLOBAL_IMAGE_TILE;
var ca = document.createElement('canvas');
ca.width = w;
ca.height = h;
si.addEventListener('load', function() {
var tw = Math.min(si.width, 64.0);
var th = Math.min(si.height, 64.0);
var cg = ca.getContext('2d');
for (x = 0; x < w; x += tw) {
for (y = 0; y < h; y += th) {
cg.drawImage(si, x, y);
}
}
cb();
}, true);
GLOBAL_IMAGE_PARAM = ca;
}
function main()
{
c = document.createElement('canvas');
video = document.getElementsByTagName('video')[0];
video_parent = video.parentElement;
video_style = video.style;
video.remove();
box = document.createElement('div');
vbox = document.createElement('div');
box.appendChild(c);
box.appendChild(vbox);
vbox.appendChild(video);
video.controls = true;
box.style = "position: fixed; top: 0; left: 0; right: 0; bottom: 0; width: 100%; height: 100%;";
c.style = "position: fixed; top: 0; left: 0; right: 0; bottom: 0; z-index: 19999999999;";
vbox.style = "position: fixed; top: 10%; left: 10%; right: 10%; bottom: 10%; z-index: 199999999999;";
var w = screen.width;
var h = screen.height;
if (w / h > 16 / 9) {
w = h * (16 / 9);
} else {
h = w / (16 / 9);
}
c.width = w;
c.height = h;
document.body.appendChild(box);
box.mozRequestFullScreen();
document.addEventListener("mozfullscreenchange", exit, true);
GLOBAL_VIDEO_PARAM = video;
genbg(w, h, function() {
gl = c.getContext('webgl');
r = new pglRender(gl);
s = new pglDemoScene(gl);
s.shdr.compileShaders(gl, SHADER_VERTEX, SHADER_FRAGMENT);
s.init(gl);
r.drawable = s;
r.startLoop();
video.play();
video.style = "";
});
}
inject_btn();
window.setInterval(inject_btn, 2000);