mirror of
https://github.com/jasonm76/GSjs.git
synced 2025-01-13 15:33:00 +00:00
Prototype files for testing NTSC filtering and barrel effects via WebGL
This commit is contained in:
parent
01195c203a
commit
7b15e28706
135
filter.js
Normal file
135
filter.js
Normal file
@ -0,0 +1,135 @@
|
||||
var yb = [ 0.00010501, 0.00042004, 0.00063006, 0.00042004, 0.00010501];
|
||||
var ya = [ 1.00000000, -3.33787709, 4.20480923, -2.36821297, 0.50296098];
|
||||
var y_zi = [ 0.99989499, -2.33840213, 1.86577704, -0.50285597, 0.00000000];
|
||||
|
||||
var cb = [ 0.70911985, -2.83647938, 4.25471907, -2.83647938, 0.70911985];
|
||||
var ca = [ 1.00000000, -3.3302056 , 4.17977946, -2.34191808, 0.49401438];
|
||||
var c_zi = [-0.70911985, 2.12735954, -2.12735954, 0.70911985, 0.00000000];
|
||||
|
||||
var ib = [ 0.00605349, 0.01210697, 0.00605349];
|
||||
var ia = [ 1.00000000, -1.76816293, 0.79237688];
|
||||
var i_zi = [ 0.99394651, -0.78632339, 0.00000000];
|
||||
|
||||
var qb = [ 0.00278987, 0.00557974, 0.00278987];
|
||||
var qa = [ 1.00000000, -1.84512919, 0.85628867];
|
||||
var q_zi = [ 0.99721013, -0.85349880, 0.00000000];
|
||||
|
||||
function add_index_range(indices, beg, end)
|
||||
{
|
||||
for (var i = beg; i <= end; ++i)
|
||||
indices.push(i);
|
||||
return indices;
|
||||
}
|
||||
|
||||
function add_index_const(indices, value, numel)
|
||||
{
|
||||
while (numel--)
|
||||
indices.push(value);
|
||||
return indices;
|
||||
}
|
||||
|
||||
function subvector_reverse(vec, idx_end, idx_start)
|
||||
{
|
||||
return vec.slice(idx_start, idx_end+1).reverse();
|
||||
}
|
||||
|
||||
inline int max_val(const vectori& vec)
|
||||
{
|
||||
return std::max_element(vec.begin(), vec.end())[0];
|
||||
}
|
||||
|
||||
function filtfilt(B, A, X, zi)
|
||||
{
|
||||
var len = X.length; // length of input
|
||||
var na = A.length;
|
||||
var nb = B.length;
|
||||
var nfilt = (nb > na) ? nb : na;
|
||||
var nfact = 3 * (nfilt - 1); // length of edge transients
|
||||
|
||||
if (len <= nfact) {
|
||||
console.log("Input data too short! Data must have length more than 3 times filter order.");
|
||||
return X;
|
||||
}
|
||||
|
||||
// set up filter's initial conditions to remove DC offset problems at the
|
||||
// beginning and end of the sequence
|
||||
|
||||
var rows, cols;
|
||||
//rows = [1:nfilt-1 2:nfilt-1 1:nfilt-2];
|
||||
rows = add_index_range([], 0, nfilt - 2);
|
||||
if (nfilt > 2)
|
||||
{
|
||||
rows = add_index_range(rows, 1, nfilt - 2);
|
||||
rows = add_index_range(rows, 0, nfilt - 3);
|
||||
}
|
||||
//cols = [ones(1,nfilt-1) 2:nfilt-1 2:nfilt-1];
|
||||
cols = add_index_const([], 0, nfilt - 1);
|
||||
if (nfilt > 2)
|
||||
{
|
||||
cols = add_index_range(cols, 1, nfilt - 2);
|
||||
cols = add_index_range(cols, 1, nfilt - 2);
|
||||
}
|
||||
// data = [1+a(2) a(3:nfilt) ones(1,nfilt-2) -ones(1,nfilt-2)];
|
||||
|
||||
var klen = rows.length;
|
||||
var data = new Array(klen);
|
||||
|
||||
data[0] = 1 + A[1]; var i, j = 1;
|
||||
if (nfilt > 2)
|
||||
{
|
||||
for (i = 2; i < nfilt; i++)
|
||||
data[j++] = A[i];
|
||||
for (i = 0; i < nfilt - 2; i++)
|
||||
data[j++] = 1.0;
|
||||
for (i = 0; i < nfilt - 2; i++)
|
||||
data[j++] = -1.0;
|
||||
}
|
||||
|
||||
var leftpad = subvector_reverse(X, nfact, 1);
|
||||
var _2x0 = 2 * X[0];
|
||||
for (i = 0; i < leftpad.length; ++i) {leftpad[i] = _2x0 - leftpad[i];}
|
||||
|
||||
var rightpad = subvector_reverse(X, len - 2, len - nfact - 1);
|
||||
var _2xl = 2 * X[len-1];
|
||||
for (i = 0; i < rightpad.length; ++i) {rightpad[i] = _2xl - rightpad[i];}
|
||||
|
||||
var y0;
|
||||
|
||||
var signal1 = leftpad.concat(X, rightpad);
|
||||
|
||||
// Do the forward and backward filtering
|
||||
y0 = signal1[0];
|
||||
zzi = zi.slice(0); // clone the array;
|
||||
for (i = 0; i < zzi.length - 1; ++i) {zzi[i] = y0 * zzi[i];}
|
||||
var signal2 = filter(B, A, signal1, zzi).reverse();
|
||||
zzi = zi.slice(0);
|
||||
y0 = signal2[0];
|
||||
for (i = 0; i < zzi.length - 1; ++i) {zzi[i] = y0 * zzi[i];}
|
||||
signal1 = filter(B, A, signal2, zzi);
|
||||
|
||||
return subvector_reverse(signal1, signal1.length - nfact - 1, nfact);
|
||||
}
|
||||
|
||||
function filter(B, A, X, zi)
|
||||
{
|
||||
|
||||
var input_size = X.length;
|
||||
var filter_order = A.length
|
||||
|
||||
// Initialize Y, the return variable
|
||||
var Y = new Array(input_size);
|
||||
for (var i = 0; i < input_size; ++i) {Y[i] = 0.0;}
|
||||
|
||||
for (i = 0; i < input_size; ++i) {
|
||||
var order = filter_order - 1;
|
||||
while(order) {
|
||||
if (i >= order) {
|
||||
zi[order - 1] = B[order] * X[i - order] - A[order] * Y[i - order] + zi[order];
|
||||
}
|
||||
--order;
|
||||
}
|
||||
Y[i] = B[0] * X[i] + zi[0];
|
||||
}
|
||||
|
||||
return Y;
|
||||
}
|
BIN
lo-res-raw.png
Normal file
BIN
lo-res-raw.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 7.9 KiB |
199
test.html
Normal file
199
test.html
Normal file
@ -0,0 +1,199 @@
|
||||
<!DOCTYPE html5>
|
||||
<html>
|
||||
<head>
|
||||
<title>JS App</title>
|
||||
<style>
|
||||
body { margin: 0; }
|
||||
canvas { width: 1024; height: 768 }
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<script type="x-shader/x-vertex" id="vertexShader" >
|
||||
varying vec2 vUv;
|
||||
|
||||
void main() {
|
||||
//vUv = uv;
|
||||
vUv = vec2(uv.x, uv.y);
|
||||
gl_Position = projectionMatrix *
|
||||
modelViewMatrix *
|
||||
vec4(position,1.0);
|
||||
}
|
||||
</script>
|
||||
<script type="x-shader/x-vertex" id="demoShader">
|
||||
uniform vec3 ShaderColor;
|
||||
|
||||
void main(void) {
|
||||
gl_FragColor = vec4(ShaderColor, 1.0);
|
||||
}
|
||||
</script>
|
||||
<script type="x-shader/x-vertex" id="textureShader">
|
||||
uniform sampler2D texture1;
|
||||
varying vec2 vUv;
|
||||
|
||||
void main(void) {
|
||||
gl_FragColor = texture2D(texture1, vUv);
|
||||
}
|
||||
</script>
|
||||
|
||||
<script type="x-shader/x-vertex" id="ntscShader">
|
||||
uniform sampler2D texture1;
|
||||
varying vec2 vUv;
|
||||
|
||||
mat3 decoderMatrix = mat3(1, 1, 1, 0, -0.3946419954299927, 2.03206205368042, 1.139883041381836, -0.5806220173835754, 0);
|
||||
vec3 decoderOffset = vec3(0, 0, 0);
|
||||
vec3 c0 = vec3(0.2794082164764404, 0.260298490524292, 0.260298490524292);
|
||||
vec3 c1 = vec3(0.2359260767698288, 0.2478785365819931, 0.2478785365819931);
|
||||
vec3 c2 = vec3(0.1346162706613541, 0.2137254178524017, 0.2137254178524017);
|
||||
vec3 c3 = vec3(0.03665492311120033, 0.1660159379243851, 0.1660159379243851);
|
||||
vec3 c4 = vec3(-0.01538314949721098, 0.11509008705616, 0.11509008705616);
|
||||
vec3 c5 = vec3(-0.02210377156734467, 0.07008156925439835, 0.07008156925439835);
|
||||
vec3 c6 = vec3(-0.009993300773203373, 0.036478441208601, 0.036478441208601);
|
||||
vec3 c7 = vec3(-0.0007181052351370454, 0.01543270610272884, 0.01543270610272884);
|
||||
vec3 c8 = vec3(0.001296989037655294, 0.005147919990122318, 0.005147919990122318);
|
||||
vec2 textureSize = vec2(1024, 256);
|
||||
float subcarrier = 0.25;
|
||||
vec2 phaseInfo = vec2(0.908333301, 0.0);
|
||||
|
||||
float PI = 3.14159265358979323846264;
|
||||
|
||||
vec3 pixel(in vec2 q)
|
||||
{
|
||||
vec3 c = texture2D(texture1, q).rgb; // c = the texture color at x,y (q)
|
||||
|
||||
float phase = 2.0 * PI * (subcarrier * textureSize.x * q.x + phaseInfo.x);
|
||||
return c * vec3(1.0, sin(phase), (1.0 - 2.0 * phaseInfo.y) * cos(phase));
|
||||
}
|
||||
|
||||
vec3 pixels(vec2 q, float i)
|
||||
{
|
||||
return pixel(vec2(q.x + i, q.y)) + pixel(vec2(q.x - i, q.y));
|
||||
}
|
||||
|
||||
void main(void)
|
||||
{
|
||||
vec3 c = pixel(vUv) * c0;
|
||||
c += pixels(vUv, 1.0 / textureSize.x) * c1;
|
||||
c += pixels(vUv, 2.0 / textureSize.x) * c2;
|
||||
c += pixels(vUv, 3.0 / textureSize.x) * c3;
|
||||
c += pixels(vUv, 4.0 / textureSize.x) * c4;
|
||||
c += pixels(vUv, 5.0 / textureSize.x) * c5;
|
||||
c += pixels(vUv, 6.0 / textureSize.x) * c6;
|
||||
c += pixels(vUv, 7.0 / textureSize.x) * c7;
|
||||
c += pixels(vUv, 8.0 / textureSize.x) * c8;
|
||||
|
||||
gl_FragColor = vec4(decoderMatrix * c + decoderOffset, 1.0);
|
||||
}
|
||||
</script>
|
||||
<script type="x-shader/x-vertex" id="barrelShader">
|
||||
uniform sampler2D texture1;
|
||||
|
||||
vec2 textureSize = vec2(1024,256);
|
||||
float barrel = 0.10;
|
||||
vec2 barrelSize = vec2(1, 0.75);
|
||||
|
||||
float scanlineLevel = 0.50;
|
||||
|
||||
float centerLighting = 0.0;
|
||||
|
||||
float luminanceGain = 1.0;
|
||||
|
||||
float PI = 3.14159265358979323846264;
|
||||
|
||||
varying vec2 vUv;
|
||||
|
||||
void main(void)
|
||||
{
|
||||
|
||||
vec2 qc = (vUv - vec2(0.5, 0.5)) * barrelSize;
|
||||
vec2 qb = barrel * qc * dot(qc, qc);
|
||||
vec2 q = vUv + qb;
|
||||
|
||||
vec3 c = texture2D(texture1, q).rgb;
|
||||
|
||||
float scanline = sin(PI * textureSize.y * q.y);
|
||||
c *= mix(1.0, scanline * scanline, scanlineLevel);
|
||||
|
||||
vec2 lighting = qc * centerLighting;
|
||||
c *= exp(-dot(lighting, lighting));
|
||||
|
||||
c *= luminanceGain;
|
||||
|
||||
gl_FragColor = vec4(c, 1.0);
|
||||
}
|
||||
</script>
|
||||
<script src="three.min.js"></script>
|
||||
|
||||
<script>
|
||||
var rttScene = new THREE.Scene();
|
||||
var renderTargetParams = {
|
||||
minFilter:THREE.LinearFilter,
|
||||
stencilBuffer:false,
|
||||
depthBuffer:false
|
||||
};
|
||||
// Load the raw image that is a copy of the Apple II display memory
|
||||
var apple_memory = THREE.ImageUtils.loadTexture( "lo-res-raw.png" );
|
||||
var imageWidth = 1024.0;
|
||||
var imageHeight = 768.0;
|
||||
// Create bugger
|
||||
var ntsc_texture = new THREE.WebGLRenderTarget( imageWidth, imageHeight, renderTargetParams );
|
||||
var ntsc_shader_material = new THREE.ShaderMaterial( {
|
||||
uniforms: {
|
||||
texture1: { type: "t", value: apple_memory }
|
||||
},
|
||||
vertexShader: document.getElementById( 'vertexShader' ).textContent,
|
||||
fragmentShader: document.getElementById( 'ntscShader' ).textContent,
|
||||
} );
|
||||
|
||||
// Setup render-to-texture scene
|
||||
var rttCamera = new THREE.OrthographicCamera( imageWidth / - 2, imageWidth / 2, imageHeight / 2, imageHeight / - 2, -10000, 10000 );
|
||||
|
||||
var rttGeometry = new THREE.PlaneGeometry( imageWidth, imageHeight );
|
||||
var rttMesh = new THREE.Mesh( rttGeometry, ntsc_shader_material );
|
||||
rttMesh.position.z = -100;
|
||||
rttScene.add( rttMesh );
|
||||
|
||||
|
||||
var scene = new THREE.Scene();
|
||||
var camera = new THREE.PerspectiveCamera(45, imageWidth/imageHeight, 1, 1000);
|
||||
//var camera = new THREE.OrthographicCamera(-384, 384, -288, 288, -1000, 1000);
|
||||
var renderer = new THREE.WebGLRenderer();
|
||||
renderer.setSize(imageWidth, imageHeight);
|
||||
document.body.appendChild(renderer.domElement);
|
||||
|
||||
var cube_geometry = new THREE.BoxGeometry( imageWidth, imageHeight, 1 );
|
||||
|
||||
var shader_material = new THREE.ShaderMaterial( {
|
||||
uniforms: {
|
||||
texture1: { type: "t", value: ntsc_texture }
|
||||
//ShaderColor: { type: "c", value: new THREE.Color( 0xFF8040 ) },
|
||||
},
|
||||
vertexShader: document.getElementById( 'vertexShader' ).textContent,
|
||||
fragmentShader: document.getElementById( 'barrelShader' ).textContent,
|
||||
} );
|
||||
|
||||
var cube = new THREE.Mesh( cube_geometry, shader_material );
|
||||
//var plane = new THREE.Mesh( plane_geometry, shader_material );
|
||||
//var plane2 = new THREE.Mesh( plane_geometry, plain_material );
|
||||
scene.add( cube );
|
||||
|
||||
//var geometry = new THREE.PlaneBufferGeometry( 592, 500);
|
||||
|
||||
//var pcolor = new THREE.Color( 0xFF8040);
|
||||
|
||||
//var material = new THREE.MeshBasicMaterial( {color: 0xff0000, side: THREE.DoubleSide} );
|
||||
//var plane = new THREE.Mesh( geometry, shader_material );
|
||||
//scene.add( plane );
|
||||
|
||||
camera.position.z = 1.5;
|
||||
cube.position.z = -695.293506+1.0;
|
||||
function render() {
|
||||
requestAnimationFrame( render );
|
||||
//cube.rotation.x += 0.01;
|
||||
//cube.rotation.y += 0.1;
|
||||
renderer.render( rttScene, rttCamera, ntsc_texture);
|
||||
renderer.render( scene, camera );
|
||||
}
|
||||
render();
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
814
three.min.js
vendored
Normal file
814
three.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
Loading…
x
Reference in New Issue
Block a user