function* run(options = {}) {
const {
width: w = 800,
height: h = 300,
resolution: res = 1/4,
noiseScale: s = .05,
layers: count = 5,
lower = 1,
upper = 0,
tx = 0,
ty = 0,
tz = .1,
sxy = 0,
sz = 0,
colorLow = '#000',
colorHigh = '#fff',
colorBg = '#000',
gco = 'source-over',
seed = '0'
} = options;
const simplex = new noise('0');
const ws = w*res|0, hs = h*res|0;
const color = d3.interpolateHsl(colorLow, colorHigh);
const layers = Array(count).fill()
.map((v,i,a)=>i/(a.length-1))
.map(t => layer(ws, hs, color2rgba(color(t))));
const cOut = html`<canvas width=${w} height=${h} style="max-width:100%">`.getContext('2d');
cOut.globalCompositeOperation = gco;
const bgFill = new ImageData(w, h);
{
const bg = color2rgba(colorBg);
for(let i=0; i<bgFill.data.length; i+=4) bgFill.data.set(bg, i);
}
let t = 0;
while(true) {
cOut.putImageData(bgFill, 0, 0);
layers.map((l,i) => {
const lt = i/layers.length;
const u = (x, y) => smoothstep(
lower, upper,
1 - simplex.noise3D(x*s + t*tx, y*s + t*ty, lt*s*sz + t*tz) ** 2
);
const p = (lt ** 2) * sxy;
cOut.drawImage(l.update(u), 0, 0, ws, hs, -p, -p, w+p*2, h+p*2);
});
yield cOut.canvas;
t += 1/30;
}
}