use crate::{ mesh::Meshable, render::{ Consts, FluidPipeline, Globals, Instances, Light, Mesh, Model, Renderer, Shadow, SpriteInstance, SpritePipeline, TerrainLocals, TerrainPipeline, Texture, }, }; use super::SceneData; use common::{ assets, figure::Segment, spiral::Spiral2d, terrain::{Block, BlockKind, TerrainChunk}, vol::{BaseVol, ReadVol, RectRasterableVol, SampleVol, Vox}, volumes::vol_grid_2d::{VolGrid2d, VolGrid2dError}, }; use crossbeam::channel; use dot_vox::DotVoxData; use hashbrown::HashMap; use std::{f32, fmt::Debug, i32, marker::PhantomData, time::Duration}; use treeculler::{BVol, Frustum, AABB}; use vek::*; struct TerrainChunkData { // GPU data load_time: f32, opaque_model: Model, fluid_model: Option>, sprite_instances: HashMap<(BlockKind, usize), Instances>, locals: Consts, visible: bool, z_bounds: (f32, f32), frustum_last_plane_index: u8, } struct ChunkMeshState { pos: Vec2, started_tick: u64, active_worker: Option, } /// A type produced by mesh worker threads corresponding to the position and /// mesh of a chunk. struct MeshWorkerResponse { pos: Vec2, z_bounds: (f32, f32), opaque_mesh: Mesh, fluid_mesh: Mesh, sprite_instances: HashMap<(BlockKind, usize), Vec>, started_tick: u64, } struct SpriteConfig { variations: usize, wind_sway: f32, // 1.0 is normal } fn sprite_config_for(kind: BlockKind) -> Option { match kind { BlockKind::Window1 => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::Window2 => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::Window3 => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::Window4 => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::LargeCactus => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::BarrelCactus => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::RoundCactus => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::ShortCactus => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::MedFlatCactus => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::ShortFlatCactus => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::BlueFlower => Some(SpriteConfig { variations: 10, wind_sway: 0.1, }), BlockKind::PinkFlower => Some(SpriteConfig { variations: 4, wind_sway: 0.1, }), BlockKind::PurpleFlower => Some(SpriteConfig { variations: 8, wind_sway: 0.1, }), BlockKind::RedFlower => Some(SpriteConfig { variations: 5, wind_sway: 0.1, }), BlockKind::WhiteFlower => Some(SpriteConfig { variations: 5, wind_sway: 0.1, }), BlockKind::YellowFlower => Some(SpriteConfig { variations: 2, wind_sway: 0.1, }), BlockKind::Sunflower => Some(SpriteConfig { variations: 2, wind_sway: 0.1, }), BlockKind::LongGrass => Some(SpriteConfig { variations: 7, wind_sway: 0.8, }), BlockKind::MediumGrass => Some(SpriteConfig { variations: 5, wind_sway: 0.5, }), BlockKind::ShortGrass => Some(SpriteConfig { variations: 5, wind_sway: 0.1, }), BlockKind::LargeGrass => Some(SpriteConfig { variations: 3, wind_sway: 0.5, }), BlockKind::Apple => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::Mushroom => Some(SpriteConfig { variations: 17, wind_sway: 0.0, }), BlockKind::Liana => Some(SpriteConfig { variations: 2, wind_sway: 0.05, }), BlockKind::Velorite => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::VeloriteFrag => Some(SpriteConfig { variations: 10, wind_sway: 0.0, }), BlockKind::Chest => Some(SpriteConfig { variations: 4, wind_sway: 0.0, }), BlockKind::Welwitch => Some(SpriteConfig { variations: 1, wind_sway: 0.1, }), BlockKind::Pumpkin => Some(SpriteConfig { variations: 7, wind_sway: 0.0, }), BlockKind::LingonBerry => Some(SpriteConfig { variations: 3, wind_sway: 0.0, }), BlockKind::LeafyPlant => Some(SpriteConfig { variations: 10, wind_sway: 0.4, }), BlockKind::Fern => Some(SpriteConfig { variations: 13, wind_sway: 0.4, }), BlockKind::DeadBush => Some(SpriteConfig { variations: 4, wind_sway: 0.1, }), BlockKind::Ember => Some(SpriteConfig { variations: 1, wind_sway: 0.8, }), BlockKind::Corn => Some(SpriteConfig { variations: 6, wind_sway: 0.4, }), BlockKind::WheatYellow => Some(SpriteConfig { variations: 10, wind_sway: 0.4, }), BlockKind::WheatGreen => Some(SpriteConfig { variations: 10, wind_sway: 0.4, }), BlockKind::Cabbage => Some(SpriteConfig { variations: 3, wind_sway: 0.0, }), BlockKind::Flax => Some(SpriteConfig { variations: 6, wind_sway: 0.4, }), BlockKind::Carrot => Some(SpriteConfig { variations: 6, wind_sway: 0.1, }), BlockKind::Tomato => Some(SpriteConfig { variations: 5, wind_sway: 0.0, }), BlockKind::Radish => Some(SpriteConfig { variations: 5, wind_sway: 0.1, }), BlockKind::Turnip => Some(SpriteConfig { variations: 6, wind_sway: 0.1, }), BlockKind::Coconut => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::Scarecrow => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::StreetLamp => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::StreetLampTall => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::Door => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::Bed => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::Bench => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::ChairSingle => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::ChairDouble => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::CoatRack => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::Crate => Some(SpriteConfig { variations: 7, wind_sway: 0.0, }), BlockKind::DrawerLarge => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::DrawerMedium => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::DrawerSmall => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::DungeonWallDecor => Some(SpriteConfig { variations: 10, wind_sway: 0.0, }), BlockKind::HangingBasket => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::HangingSign => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::WallLamp => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::Planter => Some(SpriteConfig { variations: 7, wind_sway: 0.0, }), BlockKind::Shelf => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::TableSide => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::TableDining => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::TableDouble => Some(SpriteConfig { variations: 1, wind_sway: 0.0, }), BlockKind::WardrobeDouble => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::WardrobeSingle => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::Pot => Some(SpriteConfig { variations: 2, wind_sway: 0.0, }), BlockKind::Stones => Some(SpriteConfig { variations: 3, wind_sway: 0.0, }), BlockKind::Twigs => Some(SpriteConfig { variations: 3, wind_sway: 0.0, }), BlockKind::ShinyGem => Some(SpriteConfig { variations: 3, wind_sway: 0.0, }), _ => None, } } /// Function executed by worker threads dedicated to chunk meshing. #[allow(clippy::or_fun_call)] // TODO: Pending review in #587 fn mesh_worker + RectRasterableVol + ReadVol + Debug>( pos: Vec2, z_bounds: (f32, f32), started_tick: u64, volume: as SampleVol>>::Sample, range: Aabb, ) -> MeshWorkerResponse { let (opaque_mesh, fluid_mesh) = volume.generate_mesh(range); MeshWorkerResponse { pos, z_bounds, opaque_mesh, fluid_mesh, // Extract sprite locations from volume sprite_instances: { let mut instances = HashMap::new(); for x in 0..V::RECT_SIZE.x as i32 { for y in 0..V::RECT_SIZE.y as i32 { for z in z_bounds.0 as i32..z_bounds.1 as i32 + 1 { let wpos = Vec3::from(pos * V::RECT_SIZE.map(|e: u32| e as i32)) + Vec3::new(x, y, z); let block = volume.get(wpos).ok().copied().unwrap_or(Block::empty()); if let Some(cfg) = sprite_config_for(block.kind()) { let seed = wpos.x as u64 * 3 + wpos.y as u64 * 7 + wpos.x as u64 * wpos.y as u64; // Awful PRNG let ori = block.get_ori().unwrap_or((seed % 4) as u8 * 2); let instance = SpriteInstance::new( Mat4::identity() .rotated_z(f32::consts::PI * 0.25 * ori as f32) .translated_3d( wpos.map(|e| e as f32) + Vec3::new(0.5, 0.5, 0.0), ), Rgb::broadcast(1.0), cfg.wind_sway, ); instances .entry((block.kind(), seed as usize % cfg.variations)) .or_insert_with(Vec::new) .push(instance); } } } } instances }, started_tick, } } pub struct Terrain { chunks: HashMap, TerrainChunkData>, // The mpsc sender and receiver used for talking to meshing worker threads. // We keep the sender component for no reason other than to clone it and send it to new // workers. mesh_send_tmp: channel::Sender, mesh_recv: channel::Receiver, mesh_todo: HashMap, ChunkMeshState>, // GPU data sprite_models: HashMap<(BlockKind, usize), Vec>>, waves: Texture, phantom: PhantomData, } impl Terrain { #[allow(clippy::float_cmp)] // TODO: Pending review in #587 pub fn new(renderer: &mut Renderer) -> Self { // Create a new mpsc (Multiple Produced, Single Consumer) pair for communicating // with worker threads that are meshing chunks. let (send, recv) = channel::unbounded(); let mut make_models = |s, offset, lod_axes: Vec3| { let scaled = [1.0, 0.8, 0.6, 0.4, 0.2]; scaled .iter() .map(|lod_scale| { if *lod_scale == 1.0 { Vec3::broadcast(1.0) } else { lod_axes * *lod_scale + lod_axes.map(|e| if e == 0.0 { 1.0 } else { 0.0 }) } }) .map(|lod_scale| { renderer .create_model( &Meshable::::generate_mesh( &Segment::from(assets::load_expect::(s).as_ref()) .scaled_by(lod_scale), (offset * lod_scale, Vec3::one() / lod_scale), ) .0, ) .unwrap() }) .collect::>() }; Self { chunks: HashMap::default(), mesh_send_tmp: send, mesh_recv: recv, mesh_todo: HashMap::default(), sprite_models: vec![ // Windows ( (BlockKind::Window1, 0), make_models( "voxygen.voxel.sprite.window.window-0", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Window2, 0), make_models( "voxygen.voxel.sprite.window.window-1", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Window3, 0), make_models( "voxygen.voxel.sprite.window.window-2", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Window4, 0), make_models( "voxygen.voxel.sprite.window.window-3", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), // Cacti ( (BlockKind::LargeCactus, 0), make_models( "voxygen.voxel.sprite.cacti.large_cactus", Vec3::new(-13.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::LargeCactus, 1), make_models( "voxygen.voxel.sprite.cacti.tall", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::BarrelCactus, 0), make_models( "voxygen.voxel.sprite.cacti.barrel_cactus", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::RoundCactus, 0), make_models( "voxygen.voxel.sprite.cacti.cactus_round", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::ShortCactus, 0), make_models( "voxygen.voxel.sprite.cacti.cactus_short", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::MedFlatCactus, 0), make_models( "voxygen.voxel.sprite.cacti.flat_cactus_med", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::ShortFlatCactus, 0), make_models( "voxygen.voxel.sprite.cacti.flat_cactus_short", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), // Fruit ( (BlockKind::Apple, 0), make_models( "voxygen.voxel.sprite.fruit.apple", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), // Flowers ( (BlockKind::BlueFlower, 0), make_models( "voxygen.voxel.sprite.flowers.flower_blue_1", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::BlueFlower, 1), make_models( "voxygen.voxel.sprite.flowers.flower_blue_2", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::BlueFlower, 2), make_models( "voxygen.voxel.sprite.flowers.flower_blue_3", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::BlueFlower, 3), make_models( "voxygen.voxel.sprite.flowers.flower_blue_4", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::BlueFlower, 4), make_models( "voxygen.voxel.sprite.flowers.flower_blue_5", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::BlueFlower, 5), make_models( "voxygen.voxel.sprite.flowers.flower_blue_6", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::BlueFlower, 6), make_models( "voxygen.voxel.sprite.flowers.flower_blue_7", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::BlueFlower, 7), make_models( "voxygen.voxel.sprite.flowers.flower_blue-8", Vec3::new(-5.5, -4.5, 0.0), Vec3::one(), ), ), ( (BlockKind::BlueFlower, 8), make_models( "voxygen.voxel.sprite.flowers.flower_blue-9", Vec3::new(-4.0, -3.0, 0.0), Vec3::one(), ), ), ( (BlockKind::BlueFlower, 9), make_models( "voxygen.voxel.sprite.flowers.flower_blue-10", Vec3::new(-1.5, -1.5, 0.0), Vec3::one(), ), ), ( (BlockKind::PinkFlower, 0), make_models( "voxygen.voxel.sprite.flowers.flower_pink_1", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::PinkFlower, 1), make_models( "voxygen.voxel.sprite.flowers.flower_pink_2", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::PinkFlower, 2), make_models( "voxygen.voxel.sprite.flowers.flower_pink_3", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::PinkFlower, 3), make_models( "voxygen.voxel.sprite.flowers.flower_pink_4", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::PurpleFlower, 0), make_models( "voxygen.voxel.sprite.flowers.flower_purple_1", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::PurpleFlower, 1), make_models( "voxygen.voxel.sprite.flowers.flower_purple-2", Vec3::new(-5.0, -2.5, 0.0), Vec3::one(), ), ), ( (BlockKind::PurpleFlower, 2), make_models( "voxygen.voxel.sprite.flowers.flower_purple-3", Vec3::new(-3.5, -2.5, 0.0), Vec3::one(), ), ), ( (BlockKind::PurpleFlower, 3), make_models( "voxygen.voxel.sprite.flowers.flower_purple-4", Vec3::new(-5.0, -4.5, 0.0), Vec3::one(), ), ), ( (BlockKind::PurpleFlower, 4), make_models( "voxygen.voxel.sprite.flowers.flower_purple-5", Vec3::new(-2.5, -2.5, 0.0), Vec3::one(), ), ), ( (BlockKind::PurpleFlower, 5), make_models( "voxygen.voxel.sprite.flowers.flower_purple-6", Vec3::new(-4.5, -4.5, 0.0), Vec3::one(), ), ), ( (BlockKind::PurpleFlower, 6), make_models( "voxygen.voxel.sprite.flowers.flower_purple-7", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::PurpleFlower, 7), make_models( "voxygen.voxel.sprite.flowers.flower_purple-8", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::RedFlower, 0), make_models( "voxygen.voxel.sprite.flowers.flower_red_1", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::RedFlower, 1), make_models( "voxygen.voxel.sprite.flowers.flower_red_2", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::RedFlower, 2), make_models( "voxygen.voxel.sprite.flowers.flower_red_3", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::RedFlower, 3), make_models( "voxygen.voxel.sprite.flowers.flower_red-4", Vec3::new(-6.5, -6.5, 0.0), Vec3::one(), ), ), ( (BlockKind::RedFlower, 4), make_models( "voxygen.voxel.sprite.flowers.flower_red-5", Vec3::new(-3.5, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::WhiteFlower, 0), make_models( "voxygen.voxel.sprite.flowers.flower_white_1", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::WhiteFlower, 1), make_models( "voxygen.voxel.sprite.flowers.flower_white_2", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::WhiteFlower, 2), make_models( "voxygen.voxel.sprite.flowers.flower_white-3", Vec3::new(-1.5, -1.5, 0.0), Vec3::one(), ), ), ( (BlockKind::WhiteFlower, 3), make_models( "voxygen.voxel.sprite.flowers.flower_white-4", Vec3::new(-5.0, -4.5, 0.0), Vec3::one(), ), ), ( (BlockKind::WhiteFlower, 4), make_models( "voxygen.voxel.sprite.flowers.flower_white-5", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::YellowFlower, 0), make_models( "voxygen.voxel.sprite.flowers.flower_yellow-1", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::YellowFlower, 1), make_models( "voxygen.voxel.sprite.flowers.flower_yellow-0", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Sunflower, 0), make_models( "voxygen.voxel.sprite.flowers.sunflower_1", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Sunflower, 1), make_models( "voxygen.voxel.sprite.flowers.sunflower_2", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), // Grass ( (BlockKind::LargeGrass, 0), make_models( "voxygen.voxel.sprite.grass.grass_large-0", Vec3::new(-2.0, -2.5, 0.0), Vec3::one(), ), ), ( (BlockKind::LongGrass, 1), make_models( "voxygen.voxel.sprite.grass.grass_large-1", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::LongGrass, 2), make_models( "voxygen.voxel.sprite.grass.grass_large-2", Vec3::new(-5.5, -5.0, 0.0), Vec3::one(), ), ), ( (BlockKind::LongGrass, 0), make_models( "voxygen.voxel.sprite.grass.grass_long_1", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::LongGrass, 1), make_models( "voxygen.voxel.sprite.grass.grass_long_2", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::LongGrass, 2), make_models( "voxygen.voxel.sprite.grass.grass_long_3", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::LongGrass, 3), make_models( "voxygen.voxel.sprite.grass.grass_long_4", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::LongGrass, 4), make_models( "voxygen.voxel.sprite.grass.grass_long_5", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::LongGrass, 5), make_models( "voxygen.voxel.sprite.grass.grass_long_6", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::LongGrass, 6), make_models( "voxygen.voxel.sprite.grass.grass_long_7", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::MediumGrass, 0), make_models( "voxygen.voxel.sprite.grass.grass_med_1", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::MediumGrass, 1), make_models( "voxygen.voxel.sprite.grass.grass_med_2", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::MediumGrass, 2), make_models( "voxygen.voxel.sprite.grass.grass_med_3", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::MediumGrass, 3), make_models( "voxygen.voxel.sprite.grass.grass_med_4", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::MediumGrass, 4), make_models( "voxygen.voxel.sprite.grass.grass_med_5", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::ShortGrass, 0), make_models( "voxygen.voxel.sprite.grass.grass_short_1", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::ShortGrass, 1), make_models( "voxygen.voxel.sprite.grass.grass_short_2", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::ShortGrass, 2), make_models( "voxygen.voxel.sprite.grass.grass_short_3", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::ShortGrass, 3), make_models( "voxygen.voxel.sprite.grass.grass_short_4", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::ShortGrass, 4), make_models( "voxygen.voxel.sprite.grass.grass_short_5", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 0), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-0", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 1), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-1", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 2), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-2", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 3), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-3", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 4), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-4", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 5), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-5", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 6), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-6", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 7), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-7", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 8), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-8", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 9), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-9", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 10), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-10", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 11), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-11", Vec3::new(-8.0, -8.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 12), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-12", Vec3::new(-5.0, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 13), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-13", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 14), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-14", Vec3::new(-2.5, -2.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 15), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-15", Vec3::new(-1.5, -1.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Mushroom, 16), make_models( "voxygen.voxel.sprite.mushrooms.mushroom-16", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Liana, 0), make_models( "voxygen.voxel.sprite.lianas.liana-0", Vec3::new(-1.5, -0.5, -88.0), Vec3::unit_z() * 0.5, ), ), ( (BlockKind::Liana, 1), make_models( "voxygen.voxel.sprite.lianas.liana-1", Vec3::new(-1.0, -0.5, -55.0), Vec3::unit_z() * 0.5, ), ), ( (BlockKind::Velorite, 0), make_models( "voxygen.voxel.sprite.velorite.velorite_ore", Vec3::new(-5.0, -5.0, -5.0), Vec3::one(), ), ), ( (BlockKind::VeloriteFrag, 0), make_models( "voxygen.voxel.sprite.velorite.velorite_1", Vec3::new(-3.0, -5.0, 0.0), Vec3::one(), ), ), ( (BlockKind::VeloriteFrag, 1), make_models( "voxygen.voxel.sprite.velorite.velorite_2", Vec3::new(-3.0, -5.0, 0.0), Vec3::one(), ), ), ( (BlockKind::VeloriteFrag, 2), make_models( "voxygen.voxel.sprite.velorite.velorite_3", Vec3::new(-3.0, -5.0, 0.0), Vec3::one(), ), ), ( (BlockKind::VeloriteFrag, 3), make_models( "voxygen.voxel.sprite.velorite.velorite_4", Vec3::new(-3.0, -5.0, 0.0), Vec3::one(), ), ), ( (BlockKind::VeloriteFrag, 4), make_models( "voxygen.voxel.sprite.velorite.velorite_5", Vec3::new(-3.0, -5.0, 0.0), Vec3::one(), ), ), ( (BlockKind::VeloriteFrag, 5), make_models( "voxygen.voxel.sprite.velorite.velorite_6", Vec3::new(-3.0, -5.0, 0.0), Vec3::one(), ), ), ( (BlockKind::VeloriteFrag, 6), make_models( "voxygen.voxel.sprite.velorite.velorite_7", Vec3::new(-3.0, -5.0, 0.0), Vec3::one(), ), ), ( (BlockKind::VeloriteFrag, 7), make_models( "voxygen.voxel.sprite.velorite.velorite_8", Vec3::new(-3.0, -5.0, 0.0), Vec3::one(), ), ), ( (BlockKind::VeloriteFrag, 8), make_models( "voxygen.voxel.sprite.velorite.velorite_9", Vec3::new(-3.0, -5.0, 0.0), Vec3::one(), ), ), ( (BlockKind::VeloriteFrag, 9), make_models( "voxygen.voxel.sprite.velorite.velorite_10", Vec3::new(-3.0, -5.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Chest, 0), make_models( "voxygen.voxel.sprite.chests.chest", Vec3::new(-7.0, -5.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Chest, 1), make_models( "voxygen.voxel.sprite.chests.chest_gold", Vec3::new(-7.0, -5.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Chest, 2), make_models( "voxygen.voxel.sprite.chests.chest_dark", Vec3::new(-7.0, -5.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Chest, 3), make_models( "voxygen.voxel.sprite.chests.chest_vines", Vec3::new(-7.0, -5.0, -0.0), Vec3::one(), ), ), //Welwitch ( (BlockKind::Welwitch, 0), make_models( "voxygen.voxel.sprite.welwitch.1", Vec3::new(-15.0, -17.0, -0.0), Vec3::unit_z() * 0.7, ), ), //Pumpkins ( (BlockKind::Pumpkin, 0), make_models( "voxygen.voxel.sprite.pumpkin.1", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Pumpkin, 1), make_models( "voxygen.voxel.sprite.pumpkin.2", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Pumpkin, 2), make_models( "voxygen.voxel.sprite.pumpkin.3", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Pumpkin, 3), make_models( "voxygen.voxel.sprite.pumpkin.4", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Pumpkin, 4), make_models( "voxygen.voxel.sprite.pumpkin.5", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Pumpkin, 5), make_models( "voxygen.voxel.sprite.pumpkin.6", Vec3::new(-7.0, -6.5, -0.0), Vec3::one(), ), ), ( (BlockKind::Pumpkin, 6), make_models( "voxygen.voxel.sprite.pumpkin.7", Vec3::new(-7.0, -9.5, -0.0), Vec3::one(), ), ), //Lingonberries ( (BlockKind::LingonBerry, 0), make_models( "voxygen.voxel.sprite.lingonberry.1", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::LingonBerry, 1), make_models( "voxygen.voxel.sprite.lingonberry.2", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::LingonBerry, 2), make_models( "voxygen.voxel.sprite.lingonberry.3", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), // Leafy Plants ( (BlockKind::LeafyPlant, 0), make_models( "voxygen.voxel.sprite.leafy_plant.1", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::LeafyPlant, 1), make_models( "voxygen.voxel.sprite.leafy_plant.2", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::LeafyPlant, 2), make_models( "voxygen.voxel.sprite.leafy_plant.3", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::LeafyPlant, 3), make_models( "voxygen.voxel.sprite.leafy_plant.4", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::LeafyPlant, 4), make_models( "voxygen.voxel.sprite.leafy_plant.5", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::LeafyPlant, 5), make_models( "voxygen.voxel.sprite.leafy_plant.6", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::LeafyPlant, 6), make_models( "voxygen.voxel.sprite.leafy_plant.7", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::LeafyPlant, 7), make_models( "voxygen.voxel.sprite.leafy_plant.8", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::LeafyPlant, 8), make_models( "voxygen.voxel.sprite.leafy_plant.9", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::LeafyPlant, 9), make_models( "voxygen.voxel.sprite.leafy_plant.10", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), // Ferns ( (BlockKind::Fern, 0), make_models( "voxygen.voxel.sprite.ferns.1", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Fern, 1), make_models( "voxygen.voxel.sprite.ferns.2", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Fern, 2), make_models( "voxygen.voxel.sprite.ferns.3", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Fern, 3), make_models( "voxygen.voxel.sprite.ferns.4", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Fern, 4), make_models( "voxygen.voxel.sprite.ferns.5", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Fern, 5), make_models( "voxygen.voxel.sprite.ferns.6", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Fern, 6), make_models( "voxygen.voxel.sprite.ferns.7", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Fern, 7), make_models( "voxygen.voxel.sprite.ferns.8", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Fern, 8), make_models( "voxygen.voxel.sprite.ferns.9", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Fern, 9), make_models( "voxygen.voxel.sprite.ferns.10", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Fern, 10), make_models( "voxygen.voxel.sprite.ferns.11", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Fern, 11), make_models( "voxygen.voxel.sprite.ferns.12", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Fern, 12), make_models( "voxygen.voxel.sprite.ferns.fern-0", Vec3::new(-6.5, -11.5, 0.0), Vec3::unit_z(), ), ), // Dead Bush ( (BlockKind::DeadBush, 0), make_models( "voxygen.voxel.sprite.dead_bush.1", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::DeadBush, 1), make_models( "voxygen.voxel.sprite.dead_bush.2", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::DeadBush, 2), make_models( "voxygen.voxel.sprite.dead_bush.3", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::DeadBush, 3), make_models( "voxygen.voxel.sprite.dead_bush.4", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), // Blueberries ( (BlockKind::Blueberry, 0), make_models( "voxygen.voxel.sprite.blueberry.1", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Blueberry, 1), make_models( "voxygen.voxel.sprite.blueberry.2", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Blueberry, 2), make_models( "voxygen.voxel.sprite.blueberry.3", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Blueberry, 3), make_models( "voxygen.voxel.sprite.blueberry.4", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Blueberry, 4), make_models( "voxygen.voxel.sprite.blueberry.5", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Blueberry, 5), make_models( "voxygen.voxel.sprite.blueberry.6", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Blueberry, 6), make_models( "voxygen.voxel.sprite.blueberry.7", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Blueberry, 7), make_models( "voxygen.voxel.sprite.blueberry.8", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), ( (BlockKind::Blueberry, 8), make_models( "voxygen.voxel.sprite.blueberry.9", Vec3::new(-6.0, -6.0, -0.0), Vec3::one(), ), ), // Ember ( (BlockKind::Ember, 0), make_models( "voxygen.voxel.sprite.ember.1", Vec3::new(-7.0, -7.0, -2.9), Vec3::new(1.0, 1.0, 0.0), ), ), // Corn ( (BlockKind::Corn, 0), make_models( "voxygen.voxel.sprite.corn.corn-0", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::Corn, 1), make_models( "voxygen.voxel.sprite.corn.corn-1", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::Corn, 2), make_models( "voxygen.voxel.sprite.corn.corn-2", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::Corn, 3), make_models( "voxygen.voxel.sprite.corn.corn-3", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::Corn, 4), make_models( "voxygen.voxel.sprite.corn.corn-4", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::Corn, 5), make_models( "voxygen.voxel.sprite.corn.corn-5", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), // Yellow Wheat ( (BlockKind::WheatYellow, 0), make_models( "voxygen.voxel.sprite.wheat_yellow.wheat-0", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatYellow, 1), make_models( "voxygen.voxel.sprite.wheat_yellow.wheat-1", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatYellow, 2), make_models( "voxygen.voxel.sprite.wheat_yellow.wheat-2", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatYellow, 3), make_models( "voxygen.voxel.sprite.wheat_yellow.wheat-3", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatYellow, 4), make_models( "voxygen.voxel.sprite.wheat_yellow.wheat-4", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatYellow, 5), make_models( "voxygen.voxel.sprite.wheat_yellow.wheat-5", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatYellow, 6), make_models( "voxygen.voxel.sprite.wheat_yellow.wheat-6", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatYellow, 7), make_models( "voxygen.voxel.sprite.wheat_yellow.wheat-7", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatYellow, 8), make_models( "voxygen.voxel.sprite.wheat_yellow.wheat-8", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatYellow, 9), make_models( "voxygen.voxel.sprite.wheat_yellow.wheat-9", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), // Green Wheat ( (BlockKind::WheatGreen, 0), make_models( "voxygen.voxel.sprite.wheat_green.wheat-0", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatGreen, 1), make_models( "voxygen.voxel.sprite.wheat_green.wheat-1", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatGreen, 2), make_models( "voxygen.voxel.sprite.wheat_green.wheat-2", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatGreen, 3), make_models( "voxygen.voxel.sprite.wheat_green.wheat-3", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatGreen, 4), make_models( "voxygen.voxel.sprite.wheat_green.wheat-4", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatGreen, 5), make_models( "voxygen.voxel.sprite.wheat_green.wheat-5", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatGreen, 6), make_models( "voxygen.voxel.sprite.wheat_green.wheat-6", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatGreen, 7), make_models( "voxygen.voxel.sprite.wheat_green.wheat-7", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatGreen, 8), make_models( "voxygen.voxel.sprite.wheat_green.wheat-8", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::WheatGreen, 9), make_models( "voxygen.voxel.sprite.wheat_green.wheat-9", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), // Cabbage ( (BlockKind::Cabbage, 0), make_models( "voxygen.voxel.sprite.cabbage.cabbage-0", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Cabbage, 1), make_models( "voxygen.voxel.sprite.cabbage.cabbage-1", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Cabbage, 2), make_models( "voxygen.voxel.sprite.cabbage.cabbage-2", Vec3::new(-6.0, -6.0, 0.0), Vec3::one(), ), ), // Flax ( (BlockKind::Flax, 0), make_models( "voxygen.voxel.sprite.flax.flax-0", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::Flax, 1), make_models( "voxygen.voxel.sprite.flax.flax-1", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::Flax, 2), make_models( "voxygen.voxel.sprite.flax.flax-2", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::Flax, 3), make_models( "voxygen.voxel.sprite.flax.flax-3", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::Flax, 4), make_models( "voxygen.voxel.sprite.flax.flax-4", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), ( (BlockKind::Flax, 5), make_models( "voxygen.voxel.sprite.flax.flax-5", Vec3::new(-6.0, -6.0, 0.0), Vec3::unit_z() * 0.7, ), ), // Carrot ( (BlockKind::Carrot, 0), make_models( "voxygen.voxel.sprite.carrot.0", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Carrot, 1), make_models( "voxygen.voxel.sprite.carrot.1", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Carrot, 2), make_models( "voxygen.voxel.sprite.carrot.2", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Carrot, 3), make_models( "voxygen.voxel.sprite.carrot.3", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Carrot, 4), make_models( "voxygen.voxel.sprite.carrot.4", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Carrot, 5), make_models( "voxygen.voxel.sprite.carrot.5", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Tomato, 0), make_models( "voxygen.voxel.sprite.tomato.0", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Tomato, 1), make_models( "voxygen.voxel.sprite.tomato.1", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Tomato, 2), make_models( "voxygen.voxel.sprite.tomato.2", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Tomato, 3), make_models( "voxygen.voxel.sprite.tomato.3", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Tomato, 4), make_models( "voxygen.voxel.sprite.tomato.4", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), // Radish ( (BlockKind::Radish, 0), make_models( "voxygen.voxel.sprite.radish.0", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Radish, 1), make_models( "voxygen.voxel.sprite.radish.1", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Radish, 2), make_models( "voxygen.voxel.sprite.radish.2", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Radish, 3), make_models( "voxygen.voxel.sprite.radish.3", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Radish, 4), make_models( "voxygen.voxel.sprite.radish.4", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), // Turnip ( (BlockKind::Turnip, 0), make_models( "voxygen.voxel.sprite.turnip.turnip-0", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Turnip, 1), make_models( "voxygen.voxel.sprite.turnip.turnip-1", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Turnip, 2), make_models( "voxygen.voxel.sprite.turnip.turnip-2", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Turnip, 3), make_models( "voxygen.voxel.sprite.turnip.turnip-3", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Turnip, 4), make_models( "voxygen.voxel.sprite.turnip.turnip-4", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), ( (BlockKind::Turnip, 5), make_models( "voxygen.voxel.sprite.turnip.turnip-5", Vec3::new(-5.5, -5.5, -0.25), Vec3::one(), ), ), // Coconut ( (BlockKind::Coconut, 0), make_models( "voxygen.voxel.sprite.fruit.coconut", Vec3::new(-6.0, -6.0, 2.0), Vec3::one(), ), ), // Scarecrow ( (BlockKind::Scarecrow, 0), make_models( "voxygen.voxel.sprite.misc.scarecrow", Vec3::new(-9.5, -3.0, -0.25), Vec3::unit_z(), ), ), // Street Light ( (BlockKind::StreetLamp, 0), make_models( "voxygen.voxel.sprite.misc.street_lamp", Vec3::new(-4.5, -4.5, 0.0), Vec3::unit_z(), ), ), ( (BlockKind::StreetLampTall, 0), make_models( "voxygen.voxel.sprite.furniture.street_lamp-0", Vec3::new(-10.5, -10.5, 0.0), Vec3::unit_z(), ), ), // Door ( (BlockKind::Door, 0), make_models( "voxygen.voxel.sprite.door.door-0", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), // Bed ( (BlockKind::Bed, 0), make_models( "voxygen.voxel.sprite.furniture.bed-0", Vec3::new(-9.5, -14.5, 0.0), Vec3::one(), ), ), // Bench ( (BlockKind::Bench, 0), make_models( "voxygen.voxel.sprite.furniture.bench-0", Vec3::new(-14.0, -4.0, 0.0), Vec3::one(), ), ), // Chair ( (BlockKind::ChairSingle, 0), make_models( "voxygen.voxel.sprite.furniture.chair_single-0", Vec3::new(-5.5, -4.5, 0.0), Vec3::one(), ), ), ( (BlockKind::ChairSingle, 1), make_models( "voxygen.voxel.sprite.furniture.chair_single-1", Vec3::new(-5.5, -4.5, 0.0), Vec3::one(), ), ), ( (BlockKind::ChairDouble, 0), make_models( "voxygen.voxel.sprite.furniture.chair_double-0", Vec3::new(-9.5, -4.5, 0.0), Vec3::one(), ), ), ( (BlockKind::ChairDouble, 1), make_models( "voxygen.voxel.sprite.furniture.chair_double-1", Vec3::new(-9.5, -4.5, 0.0), Vec3::one(), ), ), // CoatRack ( (BlockKind::CoatRack, 0), make_models( "voxygen.voxel.sprite.furniture.coatrack-0", Vec3::new(-6.5, -6.5, 0.0), Vec3::one(), ), ), ( (BlockKind::CoatRack, 1), make_models( "voxygen.voxel.sprite.furniture.coatrack-1", Vec3::new(-6.5, -6.5, 0.0), Vec3::one(), ), ), // Crate ( (BlockKind::Crate, 0), make_models( "voxygen.voxel.sprite.furniture.crate-0", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Crate, 1), make_models( "voxygen.voxel.sprite.furniture.crate-1", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Crate, 2), make_models( "voxygen.voxel.sprite.furniture.crate-2", Vec3::new(-3.0, -3.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Crate, 3), make_models( "voxygen.voxel.sprite.furniture.crate-3", Vec3::new(-6.0, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Crate, 4), make_models( "voxygen.voxel.sprite.furniture.crate-4", Vec3::new(-6.0, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Crate, 5), make_models( "voxygen.voxel.sprite.furniture.crate-5", Vec3::new(-5.5, -3.0, 0.0), Vec3::one(), ), ), ( (BlockKind::Crate, 6), make_models( "voxygen.voxel.sprite.furniture.crate-6", Vec3::new(-4.5, -3.0, 0.0), Vec3::one(), ), ), // DrawerLarge ( (BlockKind::DrawerLarge, 0), make_models( "voxygen.voxel.sprite.furniture.drawer_large-0", Vec3::new(-11.5, -5.0, 0.0), Vec3::one(), ), ), ( (BlockKind::DrawerLarge, 1), make_models( "voxygen.voxel.sprite.furniture.drawer_large-1", Vec3::new(-11.5, -5.0, 0.0), Vec3::one(), ), ), // DrawerMedium ( (BlockKind::DrawerMedium, 0), make_models( "voxygen.voxel.sprite.furniture.drawer_medium-0", Vec3::new(-11.0, -5.0, 0.0), Vec3::one(), ), ), ( (BlockKind::DrawerMedium, 1), make_models( "voxygen.voxel.sprite.furniture.drawer_medium-1", Vec3::new(-11.0, -5.0, 0.0), Vec3::one(), ), ), // DrawerSmall ( (BlockKind::DrawerSmall, 0), make_models( "voxygen.voxel.sprite.furniture.drawer_small-0", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::DrawerSmall, 1), make_models( "voxygen.voxel.sprite.furniture.drawer_small-1", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), // DungeonWallDecor ( (BlockKind::DungeonWallDecor, 0), make_models( "voxygen.voxel.sprite.furniture.dungeon_wall-0", Vec3::new(-5.5, -1.0, 0.0), Vec3::one(), ), ), ( (BlockKind::DungeonWallDecor, 1), make_models( "voxygen.voxel.sprite.furniture.dungeon_wall-1", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::DungeonWallDecor, 2), make_models( "voxygen.voxel.sprite.furniture.dungeon_wall-2", Vec3::new(-5.5, -3.0, 0.0), Vec3::one(), ), ), ( (BlockKind::DungeonWallDecor, 3), make_models( "voxygen.voxel.sprite.furniture.dungeon_wall-3", Vec3::new(-1.5, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::DungeonWallDecor, 4), make_models( "voxygen.voxel.sprite.furniture.dungeon_wall-4", Vec3::new(-5.5, -4.5, 0.0), Vec3::one(), ), ), ( (BlockKind::DungeonWallDecor, 5), make_models( "voxygen.voxel.sprite.furniture.dungeon_wall-5", Vec3::new(-5.5, -0.5, 0.0), Vec3::one(), ), ), ( (BlockKind::DungeonWallDecor, 6), make_models( "voxygen.voxel.sprite.furniture.dungeon_wall-6", Vec3::new(-5.5, -1.0, 0.0), Vec3::one(), ), ), ( (BlockKind::DungeonWallDecor, 7), make_models( "voxygen.voxel.sprite.furniture.dungeon_wall-7", Vec3::new(-5.5, -1.0, 0.0), Vec3::one(), ), ), ( (BlockKind::DungeonWallDecor, 8), make_models( "voxygen.voxel.sprite.furniture.dungeon_wall-8", Vec3::new(-5.5, -1.0, 0.0), Vec3::one(), ), ), ( (BlockKind::DungeonWallDecor, 9), make_models( "voxygen.voxel.sprite.furniture.dungeon_wall-9", Vec3::new(-1.5, -5.5, 0.0), Vec3::one(), ), ), // HangingBasket ( (BlockKind::HangingBasket, 0), make_models( "voxygen.voxel.sprite.furniture.hanging_basket-0", Vec3::new(-6.5, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::HangingBasket, 1), make_models( "voxygen.voxel.sprite.furniture.hanging_basket-1", Vec3::new(-9.5, -5.5, 0.0), Vec3::one(), ), ), // HangingSign ( (BlockKind::HangingSign, 0), make_models( "voxygen.voxel.sprite.furniture.hanging_sign-0", Vec3::new(-3.5, -17.0, 0.0), Vec3::one(), ), ), // WallLamp ( (BlockKind::WallLamp, 0), make_models( "voxygen.voxel.sprite.furniture.lamp_wall-0", Vec3::new(-5.5, -2.5, 0.0), Vec3::one(), ), ), ( (BlockKind::WallLamp, 1), make_models( "voxygen.voxel.sprite.furniture.lamp_wall-1", Vec3::new(-10.5, -9.0, 0.0), Vec3::one(), ), ), // Planter ( (BlockKind::Planter, 0), make_models( "voxygen.voxel.sprite.furniture.planter-0", Vec3::new(-6.0, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Planter, 1), make_models( "voxygen.voxel.sprite.furniture.planter-1", Vec3::new(-13.0, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Planter, 2), make_models( "voxygen.voxel.sprite.furniture.planter-2", Vec3::new(-6.0, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Planter, 3), make_models( "voxygen.voxel.sprite.furniture.planter-3", Vec3::new(-6.0, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Planter, 4), make_models( "voxygen.voxel.sprite.furniture.planter-4", Vec3::new(-6.0, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Planter, 5), make_models( "voxygen.voxel.sprite.furniture.planter-5", Vec3::new(-6.0, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Planter, 6), make_models( "voxygen.voxel.sprite.furniture.planter-6", Vec3::new(-7.5, -3.5, 0.0), Vec3::one(), ), ), //Pot ( (BlockKind::Pot, 0), make_models( "voxygen.voxel.sprite.furniture.pot-0", Vec3::new(-3.5, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Pot, 1), make_models( "voxygen.voxel.sprite.furniture.pot-1", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), // Shelf ( (BlockKind::Shelf, 0), make_models( "voxygen.voxel.sprite.furniture.shelf-0", Vec3::new(-14.5, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Shelf, 1), make_models( "voxygen.voxel.sprite.furniture.shelf-1", Vec3::new(-13.5, -3.5, 0.0), Vec3::one(), ), ), // TableSide ( (BlockKind::TableSide, 0), make_models( "voxygen.voxel.sprite.furniture.table_side-0", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::TableSide, 1), make_models( "voxygen.voxel.sprite.furniture.table_side-1", Vec3::new(-5.5, -5.5, 0.0), Vec3::one(), ), ), // TableDining ( (BlockKind::TableDining, 0), make_models( "voxygen.voxel.sprite.furniture.table_dining-0", Vec3::new(-8.5, -8.5, 0.0), Vec3::one(), ), ), ( (BlockKind::TableDining, 1), make_models( "voxygen.voxel.sprite.furniture.table_dining-1", Vec3::new(-8.5, -8.5, 0.0), Vec3::one(), ), ), // TableDouble ( (BlockKind::TableDouble, 0), make_models( "voxygen.voxel.sprite.furniture.table_double-0", Vec3::new(-18.5, -11.5, 0.0), Vec3::one(), ), ), // WardrobeSingle ( (BlockKind::WardrobeSingle, 0), make_models( "voxygen.voxel.sprite.furniture.wardrobe_single-0", Vec3::new(-5.5, -6.0, 0.0), Vec3::one(), ), ), ( (BlockKind::WardrobeSingle, 1), make_models( "voxygen.voxel.sprite.furniture.wardrobe_single-1", Vec3::new(-5.5, -6.5, 0.0), Vec3::one(), ), ), //WardrobeDouble ( (BlockKind::WardrobeDouble, 0), make_models( "voxygen.voxel.sprite.furniture.wardrobe_double-0", Vec3::new(-10.5, -6.5, 0.0), Vec3::one(), ), ), ( (BlockKind::WardrobeDouble, 1), make_models( "voxygen.voxel.sprite.furniture.wardrobe_double-1", Vec3::new(-10.5, -6.0, 0.0), Vec3::one(), ), ), /* Stones */ ( (BlockKind::Stones, 0), make_models( "voxygen.voxel.sprite.rocks.rock-0", Vec3::new(-3.0, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Stones, 1), make_models( "voxygen.voxel.sprite.rocks.rock-1", Vec3::new(-4.5, -5.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Stones, 2), make_models( "voxygen.voxel.sprite.rocks.rock-2", Vec3::new(-4.5, -4.5, 0.0), Vec3::one(), ), ), /* Twigs */ ( (BlockKind::Twigs, 0), make_models( "voxygen.voxel.sprite.twigs.twigs-0", Vec3::new(-3.5, -3.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Twigs, 1), make_models( "voxygen.voxel.sprite.twigs.twigs-1", Vec3::new(-2.0, -1.5, 0.0), Vec3::one(), ), ), ( (BlockKind::Twigs, 2), make_models( "voxygen.voxel.sprite.twigs.twigs-2", Vec3::new(-4.0, -4.0, 0.0), Vec3::one(), ), ), // Shiny Gems ( (BlockKind::ShinyGem, 0), make_models( "voxygen.voxel.sprite.gem.gem_blue", Vec3::new(-2.0, -3.0, 0.0), Vec3::one(), ), ), ( (BlockKind::ShinyGem, 1), make_models( "voxygen.voxel.sprite.gem.gem_green", Vec3::new(-2.0, -3.0, 0.0), Vec3::one(), ), ), ( (BlockKind::ShinyGem, 2), make_models( "voxygen.voxel.sprite.gem.gem_red", Vec3::new(-3.0, -2.0, -2.0), Vec3::one(), ), ), ] .into_iter() .collect(), waves: renderer .create_texture( &assets::load_expect("voxygen.texture.waves"), Some(gfx::texture::FilterMethod::Trilinear), Some(gfx::texture::WrapMode::Tile), ) .expect("Failed to create wave texture"), phantom: PhantomData, } } /// Maintain terrain data. To be called once per tick. #[allow(clippy::for_loops_over_fallibles)] // TODO: Pending review in #587 #[allow(clippy::len_zero)] // TODO: Pending review in #587 pub fn maintain( &mut self, renderer: &mut Renderer, scene_data: &SceneData, focus_pos: Vec3, loaded_distance: f32, view_mat: Mat4, proj_mat: Mat4, ) { let current_tick = scene_data.tick; let current_time = scene_data.state.get_time(); // Add any recently created or changed chunks to the list of chunks to be // meshed. for (modified, pos) in scene_data .state .terrain_changes() .modified_chunks .iter() .map(|c| (true, c)) .chain( scene_data .state .terrain_changes() .new_chunks .iter() .map(|c| (false, c)), ) { // TODO: ANOTHER PROBLEM HERE! // What happens if the block on the edge of a chunk gets modified? We need to // spawn a mesh worker to remesh its neighbour(s) too since their // ambient occlusion and face elision information changes too! for i in -1..2 { for j in -1..2 { let pos = pos + Vec2::new(i, j); if !self.chunks.contains_key(&pos) || modified { let mut neighbours = true; for i in -1..2 { for j in -1..2 { neighbours &= scene_data .state .terrain() .get_key(pos + Vec2::new(i, j)) .is_some(); } } if neighbours { self.mesh_todo.insert(pos, ChunkMeshState { pos, started_tick: current_tick, active_worker: None, }); } } } } } // Add the chunks belonging to recently changed blocks to the list of chunks to // be meshed for pos in scene_data .state .terrain_changes() .modified_blocks .iter() .map(|(p, _)| *p) { // Handle block changes on chunk borders // Remesh all neighbours because we have complex lighting now // TODO: if lighting is on the server this can be updated to only remesh when // lighting changes in that neighbouring chunk or if the block // change was on the border for x in -1..2 { for y in -1..2 { let neighbour_pos = pos + Vec3::new(x, y, 0); let neighbour_chunk_pos = scene_data.state.terrain().pos_key(neighbour_pos); // Only remesh if this chunk has all its neighbors let mut neighbours = true; for i in -1..2 { for j in -1..2 { neighbours &= scene_data .state .terrain() .get_key(neighbour_chunk_pos + Vec2::new(i, j)) .is_some(); } } if neighbours { self.mesh_todo.insert(neighbour_chunk_pos, ChunkMeshState { pos: neighbour_chunk_pos, started_tick: current_tick, active_worker: None, }); } } } } // Remove any models for chunks that have been recently removed. for pos in &scene_data.state.terrain_changes().removed_chunks { self.chunks.remove(pos); self.mesh_todo.remove(pos); } for todo in self .mesh_todo .values_mut() .filter(|todo| { todo.active_worker .map(|worker_tick| worker_tick < todo.started_tick) .unwrap_or(true) }) .min_by_key(|todo| todo.active_worker.unwrap_or(todo.started_tick)) { // TODO: find a alternative! if scene_data.thread_pool.queued_jobs() > 0 { break; } // Find the area of the terrain we want. Because meshing needs to compute things // like ambient occlusion and edge elision, we also need the borders // of the chunk's neighbours too (hence the `- 1` and `+ 1`). let aabr = Aabr { min: todo .pos .map2(VolGrid2d::::chunk_size(), |e, sz| e * sz as i32 - 1), max: todo.pos.map2(VolGrid2d::::chunk_size(), |e, sz| { (e + 1) * sz as i32 + 1 }), }; // Copy out the chunk data we need to perform the meshing. We do this by taking // a sample of the terrain that includes both the chunk we want and // its neighbours. let volume = match scene_data.state.terrain().sample(aabr) { Ok(sample) => sample, // Either this chunk or its neighbours doesn't yet exist, so we keep it in the // queue to be processed at a later date when we have its neighbours. Err(VolGrid2dError::NoSuchChunk) => return, _ => panic!("Unhandled edge case"), }; // The region to actually mesh let min_z = volume .iter() .fold(i32::MAX, |min, (_, chunk)| chunk.get_min_z().min(min)); let max_z = volume .iter() .fold(i32::MIN, |max, (_, chunk)| chunk.get_max_z().max(max)); let aabb = Aabb { min: Vec3::from(aabr.min) + Vec3::unit_z() * (min_z - 2), max: Vec3::from(aabr.max) + Vec3::unit_z() * (max_z + 2), }; // Clone various things so that they can be moved into the thread. let send = self.mesh_send_tmp.clone(); let pos = todo.pos; // Queue the worker thread. let started_tick = todo.started_tick; scene_data.thread_pool.execute(move || { let _ = send.send(mesh_worker( pos, (min_z as f32, max_z as f32), started_tick, volume, aabb, )); }); todo.active_worker = Some(todo.started_tick); } // Receive a chunk mesh from a worker thread and upload it to the GPU, then // store it. Only pull out one chunk per frame to avoid an unacceptable // amount of blocking lag due to the GPU upload. That still gives us a // 60 chunks / second budget to play with. if let Ok(response) = self.mesh_recv.recv_timeout(Duration::new(0, 0)) { match self.mesh_todo.get(&response.pos) { // It's the mesh we want, insert the newly finished model into the terrain model // data structure (convert the mesh to a model first of course). Some(todo) if response.started_tick <= todo.started_tick => { let load_time = self .chunks .get(&response.pos) .map(|chunk| chunk.load_time) .unwrap_or(current_time as f32); self.chunks.insert(response.pos, TerrainChunkData { load_time, opaque_model: renderer .create_model(&response.opaque_mesh) .expect("Failed to upload chunk mesh to the GPU!"), fluid_model: if response.fluid_mesh.vertices().len() > 0 { Some( renderer .create_model(&response.fluid_mesh) .expect("Failed to upload chunk mesh to the GPU!"), ) } else { None }, sprite_instances: response .sprite_instances .into_iter() .map(|(kind, instances)| { ( kind, renderer.create_instances(&instances).expect( "Failed to upload chunk sprite instances to the GPU!", ), ) }) .collect(), locals: renderer .create_consts(&[TerrainLocals { model_offs: Vec3::from( response.pos.map2(VolGrid2d::::chunk_size(), |e, sz| { e as f32 * sz as f32 }), ) .into_array(), load_time, }]) .expect("Failed to upload chunk locals to the GPU!"), visible: false, z_bounds: response.z_bounds, frustum_last_plane_index: 0, }); if response.started_tick == todo.started_tick { self.mesh_todo.remove(&response.pos); } }, // Chunk must have been removed, or it was spawned on an old tick. Drop the mesh // since it's either out of date or no longer needed. _ => {}, } } // Construct view frustum let frustum = Frustum::from_modelview_projection((proj_mat * view_mat).into_col_arrays()); // Update chunk visibility let chunk_sz = V::RECT_SIZE.x as f32; for (pos, chunk) in &mut self.chunks { let chunk_pos = pos.map(|e| e as f32 * chunk_sz); // Limit focus_pos to chunk bounds and ensure the chunk is within the fog // boundary let nearest_in_chunk = Vec2::from(focus_pos).clamped(chunk_pos, chunk_pos + chunk_sz); let in_range = Vec2::::from(focus_pos).distance_squared(nearest_in_chunk) < loaded_distance.powf(2.0); if !in_range { chunk.visible = in_range; continue; } // Ensure the chunk is within the view frustum let chunk_min = [chunk_pos.x, chunk_pos.y, chunk.z_bounds.0]; let chunk_max = [ chunk_pos.x + chunk_sz, chunk_pos.y + chunk_sz, chunk.z_bounds.1, ]; let (in_frustum, last_plane_index) = AABB::new(chunk_min, chunk_max) .coherent_test_against_frustum(&frustum, chunk.frustum_last_plane_index); chunk.frustum_last_plane_index = last_plane_index; chunk.visible = in_frustum; } } pub fn chunk_count(&self) -> usize { self.chunks.len() } pub fn visible_chunk_count(&self) -> usize { self.chunks.iter().filter(|(_, c)| c.visible).count() } pub fn render( &self, renderer: &mut Renderer, globals: &Consts, lights: &Consts, shadows: &Consts, focus_pos: Vec3, ) { let focus_chunk = Vec2::from(focus_pos).map2(TerrainChunk::RECT_SIZE, |e: f32, sz| { (e as i32).div_euclid(sz as i32) }); let chunks = &self.chunks; let chunk_iter = Spiral2d::new() .scan(0, |n, rpos| { if *n >= chunks.len() { None } else { let pos = focus_chunk + rpos; Some(chunks.get(&pos).map(|c| { *n += 1; (pos, c) })) } }) .filter_map(|x| x); // Opaque for (_, chunk) in chunk_iter.clone() { if chunk.visible { renderer.render_terrain_chunk( &chunk.opaque_model, globals, &chunk.locals, lights, shadows, ); } } } pub fn render_translucent( &self, renderer: &mut Renderer, globals: &Consts, lights: &Consts, shadows: &Consts, focus_pos: Vec3, sprite_render_distance: f32, ) { let focus_chunk = Vec2::from(focus_pos).map2(TerrainChunk::RECT_SIZE, |e: f32, sz| { (e as i32).div_euclid(sz as i32) }); let chunks = &self.chunks; let chunk_iter = Spiral2d::new() .scan(0, |n, rpos| { if *n >= chunks.len() { None } else { let pos = focus_chunk + rpos; Some(chunks.get(&pos).map(|c| { *n += 1; (pos, c) })) } }) .filter_map(|x| x); // Terrain sprites for (pos, chunk) in chunk_iter.clone() { if chunk.visible { let sprite_low_detail_distance = sprite_render_distance * 0.75; let sprite_mid_detail_distance = sprite_render_distance * 0.5; let sprite_hid_detail_distance = sprite_render_distance * 0.35; let sprite_high_detail_distance = sprite_render_distance * 0.15; let chunk_center = pos.map2(V::RECT_SIZE, |e, sz: u32| (e as f32 + 0.5) * sz as f32); let dist_sqrd = Vec2::from(focus_pos).distance_squared(chunk_center); if dist_sqrd < sprite_render_distance.powf(2.0) { for (kind, instances) in &chunk.sprite_instances { renderer.render_sprites( if dist_sqrd < sprite_high_detail_distance.powf(2.0) { &self.sprite_models[&kind][0] } else if dist_sqrd < sprite_hid_detail_distance.powf(2.0) { &self.sprite_models[&kind][1] } else if dist_sqrd < sprite_mid_detail_distance.powf(2.0) { &self.sprite_models[&kind][2] } else if dist_sqrd < sprite_low_detail_distance.powf(2.0) { &self.sprite_models[&kind][3] } else { &self.sprite_models[&kind][4] }, globals, &instances, lights, shadows, ); } } } } // Translucent chunk_iter .clone() .filter(|(_, chunk)| chunk.visible) .filter_map(|(_, chunk)| { chunk .fluid_model .as_ref() .map(|model| (model, &chunk.locals)) }) .collect::>() .into_iter() .rev() // Render back-to-front .for_each(|(model, locals)| { renderer.render_fluid_chunk(model, globals, locals, lights, shadows, &self.waves) }); } }