pub mod cell; pub mod mat_cell; pub use mat_cell::Material; // Reexport pub use self::{ cell::{Cell, CellData}, mat_cell::MatCell, }; use self::cell::{GLOWY, SHINY}; use crate::{ vol::{IntoFullPosIterator, IntoFullVolIterator, ReadVol, SizedVol, Vox, WriteVol}, volumes::dyna::Dyna, }; use dot_vox::DotVoxData; use vek::*; /// A type representing a volume that may be part of an animated figure. /// /// Figures are used to represent things like characters, NPCs, mobs, etc. pub type Segment = Dyna; impl From<&DotVoxData> for Segment { fn from(dot_vox_data: &DotVoxData) -> Self { Segment::from_vox(dot_vox_data, false) } } impl Segment { /// Take a list of voxel data, offsets, and x-mirror flags, and assembled /// them into a combined segment pub fn from_voxes(data: &[(&DotVoxData, Vec3, bool)]) -> (Self, Vec3) { let mut union = DynaUnionizer::new(); for (datum, offset, xmirror) in data.iter() { union = union.add(Segment::from_vox(datum, *xmirror), *offset); } union.unify() } pub fn from_vox(dot_vox_data: &DotVoxData, flipped: bool) -> Self { if let Some(model) = dot_vox_data.models.get(0) { let palette = dot_vox_data .palette .iter() .map(|col| Rgba::from(col.to_ne_bytes()).into()) .collect::>(); let mut segment = Segment::filled( Vec3::new(model.size.x, model.size.y, model.size.z), Cell::empty(), (), ); for voxel in &model.voxels { if let Some(&color) = palette.get(voxel.i as usize) { segment .set( Vec3::new( if flipped { model.size.x as u8 - 1 - voxel.x } else { voxel.x }, voxel.y, voxel.z, ) .map(i32::from), Cell::new( color, (13..16).contains(&voxel.i), // Glowy (8..13).contains(&voxel.i), // Shiny ), ) .unwrap(); }; } segment } else { Segment::filled(Vec3::zero(), Cell::empty(), ()) } } /// Transform cells pub fn map(mut self, transform: impl Fn(Cell) -> Option) -> Self { for pos in self.full_pos_iter() { if let Some(new) = transform(*self.get(pos).unwrap()) { self.set(pos, new).unwrap(); } } self } /// Transform cell colors pub fn map_rgb(self, transform: impl Fn(Rgb) -> Rgb) -> Self { self.map(|cell| { cell.get_color() .map(|rgb| Cell::new(transform(rgb), cell.is_glowy(), cell.is_shiny())) }) } } // TODO: move /// A `Dyna` builder that combines Dynas pub struct DynaUnionizer(Vec<(Dyna, Vec3)>); impl DynaUnionizer { #[allow(clippy::new_without_default)] // TODO: Pending review in #587 pub fn new() -> Self { DynaUnionizer(Vec::new()) } pub fn add(mut self, dyna: Dyna, offset: Vec3) -> Self { self.0.push((dyna, offset)); self } pub fn maybe_add(self, maybe: Option<(Dyna, Vec3)>) -> Self { match maybe { Some((dyna, offset)) => self.add(dyna, offset), None => self, } } #[allow(clippy::neg_multiply)] // TODO: Pending review in #587 pub fn unify(self) -> (Dyna, Vec3) { if self.0.is_empty() { return (Dyna::filled(Vec3::zero(), V::empty(), ()), Vec3::zero()); } // Determine size of the new Dyna let mut min_point = self.0[0].1; let mut max_point = self.0[0].1 + self.0[0].0.size().map(|e| e as i32); for (dyna, offset) in self.0.iter().skip(1) { let size = dyna.size().map(|e| e as i32); min_point = min_point.map2(*offset, std::cmp::min); max_point = max_point.map2(offset + size, std::cmp::max); } let new_size = (max_point - min_point).map(|e| e as u32); // Allocate new segment let mut combined = Dyna::filled(new_size, V::empty(), ()); // Copy segments into combined let origin = min_point.map(|e| e * -1); for (dyna, offset) in self.0 { for (pos, vox) in dyna.full_vol_iter() { if !vox.is_empty() { combined.set(origin + offset + pos, *vox).unwrap(); } } } (combined, origin) } } pub type MatSegment = Dyna; impl MatSegment { pub fn to_segment(&self, map: impl Fn(Material) -> Rgb) -> Segment { let mut vol = Dyna::filled(self.size(), Cell::empty(), ()); for (pos, vox) in self.full_vol_iter() { let data = match vox { MatCell::None => continue, MatCell::Mat(mat) => CellData { col: map(*mat), attr: 0, }, MatCell::Normal(data) => *data, }; vol.set(pos, Cell::Filled(data)).unwrap(); } vol } /// Transform cells pub fn map(mut self, transform: impl Fn(MatCell) -> Option) -> Self { for pos in self.full_pos_iter() { if let Some(new) = transform(*self.get(pos).unwrap()) { self.set(pos, new).unwrap(); } } self } /// Transform cell colors pub fn map_rgb(self, transform: impl Fn(Rgb) -> Rgb) -> Self { self.map(|cell| match cell { MatCell::Normal(data) => Some(MatCell::Normal(CellData { col: transform(data.col), ..data })), _ => None, }) } #[allow(clippy::identity_op)] pub fn from_vox(dot_vox_data: &DotVoxData, flipped: bool) -> Self { if let Some(model) = dot_vox_data.models.get(0) { let palette = dot_vox_data .palette .iter() .map(|col| Rgba::from(col.to_ne_bytes()).into()) .collect::>(); let mut vol = Dyna::filled( Vec3::new(model.size.x, model.size.y, model.size.z), MatCell::empty(), (), ); for voxel in &model.voxels { let block = match voxel.i { 0 => MatCell::Mat(Material::Skin), 1 => MatCell::Mat(Material::Hair), 2 => MatCell::Mat(Material::EyeDark), 3 => MatCell::Mat(Material::EyeLight), 4 => MatCell::Mat(Material::SkinDark), 5 => MatCell::Mat(Material::SkinLight), 7 => MatCell::Mat(Material::EyeWhite), //6 => MatCell::Mat(Material::Clothing), index => { let color = palette .get(index as usize) .copied() .unwrap_or_else(|| Rgb::broadcast(0)); MatCell::Normal(CellData { col: color, attr: 0 | ((13..16).contains(&index) as u8 * GLOWY) | ((8..13).contains(&index) as u8 * SHINY), }) }, }; vol.set( Vec3::new( if flipped { model.size.x as u8 - 1 - voxel.x } else { voxel.x }, voxel.y, voxel.z, ) .map(i32::from), block, ) .unwrap(); } vol } else { Dyna::filled(Vec3::zero(), MatCell::empty(), ()) } } } impl From<&DotVoxData> for MatSegment { fn from(dot_vox_data: &DotVoxData) -> Self { Self::from_vox(dot_vox_data, false) } }