当前示例源码github地址:
https://github.com/vilyLei/voxwebgpu/blob/main/src/voxgpu/sample/ScreenPostEffect.ts
此示例渲染系统实现的特性:
1. 用户态与系统态隔离。
细节请见:引擎系统设计思路 - 用户态与系统态隔离-CSDN博客
2. 高频调用与低频调用隔离。
3. 面向用户的易用性封装。
4. 渲染数据(内外部相关资源)和渲染机制分离。
5. 用户操作和渲染系统调度并行机制。
6. 数据/语义驱动。
当前示例运行效果:
WGSL片段shader:
@group(0) @binding(0) var<uniform> color: vec4f;
@group(0) @binding(1) var<storage> params: array<vec4<f32>>;
fn calcColor3(vtxUV: vec2f) -> vec3f{
let stSize = params[1].zw;
let time = params[2].x;
let fragCoord = vtxUV * stSize;
let color1 = vec3f(1.7, 0.25, 0.5);
let color2 = vec3f(0.5, 0.7, 0.25);
let color3 = vec3f(0.25, 0.5, 0.7);
let point1 = stSize * 0.45 + vec2f(sin(time*2.0) * 10.0, cos(time*2.0) * 5.0);
let point2 = stSize * 0.5 + vec2f(sin(time) * 75.0, cos(time)*50.0);
let point3 = stSize * 0.55 + vec2f(sin(time) * 25.0, sin(time*2.0)*50.0)*2.0;
let dist1 = fragCoord - point1;
let intensity1 = pow(32.0/(0.01+length(dist1)), 2.0);
let dist2 = fragCoord - point2;
let intensity2 = pow(3.0/(0.01+length(dist2)), 2.0);
let dist3 = fragCoord - point3;
let intensity3 = pow(80.0/(0.01+length(dist3)), 1.0);
return vec3f((color1*intensity1 + color2*intensity2 + color3*intensity3)*modf(fragCoord.y/2.0).fract);
}
@fragment
fn main(
@location(0) uv: vec2f
) -> @location(0) vec4f {
let c3 = calcColor3(uv);
let c4 = vec4f(c3.xyz * color.xyz, color.w);
return c4;
}
此示例基于此渲染系统实现,当前示例TypeScript源码如下:
export class ScreenPostEffect {
private mRscene = new RendererScene();
initialize(): void {
console.log("ScreenPostEffect::initialize() ...");
const rc = this.mRscene;
rc.initialize();
this.initEvent();
this.initScene();
}
private initEvent(): void {
const rc = this.mRscene;
rc.addEventListener(MouseEvent.MOUSE_DOWN, this.mouseDown);
new MouseInteraction().initialize(rc, 0, false).setAutoRunning(true);
}
private mouseDown = (evt: MouseEvent): void => {};
private mColorV = new WGRUniformValue({ data: new Float32Array([1.0, 0.1, 0.2, 1.0]) });
private mParamsV = new WGRStorageValue({ data: new Float32Array(4 * 3), arrayStride: 16 });
private initScene(): void {
const rc = this.mRscene;
let param0 = this.mParamsV.data as Float32Array;
param0.set([1.0, 1.0, 1.0, 1.0]);
param0.set([0.5, 0.5, 512.0, 512.0], 4);
let shaderSrc = {
vertShaderSrc: { code: vertWGSL, uuid: "vert-screenPostEffect" },
fragShaderSrc: {
code: frag1WGSL,
uuid: "frag-screenPostEffect"
}
};
let uniformValues: WGRUniformValue[] = [this.mColorV, this.mParamsV];
let entity = new FixScreenPlaneEntity({ shaderSrc, uniformValues });
rc.addEntity(entity);
}
private mTime = 0;
run(): void {
let vs = this.mParamsV.data as Float32Array;
vs[8] += 0.01;
vs[4] = Math.cos(this.mTime) * 0.5 + 0.5;
this.mTime += 0.01;
this.mParamsV.upate();
this.mRscene.run();
}
}