use crate::{ legion::{ borrow::RefMap, prelude::{Entity, World}, }, render::render_graph_2::{UniformPropertyType, BindType}, math::Vec4, }; use zerocopy::AsBytes; use legion::storage::Component; pub type ShaderUniformSelector = fn(Entity, &World) -> Option>; pub struct ShaderUniforms { // used for distinguishing pub uniform_selectors: Vec, } impl<'a> ShaderUniforms { pub fn new() -> Self { ShaderUniforms { uniform_selectors: Vec::new(), } } pub fn add(&mut self, selector: ShaderUniformSelector) { self.uniform_selectors.push(selector); } } pub struct StandardMaterial { pub albedo: Vec4, } pub trait GetBytes { fn get_bytes(&self) -> Vec; fn get_bytes_ref(&self) -> Option<&[u8]>; } // TODO: might need to add zerocopy to this crate to impl AsBytes for external crates // impl GetBytes for dyn AsBytes { // fn get_bytes(&self) -> Vec { // self.as_bytes().into() // } // fn get_bytes_ref(&self) -> Option<&[u8]> { // Some(self.as_bytes()) // } // } impl GetBytes for Vec4 { fn get_bytes(&self) -> Vec { let vec4_array: [f32; 4] = (*self).into(); vec4_array.as_bytes().into() } fn get_bytes_ref(&self) -> Option<&[u8]> { None } } pub trait AsUniforms { fn get_uniform_info(&self) -> &[UniformInfo]; fn get_uniform_layouts(&self) -> &[&[UniformPropertyType]]; fn get_uniform_value(&self, index: usize) -> Vec; // TODO: support zero-copy uniforms // fn get_uniform_value_ref(&self, index: usize) -> &[u8]; } // pub struct UniformInfo<'a> { // pub name: &'a str, // pub // } // create this from a derive macro const STANDARD_MATERIAL_UNIFORM_INFO: &[UniformInfo] = &[ UniformInfo { name: "StandardMaterial", bind_type: BindType::Uniform { dynamic: false, properties: Vec::new() }, } ]; // these are separate from BindType::Uniform{properties} because they need to be const const STANDARD_MATERIAL_UNIFORM_LAYOUTS: &[&[UniformPropertyType]] = &[&[]]; pub struct UniformInfo<'a> { pub name: &'a str, pub bind_type: BindType, } pub fn standard_material_selector(entity: Entity, world: &World) -> Option> where T: AsUniforms + Component { world.get_component::(entity).map( |c| { c.map_into(|s| { s as &dyn AsUniforms }) }) } // const ST impl AsUniforms for StandardMaterial { fn get_uniform_info(&self) -> &[UniformInfo] { STANDARD_MATERIAL_UNIFORM_INFO } fn get_uniform_layouts(&self) -> &[&[UniformPropertyType]] { STANDARD_MATERIAL_UNIFORM_LAYOUTS } fn get_uniform_value(&self, index: usize) -> Vec { match index { 0 => self.albedo.get_bytes(), _ => panic!("index out of bounds"), } } // fn iter_properties(&self) -> std::slice::Iter<&'static str> { // STANDARD_MATERIAL_PROPERTIES.iter() // } // fn get_property(&self, name: &str) -> Option { // match name { // "albedo" => Some(match self.albedo { // Albedo::Color(color) => ShaderValue::Vec4(color), // Albedo::Texture(ref texture) => ShaderValue::Texture(texture) // }), // _ => None, // } // } // fn get_selector(&self) -> ShaderMaterialSelector { // |entity, world| { // world.get_component::(entity).map( // |c: Ref| { // c.map_into(|s| { // s as &dyn ShaderMaterial // }) // } // ) // } // } }