import { vec3, mat4 } from 'glm'; import * as WebGPU from 'engine/WebGPU.js'; import { Camera, Transform } from 'engine/core.js'; import { getLocalModelMatrix, getGlobalModelMatrix, getGlobalViewMatrix, getProjectionMatrix, getModels, } from 'engine/core/SceneUtils.js'; import { BaseRenderer } from 'engine/renderers/BaseRenderer.js'; import { Light } from './Light.js'; const vertexBufferLayout = { arrayStride: 32, attributes: [ { name: 'position', shaderLocation: 0, offset: 0, format: 'float32x3', }, { name: 'texcoords', shaderLocation: 1, offset: 12, format: 'float32x2', }, { name: 'normal', shaderLocation: 2, offset: 20, format: 'float32x3', }, ], }; export class Renderer extends BaseRenderer { constructor(canvas) { super(canvas); } async initialize() { await super.initialize(); const code = await fetch(new URL('shader.wgsl', import.meta.url)).then( (response) => response.text(), ); const module = this.device.createShaderModule({ code }); this.pipeline = await this.device.createRenderPipelineAsync({ layout: 'auto', vertex: { module, buffers: [vertexBufferLayout], }, fragment: { module, targets: [{ format: this.format }], }, depthStencil: { format: 'depth24plus', depthWriteEnabled: true, depthCompare: 'less', }, }); this.recreateDepthTexture(); } recreateDepthTexture() { this.depthTexture?.destroy(); this.depthTexture = this.device.createTexture({ format: 'depth24plus', size: [this.canvas.width, this.canvas.height], usage: GPUTextureUsage.RENDER_ATTACHMENT, }); } prepareNode(node) { if (this.gpuObjects.has(node)) { return this.gpuObjects.get(node); } const modelUniformBuffer = this.device.createBuffer({ label: 'modelUniformBuffer', size: 128, usage: GPUBufferUsage.UNIFORM | GPUBufferUsage.COPY_DST, }); const modelBindGroup = this.device.createBindGroup({ label: 'modelBindGroup', layout: this.pipeline.getBindGroupLayout(1), entries: [{ binding: 0, resource: { buffer: modelUniformBuffer } }], }); const gpuObjects = { modelUniformBuffer, modelBindGroup }; this.gpuObjects.set(node, gpuObjects); return gpuObjects; } prepareCamera(camera) { if (this.gpuObjects.has(camera)) { return this.gpuObjects.get(camera); } const cameraUniformBuffer = this.device.createBuffer({ label: 'cameraUniformBuffer', size: 144, usage: GPUBufferUsage.UNIFORM | GPUBufferUsage.COPY_DST, }); const cameraBindGroup = this.device.createBindGroup({ label: 'cameraBindGroup', layout: this.pipeline.getBindGroupLayout(0), entries: [ { binding: 0, resource: { buffer: cameraUniformBuffer } }, ], }); const gpuObjects = { cameraUniformBuffer, cameraBindGroup }; this.gpuObjects.set(camera, gpuObjects); return gpuObjects; } prepareLight(light) { if (this.gpuObjects.has(light)) { return this.gpuObjects.get(light); } const lightUniformBuffer = this.device.createBuffer({ label: 'lightUniformBuffer', size: 48, usage: GPUBufferUsage.UNIFORM | GPUBufferUsage.COPY_DST, }); const lightBindGroup = this.device.createBindGroup({ label: 'lightBindGroup', layout: this.pipeline.getBindGroupLayout(3), entries: [{ binding: 0, resource: { buffer: lightUniformBuffer } }], }); const gpuObjects = { lightUniformBuffer, lightBindGroup }; this.gpuObjects.set(light, gpuObjects); return gpuObjects; } prepareMaterial(material) { if (this.gpuObjects.has(material)) { return this.gpuObjects.get(material); } const baseTexture = this.prepareImage( material.baseTexture.image, ).gpuTexture; const baseSampler = this.prepareSampler( material.baseTexture.sampler, ).gpuSampler; const materialUniformBuffer = this.device.createBuffer({ size: 64, usage: GPUBufferUsage.UNIFORM | GPUBufferUsage.COPY_DST, }); const materialBindGroup = this.device.createBindGroup({ label: 'materialBindGroup', layout: this.pipeline.getBindGroupLayout(2), entries: [ { binding: 0, resource: { buffer: materialUniformBuffer } }, { binding: 1, resource: baseTexture.createView() }, { binding: 2, resource: baseSampler }, ], }); const gpuObjects = { materialUniformBuffer, materialBindGroup }; this.gpuObjects.set(material, gpuObjects); return gpuObjects; } calculateLightDirection(lightMatrix) { // const lightDirection = mat4.getTranslation(vec3.create(), lightMatrix); const lightDirection = vec3.fromValues( lightMatrix[4], lightMatrix[5], lightMatrix[6], ); vec3.negate(lightDirection, lightDirection); vec3.normalize(lightDirection, lightDirection); return lightDirection; } render(scene, camera) { if ( this.depthTexture.width !== this.canvas.width || this.depthTexture.height !== this.canvas.height ) { this.recreateDepthTexture(); } const encoder = this.device.createCommandEncoder(); this.renderPass = encoder.beginRenderPass({ colorAttachments: [ { view: this.context.getCurrentTexture().createView(), clearValue: [1, 1, 1, 1], loadOp: 'clear', storeOp: 'store', }, ], depthStencilAttachment: { view: this.depthTexture.createView(), depthClearValue: 1, depthLoadOp: 'clear', depthStoreOp: 'discard', }, }); this.renderPass.setPipeline(this.pipeline); const cameraComponent = camera.getComponentOfType(Camera); const cameraMatrix = getGlobalModelMatrix(camera); const cameraPosition = mat4.getTranslation(vec3.create(), cameraMatrix); const viewMatrix = getGlobalViewMatrix(camera); const projectionMatrix = getProjectionMatrix(camera); const { cameraUniformBuffer, cameraBindGroup } = this.prepareCamera(cameraComponent); this.device.queue.writeBuffer(cameraUniformBuffer, 0, viewMatrix); this.device.queue.writeBuffer( cameraUniformBuffer, 64, projectionMatrix, ); this.device.queue.writeBuffer(cameraUniformBuffer, 128, cameraPosition); this.renderPass.setBindGroup(0, cameraBindGroup); const light = scene.find((node) => node.getComponentOfType(Light)); const lightComponent = light.getComponentOfType(Light); const lightMatrix = getGlobalModelMatrix(light); const lightPosition = mat4.getTranslation(vec3.create(), lightMatrix); const lightDirection = this.calculateLightDirection(lightMatrix); const { lightUniformBuffer, lightBindGroup } = this.prepareLight(lightComponent); this.device.queue.writeBuffer(lightUniformBuffer, 0, lightPosition); this.device.queue.writeBuffer(lightUniformBuffer, 12, lightDirection); this.device.queue.writeBuffer( lightUniformBuffer, 24, new Float32Array([ 1.0, Math.cos(lightComponent.cutoffAngle), lightComponent.ambient, lightComponent.intensity, ]), ); this.renderPass.setBindGroup(3, lightBindGroup); this.renderNode(scene); this.renderPass.end(); this.device.queue.submit([encoder.finish()]); } renderNode(node, modelMatrix = mat4.create()) { const localMatrix = getLocalModelMatrix(node); modelMatrix = mat4.multiply(mat4.create(), modelMatrix, localMatrix); const { modelUniformBuffer, modelBindGroup } = this.prepareNode(node); const normalMatrix = mat4.normalFromMat4(mat4.create(), modelMatrix); this.device.queue.writeBuffer(modelUniformBuffer, 0, modelMatrix); this.device.queue.writeBuffer(modelUniformBuffer, 64, normalMatrix); this.renderPass.setBindGroup(1, modelBindGroup); for (const model of getModels(node)) { this.renderModel(model); } for (const child of node.children) { this.renderNode(child, modelMatrix); } } renderModel(model) { for (const primitive of model.primitives) { this.renderPrimitive(primitive); } } renderPrimitive(primitive) { const { materialUniformBuffer, materialBindGroup } = this.prepareMaterial(primitive.material); this.device.queue.writeBuffer( materialUniformBuffer, 0, new Float32Array(primitive.material.baseFactor), ); this.device.queue.writeBuffer( materialUniformBuffer, 16, new Float32Array([1, 1]), ); this.renderPass.setBindGroup(2, materialBindGroup); const { vertexBuffer, indexBuffer } = this.prepareMesh( primitive.mesh, vertexBufferLayout, ); this.renderPass.setVertexBuffer(0, vertexBuffer); this.renderPass.setIndexBuffer(indexBuffer, 'uint32'); this.renderPass.drawIndexed(primitive.mesh.indices.length); } }