Improve drawing
This commit is contained in:
parent
9f01a9e236
commit
de7fcc15d0
6 changed files with 221 additions and 56 deletions
143
src/pipelines/brush/brush-pipeline.ts
Normal file
143
src/pipelines/brush/brush-pipeline.ts
Normal file
|
|
@ -0,0 +1,143 @@
|
||||||
|
import shader from './brush.wgsl';
|
||||||
|
|
||||||
|
import { vec2 } from 'gl-matrix';
|
||||||
|
|
||||||
|
export class BrushPipeline {
|
||||||
|
private static readonly UNIFORM_COUNT = 2;
|
||||||
|
private static readonly MAX_LINE_COUNT = 100;
|
||||||
|
private static readonly VERTICES_PER_LINE_SEGMENT = 6;
|
||||||
|
|
||||||
|
private readonly pipeline: GPURenderPipeline;
|
||||||
|
private readonly uniforms: GPUBuffer;
|
||||||
|
private readonly vertexBuffer: GPUBuffer;
|
||||||
|
private readonly linePoints: Array<vec2> = [];
|
||||||
|
private bindGroup: GPUBindGroup;
|
||||||
|
|
||||||
|
public constructor(private readonly device: GPUDevice) {
|
||||||
|
this.vertexBuffer = device.createBuffer({
|
||||||
|
size:
|
||||||
|
BrushPipeline.MAX_LINE_COUNT *
|
||||||
|
BrushPipeline.VERTICES_PER_LINE_SEGMENT *
|
||||||
|
2 *
|
||||||
|
Float32Array.BYTES_PER_ELEMENT,
|
||||||
|
usage: GPUBufferUsage.VERTEX | GPUBufferUsage.COPY_DST,
|
||||||
|
});
|
||||||
|
|
||||||
|
this.pipeline = device.createRenderPipeline({
|
||||||
|
layout: 'auto',
|
||||||
|
vertex: {
|
||||||
|
module: device.createShaderModule({
|
||||||
|
code: shader,
|
||||||
|
}),
|
||||||
|
entryPoint: 'vertex',
|
||||||
|
buffers: [
|
||||||
|
{
|
||||||
|
arrayStride: Float32Array.BYTES_PER_ELEMENT * 2,
|
||||||
|
attributes: [
|
||||||
|
{
|
||||||
|
shaderLocation: 0,
|
||||||
|
format: 'float32x2',
|
||||||
|
offset: 0,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
fragment: {
|
||||||
|
module: device.createShaderModule({
|
||||||
|
code: shader,
|
||||||
|
}),
|
||||||
|
entryPoint: 'fragment',
|
||||||
|
targets: [
|
||||||
|
{
|
||||||
|
format: 'rgba16float',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
primitive: {
|
||||||
|
topology: 'triangle-list',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
this.uniforms = this.device.createBuffer({
|
||||||
|
size: BrushPipeline.UNIFORM_COUNT * Float32Array.BYTES_PER_ELEMENT,
|
||||||
|
usage: GPUBufferUsage.UNIFORM | GPUBufferUsage.COPY_DST,
|
||||||
|
});
|
||||||
|
|
||||||
|
this.bindGroup = this.device.createBindGroup({
|
||||||
|
layout: this.pipeline.getBindGroupLayout(0),
|
||||||
|
entries: [
|
||||||
|
{
|
||||||
|
binding: 0,
|
||||||
|
resource: {
|
||||||
|
buffer: this.uniforms,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public addSwipe(position: vec2) {
|
||||||
|
this.linePoints.push(position);
|
||||||
|
}
|
||||||
|
|
||||||
|
public clearSwipes() {
|
||||||
|
this.linePoints.length = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public setParameters({ width, height }: { width: number; height: number }) {
|
||||||
|
this.device.queue.writeBuffer(this.uniforms, 0, new Float32Array([width, height]));
|
||||||
|
|
||||||
|
this.device.queue.writeBuffer(
|
||||||
|
this.vertexBuffer,
|
||||||
|
0,
|
||||||
|
new Float32Array(
|
||||||
|
new Array(this.lineCount).fill(0).flatMap((_, i) => {
|
||||||
|
const from = this.linePoints[i];
|
||||||
|
const to = this.linePoints[i + 1];
|
||||||
|
const [a, b, c, d] = this.lineToRectangle(from, to, 0.01);
|
||||||
|
return [...a, ...b, ...c, ...b, ...c, ...d];
|
||||||
|
})
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private get lineCount() {
|
||||||
|
return Math.max(0, this.linePoints.length - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
private lineToRectangle(from: vec2, to: vec2, width: number): [vec2, vec2, vec2, vec2] {
|
||||||
|
const dir = vec2.sub(vec2.create(), to, from);
|
||||||
|
const perp = vec2.fromValues(dir[1], -dir[0]);
|
||||||
|
vec2.normalize(perp, perp);
|
||||||
|
vec2.scale(perp, perp, width / 2);
|
||||||
|
return [
|
||||||
|
vec2.add(vec2.create(), from, perp),
|
||||||
|
vec2.sub(vec2.create(), from, perp),
|
||||||
|
vec2.add(vec2.create(), to, perp),
|
||||||
|
vec2.sub(vec2.create(), to, perp),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public execute(commandEncoder: GPUCommandEncoder, trailMapOut: GPUTexture) {
|
||||||
|
const renderPassDescriptor: GPURenderPassDescriptor = {
|
||||||
|
colorAttachments: [
|
||||||
|
{
|
||||||
|
view: trailMapOut.createView(),
|
||||||
|
clearValue: { r: 1.0, g: 1.0, b: 1.0, a: 1.0 },
|
||||||
|
loadOp: 'load',
|
||||||
|
storeOp: 'store',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
|
|
||||||
|
const passEncoder = commandEncoder.beginRenderPass(renderPassDescriptor);
|
||||||
|
passEncoder.setPipeline(this.pipeline);
|
||||||
|
passEncoder.setBindGroup(0, this.bindGroup);
|
||||||
|
passEncoder.setVertexBuffer(0, this.vertexBuffer);
|
||||||
|
passEncoder.draw(this.lineCount * BrushPipeline.VERTICES_PER_LINE_SEGMENT, 1);
|
||||||
|
passEncoder.end();
|
||||||
|
|
||||||
|
this.linePoints.splice(0, this.linePoints.length - 1); // clear the array
|
||||||
|
}
|
||||||
|
}
|
||||||
23
src/pipelines/brush/brush.wgsl
Normal file
23
src/pipelines/brush/brush.wgsl
Normal file
|
|
@ -0,0 +1,23 @@
|
||||||
|
struct VertexOutput {
|
||||||
|
@builtin(position) position : vec4<f32>,
|
||||||
|
@location(0) uv : vec2<f32>
|
||||||
|
}
|
||||||
|
|
||||||
|
@vertex
|
||||||
|
fn vertex(
|
||||||
|
@location(0) uv : vec2<f32>
|
||||||
|
) -> VertexOutput {
|
||||||
|
let position = uv * 2.0 - 1.0;
|
||||||
|
return VertexOutput(vec4(position, 0.0, 1.0), uv);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Settings {
|
||||||
|
size : vec2<f32>
|
||||||
|
};
|
||||||
|
|
||||||
|
@group(0) @binding(0) var<uniform> settings : Settings;
|
||||||
|
|
||||||
|
@fragment
|
||||||
|
fn fragment(@location(0) uv: vec2<f32>) -> @location(0) vec4<f32> {
|
||||||
|
return vec4(1, settings.size.x * 0, 0, 1);
|
||||||
|
}
|
||||||
|
|
@ -1,14 +1,11 @@
|
||||||
struct Settings {
|
struct Settings {
|
||||||
size : vec2<f32>,
|
size : vec2<f32>,
|
||||||
swipePrevious : vec2<f32>,
|
|
||||||
swipeCurrent : vec2<f32>,
|
|
||||||
diffusionRate : f32,
|
diffusionRate : f32,
|
||||||
decayRate : f32,
|
decayRate : f32,
|
||||||
deltaTime : f32,
|
deltaTime : f32,
|
||||||
time : f32,
|
time : f32,
|
||||||
swipeRadius : f32,
|
swipeRadius : f32,
|
||||||
swipeBlur : f32,
|
swipeBlur : f32,
|
||||||
isSwipeActive : f32
|
|
||||||
};
|
};
|
||||||
|
|
||||||
@group(0) @binding(0) var<uniform> settings : Settings;
|
@group(0) @binding(0) var<uniform> settings : Settings;
|
||||||
|
|
@ -26,18 +23,6 @@ fn fragment(@location(0) uv: vec2<f32>) -> @location(0) vec4<f32> {
|
||||||
+ textureSample(trailMap, Sampler, uv + vec2<f32>(1, 0) / settings.size)
|
+ textureSample(trailMap, Sampler, uv + vec2<f32>(1, 0) / settings.size)
|
||||||
);
|
);
|
||||||
|
|
||||||
if (settings.isSwipeActive == 1.0) {
|
|
||||||
let pa = (uv - settings.swipePrevious) * normalize(settings.size);
|
|
||||||
let direction = (settings.swipeCurrent - settings.swipePrevious) * normalize(settings.size);
|
|
||||||
let q = clamp(dot(pa, direction) / dot(direction, direction), 0, 1);
|
|
||||||
let distance = length(pa - direction * q) - settings.swipeRadius;
|
|
||||||
|
|
||||||
if(distance < 0) {
|
|
||||||
let opacity = -distance / settings.swipeBlur;
|
|
||||||
return clamp(vec4(1), current, vec4(1));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return mix(
|
return mix(
|
||||||
current,
|
current,
|
||||||
neighbours / 4.0,
|
neighbours / 4.0,
|
||||||
|
|
|
||||||
|
|
@ -1,19 +1,13 @@
|
||||||
import { setUpFullScreenQuad } from '../../utils/full-screen-quad';
|
import { setUpFullScreenQuad } from '../../utils/full-screen-quad';
|
||||||
import shader from './diffuse.wgsl';
|
import shader from './diffuse.wgsl';
|
||||||
|
|
||||||
import { vec2 } from 'gl-matrix';
|
|
||||||
|
|
||||||
export class DiffusionPipeline {
|
export class DiffusionPipeline {
|
||||||
private static readonly UNIFORM_COUNT = 14;
|
private static readonly UNIFORM_COUNT = 16;
|
||||||
|
|
||||||
private readonly pipeline: GPURenderPipeline;
|
private readonly pipeline: GPURenderPipeline;
|
||||||
private readonly uniforms: GPUBuffer;
|
private readonly uniforms: GPUBuffer;
|
||||||
private readonly quadVertexBuffer: GPUBuffer;
|
private readonly quadVertexBuffer: GPUBuffer;
|
||||||
|
|
||||||
private swipes: Array<vec2> = [
|
|
||||||
vec2.fromValues(Number.NaN, Number.NaN),
|
|
||||||
vec2.fromValues(Number.NaN, Number.NaN),
|
|
||||||
];
|
|
||||||
private bindGroup?: GPUBindGroup;
|
private bindGroup?: GPUBindGroup;
|
||||||
private previousTrailMapIn?: GPUTexture;
|
private previousTrailMapIn?: GPUTexture;
|
||||||
|
|
||||||
|
|
@ -53,40 +47,30 @@ export class DiffusionPipeline {
|
||||||
decayRate,
|
decayRate,
|
||||||
deltaTime,
|
deltaTime,
|
||||||
time,
|
time,
|
||||||
swipe,
|
|
||||||
swipeRadius,
|
swipeRadius,
|
||||||
swipeBlur,
|
swipeBlur,
|
||||||
isSwipeActive,
|
|
||||||
}: {
|
}: {
|
||||||
width: number;
|
width: number;
|
||||||
height: number;
|
height: number;
|
||||||
swipe: vec2;
|
|
||||||
diffusionRate: number;
|
diffusionRate: number;
|
||||||
decayRate: number;
|
decayRate: number;
|
||||||
deltaTime: number;
|
deltaTime: number;
|
||||||
time: number;
|
time: number;
|
||||||
swipeRadius: number;
|
swipeRadius: number;
|
||||||
swipeBlur: number;
|
swipeBlur: number;
|
||||||
isSwipeActive: boolean;
|
|
||||||
}) {
|
}) {
|
||||||
if (swipe) {
|
|
||||||
this.swipes = [...this.swipes.slice(-1), swipe];
|
|
||||||
}
|
|
||||||
|
|
||||||
this.device.queue.writeBuffer(
|
this.device.queue.writeBuffer(
|
||||||
this.uniforms,
|
this.uniforms,
|
||||||
0,
|
0,
|
||||||
new Float32Array([
|
new Float32Array([
|
||||||
width,
|
width,
|
||||||
height,
|
height,
|
||||||
...this.swipes.flatMap((s) => [s[0], s[1]]),
|
|
||||||
diffusionRate,
|
diffusionRate,
|
||||||
decayRate,
|
decayRate,
|
||||||
deltaTime,
|
deltaTime,
|
||||||
time,
|
time,
|
||||||
swipeRadius,
|
swipeRadius,
|
||||||
swipeBlur,
|
swipeBlur,
|
||||||
isSwipeActive ? 1.0 : 0.0,
|
|
||||||
])
|
])
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,8 +1,10 @@
|
||||||
import { Agent } from './pipelines/agents/agent';
|
import { Agent } from './pipelines/agents/agent';
|
||||||
import { AgentPipeline } from './pipelines/agents/agent-pipeline';
|
import { AgentPipeline } from './pipelines/agents/agent-pipeline';
|
||||||
|
import { BrushPipeline } from './pipelines/brush/brush-pipeline';
|
||||||
import { DiffusionPipeline } from './pipelines/diffusion/diffusion-pipeline';
|
import { DiffusionPipeline } from './pipelines/diffusion/diffusion-pipeline';
|
||||||
import { RenderPipeline } from './pipelines/render/render-pipeline';
|
import { RenderPipeline } from './pipelines/render/render-pipeline';
|
||||||
import { settings } from './settings';
|
import { settings } from './settings';
|
||||||
|
import { DeltaTimeCalculator } from './utils/delta-time-calculator';
|
||||||
import { randomBetween } from './utils/random-between';
|
import { randomBetween } from './utils/random-between';
|
||||||
import { sleep } from './utils/sleep';
|
import { sleep } from './utils/sleep';
|
||||||
|
|
||||||
|
|
@ -16,15 +18,15 @@ export default class Renderer {
|
||||||
|
|
||||||
private agentPipeline: AgentPipeline;
|
private agentPipeline: AgentPipeline;
|
||||||
private renderPipeline: RenderPipeline;
|
private renderPipeline: RenderPipeline;
|
||||||
|
private brushPipeline: BrushPipeline;
|
||||||
private diffusionPipeline: DiffusionPipeline;
|
private diffusionPipeline: DiffusionPipeline;
|
||||||
|
|
||||||
private preferredCanvasFormat: GPUTextureFormat;
|
private preferredCanvasFormat: GPUTextureFormat;
|
||||||
private trailMapA?: GPUTexture;
|
private trailMapA?: GPUTexture;
|
||||||
private trailMapB?: GPUTexture;
|
private trailMapB?: GPUTexture;
|
||||||
|
|
||||||
private previousTime?: DOMHighResTimeStamp = null;
|
|
||||||
private swipeLocation?: vec2;
|
|
||||||
private isSwipeActive = false;
|
private isSwipeActive = false;
|
||||||
|
private readonly deltaTimeCalculator = new DeltaTimeCalculator();
|
||||||
|
|
||||||
public constructor(private canvas: HTMLCanvasElement) {}
|
public constructor(private canvas: HTMLCanvasElement) {}
|
||||||
|
|
||||||
|
|
@ -32,10 +34,6 @@ export default class Renderer {
|
||||||
await this.initializeDevice();
|
await this.initializeDevice();
|
||||||
|
|
||||||
this.resize();
|
this.resize();
|
||||||
window.addEventListener('resize', this.resize.bind(this));
|
|
||||||
window.addEventListener('mousemove', this.onSwipe.bind(this));
|
|
||||||
window.addEventListener('mousedown', (_) => (this.isSwipeActive = true));
|
|
||||||
window.addEventListener('mouseup', (_) => (this.isSwipeActive = false));
|
|
||||||
|
|
||||||
this.agentPipeline = new AgentPipeline(this.device, this.spawnAgents());
|
this.agentPipeline = new AgentPipeline(this.device, this.spawnAgents());
|
||||||
this.renderPipeline = new RenderPipeline(
|
this.renderPipeline = new RenderPipeline(
|
||||||
|
|
@ -43,18 +41,31 @@ export default class Renderer {
|
||||||
this.device,
|
this.device,
|
||||||
this.preferredCanvasFormat
|
this.preferredCanvasFormat
|
||||||
);
|
);
|
||||||
|
this.brushPipeline = new BrushPipeline(this.device);
|
||||||
this.diffusionPipeline = new DiffusionPipeline(this.device);
|
this.diffusionPipeline = new DiffusionPipeline(this.device);
|
||||||
|
|
||||||
|
window.addEventListener('resize', this.resize.bind(this));
|
||||||
|
window.addEventListener('mousemove', this.onSwipe.bind(this));
|
||||||
|
window.addEventListener('mousedown', (_) => (this.isSwipeActive = true));
|
||||||
|
window.addEventListener('mouseup', (_) => {
|
||||||
|
this.isSwipeActive = false;
|
||||||
|
this.brushPipeline.clearSwipes();
|
||||||
|
});
|
||||||
|
|
||||||
requestAnimationFrame(this.render.bind(this));
|
requestAnimationFrame(this.render.bind(this));
|
||||||
}
|
}
|
||||||
|
|
||||||
private onSwipe(event: MouseEvent) {
|
private onSwipe(event: MouseEvent) {
|
||||||
const position = vec2.fromValues(event.clientX, event.clientY);
|
if (!this.isSwipeActive) {
|
||||||
this.swipeLocation = vec2.divide(
|
return;
|
||||||
position,
|
}
|
||||||
position,
|
|
||||||
vec2.fromValues(this.canvas.width, this.canvas.height)
|
const uv = vec2.fromValues(
|
||||||
|
event.clientX / this.canvas.width,
|
||||||
|
1 - event.clientY / this.canvas.height
|
||||||
);
|
);
|
||||||
|
|
||||||
|
this.brushPipeline.addSwipe(uv);
|
||||||
}
|
}
|
||||||
|
|
||||||
private spawnAgents(): Array<Agent> {
|
private spawnAgents(): Array<Agent> {
|
||||||
|
|
@ -129,7 +140,7 @@ export default class Renderer {
|
||||||
}
|
}
|
||||||
|
|
||||||
private async render(time: DOMHighResTimeStamp) {
|
private async render(time: DOMHighResTimeStamp) {
|
||||||
const deltaTime = this.calculateDeltaTime(time);
|
const deltaTime = this.deltaTimeCalculator.calculateDeltaTimeInSeconds(time);
|
||||||
|
|
||||||
this.agentPipeline.setParameters({
|
this.agentPipeline.setParameters({
|
||||||
...settings,
|
...settings,
|
||||||
|
|
@ -138,19 +149,22 @@ export default class Renderer {
|
||||||
time,
|
time,
|
||||||
deltaTime,
|
deltaTime,
|
||||||
});
|
});
|
||||||
|
this.brushPipeline.setParameters({
|
||||||
|
width: this.canvas.width,
|
||||||
|
height: this.canvas.height,
|
||||||
|
});
|
||||||
this.diffusionPipeline.setParameters({
|
this.diffusionPipeline.setParameters({
|
||||||
...settings,
|
...settings,
|
||||||
width: this.canvas.width,
|
width: this.canvas.width,
|
||||||
height: this.canvas.height,
|
height: this.canvas.height,
|
||||||
deltaTime,
|
deltaTime,
|
||||||
time,
|
time,
|
||||||
isSwipeActive: this.isSwipeActive,
|
|
||||||
swipe: this.swipeLocation,
|
|
||||||
});
|
});
|
||||||
const commandEncoder = this.device.createCommandEncoder();
|
const commandEncoder = this.device.createCommandEncoder();
|
||||||
|
|
||||||
for (let i = 0; i < settings.renderSpeed; i++) {
|
for (let i = 0; i < settings.renderSpeed; i++) {
|
||||||
this.agentPipeline.execute(commandEncoder, this.trailMapA, this.trailMapB);
|
this.agentPipeline.execute(commandEncoder, this.trailMapA, this.trailMapB);
|
||||||
|
this.brushPipeline.execute(commandEncoder, this.trailMapB);
|
||||||
this.diffusionPipeline.execute(commandEncoder, this.trailMapB, this.trailMapA);
|
this.diffusionPipeline.execute(commandEncoder, this.trailMapB, this.trailMapA);
|
||||||
this.renderPipeline.execute(commandEncoder, this.trailMapA);
|
this.renderPipeline.execute(commandEncoder, this.trailMapA);
|
||||||
[this.trailMapA, this.trailMapB] = [this.trailMapB, this.trailMapA];
|
[this.trailMapA, this.trailMapB] = [this.trailMapB, this.trailMapA];
|
||||||
|
|
@ -158,16 +172,7 @@ export default class Renderer {
|
||||||
|
|
||||||
this.queue.submit([commandEncoder.finish()]);
|
this.queue.submit([commandEncoder.finish()]);
|
||||||
|
|
||||||
// await sleep(1000);
|
await sleep(200);
|
||||||
requestAnimationFrame(this.render.bind(this));
|
requestAnimationFrame(this.render.bind(this));
|
||||||
}
|
}
|
||||||
|
|
||||||
private calculateDeltaTime(time: DOMHighResTimeStamp): number {
|
|
||||||
if (this.previousTime === null) {
|
|
||||||
this.previousTime = time;
|
|
||||||
}
|
|
||||||
const deltaTime = time - this.previousTime;
|
|
||||||
this.previousTime = time;
|
|
||||||
return deltaTime / 1000;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
25
src/utils/delta-time-calculator.ts
Normal file
25
src/utils/delta-time-calculator.ts
Normal file
|
|
@ -0,0 +1,25 @@
|
||||||
|
export class DeltaTimeCalculator {
|
||||||
|
private previousTime: DOMHighResTimeStamp | null = null;
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
document.addEventListener('visibilitychange', this.handleVisibilityChange.bind(this));
|
||||||
|
}
|
||||||
|
|
||||||
|
public calculateDeltaTimeInSeconds(
|
||||||
|
currentTime: DOMHighResTimeStamp
|
||||||
|
): DOMHighResTimeStamp {
|
||||||
|
if (this.previousTime === null) {
|
||||||
|
this.previousTime = currentTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
const delta = currentTime - this.previousTime;
|
||||||
|
this.previousTime = currentTime;
|
||||||
|
return delta / 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
private handleVisibilityChange() {
|
||||||
|
if (!document.hidden) {
|
||||||
|
this.previousTime = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
Add table
Add a link
Reference in a new issue