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> { self.buf.slice(map_range(&self.vertex_range)) } 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, device: &wgpu::Device, queue: &wgpu::Queue, mesh: &Mesh, offset: usize, ) { self.vbuf.update(device, 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() } } fn map_range(range: &Range) -> Range { (range.start as u64)..(range.end as u64) }