use super::{ buffer::{Buffer, DynamicBuffer}, mesh::Mesh, Vertex, }; use std::ops::Range; /// Represents a mesh that has been sent to the GPU. pub struct SubModel<'a, V: Vertex> { pub vertex_range: Range, buf: &'a wgpu::Buffer, phantom_data: std::marker::PhantomData, } impl<'a, V: Vertex> SubModel<'a, V> { pub(super) fn buf(&self) -> wgpu::BufferSlice<'a> { let start = self.vertex_range.start as wgpu::BufferAddress * V::STRIDE; let end = self.vertex_range.end as wgpu::BufferAddress * V::STRIDE; self.buf.slice(start..end) } pub fn len(&self) -> u32 { self.vertex_range.end - self.vertex_range.start } } /// Represents a mesh that has been sent to the GPU. pub struct Model { vbuf: Buffer, } impl Model { pub fn new(device: &wgpu::Device, mesh: &Mesh) -> Self { Self { vbuf: Buffer::new(device, wgpu::BufferUsage::VERTEX, mesh.vertices()), } } /// Create a model with a slice of a portion of this model to send to the /// renderer. pub fn submodel(&self, vertex_range: Range) -> SubModel { SubModel { vertex_range, buf: self.buf(), phantom_data: std::marker::PhantomData, } } pub(super) fn buf(&self) -> &wgpu::Buffer { &self.vbuf.buf } pub fn len(&self) -> usize { self.vbuf.len() } } /// Represents a mesh that has been sent to the GPU. pub struct DynamicModel { vbuf: DynamicBuffer, } impl DynamicModel { pub fn new(device: &wgpu::Device, size: usize) -> Self { Self { vbuf: DynamicBuffer::new(device, size, wgpu::BufferUsage::VERTEX), } } pub fn update(&self, queue: &wgpu::Queue, mesh: &Mesh, offset: usize) { self.vbuf.update(queue, mesh.vertices(), offset) } /// Create a model with a slice of a portion of this model to send to the /// renderer. pub fn submodel(&self, vertex_range: Range) -> SubModel { SubModel { vertex_range, buf: self.buf(), phantom_data: std::marker::PhantomData, } } pub fn buf(&self) -> &wgpu::Buffer { &self.vbuf.buf } pub fn len(&self) -> usize { self.vbuf.len() } }