use crate::{ all::ForestKind, block::block_from_structure, column::ColumnGen, util::{RandomPerm, Sampler, UnitChooser}, Canvas, CONFIG, }; use common::{ terrain::{structure::Structure, Block, BlockKind}, vol::ReadVol, }; use hashbrown::HashMap; use lazy_static::lazy_static; use std::{f32, sync::Arc}; use vek::*; lazy_static! { pub static ref OAKS: Vec> = Structure::load_group("oaks"); pub static ref OAK_STUMPS: Vec> = Structure::load_group("oak_stumps"); pub static ref PINES: Vec> = Structure::load_group("pines"); pub static ref PALMS: Vec> = Structure::load_group("palms"); pub static ref ACACIAS: Vec> = Structure::load_group("acacias"); pub static ref BAOBABS: Vec> = Structure::load_group("baobabs"); pub static ref FRUIT_TREES: Vec> = Structure::load_group("fruit_trees"); pub static ref BIRCHES: Vec> = Structure::load_group("birch"); pub static ref MANGROVE_TREES: Vec> = Structure::load_group("mangrove_trees"); pub static ref QUIRKY: Vec> = Structure::load_group("quirky"); pub static ref QUIRKY_DRY: Vec> = Structure::load_group("quirky_dry"); pub static ref SWAMP_TREES: Vec> = Structure::load_group("swamp_trees"); } static MODEL_RAND: RandomPerm = RandomPerm::new(0xDB21C052); static UNIT_CHOOSER: UnitChooser = UnitChooser::new(0x700F4EC7); static QUIRKY_RAND: RandomPerm = RandomPerm::new(0xA634460F); #[allow(clippy::if_same_then_else)] pub fn apply_trees_to(canvas: &mut Canvas) { struct Tree { pos: Vec3, model: Arc, seed: u32, units: (Vec2, Vec2), } let mut tree_cache = HashMap::new(); let info = canvas.info(); canvas.foreach_col(|canvas, wpos2d, col| { let trees = info.land().get_near_trees(wpos2d); for (tree_wpos, seed) in trees { let tree = if let Some(tree) = tree_cache.entry(tree_wpos).or_insert_with(|| { let col = ColumnGen::new(info.land()).get((tree_wpos, info.index()))?; let is_quirky = QUIRKY_RAND.chance(seed, 1.0 / 500.0); // Ensure that it's valid to place a *thing* here if col.alt < col.water_level || col.spawn_rate < 0.9 || col.water_dist.map(|d| d < 8.0).unwrap_or(false) || col.path.map(|(d, _, _, _)| d < 12.0).unwrap_or(false) { return None; } // Ensure that it's valid to place a tree here if !is_quirky && ((seed.wrapping_mul(13)) & 0xFF) as f32 / 256.0 > col.tree_density { return None; } Some(Tree { pos: Vec3::new(tree_wpos.x, tree_wpos.y, col.alt as i32), model: { let models: &'static [_] = if is_quirky { if col.temp > CONFIG.desert_temp { &QUIRKY_DRY } else { &QUIRKY } } else { match col.forest_kind { ForestKind::Oak if QUIRKY_RAND.chance(seed + 1, 1.0 / 16.0) => { &OAK_STUMPS }, ForestKind::Oak if QUIRKY_RAND.chance(seed + 2, 1.0 / 20.0) => { &FRUIT_TREES }, ForestKind::Palm => &PALMS, ForestKind::Acacia => &ACACIAS, ForestKind::Baobab => &BAOBABS, ForestKind::Oak => &OAKS, ForestKind::Pine => &PINES, ForestKind::Birch => &BIRCHES, ForestKind::Mangrove => &MANGROVE_TREES, ForestKind::Swamp => &SWAMP_TREES, } }; Arc::clone( &models[(MODEL_RAND.get(seed.wrapping_mul(17)) / 13) as usize % models.len()], ) }, seed, units: UNIT_CHOOSER.get(seed), }) }) { tree } else { continue; }; let bounds = tree.model.get_bounds(); let mut is_top = true; let mut is_leaf_top = true; for z in (bounds.min.z..bounds.max.z).rev() { let wpos = Vec3::new(wpos2d.x, wpos2d.y, tree.pos.z + z); let model_pos = Vec3::from( (wpos - tree.pos) .xy() .map2(Vec2::new(tree.units.0, tree.units.1), |rpos, unit| { unit * rpos }) .sum(), ) + Vec3::unit_z() * (wpos.z - tree.pos.z); block_from_structure( info.index(), if let Some(block) = tree.model.get(model_pos).ok().copied() { block } else { // If we hit an inaccessible block, we're probably outside the model bounds. // Skip this column. break; }, wpos, tree.pos.xy(), tree.seed, col, Block::air, ) .map(|block| { // Add a snow covering to the block above under certain circumstances if col.snow_cover && ((block.kind() == BlockKind::Leaves && is_leaf_top) || (is_top && block.is_filled())) { canvas.set( wpos + Vec3::unit_z(), Block::new(BlockKind::Snow, Rgb::new(210, 210, 255)), ); } canvas.set(wpos, block); is_leaf_top = false; is_top = false; }) .unwrap_or_else(|| { is_leaf_top = true; }); } } }); }