Last active
May 26, 2021 10:53
-
-
Save sporsh/de5dd62b66a18f5c6a78 to your computer and use it in GitHub Desktop.
k-d tree / octree metaball https://goo.gl/km4snW
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<!DOCTYPE html> | |
<html> | |
<head> | |
<meta charset="utf-8"> | |
<title>Astigmatism eye test</title> | |
<style> | |
canvas { | |
position: fixed; | |
top: 0; | |
left: 0; | |
width: 100%; | |
height: 100%; | |
} | |
</style> | |
<script> | |
var angle = 0; | |
var radius = 20; | |
var tile, tileCtx, ctx; | |
onload = function() { | |
tile = document.createElement('canvas'); | |
tileCtx = tile.getContext('2d'); | |
ctx = canvas.getContext('2d'); | |
draw(); | |
} | |
function draw() { | |
tile.width = tile.height = radius * 2; | |
tileCtx.lineWidth = Math.max(1, radius / 3); | |
var cos = Math.cos(angle / 360 * Math.PI); | |
var sin = Math.sin(angle / 360 * Math.PI); | |
var x1 = (cos * radius) + radius; | |
var y1 = (sin * radius) + radius; | |
var x2 = tile.width - x1; | |
var y2 = tile.height - y1; | |
tileCtx.beginPath(); | |
tileCtx.moveTo(x1, y1); | |
tileCtx.lineTo(x2, y2); | |
tileCtx.stroke(); | |
var pattern = ctx.createPattern(tile, 'repeat'); | |
canvas.width = window.innerWidth; | |
canvas.height = window.innerHeight; | |
ctx.save(); | |
var offsetX = canvas.width / 2 + radius; | |
var offsetY = canvas.height / 2 + radius; | |
ctx.translate(offsetX, offsetY); | |
ctx.fillStyle = pattern; | |
// ctx.rotate(angle * Math.PI); | |
ctx.fillRect(-offsetX, -offsetY, canvas.width, canvas.height); | |
ctx.restore(); | |
} | |
// onmousemove = function(event) { | |
// angle = event.offsetX / window.innerWidth * 2 * Math.PI; | |
// draw(); | |
// } | |
onwheel = function(event) { | |
console.log(event); | |
} | |
onkeyup = function(event) { | |
switch (event.keyIdentifier) { | |
case "Up": | |
case "U+004B": | |
radius *= 1.5; | |
break; | |
case "Down": | |
case "U+004D": | |
radius /= 1.5; | |
break; | |
case "Left": | |
angle -= 5; | |
break; | |
case "Right": | |
angle += 5; | |
break; | |
default: | |
} | |
draw(); | |
} | |
</script> | |
</head> | |
<body> | |
<canvas id="canvas" /> | |
</body> | |
</html> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<!DOCTYPE html> | |
<html> | |
<head> | |
<meta charset="utf-8"> | |
<style> | |
canvas { | |
position: fixed; | |
top: 0; | |
left: 0; | |
width: 100%; | |
height: 100%; | |
} | |
</style> | |
<script> | |
var angle = 0; | |
var radius = 10; | |
var tile, tileCtx, ctx; | |
onload = function() { | |
tile = document.createElement('canvas'); | |
tileCtx = tile.getContext('2d'); | |
ctx = canvas.getContext('2d'); | |
draw(); | |
} | |
function draw() { | |
tile.width = tile.height = radius * 2; | |
tileCtx.lineWidth = Math.max(1, radius / 3); | |
var cos = Math.cos(angle); | |
var sin = Math.sin(angle); | |
var x1 = (cos * radius); | |
var y1 = (sin * radius); | |
// var x1 = (cos * radius) + (tile.width / 2); | |
// var y1 = (sin * radius) + (tile.height / 2); | |
var x2 = tile.width - x1; | |
var y2 = tile.height - y1; | |
tileCtx.beginPath(); | |
tileCtx.moveTo(x1, y1); | |
tileCtx.lineTo(x2, y2); | |
tileCtx.stroke(); | |
tileCtx.beginPath(); | |
tileCtx.moveTo(x1 + tile.width/4, y1 + tile.height/4); | |
tileCtx.lineTo(x2 + tile.width/4, y2 + tile.height/4); | |
tileCtx.stroke(); | |
var pattern = ctx.createPattern(tile, 'repeat'); | |
canvas.width = window.innerWidth; | |
canvas.height = window.innerHeight; | |
var offsetX = canvas.width / 2 + radius; | |
var offsetY = canvas.height / 2 + radius; | |
ctx.save(); | |
ctx.translate(offsetX, offsetY); | |
ctx.fillStyle = pattern; | |
ctx.fillRect(-offsetX, -offsetY, canvas.width, canvas.height); | |
ctx.restore(); | |
} | |
onmousemove = function(event) { | |
angle = event.offsetX / window.innerWidth * 2 * Math.PI; | |
draw(); | |
} | |
onkeyup = function(event) { | |
console.log(event); | |
switch (event.keyIdentifier) { | |
case "U+004B": | |
radius *= 1.5; | |
break; | |
case "U+004D": | |
radius /= 1.5; | |
break; | |
default: | |
} | |
draw(); | |
} | |
</script> | |
</head> | |
<body> | |
<canvas id="canvas"> | |
</body> | |
</html> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<style> | |
#canvas { | |
background: linear-gradient(#a25b5b, #2d1821); | |
} | |
</style> | |
<canvas id="canvas"></canvas> | |
<button onclick="toggle()">Start/Stop</button> | |
<script> | |
canvas.width = canvas.height = 0xff; | |
canvas.width = canvas.height; //0xff * 2; | |
halfWidth = canvas.width / 2; | |
halfHeight = canvas.height / 2; | |
ctx = canvas.getContext('2d'); | |
var id = ctx.getImageData(0, 0, canvas.width, canvas.height); | |
id32 = new Uint32Array(id.data.buffer); | |
a = .25 * Math.PI; | |
fov = 180 / 180 * Math.PI; | |
// da = fov / canvas.width; a0 = a - camera.fov / 2; | |
cosa = Math.cos(a); | |
sina = Math.sin(a); | |
d = canvas.width / fov; | |
// d = 128; d = 0xff / 2; | |
frames = 0; | |
origin = [0, 0, -1]; | |
getDirection = function(x, y) { | |
return [ | |
origin[0] + (x - halfWidth) / halfWidth, | |
origin[1] + (y - halfHeight) / halfHeight, | |
1 | |
] | |
} | |
function length(v) { | |
return Math.sqrt(v[0] * v[0] + v[1] * v[1] + v[2] * v[2]); | |
} | |
function distanceSphere(x, y, z) { | |
let radius = .5; | |
return length([x, y, z]) - radius; | |
} | |
function distanceTorus(x, y, z) { | |
var t = [.5, .2] | |
// var l = [Math.sqrt(x * x + z * z) - t[0], y]; | |
var l = [Math.sqrt(x * x + y * y) - t[0], z]; | |
return Math.sqrt(l[0] * l[0] + l[1] * l[1]) - t[1]; | |
} | |
// normal(p, f) { | |
// var epsilon = 0.001; | |
// return [ | |
// f(p[0]+epsilon, p[1], p[2]) | |
// ] | |
// } | |
function cast(trace) { | |
for (; trace.t < 1; trace.t += .0051) { | |
// if (distanceSphere( | |
if (distanceTorus( | |
trace.origin[0] + trace.direction[0] * trace.t, | |
trace.origin[1] + trace.direction[1] * trace.t, | |
trace.origin[2] + trace.direction[2] * trace.t) <= 0) { | |
return true; | |
} | |
} | |
return false; | |
} | |
function update(time) { | |
canvas.width = canvas.width; | |
ox = time / 100 | |
for (x = 0; x < canvas.width; x++) { | |
for (y = 0; y < canvas.height; y++) { | |
i = ((y * canvas.width) + x); | |
let trace = { | |
origin: origin, | |
direction: getDirection(x, y), | |
t: 0, | |
} | |
if (cast(trace)) { | |
// id32[i] = getColor(trace); | |
// id32[i] = 0xff00ffff; | |
i *= 4; | |
id.data[i] = id.data[i + 1] = id.data[i + 2] = 0xff - trace.t * 0xff; | |
id.data[i + 3] = 0xff; | |
} else { | |
id32[i] = 0x00ffffff; | |
} | |
// i = ((y * canvas.width) + x) * 4; | |
} | |
} | |
ctx.putImageData(id, 0, 0); | |
dt = performance.now() - time; | |
ctx.fillStyle = '#fff' | |
// ctx.fillText('fps: ' + parseInt(1 / (dt / 1000)) + ' frames: ' + ++frames, 10, 10) | |
ctx.fillText('fps: ' + parseInt(1 / (dt / 1000)), 10, 10); | |
ctx.fillText('frame: ' + ++frames, 10, 20); | |
animationFrameRequestId = requestAnimationFrame(update); | |
} | |
var animationFrameRequestId; | |
toggle = function() { | |
if (animationFrameRequestId) { | |
cancelAnimationFrame(animationFrameRequestId) | |
animationFrameRequestId = null; | |
} else { | |
animationFrameRequestId = requestAnimationFrame(update); | |
} | |
} | |
</script> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<style> | |
#canvas { | |
/*background: #000;*/ | |
} | |
</style> | |
<canvas id="canvas"></canvas> | |
<button onclick="toggle()">Start/Stop</button> | |
<script src="https://cdnjs.cloudflare.com/ajax/libs/dat-gui/0.5.1/dat.gui.js"></script> | |
<script> | |
canvas.width = canvas.height = 0xff; | |
canvas.width = canvas.height; //0xff * 2; | |
halfWidth = canvas.width / 2; | |
halfHeight = canvas.height / 2; | |
ctx = canvas.getContext('2d'); | |
var id = ctx.getImageData(0, 0, canvas.width, canvas.height); | |
id32 = new Uint32Array(id.data.buffer); | |
id32.fill(0xff000000); | |
let i = (10 + 10 * canvas.width) * 4; | |
id.data[i] += 0x30; | |
ctx.putImageData(id, 0, 0); | |
var cloneFittest = true; | |
var maxPopulation = 100; | |
var crossoverProbability = .5; | |
var mutationProbability = .5; | |
var mutationStrength = 20; | |
var population = []; | |
var selectA = 'randomLinearRank' | |
var selectB = 'sequential' | |
var gui = new dat.GUI(); | |
gui.add(window, "cloneFittest"); | |
gui.add(window, "maxPopulation", 0, 1000); | |
gui.add(window, "crossoverProbability", 0, 1); | |
gui.add(window, "mutationProbability", 0, 1); | |
gui.add(window, "mutationStrength", 0, 100); | |
gui.add(window, "selectA", ['fittest', 'random', 'randomLinearRank', 'sequential']); | |
gui.add(window, "selectB", ['fittest', 'random', 'randomLinearRank', 'sequential']); | |
function seed() { | |
return [ | |
Math.round(Math.random() * canvas.width), | |
Math.round(Math.random() * canvas.height) | |
]; | |
} | |
function fitness(p) { | |
let i = p[0] + p[1] * canvas.width; | |
return id.data[i * 4]; | |
} | |
function init(population, num) { | |
for (i = 0; i < num; i++) { | |
population.push(seed()); | |
} | |
} | |
init(population, maxPopulation); | |
console.log(population) | |
render(population) | |
function render(population) { | |
population.forEach(p => { | |
let i = (p[0] + p[1] * canvas.width) * 4; | |
id.data[i] += 0x30; | |
}); | |
ctx.putImageData(id, 0, 0); | |
} | |
var select = { | |
fittest: (generation) => generation[0].individual, | |
random: (generation) => generation[Math.floor(Math.random() * generation.length)].individual, | |
randomLinearRank: (generation, n) => generation[Math.floor(Math.random() * Math.min(generation.length, n))].individual, | |
sequential: (generation, n) => generation[n % generation.length].individual | |
} | |
function crossover(a, b) { | |
return [a[0], b[1]] | |
} | |
function mutate(i) { | |
return [ | |
Math.min(canvas.width - 1, Math.max(0, i[0] + Math.round((Math.random() - .5) * mutationStrength))), | |
Math.min(canvas.height - 1, Math.max(0, i[1] + Math.round((Math.random() - .5) * mutationStrength))) | |
] | |
} | |
function evolve() { | |
render(population); | |
// Mesaure fitness of individuals | |
let generation = population | |
.map(p => { | |
return { | |
fitness: fitness(p), | |
individual: p | |
} | |
}) | |
.sort((a, b) => a.fitness - b.fitness); | |
var newPopulation = []; | |
if (cloneFittest) { | |
newPopulation.push(generation[0].individual); | |
} | |
// Crossover and mutate to build new population | |
let n = 0; | |
while (newPopulation.length < maxPopulation) { | |
let a = select[selectA](generation, ++n); | |
if (Math.random() < crossoverProbability) { | |
let b = select[selectB](generation, ++n); | |
let ab = crossover(a, b); | |
let ba = crossover(b, a); | |
newPopulation.push(Math.random() < mutationProbability ? mutate(ab) : ab); | |
newPopulation.push(Math.random() < mutationProbability ? mutate(ba) : ba); | |
newPopulation.push(Math.random() < mutationProbability ? mutate(ba) : ba); | |
} else { | |
newPopulation.push(Math.random() < mutationProbability ? mutate(a) : a); | |
} | |
} | |
population = newPopulation; | |
// console.log(generation) | |
animationFrameRequestId = requestAnimationFrame(evolve); | |
} | |
var animationFrameRequestId; | |
toggle = function() { | |
if (animationFrameRequestId) { | |
cancelAnimationFrame(animationFrameRequestId) | |
animationFrameRequestId = null; | |
} else { | |
animationFrameRequestId = requestAnimationFrame(evolve); | |
} | |
} | |
</script> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<script src="https://cdnjs.cloudflare.com/ajax/libs/dat-gui/0.5.1/dat.gui.js"></script> | |
<style> | |
#canvas { | |
background: #000; | |
} | |
</style> | |
<canvas id="canvas"></canvas> | |
<button onclick="evolve()">Start/Stop</button> | |
<script> | |
canvas.width = canvas.height = 0xff; | |
canvas.width = canvas.height; //0xff * 2; | |
halfWidth = canvas.width / 2; | |
halfHeight = canvas.height / 2; | |
var gui = new dat.GUI(); | |
gui.add(canvas, "width", 0, 400); | |
gui.add(canvas, "height", 0, 400) | |
ctx = canvas.getContext('2d'); | |
var id = ctx.getImageData(0, 0, canvas.width, canvas.height); | |
id32 = new Uint32Array(id.data.buffer); | |
var population = []; | |
function seed() { | |
return [ | |
Math.random() * canvas.width, | |
Math.random() * canvas.height | |
]; | |
} | |
function fitness(p) { | |
return Math.sqrt(p[0] * p[0] + p[1] * p[1]) | |
} | |
function init(population, num) { | |
for (i = 0; i < num; i++) { | |
population.push(seed()); | |
} | |
} | |
init(population, 10); | |
function render(population) { | |
ctx.fillStyle = '#000'; | |
ctx.globalAlpha = 0.1; | |
ctx.fillRect(0, 0, canvas.width, canvas.height); | |
ctx.globalAlpha = 1; | |
ctx.fillStyle = '#fff' | |
population.forEach(p => { | |
ctx.beginPath(); | |
ctx.arc(p[0], p[1], 1.5, 0, 2 * Math.PI) | |
ctx.fill(); | |
}); | |
} | |
function cross(p1, p2) { | |
return [ | |
(p1[0] + p2[0]) / 2, | |
(p1[1] + p2[1]) / 2 | |
]; | |
} | |
var select = { | |
fittest: (generation) => generation[0].individual, | |
random: (generation) => generation[Math.floor(Math.random() * generation.length)].individual, | |
randomLinearRank: (generation, n) => generation[Math.floor(Math.random() * Math.min(generation.length, n))].individual, | |
sequential: (generation, n) => generation[n % generation.length].individual | |
} | |
function crossover(a, b) { | |
return [a[0], b[1]] | |
} | |
function mutate(i) { | |
return [ | |
i[0] + i[0] * (Math.random() - .5), | |
i[1] + i[1] * (Math.random() - .5), | |
] | |
} | |
var cloneFittest = true; | |
var maxPopulation = 20; | |
var mutationProbability = 0.2; | |
function evolve() { | |
render(population); | |
// Mesaure fitness of individuals | |
let generation = population | |
.map(p => { | |
return { | |
fitness: fitness(p), | |
individual: p | |
} | |
}) | |
.sort((a, b) => a.fitness - b.fitness); | |
var newPopulation = []; | |
if (cloneFittest) { | |
newPopulation.push(generation[0].individual); | |
} | |
// Crossover and mutate to build new population | |
let n = 0; | |
while (newPopulation.length < maxPopulation) { | |
let a = select.randomLinearRank(generation, ++n); | |
let b = select.randomLinearRank(generation, ++n); | |
let ab = crossover(a, b); | |
let ba = crossover(b, a); | |
newPopulation.push(Math.random() < mutationProbability ? mutate(ab) : ab); | |
newPopulation.push(Math.random() < mutationProbability ? mutate(ba) : ba); | |
} | |
population = newPopulation; | |
console.log(generation[0].individual) | |
} | |
</script> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<style> | |
#canvas { | |
background: linear-gradient(#22a, #aae); | |
transform: scaleY(-1); | |
} | |
img { | |
display: none; | |
} | |
</style> | |
<canvas id="canvas"></canvas> | |
<canvas id="map"></canvas> | |
<img id="mapImg" src=""> | |
<script> | |
map.width = map.height = 0xff; | |
mapCtx = map.getContext('2d'); | |
mapCtx.drawImage(mapImg, 0, 0, map.width, map.height); | |
heightmap = mapCtx.getImageData(0, 0, map.width, map.height); | |
canvas.width = canvas.height = 0xff; | |
canvas.width = 0xff * 2; | |
ctx = canvas.getContext('2d'); | |
mapCtx.fillStyle = mapCtx.strokeStyle = '#ff0' | |
x0 = map.width / 4; | |
y0 = map.height / 4; | |
a = .25 * Math.PI; | |
fov = 180 / 180 * Math.PI; | |
// da = fov / canvas.width; a0 = a - camera.fov / 2; | |
cosa = Math.cos(a); | |
sina = Math.sin(a); | |
d = canvas.width / fov; | |
// d = 128; d = 0xff / 2; | |
x1 = x0 + d * cosa; | |
y1 = y0 + d * sina; | |
// Draw camera position | |
mapCtx.beginPath(); | |
mapCtx.arc(x0, y0, 5, 0, 2 * Math.PI); | |
mapCtx.stroke(); | |
// Draw camera direction | |
mapCtx.beginPath(); | |
mapCtx.moveTo(x0, y0); | |
mapCtx.lineTo(x1, y1) | |
// mapCtx.lineTo(camera.x + Math.cos(camera.a + camera.fov / 2) * 50, camera.y + Math.sin(camera.a + camera.fov / 2) * 50) mapCtx.lineTo(camera.x, camera.y) | |
mapCtx.stroke(); | |
ox = x0; | |
oy = y0; | |
frames = 0; | |
function update(time) { | |
canvas.width = canvas.width; | |
ox = time / 100 | |
// oy -=1 | |
// Loop through each vertical span | |
for (x = 0; x < canvas.width; x++) { | |
// Start at the camera position | |
x0 = ox; | |
y0 = oy; | |
// Calculate the end position | |
var x3d = (x - canvas.width / 2); | |
x1 = x0 + (cosa * x3d + sina * d); | |
y1 = y0 + (-sina * x3d + cosa * d); | |
// x1 = 0 | x1; | |
// y1 = 0 | y1; | |
// x1 %= map.width; | |
// y1 %= map.height; | |
// i = ((y1 * map.width + x1) * 4); | |
// ctx.fillStyle = 'rgb(' + heightmap.data[i + 0] + ',' + heightmap.data[i + 1] + ',' + heightmap.data[i + 2] + ')'; // if (x == 10) | |
// console.log(ctx.fillStyle, i, x1, y1) ctx.fillRect(x, canvas.height / 2, 1, 10); | |
// mapCtx.fillRect(x1, y1, 1, 1); | |
// a += da; | |
// if (x == 0) { | |
// Reset the y-buffer | |
ymin = 0; | |
// Trace the ray in `tmax` steps | |
tmax = 128; | |
dx = (x1 - x0) / tmax; | |
dy = (y1 - y0) / tmax; | |
// console.log(d, x1-x0, dx) | |
for (t = 1; t < tmax; t++) { | |
x0 += dx; | |
y0 += dy; | |
// mapCtx.fillRect(x0, y0, 1, 1); | |
i = (((0 | y0) * map.width + (0 | x0)) * 4); | |
h = heightmap.data[i + 3] - 128; | |
h = (h / t * 20) + 128; | |
h = Math.max(ymin, h) | 0; | |
if (h > ymin) { | |
ctx.fillStyle = 'rgb(' + heightmap.data[i + 0] + ',' + heightmap.data[i + 1] + ',' + heightmap.data[i + 2] + ')'; | |
ctx.fillRect(x, ymin, 1, h - ymin); | |
} | |
ymin = Math.max(h, ymin) | |
} | |
// } | |
} | |
dt = performance.now() - time; | |
ctx.fillStyle = '#000' | |
ctx.fillText('fps: ' + (1 / (dt / 1000)) + 'frames: ' + ++frames, 10, 10) | |
animationFrameRequestId = requestAnimationFrame(update); | |
} | |
animationFrameRequestId = requestAnimationFrame(update); | |
onclick = function() { | |
cancelAnimationFrame(animationFrameRequestId) | |
} | |
</script> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<!DOCTYPE html> | |
<html> | |
<head> | |
<meta charset="utf-8"> | |
<title></title> | |
<style> | |
#canvas { | |
background: linear-gradient(#22a, #aae); | |
} | |
</style> | |
<script> | |
var heigtmap, | |
heightBuf32; | |
var ctx, | |
imageData, | |
buf32, | |
animationFrame, | |
prevTime, | |
frames = 0; | |
u0 = 0; | |
v0 = 0; | |
function recurse(x0, x1) { | |
'use strict'; | |
let d = x1-x0; | |
if (d > 2) { | |
let u = x0 + (d / 2)|0; | |
let y = 0 | (Math.random() * 10 + canvas.height / 2); | |
for (let h = 0; h < 10; h++) { | |
let i = (y + h) * canvas.width + u; | |
// buf32[i] = 0xffff00ff; // ABGR | |
buf32[i] = heightBuf32[i] | 0xff000000; | |
} | |
recurse(x0, u); | |
recurse(u, x1); | |
} | |
} | |
function render(time) { | |
// Clear image buffer | |
buf32.fill(0x00); | |
// Render new image data | |
for (x = 0; x < canvas.width; x++) { | |
// u = u0; v = v0; data = heightmap.data[u + v * heightmap.width]; w = data >> 3; // Alpha channel w0 = w / 0xff * 2 - 1; // Convert height map data byte to [-1,1] range c = data | 0xff000000; // Color data + solid alpha | |
y = 0 | (Math.random() * 10 + canvas.height / 2); | |
for (h = 0; h < 10; h++) { | |
i = (y + h) * canvas.width + x; | |
// buf32[i] = 0xffff00ff; // ABGR | |
buf32[i] = heightBuf32[i] | 0xff000000; | |
} | |
} | |
frames += 1; | |
// Display new image data | |
ctx.putImageData(imageData, 0, 0); | |
if (prevTime) { | |
ctx.fillText('FPS ' + (1000 / (time - prevTime) | 0), 4, 12) | |
} | |
// Prepare next frame | |
prevTime = time; | |
animationFrame = window.requestAnimationFrame(render); | |
} | |
// Load heigtmap data | |
heightMapImg = new Image(); | |
heightMapImg.src = 'heightmap.png'; | |
heightMapImg.onload = function () { | |
ctx = canvas.getContext('2d'); | |
// Grab heightmap data | |
canvas.width = heightMapImg.width; | |
canvas.height = heightMapImg.height; | |
ctx.drawImage(heightMapImg, 0, 0); | |
heightmap = ctx.getImageData(0, 0, canvas.width, canvas.height); | |
heightBuf32 = new Uint32Array(heightmap.data.buffer); | |
// Initialize canvas and image data buffer | |
canvas.width = canvas.height = 0xff; | |
ctx.font = '8px monospace' | |
imageData = ctx.createImageData(canvas.width, canvas.height); | |
buf32 = new Uint32Array(imageData.data.buffer); | |
// Enter render loop | |
// animationFrame = window.requestAnimationFrame(render); | |
// Experiment... | |
console.log(0, canvas.width) | |
recurse(0, canvas.width) | |
// Display new image data | |
ctx.putImageData(imageData, 0, 0); | |
if (prevTime) { | |
ctx.fillText('FPS ' + (1000 / (time - prevTime) | 0), 4, 12) | |
} | |
} | |
onclick = function () { | |
// Stop render loop | |
window.cancelAnimationFrame(animationFrame); | |
} | |
</script> | |
</head> | |
<body> | |
<canvas id="canvas"></canvas> | |
<img src="heightmap.png"/> | |
</body> | |
</html> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<style> | |
#canvas { | |
object-fit: contain; | |
position: fixed; | |
top: 0; | |
left: 0; | |
width: 100%; | |
height: 100%; | |
background: linear-gradient(#456, #200); | |
} | |
</style> | |
<canvas id="canvas" /> | |
<script> | |
audioContext = new AudioContext(); | |
gain = audioContext.createGain(); | |
gain.gain.value = .1; | |
gain.connect(audioContext.destination); | |
context = canvas.getContext('2d'); | |
MIN_F = 50; | |
MAX_F = 800; | |
MIN_D = 100; | |
MAX_D = 1000; | |
MIN_A = 5; | |
MAX_A = 20; | |
minFps = 999; | |
maxFps = 0; | |
function mutateNote(note) { | |
note.frequency.value = Math.random() * (MAX_F - MIN_F) + MIN_F; | |
// note.frequency = Math.random() * (MAX_F - MIN_F) + MIN_F; | |
note.duration = Math.random() * (MAX_D - MIN_D) + MIN_D; | |
// note.amplitude = Math.random() * (MAX_A - MIN_A) + MIN_A; | |
} | |
function generateNote(type) { | |
note = audioContext.createOscillator(); | |
note.type = type || 'sine'; | |
mutateNote(note); | |
note.start(); | |
note.connect(gain); | |
return note; | |
} | |
notes = [ | |
// generateNote('sawtooth'), | |
// generateNote('sine'), | |
// generateNote('triangle'), | |
generateNote('square'), | |
]; | |
function update(time) { | |
if (!window.currentTime) { | |
currentTime = time - 1000 / 60; | |
} | |
previousTime = currentTime; | |
currentTime = time; | |
deltaTime = currentTime - previousTime; | |
fps = 1000 / deltaTime; | |
minFps = Math.min(fps, minFps); | |
maxFps = Math.max(fps, maxFps); | |
// update stuff | |
// console.log(time) | |
notes.forEach(note => { | |
note.duration -= deltaTime; | |
if (note.duration <= 0) { | |
mutateNote(note); | |
} | |
}); | |
// reset canvas | |
height = canvas.height = 512; | |
width = canvas.width = 512; | |
// width = canvas.width = 0 | height * innerWidth / innerHeight; | |
// render visuals | |
context.shadowColor = '#fff'; | |
// context.shadowBlur = 24; | |
context.shadowBlur = 10; | |
context.strokeStyle = '#fff'; | |
notes.forEach(note => { | |
context.beginPath(); | |
y = height / MAX_F * (MAX_F - note.frequency.value) + Math.sin((currentTime/20000) * note.frequency.value) * 10; | |
// y = ((note.frequency.value - MIN_F) / (MAX_F - MIN_F)) * height + Math.sin(currentTime / (note.frequency.value / 4)) * 10; | |
context.moveTo(0, y); | |
context.lineTo(width, y); | |
context.stroke(); | |
}); | |
// render stats | |
context.shadowBlur = 0; | |
context.strokeStyle = '#fff'; | |
context.strokeText(['FPS: ' + (0 | fps), 0 | minFps, 0 | maxFps], 10, 20); | |
requestAnimationFrame(update); | |
} | |
requestAnimationFrame(update); | |
onclick = function() { | |
audioContext.suspend(); | |
update = function() {} | |
} | |
</script> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<style> | |
#b { | |
position: fixed; | |
top: 0; | |
left: 0; | |
width: 100%; | |
height: 100%; | |
background: radial-gradient(circle, #345, #201); | |
} | |
</style><audio id="a"></audio><canvas id="b"></canvas><script> | |
str=''; | |
notesFreq = [155,195,261,155,233,155,195,207,195,207,195,207,195,207,195,261,155,233,155,233,261,155,233,155,233,155,233,261,155,195,261,195,207,195,261,155,233,155,233,155,195,207,233,155,233,155,195,207,195,207,233,261,155,233,261,233,261,155,195,207,195,207,195,207,195,261,233,155,195,207,233,261,233,155,233,261,155,195,207,195,261,155,233,261,155,233,261,195,261,233,261,155,233,155,195,207,233,261,233,155,195,261,233,155,195,261,195,261,155,233,261,195,207,233,261,195,207,195,261,233,155,195,261,155,195,207,195,261]; | |
notesIndex = 0; | |
for(time=0;time<60;time+=1/16384) { | |
frac= (time&4?time*6:time*4)%1; | |
notesIndex += frac==0; | |
sample = (time * notesFreq[notesIndex%notesFreq.length] % 1) * (1 - frac) * 16; | |
frac = time*2%1; | |
sample += (Math.random()) * Math.pow(1-frac, 8) * 16; | |
str += String.fromCharCode(sample + 127); | |
} | |
a.src = 'data:audio/wav;base64,UklGRiQAAABXQVZFZm10IBAAAAABAAEAA' | |
+ 'EAAAABA' // 16khz | |
+ 'AAABAAgAZGF0YQAAAAAA' + btoa(str); | |
a.play(); | |
c=b.getContext('2d'); | |
ps = []; | |
COUNT = 2560; | |
for(i=0;i<COUNT; i++) { | |
ps.push({ | |
x: (Math.random()-.5)*256 * 2, | |
y: (Math.random()-.5)*256 * 2, | |
z: (Math.random()-.5)*256 * 2, | |
u: i / 48 % 1, | |
v: i / 48 / 48, | |
col: i & 1 ? '#eee' : '#ddd' | |
}) | |
}; | |
render = function() { | |
requestAnimationFrame(render); | |
time = a.currentTime; | |
H=b.height=512; | |
W=b.width=0|innerWidth/innerHeight*H; | |
c.translate(W/2,H/2); | |
angle = time; | |
c.rotate((angle&13)/32); | |
ps.forEach(p => { | |
if (p.v < 1) { | |
an = p.u*Math.PI*2; | |
frac = time*2%1; | |
frac = Math.pow(1-frac, 2); | |
d = 64 + frac * 32 * Math.cos(an*4)*Math.cos(p.v*6+time*4); | |
p.x = d*Math.cos(an); | |
p.z = d*Math.sin(an); | |
p.y = 384*(p.v-.5); | |
p.col = 'rgb('+ | |
[255-d*4|0, | |
255-d*4|0, | |
255-d*4|0] +')' | |
} | |
p.rx = Math.cos(angle) * p.x - Math.sin(angle) * p.z; | |
p.rz = Math.sin(angle) * p.x + Math.cos(angle) * p.z; | |
p.ry = p.y; | |
p.p = (384+p.rz)/256; | |
p.sx = p.p*p.rx; | |
p.sy = p.p*p.ry; | |
}); | |
ps.sort((a, b) => a.rz - b.rz) | |
ps.forEach(p => { | |
c.shadowColor = c.fillStyle = p.col; | |
s = p.p*8; | |
c.shadowBlur = p.v < 1 ? 0 : s; | |
c.fillRect(p.sx,p.sy,s,s); | |
}) | |
} | |
render(); | |
</script> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<canvas id="canvas"> | |
<script> | |
var context = canvas.getContext("2d"), | |
a, b, position; | |
function surface(a, b) { | |
var angle = a * Math.PI * 2, | |
radius = 100, | |
length = 400, | |
x = a * 400, | |
y = b * 400, | |
z = a * b * 200, | |
yAxisRotationAngle = -.40, // in radians! | |
rotatedX = x * Math.cos(yAxisRotationAngle) + z * Math.sin(yAxisRotationAngle), | |
rotatedZ = x * -Math.sin(yAxisRotationAngle) + z * Math.cos(yAxisRotationAngle); | |
return { | |
x: rotatedX, | |
y: y, | |
z: rotatedZ, | |
r: 0 | a * 255, | |
g: 0 | b * 255, | |
b: 0 | |
}; | |
} | |
// function surface(a, b) { | |
// var angle = a * Math.PI * 2, | |
// radius = 100, | |
// length = 400, | |
// x = Math.cos(angle) * radius, | |
// y = Math.sin(angle) * radius, | |
// z = b * length - length / 2, | |
// yAxisRotationAngle = -.4, // in radians! | |
// rotatedX = x * Math.cos(yAxisRotationAngle) + z * Math.sin(yAxisRotationAngle), | |
// rotatedZ = x * -Math.sin(yAxisRotationAngle) + z * Math.cos(yAxisRotationAngle); | |
// | |
// return { | |
// x: rotatedX, | |
// y: y, | |
// z: rotatedZ, | |
// r: 0, | |
// g: 0 | b * 255, | |
// b: 0 | |
// }; | |
// } | |
var pX, pY, // projected on canvas x and y coordinates | |
perspective = 350, | |
halfHeight = canvas.height / 2, | |
halfWidth = canvas.width / 2, | |
cameraZ = -700; | |
var zBuffer = [], | |
zBufferIndex; | |
// for (a = 0; a < 1; a += .001) { | |
// for (b = 0; b < 1; b += .01) { | |
// if (point = surface(a, b)) { | |
// pX = Math.floor((point.x * perspective) / (point.z - cameraZ) + halfWidth); | |
// pY = Math.floor((point.y * perspective) / (point.z - cameraZ) + halfHeight); | |
// zBufferIndex = pY * canvas.width + pX; | |
// if ((typeof zBuffer[zBufferIndex] === "undefined") || (point.z < zBuffer[zBufferIndex])) { | |
// zBuffer[zBufferIndex] = point.z; | |
// context.fillStyle = "rgb(" + point.r + "," + point.g + "," + point.b + ")"; | |
// context.fillRect(pX, pY, 1, 1); | |
// } | |
// } | |
// } | |
// } | |
var i; | |
// window.setInterval(function() { | |
for (i = 0; i < 50000; i++) { | |
if (point = surface(Math.random(), Math.random())) { | |
pX = Math.floor((point.x * perspective) / (point.z - cameraZ) + halfWidth); | |
pY = Math.floor((point.y * perspective) / (point.z - cameraZ) + halfHeight); | |
zBufferIndex = pY * canvas.width + pX; | |
if ((typeof zBuffer[zBufferIndex] === "undefined") || (point.z < zBuffer[zBufferIndex])) { | |
zBuffer[zBufferIndex] = point.z; | |
context.fillStyle = "rgb(" + point.r + "," + point.g + "," + point.b + ")"; | |
context.fillRect(pX, pY, 1, 1); | |
} | |
} | |
} | |
// }, 0); | |
</script> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<script> | |
class KDNode { | |
constructor(axis, d, children) { | |
this.axis = axis; | |
this.d = d; | |
this.children = children || []; | |
} | |
intersect(origin, direction) { | |
// Determine wich side of the splitting plane the origin lies within | |
let originChildIndex = +(origin[this.axis] > this.d); | |
if (direction[this.axis] === 0) { | |
// Direction parallel to splitting plane, and will never intersect splitting plane | |
// Only traverse origin side of splitting plane | |
this.children[originChildIndex].intersect(origin, direction); | |
} else { | |
// Distance from origin to splitting plane | |
let t = (this.d - origin[this.axis]) / direction[thid.axis]; | |
if (t >= 0) { | |
// Direction points towards splitting plane, so both sides must be considered | |
// Traverse origin side of splitting plane first, then the other side | |
this.children[originChildIndex].intersect(origin, direction); | |
this.children[originChildIndex ^ 1].intersect(origin, direction); | |
} else { | |
// Direction points away from splitting plane, so will never intersect splitting plane | |
// Only traverse origin side of splitting plane | |
this.children[originChildIndex].intersect(origin, direction); | |
} | |
} | |
} | |
} | |
</script> | |
<script> | |
function createVoxelData(width, height, depth) { | |
return { | |
width: width, | |
height: height, | |
depth: depth, | |
data: new Uint32Array(width * height * depth) | |
} | |
} | |
voxelData = createVoxelData(3, 3, 3); | |
for (let x = 0; x < voxelData.width; x++) { | |
for (let y = 0; y < voxelData.height; y++) { | |
for (let z = 0; z < voxelData.depth; z++) { | |
let i = z * voxelData.width * voxelData.height + y * voxelData.width + x; | |
voxelData[i] = 0xff; | |
} | |
} | |
} | |
</script> | |
<style> | |
#canvas { | |
background: linear-gradient(#a25b5b, #2d1821); | |
width: 20px; | |
height: 200px; | |
} | |
#circle { | |
fill: #f00; | |
} | |
</style> | |
<svg width="300" height="200" viewBox="-2 -1 3 2" style="background: black"> | |
<polyline id="view" points="-2,0 -1,1 -1,-1" fill="rgba(255, 255, 255,.25)" /> | |
<circle id="circle" cx="0" cy="0" r="0.5" fill="#ff0000"/> | |
<line id="ray" x1="-2" y1="0" | |
x2="-1" y2="1" | |
stroke="yellow" | |
stroke-width=".01"/> | |
</svg> | |
<canvas id="canvas"></canvas> | |
<button onclick="toggle()">Start/Stop</button> | |
<script> | |
console.log(ray.x2.baseVal.value) | |
origin = [view.points[0].x, view.points[0].y]; | |
console.log(origin) | |
top = [view.points[1].x, view.points[1].y]; | |
bottom = [view.points[2].x, view.points[2].y]; | |
circle.origin = [circle.cx.baseVal.value, circle.cy.baseVal.value]; | |
circle.radius = circle.r.baseVal.value; | |
circle.color = parseInt(circle.getAttribute('fill').replace('#', '0x')); | |
canvas.width = 1; | |
canvas.height = 200; | |
ctx = canvas.getContext('2d'); | |
var id = ctx.getImageData(0, 0, canvas.width, canvas.height); | |
id32 = new Uint32Array(id.data.buffer); | |
function getDirection(y) { | |
let d = normalize([-origin[0], y]); | |
ray.x2.baseVal.value = d[0]; | |
ray.y2.baseVal.value = d[1]; | |
return d; | |
} | |
function dot(a, b) { | |
return a[0] * b[0] + a[1] * b[1]; | |
} | |
function length(v) { | |
return Math.sqrt(v[0] * v[0] + v[1] * v[1]); | |
} | |
function normalize(v) { | |
let l = length(v); | |
return [v[0] / l, v[1] / l]; | |
} | |
vec = [-1, -1] | |
console.log("VEC", vec) | |
console.log("LEN", length(vec)) | |
console.log("NORM", normalize(vec)) | |
console.log(circle.origin) | |
function cast(trace) { | |
let m = [ | |
trace.origin[0] - circle.origin[0], | |
trace.origin[1] - circle.origin[1] | |
]; | |
let b = dot(m, trace.direction); | |
let c = dot(m, m) - circle.radius * circle.radius; | |
console.log(trace, m, b, c) | |
if (c > 0 && b > 0) { | |
// Origin outside circle and pointing away | |
return false; | |
} | |
let discr = b * b - c; | |
if (discr < 0) { | |
// Negative discriminant indicates ray missing sphere | |
return false; | |
} | |
trace.t = -b - Math.sqrt(discr); | |
if (trace.t < 0) { | |
// Negative t indicates ray origin within circle | |
// Clamp to 0 | |
trace.t = 0; | |
} | |
return true; | |
} | |
function update(time) { | |
canvas.width = canvas.width; | |
ox = time / 100 | |
for (i = 0; i < canvas.height; i++) { | |
let y = ((i / canvas.height) * 2) - 1; | |
let trace = { | |
origin: origin, | |
direction: getDirection(y), | |
t: 0, | |
} | |
if (cast(trace)) { | |
// let i = y * 4; | |
// id.data[i] = id.data[i + 1] = id.data[i + 2] = 0xff - trace.t * 0xff; | |
// id.data[i + 3] = 0xff; | |
id32[i] = 0xff0000ff; | |
} else { | |
id32[i] = 0x00ffffff; | |
} | |
// i = ((y * canvas.width) + x) * 4; | |
} | |
ctx.putImageData(id, 0, 0); | |
animationFrameRequestId = requestAnimationFrame(update); | |
} | |
var animationFrameRequestId; | |
toggle = function() { | |
if (animationFrameRequestId) { | |
cancelAnimationFrame(animationFrameRequestId) | |
animationFrameRequestId = null; | |
} else { | |
animationFrameRequestId = requestAnimationFrame(update); | |
} | |
} | |
</script> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<style> | |
#canvas { | |
object-fit: contain; | |
position: fixed; | |
top: 0; | |
left: 0; | |
width: 100%; | |
height: 100%; | |
/*background: radial-gradient(#456, #200);*/ | |
/*background: linear-gradient(#456, #200);*/ | |
background-color: black; | |
} | |
</style> | |
<canvas id="canvas" /> | |
<script> | |
context = canvas.getContext('2d'); | |
minFps = 999; | |
maxFps = 0; | |
function ClosestPtPointAABB(p, b, q) { | |
for (var i = 0; i < 3; i++) { | |
var v = p[i]; | |
if (v < b.min[i]) v = b.max[i]; | |
if (v > b.max[i]) v = b.min[i]; | |
q[i] = v; | |
} | |
} | |
// Computes the square distance between a point p and an AABB b | |
function sqDistPointAABB(point, aabb) { | |
var sqDist = 0.0; | |
for (var i = 0; i < 3; i++) { | |
// For each axis count any excess distance outside box extents | |
var v = point[i]; | |
if (v < aabb.min[i]) sqDist += (aabb.min[i] - v) * (aabb.min[i] - v); | |
if (v > aabb.max[i]) sqDist += (v - aabb.max[i]) * (v - aabb.max[i]); | |
} | |
return sqDist; | |
} | |
function testSphereAABB(sphere, aabb) { | |
// Compute squared distance between sphere center and AABB | |
var sqDist = sqDistPointAABB(sphere.center, aabb); | |
// Sphere and AABB intersect if the (squared) distance | |
// between them is less than the (squared) sphere radius | |
return sqDist <= sphere.radius * sphere.radius; | |
} | |
function testSceneAABB(aabb) { | |
return spheres.some(sphere => { | |
return testSphereAABB(sphere, aabb); | |
}) | |
} | |
MIN_W = 0.05; | |
var spheres = [{ | |
center: [0.5, 0.5, 0.5], | |
radius: 0.1 | |
}, { | |
center: [0.5, 0.5, 0.5], | |
radius: 0.25, | |
}]; | |
var pX, pY, perspective = 1, | |
halfHeight = canvas.height / 2, | |
halfWidth = canvas.width / 2, | |
cameraZ = -1; | |
function traverseOct(min, w, k) { | |
var aabb = { | |
min: min.slice(0), | |
max: [min[0] + w[0], min[1] + w[1], min[2] + w[2]] | |
} | |
if (testSceneAABB(aabb)) { | |
// if (testSphereAABB(sphere, aabb)) { | |
// if (w <= MIN_W) { | |
if (k >= 15) { | |
// render hit | |
pX = 0 | ((min[0] - .5) * perspective / (min[2] - cameraZ) + .5) * canvas.width; | |
pY = 0 | ((min[1] - .5) * perspective / (min[2] - cameraZ) + .5) * canvas.height; | |
// context.fillStyle = "green"; | |
// context.fillStyle = 'rgb(' + (0 | ((1 - min[2]) * 255)) + ',' + (0 | (1 - min[2]) * 255) + ',' + (0 | (1 - min[2]) * 255) + ')'; | |
// context.strokeStyle = 'rgb(' + (0 | ((1 - min[2]) * 255)) + ',' + (0 | (1 - min[2]) * 255) + ',' + (0 | (1 - min[2]) * 255) + ')'; | |
context.fillStyle = context.strokeStyle = '#fff'; | |
context.globalAlpha = .05; | |
// context.fillRect(pX, pY, 2, 2); | |
// context.beginPath(); | |
// context.arc(pX, pY, 2, 0, 2 * Math.PI); | |
// context.fill(); | |
context.strokeRect(pX, pY, w[0] * canvas.width, w[1] * canvas.height); | |
context.fillRect(pX, pY, w[0] * canvas.width, w[1] * canvas.height); | |
} else { | |
// subdivide | |
var i = k%3; | |
var w2 = w.slice(0); | |
var min2 = min.slice(0); | |
w2[i] = w2[i] / 2; | |
traverseOct(min2, w2, k+1); | |
min2[i] += w2[i]; | |
traverseOct(min2, w2, k+1); | |
// traverseOct(min[i] + w[i], w, k+1); | |
// w = w / 2; | |
// traverseOct(x + w, y + w, z + w, w) | |
// traverseOct(x + w, y, z + w, w) | |
// traverseOct(x, y + w, z + w, w) | |
// traverseOct(x, y, z + w, w) | |
// traverseOct(x + w, y + w, z, w) | |
// traverseOct(x, y + w, z, w) | |
// traverseOct(x + w, y, z, w) | |
// traverseOct(x, y, z, w) | |
} | |
} | |
} | |
function update(time) { | |
if (!window.currentTime) { | |
currentTime = time - 1000 / 60; | |
} | |
previousTime = currentTime; | |
currentTime = time; | |
deltaTime = currentTime - previousTime; | |
fps = 1000 / deltaTime; | |
minFps = Math.min(fps, minFps); | |
maxFps = Math.max(fps, maxFps); | |
// update stuff | |
spheres[0].center[0] = 0.5 + (Math.cos(currentTime / 1000)) / 4; | |
spheres[0].center[1] = 0.5 + (Math.sin(currentTime / 2000)) / 4; | |
spheres[0].center[2] = 0.5 + (Math.sin(currentTime / 1000)) / 4; | |
spheres[1].center[0] = 0.5 + (Math.sin(currentTime / 500)) / 3; | |
spheres[1].center[1] = 0.5 + (Math.cos(currentTime / 2000)) / 3; | |
spheres[1].center[2] = 0.5 + (Math.cos(currentTime / 1000)) / 2; | |
// spheres[1].radius = 0.2 + 0.1 * Math.sin(currentTime / 1000 * 4); | |
// reset canvas | |
height = canvas.height = 512; | |
width = canvas.width = 512; | |
// width = canvas.width = 0 | height * innerWidth / innerHeight; | |
// render visuals | |
// context.shadowColor = '#fff'; | |
// context.shadowBlur = 24; | |
context.shadowBlur = 10; | |
context.strokeStyle = '#fff'; | |
traverseOct([0, 0, 0], [1, 1, 1], 0); | |
// render stats | |
context.shadowBlur = 0; | |
context.strokeStyle = '#fff'; | |
context.strokeText(['FPS: ' + (0 | fps), 0 | minFps, 0 | maxFps], 10, 20); | |
requestAnimationFrame(update); | |
} | |
requestAnimationFrame(update); | |
onclick = function() { | |
update = function() {} | |
} | |
</script> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<script> | |
class V3 { | |
static new() { | |
return new Float32Array(3); | |
} | |
static fromValues(x, y, z) { | |
return Float32Array.of(x, y, z); | |
} | |
static add(a, b, result) { | |
result = result || V3.new(); | |
result[0] = a[0] + b[0]; | |
result[1] = a[1] + b[1]; | |
result[2] = a[2] + b[2]; | |
return result; | |
} | |
static sub(a, b, result) { | |
result = result || V3.new(); | |
result[0] = a[0] - b[0]; | |
result[1] = a[1] - b[1]; | |
result[2] = a[2] - b[2]; | |
return result; | |
} | |
static scale(v, s, result) { | |
result = result || V3.new(); | |
result[0] = v[0] * s; | |
result[1] = v[1] * s; | |
result[2] = v[2] * s; | |
return result; | |
} | |
static dot(a, b) { | |
return a[0] * b[0] + a[1] * b[1] + a[2] * b[2]; | |
} | |
static length(v) { | |
return Math.sqrt(V3.dot(v, v)); | |
} | |
static normalize(v, result) { | |
result = result || V3.new(); | |
let scale = 1 / V3.length(v); | |
return V3.scale(v, scale, result); | |
} | |
} | |
</script> | |
<script> | |
class Sphere { | |
constructor(center, radius) { | |
this.center = center; | |
this.radius = radius; | |
this.radius2 = radius * radius; | |
} | |
intersect(ray, test, epsilon) { | |
let m = V3.sub(ray.origin, this.center); | |
let c = V3.dot(m, m) - this.radius2; | |
if (test && c < epsilon) { | |
return true; | |
} | |
let b = V3.dot(m, ray.direction); | |
if (b > 0) { | |
return false | |
} | |
let discr = b * b - c; | |
if (discr < 0) { | |
return false; | |
} | |
if (test) { | |
return true; | |
} | |
let inside = false; | |
let sqrtDiscr = Math.sqrt(discr); | |
let t = -b - sqrtDiscr; | |
if (t < epsilon) { | |
inside = true; | |
t = -b + sqrtDiscr; | |
if (t < epsilon) { | |
return false; | |
} | |
} | |
let point = V3.add(ray.origin, V3.scale(ray.direction, t)); | |
return { | |
point: point, | |
t: t, | |
inside: inside, | |
normal: V3.normalize(V3.sub(point, this.center)) | |
}; | |
} | |
} | |
</script> | |
<script> | |
class GeometryGroup { | |
constructor(...geometries) { | |
this.geometries = geometries; | |
} | |
intersect(ray, test, epsilon) { | |
return this.geometries | |
.map(geometry => geometry.intersect(ray, test, epsilon)) | |
.reduce((previous, current) => { | |
if (!previous) { | |
return current; | |
} | |
if (current) { | |
if (current.t < previous.t) { | |
return current; | |
} else { | |
return previous; | |
} | |
} else { | |
return previous; | |
} | |
}); | |
} | |
} | |
</script> | |
<script src="https://cdnjs.cloudflare.com/ajax/libs/dat-gui/0.5.1/dat.gui.js"></script> | |
<script> | |
class PathTracer { | |
constructor() { | |
this.scene = new GeometryGroup( | |
new Sphere(V3.fromValues(0, 0, 100), 99), | |
new Sphere(V3.fromValues(-100, 0, 0), 99), | |
new Sphere(V3.fromValues(100, 0, 0), 99), | |
new Sphere(V3.fromValues(0, 100, 0), 99), | |
new Sphere(V3.fromValues(0, -100, 0), 99), | |
// new Sphere(V3.fromValues(-1, -1, 1), .2), | |
// new Sphere(V3.fromValues(1, 1, 1), .2), | |
// new Sphere(V3.fromValues(-1, 1, 1), .2), | |
// new Sphere(V3.fromValues(1, -1, 1), .2) | |
// new Sphere(V3.fromValues(.5, 0, 0), .5), | |
new Sphere(V3.fromValues(0, 0, 0), .25) | |
); | |
this.light = new Sphere(V3.fromValues(-.25, .25, -.5), .2); | |
this.epsilon = 0.001; | |
} | |
sample(ray) { | |
let intersection = this.scene.intersect(ray, false, this.epsilon); | |
if (intersection) { | |
// Intensity color | |
let incidence = V3.normalize(V3.sub(this.light.center, intersection.point)); | |
let lambert = V3.dot(intersection.normal, incidence); | |
if (lambert < 0) { | |
return 0xff000000; | |
} | |
let c = Math.min(0xff, ~~(lambert * this.light.radius * 0xff)); | |
// return 0xff000000 | c << 16 | c << 8 | c; | |
let r = ((intersection.normal[0]) + 1) / 2 * c; | |
let g = ((intersection.normal[1]) + 1) / 2 * c; | |
let b = ((intersection.normal[2]) + 1) / 2 * c; | |
return 0xff000000 | r << 16 | g << 8 | b; | |
// Normal color | |
// let r = ((intersection.normal[0]) + 1) / 2 * 0xff; | |
// let g = ((intersection.normal[1]) + 1) / 2 * 0xff; | |
// let b = ((intersection.normal[2]) + 1) / 2 * 0xff; | |
// return 0xff000000 | r << 16 | g << 8 | b; | |
} else { | |
return 0xff000000; | |
} | |
} | |
} | |
class ViewPort { | |
constructor(canvas) { | |
this.ctx = canvas.getContext('2d'); | |
this.imageData = this.ctx.getImageData(0, 0, canvas.width, canvas.height); | |
console.log(this.imageData) | |
this.id8 = this.imageData.data.buffer; | |
this.id32 = new Uint32Array(this.id8); | |
this.halfWidth = canvas.width / 2; | |
this.halfHeight = canvas.height / 2; | |
this.origin = V3.fromValues(0, 0, -1); | |
} | |
render(renderer, time) { | |
for (let y = 0; y < this.imageData.height; y++) { | |
for (let x = 0; x < this.imageData.width; x++) { | |
let direction = this.getRayThrough(x, y); | |
let color = renderer.sample({ | |
origin: this.origin, | |
direction: direction | |
}) | |
this.setColor(x, y, color); | |
} | |
} | |
this.updateCanvas(); | |
} | |
getRayThrough(x, y) { | |
let target = V3.fromValues( | |
2 * x / this.imageData.width - 1, | |
2 * y / this.imageData.height - 1, | |
0 | |
) | |
return V3.normalize(V3.sub(target, this.origin)); | |
} | |
setColor(x, y, color) { | |
let i = x + y * this.imageData.width; | |
this.id32[i] = color; | |
} | |
updateCanvas() { | |
this.ctx.putImageData(this.imageData, 0, 0); | |
} | |
} | |
function onLoad() { | |
let pathTracer = new PathTracer(); | |
canvas.width = canvas.height = 0xff; | |
let viewPort = new ViewPort(canvas); | |
canvas.addEventListener('mousemove', ({ | |
offsetX, | |
offsetY | |
}) => { | |
let x = offsetX / canvas.width * 2 - 1; | |
let y = offsetY / canvas.height * 2 - 1; | |
pathTracer.light.center[0] = x; | |
pathTracer.light.center[1] = y; | |
}) | |
var animationFrameRequestId; | |
let ui = { | |
run: function() { | |
viewPort.render(pathTracer); | |
animationFrameRequestId = requestAnimationFrame(ui.run); | |
}, | |
toggle: function() { | |
if (animationFrameRequestId) { | |
cancelAnimationFrame(animationFrameRequestId) | |
animationFrameRequestId = null; | |
} else { | |
animationFrameRequestId = requestAnimationFrame(ui.run); | |
} | |
} | |
} | |
var gui = new dat.GUI(); | |
gui.add(ui, 'toggle'); | |
gui.add(pathTracer.light, 'radius', 0, 10); | |
} | |
</script> | |
<body onload="onLoad()"> | |
<canvas id="canvas"></canvas> | |
</body> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
var seed = function(s) { | |
return function() { | |
s = Math.sin(s) * 10000; | |
return s - Math.floor(s); | |
}; | |
}; | |
function Random(seed) { | |
this.seed = seed; | |
} | |
Random.prototype.next = function() { | |
this.seed = Math.sin(this.seed) * 10000; | |
return this.seed - Math.floor(this.seed); | |
}; | |
random = new Random(23); | |
random.next(); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<style> | |
#canvas { | |
/*object-fit: contain; | |
position: fixed;*/ | |
/*top: 0; | |
left: 0;*/ | |
/*width: 256px; | |
height: 256px;*/ | |
/*background: radial-gradient(#456, #200);*/ | |
/*background: linear-gradient(#456, #200);*/ | |
background-color: black; | |
} | |
</style> | |
a | |
<canvas id="heightmap" height="256" width="256"></canvas> | |
b | |
<canvas id="canvas" height="256" width="256"></canvas> | |
c | |
<script> | |
var ctx = heightmap.getContext('2d'); | |
var id = ctx.getImageData(0, 0, heightmap.width, heightmap.height); | |
function generateHeightMap() { | |
id.data.fill(0xff); | |
console.log(heightmap.width) | |
for (var y = 0; y < heightmap.height; y++) { | |
var y_ = y / heightmap.height; | |
for (var x = 0; x < heightmap.width; x++) { | |
var x_ = x / heightmap.width; | |
i = ((y * heightmap.width) + x) * 4; | |
var v = 0 | volume(x_, y_) * 0xff; | |
id.data[i] = id.data[i + 1] = id.data[i + 2] = v; | |
} | |
} | |
ctx.putImageData(id, 0, 0); | |
} | |
context = canvas.getContext('2d'); | |
imageData = context.getImageData(0, 0, canvas.width, canvas.height); | |
imageData.data.fill(0xff); | |
// function volume(x, y) { | |
// return Math.random() | |
// } | |
// function volume(x, y) { | |
// x -= .5; | |
// y -= .5; | |
// x *= 3; | |
// y *= 2; | |
// if (x && y) { | |
// return 1 / (x * x + y * y); | |
// } else { | |
// return 1; | |
// } | |
// } | |
// function volume(x, y) { | |
// return Math.sin(5 * y * Math.PI) / 2 - Math.cos(5 * x * Math.PI) / 2; | |
// } | |
function volume(x, y) { | |
return .7 - y * .5; | |
} | |
// function draw(time) { | |
// // update stuff | |
// | |
// // reset canvas | |
// height = canvas.height = 256; | |
// width = canvas.width = 256; | |
// | |
// // render visuals | |
// context.fillStyle = context.strokeStyle = context.shadowColor = '#fff'; | |
// // context.shadowBlur = 24; | |
// // march([0, 0, 0], [1, 1, 1], 0); | |
// for (y = 0; y < imageData.height; y++) { | |
// y_ = (y / imageData.height) - .5; | |
// for (x = 0; x < imageData.width; x++) { | |
// x_ = (x / imageData.width) - .5; | |
// i = ((y * imageData.width) + x) * 4; | |
// v = volume(x_, y_) / 100 * 0xff; | |
// imageData.data[i] = v; | |
// imageData.data[i + 1] = v; | |
// imageData.data[i + 2] = v; | |
// // imageData.data[x * 4 + y * canvas.width + 0] = 0 | volume(x_, y_) * 255; | |
// // imageData.data[x * 4 + y * canvas.width + 1] = 0 | volume(x_, y_) * 255; | |
// // imageData.data[x * 4 + y * canvas.width + 2] = 0 | volume(x_, y_) * 255; | |
// } | |
// console.log(x_, y_, volume(x_, y_)) | |
// } | |
// context.putImageData(imageData, 0, 0); | |
// | |
// // render stats | |
// context.strokeText([((performance.now() - time) / 1000)], 10, 20); | |
// } | |
var camera = { | |
x: 256/2, | |
y: 256/2, | |
z: 0, | |
} | |
yBuffer = []; | |
function draw(time) { | |
for (x = 0; x < imageData.width; x++) { | |
x_ = (x / imageData.width); | |
ymin = 0; | |
dx = x - camera.x; | |
for (z = imageData.height; z > 0; z--) { | |
// for (z = 0; z < imageData.height; z++) { | |
z_ = z / imageData.height; | |
y_ = volume(x_, z_); | |
// yBuffer[x] = Math.max(yBuffer[x] || 0, y_); | |
ymax = y_ * imageData.height; | |
dy = ymax - ymin; | |
if (ymin < ymax) { | |
// console.log(ymin, ymax, dy) | |
context.fillStyle = context.strokeStyle = context.shadowColor = 'rgb(' + (0 | y_ * 0xff) + ',' + (0 | y_ * 0xff) + ',' + (0 | y_ * 0xff) + ')'; | |
context.fillRect(x, imageData.height - ymin, 1, -dy) | |
// context.fillRect(x, y_ * imageData.height, 1, 1) | |
ymin = ymax; | |
} | |
} | |
} | |
} | |
generateHeightMap(); | |
draw(performance.now()); | |
</script> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<style> | |
body { | |
background-color: black; | |
} | |
#canvas { | |
background-color: #333; | |
transform: scaleY(-1); | |
} | |
</style> | |
<canvas id="heightmap" height="256" width="256"></canvas> | |
<canvas id="canvas" height="256" width="256"></canvas> | |
<script> | |
var ctx = heightmap.getContext('2d'); | |
var id = ctx.getImageData(0, 0, heightmap.width, heightmap.height); | |
function generateHeightMap() { | |
id.data.fill(0xff); | |
for (var y = 0; y < heightmap.height; y++) { | |
var y_ = y / heightmap.height; | |
for (var x = 0; x < heightmap.width; x++) { | |
var x_ = x / heightmap.width; | |
i = ((y * heightmap.width) + x) * 4; | |
var v = 0 | volume(x_, y_) * 0xff; | |
id.data[i] = id.data[i + 1] = id.data[i + 2] = v; | |
} | |
} | |
ctx.putImageData(id, 0, 0); | |
ctx.fillStyle = '#00f'; | |
ctx.fillRect(0, camera.z * heightmap.height, heightmap.width, 1); | |
} | |
function volume(x, y) { | |
return Math.sin(3 * y * 2 * Math.PI) * Math.cos(3 * x * 2 * Math.PI); | |
} | |
var camera = { | |
x: 256 / 2, | |
y: 256 / 2, | |
z: 0, | |
} | |
yBuffer = []; | |
context = canvas.getContext('2d'); | |
function draw(time) { | |
// canvas.width = canvas.width; | |
for (x = 0; x < canvas.width; x++) { | |
x_ = x / canvas.width; | |
y_ = volume(x_, camera.z); | |
y = (1 + y_) * (canvas.height / 2); | |
yBuffer[x] = yBuffer[x] || 0; | |
if (y > yBuffer[x]) { | |
color = (0 | y_ * 0xff).toString(16); | |
color = '00'.substring(0, 2 - color.length) + color; | |
context.fillStyle = '#' + color + color + color; | |
context.fillRect(x, yBuffer[x], 1, y - yBuffer[x]); | |
} | |
yBuffer[x] = Math.max(yBuffer[x] || 0, y); | |
} | |
} | |
heightmap.onclick = function(event) { | |
camera.z = event.offsetY / this.height; | |
generateHeightMap(); | |
draw(performance.now()); | |
} | |
// for (var z = 0xff; z > 0; z--) { | |
// // for (var z = 0; z > 0xff; z++) { | |
// camera.z = z / 0xff; | |
// draw(performance.now()); | |
// } | |
generateHeightMap(); | |
console.log("DONE") | |
</script> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<style> | |
body { | |
background-color: black; | |
} | |
#canvas { | |
background-color: #abc; | |
transform: scaleY(-1); | |
} | |
</style> | |
<canvas id="heightmap" height="256" width="256"></canvas> | |
<canvas id="canvas" height="256" width="256"></canvas> | |
<script> | |
var ctx = heightmap.getContext('2d'); | |
var id = ctx.getImageData(0, 0, heightmap.width, heightmap.height); | |
function generateHeightMap() { | |
id.data.fill(0xff); | |
for (var y = 0; y < heightmap.height; y++) { | |
var y_ = y / heightmap.height; | |
for (var x = 0; x < heightmap.width; x++) { | |
var x_ = x / heightmap.width; | |
i = ((y * heightmap.width) + x) * 4; | |
var v = 0 | volume(x_, y_) * 0xff; | |
id.data[i] = id.data[i + 1] = id.data[i + 2] = v; | |
} | |
} | |
ctx.putImageData(id, 0, 0); | |
ctx.fillStyle = '#00f'; | |
ctx.fillRect(camera.u * heightmap.width, 0, 1, heightmap.height); | |
ctx.fillRect(0, camera.v * heightmap.height, heightmap.width, 1); | |
} | |
function volume(x, y) { | |
return Math.sin(3 * y * 2 * Math.PI) * Math.cos(3 * x * 2 * Math.PI) - .5; | |
// return (0 | (v * 50)) / 50; | |
} | |
var camera = { | |
u: .5, | |
v: 0, | |
r: 1.25 * Math.PI, | |
} | |
yBuffer = []; | |
context = canvas.getContext('2d'); | |
function draw(time) { | |
canvas.width = canvas.width; | |
for (x = 0; x < canvas.width; x++) { | |
r = camera.r + ((x / canvas.width) * Math.PI / 2); | |
// console.log(camera.r,x, canvas.width) | |
cos = Math.cos(r); | |
sin = Math.sin(r); | |
u1 = camera.u; | |
v1 = camera.v; | |
u2 = u1 + cos; | |
v2 = v1 + sin; | |
tMax = 100; | |
du = (u2 - u1) / tMax; | |
dv = (v2 - v1) / tMax; | |
minY = 0; | |
for (t = 0; t < tMax; t++) { | |
t_ = t / tMax; | |
u1 += du; | |
v1 += dv; | |
y_ = volume(u1, v1); | |
y = (y_ * canvas.height * (1 - t_)) + (canvas.height / 2); | |
if (y > minY) { | |
color = (0 | (1 + y_) / 2 * 0xff).toString(16); | |
color = '00'.substring(0, 2 - color.length) + color; | |
context.fillStyle = '#' + color + color + color; | |
context.fillRect(x, minY, 1, y - minY); | |
} | |
minY = Math.max(minY, y); | |
} | |
} | |
} | |
heightmap.onclick = function(event) { | |
camera.u = event.offsetX / this.width; | |
camera.v = event.offsetY / this.height; | |
generateHeightMap(); | |
draw(performance.now()); | |
} | |
</script> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
function length2(v) { | |
return v[0] * v[0] + v[1] * v[1] + v[2] * v[2]; | |
} | |
function length(v) { | |
return Math.sqrt(length2(v)); | |
} | |
function dot(v) {} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment