bevy/src/render/render_graph_2/uniform.rs

133 lines
3.4 KiB
Rust
Raw Normal View History

2020-01-23 08:31:56 +00:00
use crate::{
legion::{
2020-01-24 07:39:56 +00:00
borrow::RefMap,
2020-01-23 08:31:56 +00:00
prelude::{Entity, World},
},
math::Vec4,
render::render_graph_2::{BindType, UniformPropertyType},
2020-01-23 08:31:56 +00:00
};
use legion::storage::Component;
2020-02-08 07:17:51 +00:00
use zerocopy::AsBytes;
2020-01-18 22:09:53 +00:00
2020-01-23 08:31:56 +00:00
pub trait GetBytes {
fn get_bytes(&self) -> Vec<u8>;
fn get_bytes_ref(&self) -> Option<&[u8]>;
2020-01-18 22:09:53 +00:00
}
2020-01-24 07:39:56 +00:00
// TODO: might need to add zerocopy to this crate to impl AsBytes for external crates
// impl<T> GetBytes for T where T : AsBytes {
2020-01-24 07:39:56 +00:00
// fn get_bytes(&self) -> Vec<u8> {
// self.as_bytes().into()
// }
2020-01-18 22:09:53 +00:00
2020-01-24 07:39:56 +00:00
// fn get_bytes_ref(&self) -> Option<&[u8]> {
// Some(self.as_bytes())
// }
// }
2020-01-18 22:09:53 +00:00
2020-01-23 08:31:56 +00:00
impl GetBytes for Vec4 {
fn get_bytes(&self) -> Vec<u8> {
let vec4_array: [f32; 4] = (*self).into();
vec4_array.as_bytes().into()
}
fn get_bytes_ref(&self) -> Option<&[u8]> {
None
}
2020-01-18 22:09:53 +00:00
}
2020-01-23 08:31:56 +00:00
pub trait AsUniforms {
fn get_uniform_infos(&self) -> &[UniformInfo];
fn get_uniform_info(&self, name: &str) -> Option<&UniformInfo>;
2020-01-23 08:31:56 +00:00
fn get_uniform_layouts(&self) -> &[&[UniformPropertyType]];
fn get_uniform_bytes(&self, name: &str) -> Option<Vec<u8>>;
2020-01-23 08:31:56 +00:00
// TODO: support zero-copy uniforms
// fn get_uniform_bytes_ref(&self, name: &str) -> Option<&[u8]>;
2020-01-18 22:09:53 +00:00
}
2020-01-23 08:31:56 +00:00
// pub struct UniformInfo<'a> {
// pub name: &'a str,
// pub
2020-01-23 08:31:56 +00:00
// }
pub struct UniformInfo<'a> {
pub name: &'a str,
pub bind_type: BindType,
}
pub fn uniform_selector<T>(entity: Entity, world: &World) -> Option<RefMap<&dyn AsUniforms>>
where
T: AsUniforms + Component,
{
world
.get_component::<T>(entity)
.map(|c| c.map_into(|s| s as &dyn AsUniforms))
}
2020-01-23 08:31:56 +00:00
2020-02-09 01:17:07 +00:00
// TODO: Remove these
2020-01-23 08:31:56 +00:00
2020-02-09 01:17:07 +00:00
pub type ShaderUniformSelector = fn(Entity, &World) -> Option<RefMap<&dyn AsUniforms>>;
2020-01-23 08:31:56 +00:00
2020-02-09 01:17:07 +00:00
pub struct ShaderUniforms {
// used for distinguishing
pub uniform_selectors: Vec<ShaderUniformSelector>,
}
2020-01-23 08:31:56 +00:00
2020-02-09 01:17:07 +00:00
impl ShaderUniforms {
pub fn new() -> Self {
ShaderUniforms {
uniform_selectors: Vec::new(),
}
}
2020-02-09 01:17:07 +00:00
pub fn add(&mut self, selector: ShaderUniformSelector) {
self.uniform_selectors.push(selector);
}
2020-02-09 01:17:07 +00:00
pub fn get_uniform_info<'a>(
&'a self,
world: &'a World,
entity: Entity,
uniform_name: &str,
) -> Option<&'a UniformInfo> {
for uniform_selector in self.uniform_selectors.iter().rev() {
let uniforms = uniform_selector(entity, world).unwrap_or_else(|| {
panic!(
"ShaderUniform selector points to a missing component. Uniform: {}",
uniform_name
)
});
2020-02-09 01:17:07 +00:00
let info = uniforms.get_uniform_info(uniform_name);
if let Some(_) = info {
return info;
}
}
2020-02-09 01:17:07 +00:00
None
}
2020-02-09 01:17:07 +00:00
pub fn get_uniform_bytes<'a>(
&'a self,
world: &'a World,
entity: Entity,
uniform_name: &str,
) -> Option<Vec<u8>> {
for uniform_selector in self.uniform_selectors.iter().rev() {
let uniforms = uniform_selector(entity, world).unwrap_or_else(|| {
panic!(
"ShaderUniform selector points to a missing component. Uniform: {}",
uniform_name
)
});
2020-02-09 01:17:07 +00:00
let bytes = uniforms.get_uniform_bytes(uniform_name);
if let Some(_) = bytes {
return bytes;
}
}
2020-02-09 01:17:07 +00:00
None
2020-01-18 22:09:53 +00:00
}
}