Skip to content

Instantly share code, notes, and snippets.

@sporsh
Last active May 26, 2021 10:53
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 sporsh/de5dd62b66a18f5c6a78 to your computer and use it in GitHub Desktop.
Save sporsh/de5dd62b66a18f5c6a78 to your computer and use it in GitHub Desktop.
k-d tree / octree metaball https://goo.gl/km4snW
<!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>
<!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>
<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>
<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>
<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>
<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>
<!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>
<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>
<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>
<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>
<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>
<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>
<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>
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();
<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>
<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>
<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>
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