veloren/world/src/layer/tree.rs

548 lines
22 KiB
Rust
Raw Normal View History

use crate::{
2021-02-07 19:55:13 +00:00
all::*,
block::block_from_structure,
column::ColumnGen,
util::{RandomPerm, Sampler, UnitChooser},
Canvas, CONFIG,
};
use common::{
2020-12-12 22:14:24 +00:00
assets::AssetHandle,
2021-02-06 23:53:25 +00:00
terrain::{
structure::{Structure, StructureBlock, StructuresGroup},
Block, BlockKind, SpriteKind,
2021-02-06 23:53:25 +00:00
},
vol::ReadVol,
};
2020-12-12 01:45:46 +00:00
use hashbrown::HashMap;
use lazy_static::lazy_static;
2021-02-06 23:53:25 +00:00
use rand::prelude::*;
use std::{f32, ops::Range};
use vek::*;
lazy_static! {
2020-12-12 22:14:24 +00:00
static ref OAKS: AssetHandle<StructuresGroup> = Structure::load_group("oaks");
static ref OAK_STUMPS: AssetHandle<StructuresGroup> = Structure::load_group("oak_stumps");
static ref PINES: AssetHandle<StructuresGroup> = Structure::load_group("pines");
static ref PALMS: AssetHandle<StructuresGroup> = Structure::load_group("palms");
static ref ACACIAS: AssetHandle<StructuresGroup> = Structure::load_group("acacias");
static ref BAOBABS: AssetHandle<StructuresGroup> = Structure::load_group("baobabs");
static ref FRUIT_TREES: AssetHandle<StructuresGroup> = Structure::load_group("fruit_trees");
static ref BIRCHES: AssetHandle<StructuresGroup> = Structure::load_group("birch");
2020-12-13 01:09:57 +00:00
static ref MANGROVE_TREES: AssetHandle<StructuresGroup> =
Structure::load_group("mangrove_trees");
2020-12-12 22:14:24 +00:00
static ref QUIRKY: AssetHandle<StructuresGroup> = Structure::load_group("quirky");
static ref QUIRKY_DRY: AssetHandle<StructuresGroup> = Structure::load_group("quirky_dry");
static ref SWAMP_TREES: AssetHandle<StructuresGroup> = 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);
2020-11-23 15:39:03 +00:00
#[allow(clippy::if_same_then_else)]
pub fn apply_trees_to(canvas: &mut Canvas, dynamic_rng: &mut impl Rng) {
2021-01-03 20:43:07 +00:00
// TODO: Get rid of this
enum TreeModel {
Structure(Structure),
Procedural(ProceduralTree, StructureBlock),
2021-01-03 20:43:07 +00:00
}
struct Tree {
pos: Vec3<i32>,
2021-01-03 20:43:07 +00:00
model: TreeModel,
seed: u32,
units: (Vec2<i32>, Vec2<i32>),
}
let mut tree_cache = HashMap::new();
let info = canvas.info();
canvas.foreach_col(|canvas, wpos2d, col| {
let trees = info.land().get_near_trees(wpos2d);
2021-02-08 11:10:47 +00:00
for TreeAttr { pos, seed, scale, forest_kind, lanterns } in trees {
2021-02-07 19:55:13 +00:00
let tree = if let Some(tree) = tree_cache.entry(pos).or_insert_with(|| {
let col = ColumnGen::new(info.land()).get((pos, info.index()))?;
2020-11-11 11:42:22 +00:00
let is_quirky = QUIRKY_RAND.chance(seed, 1.0 / 500.0);
2020-11-23 15:39:03 +00:00
// Ensure that it's valid to place a *thing* here
2020-11-11 11:42:22 +00:00
if col.alt < col.water_level
2020-11-22 01:37:20 +00:00
|| 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;
2020-11-23 15:39:03 +00:00
}
// 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
2020-11-15 01:40:23 +00:00
{
2020-11-11 11:42:22 +00:00
return None;
}
Some(Tree {
2021-02-07 19:55:13 +00:00
pos: Vec3::new(pos.x, pos.y, col.alt as i32),
2021-01-03 20:43:07 +00:00
model: 'model: {
2020-12-12 22:14:24 +00:00
let models: AssetHandle<_> = if is_quirky {
if col.temp > CONFIG.desert_temp {
2020-12-12 22:14:24 +00:00
*QUIRKY_DRY
} else {
2020-12-12 22:14:24 +00:00
*QUIRKY
}
} else {
2021-02-07 19:55:13 +00:00
match forest_kind {
2020-11-15 01:40:23 +00:00
ForestKind::Oak if QUIRKY_RAND.chance(seed + 1, 1.0 / 16.0) => {
2020-12-12 22:14:24 +00:00
*OAK_STUMPS
2020-11-15 01:40:23 +00:00
},
ForestKind::Oak if QUIRKY_RAND.chance(seed + 2, 1.0 / 20.0) => {
2020-12-12 22:14:24 +00:00
*FRUIT_TREES
2020-11-15 01:40:23 +00:00
},
2020-12-12 22:14:24 +00:00
ForestKind::Palm => *PALMS,
ForestKind::Acacia => *ACACIAS,
ForestKind::Baobab => *BAOBABS,
2021-01-03 20:43:07 +00:00
// ForestKind::Oak => *OAKS,
2021-02-06 23:53:25 +00:00
ForestKind::Oak => {
break 'model TreeModel::Procedural(
ProceduralTree::generate(
2021-02-07 19:55:13 +00:00
TreeConfig::oak(&mut RandomPerm::new(seed), scale),
2021-02-06 23:53:25 +00:00
&mut RandomPerm::new(seed),
),
StructureBlock::TemperateLeaves,
);
},
//ForestKind::Pine => *PINES,
2021-02-06 23:53:25 +00:00
ForestKind::Pine => {
break 'model TreeModel::Procedural(
ProceduralTree::generate(
2021-02-07 19:55:13 +00:00
TreeConfig::pine(&mut RandomPerm::new(seed), scale),
2021-02-06 23:53:25 +00:00
&mut RandomPerm::new(seed),
),
StructureBlock::PineLeaves,
);
},
2020-12-12 22:14:24 +00:00
ForestKind::Birch => *BIRCHES,
ForestKind::Mangrove => *MANGROVE_TREES,
ForestKind::Swamp => *SWAMP_TREES,
2021-02-07 23:50:05 +00:00
ForestKind::Giant => {
break 'model TreeModel::Procedural(
ProceduralTree::generate(
TreeConfig::giant(&mut RandomPerm::new(seed), scale),
&mut RandomPerm::new(seed),
),
StructureBlock::TemperateLeaves,
);
},
}
};
2020-12-12 22:14:24 +00:00
let models = models.read();
2021-02-06 23:53:25 +00:00
TreeModel::Structure(
models[(MODEL_RAND.get(seed.wrapping_mul(17)) / 13) as usize
% models.len()]
.clone(),
)
},
seed,
units: UNIT_CHOOSER.get(seed),
})
}) {
tree
} else {
continue;
};
2021-01-03 20:43:07 +00:00
let bounds = match &tree.model {
TreeModel::Structure(s) => s.get_bounds(),
TreeModel::Procedural(t, _) => t.get_bounds().map(|e| e as i32),
2021-01-03 20:43:07 +00:00
};
let rpos2d = (wpos2d - tree.pos.xy())
2021-02-06 23:53:25 +00:00
.map2(Vec2::new(tree.units.0, tree.units.1), |p, unit| unit * p)
.sum();
if !Aabr::from(bounds).contains_point(rpos2d) {
2021-01-03 20:43:07 +00:00
// Skip this column
continue;
2021-01-03 20:43:07 +00:00
}
2020-11-09 15:06:37 +00:00
let mut is_top = true;
2020-11-09 17:09:33 +00:00
let mut is_leaf_top = true;
let mut last_block = Block::empty();
2020-11-09 15:06:37 +00:00
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(),
2021-01-03 20:43:07 +00:00
if let Some(block) = match &tree.model {
TreeModel::Structure(s) => s.get(model_pos).ok().copied(),
2021-02-06 23:53:25 +00:00
TreeModel::Procedural(t, leaf_block) => Some(
match t.is_branch_or_leaves_at(model_pos.map(|e| e as f32 + 0.5)) {
(true, _) => StructureBlock::Log,
2021-02-06 23:53:25 +00:00
(_, true) => *leaf_block,
(_, _) => StructureBlock::None,
},
),
2021-01-03 20:43:07 +00:00
} {
block
} else {
2021-02-06 23:53:25 +00:00
break;
},
wpos,
tree.pos.xy(),
tree.seed,
col,
Block::air,
)
2020-11-09 15:06:37 +00:00
.map(|block| {
// Add mushrooms to the tree
2021-02-08 11:10:47 +00:00
if lanterns && last_block.is_air() && block.kind() == BlockKind::Wood && dynamic_rng.gen_range(0..48) == 0 {
canvas.set(wpos + Vec3::unit_z(), Block::air(SpriteKind::Lantern));
2020-11-09 17:09:33 +00:00
// Add a snow covering to the block above under certain circumstances
} else if col.snow_cover
2020-11-09 17:09:33 +00:00
&& ((block.kind() == BlockKind::Leaves && is_leaf_top)
|| (is_top && block.is_filled()))
{
2020-11-09 15:06:37 +00:00
canvas.set(
wpos + Vec3::unit_z(),
Block::new(BlockKind::Snow, Rgb::new(210, 210, 255)),
);
}
canvas.set(wpos, block);
2020-11-09 17:09:33 +00:00
is_leaf_top = false;
2020-11-09 15:06:37 +00:00
is_top = false;
last_block = block;
2020-11-09 15:06:37 +00:00
})
.unwrap_or_else(|| {
if last_block.kind() == BlockKind::Wood && dynamic_rng.gen_range(0..512) == 0 {
canvas.set(wpos, Block::air(SpriteKind::Beehive));
}
2020-11-09 17:09:33 +00:00
is_leaf_top = true;
last_block = Block::empty();
2020-11-09 15:06:37 +00:00
});
}
}
});
}
2021-01-03 20:43:07 +00:00
/// A type that specifies the generation properties of a tree.
pub struct TreeConfig {
/// Length of trunk, also scales other branches.
2021-01-06 12:59:16 +00:00
pub trunk_len: f32,
/// Radius of trunk, also scales other branches.
pub trunk_radius: f32,
// The scale that child branch lengths should be compared to their parents.
2021-01-06 12:59:16 +00:00
pub branch_child_len: f32,
// The scale that child branch radii should be compared to their parents.
2021-01-06 12:59:16 +00:00
pub branch_child_radius: f32,
/// The range of radii that leaf-emitting branches might have.
pub leaf_radius: Range<f32>,
/// 0 - 1 (0 = chaotic, 1 = straight).
pub straightness: f32,
/// Maximum number of branch layers (not including trunk).
pub max_depth: usize,
/// The number of branches that form from each branch.
2021-02-07 19:55:13 +00:00
pub splits: Range<f32>,
2021-02-06 23:53:25 +00:00
/// The range of proportions along a branch at which a split into another
/// branch might occur. This value is clamped between 0 and 1, but a
/// wider range may bias the results towards branch ends.
pub split_range: Range<f32>,
2021-02-06 23:53:25 +00:00
/// The bias applied to the length of branches based on the proportion along
/// their parent that they eminate from. -1.0 = negative bias (branches
/// at ends are longer, branches at the start are shorter) 0.0 = no bias
/// (branches do not change their length with regard to parent branch
/// proportion) 1.0 = positive bias (branches at ends are shorter,
/// branches at the start are longer)
pub branch_len_bias: f32,
2021-02-06 23:53:25 +00:00
/// The scale of leaves in the vertical plane. Less than 1.0 implies a
/// flattening of the leaves.
pub leaf_vertical_scale: f32,
/// How evenly spaced (vs random) sub-branches are along their parent.
pub proportionality: f32,
}
impl TreeConfig {
2021-02-07 19:55:13 +00:00
pub fn oak(rng: &mut impl Rng, scale: f32) -> Self {
let scale = scale * (0.9 + rng.gen::<f32>().powi(4));
2021-02-07 19:55:13 +00:00
let log_scale = 1.0 + scale.log2().max(0.0);
2021-02-06 23:53:25 +00:00
Self {
2021-02-07 19:55:13 +00:00
trunk_len: 9.0 * scale,
trunk_radius: 2.0 * scale,
branch_child_len: 0.9,
2021-02-07 19:55:13 +00:00
branch_child_radius: 0.75,
leaf_radius: 2.5 * log_scale..3.25 * log_scale,
straightness: 0.45,
max_depth: 4,
splits: 2.25..3.25,
2021-02-07 19:55:13 +00:00
split_range: 0.75..1.5,
2021-02-06 23:53:25 +00:00
branch_len_bias: 0.0,
leaf_vertical_scale: 1.0,
proportionality: 0.0,
}
}
2021-02-07 19:55:13 +00:00
pub fn pine(rng: &mut impl Rng, scale: f32) -> Self {
let scale = scale * (1.0 + rng.gen::<f32>().powi(4));
let log_scale = 1.0 + scale.log2().max(0.0);
2021-02-06 23:53:25 +00:00
Self {
trunk_len: 32.0 * scale,
2021-02-07 19:55:13 +00:00
trunk_radius: 1.25 * scale,
branch_child_len: 0.3 / scale,
2021-02-06 23:53:25 +00:00
branch_child_radius: 0.0,
2021-02-07 19:55:13 +00:00
leaf_radius: 1.5 * log_scale..2.0 * log_scale,
2021-02-06 23:53:25 +00:00
straightness: 0.0,
max_depth: 1,
2021-02-07 19:55:13 +00:00
splits: 50.0..70.0,
2021-02-07 23:50:05 +00:00
split_range: 0.1..1.2,
2021-02-06 23:53:25 +00:00
branch_len_bias: 0.75,
leaf_vertical_scale: 0.3,
proportionality: 1.0,
}
}
2021-02-07 23:50:05 +00:00
pub fn giant(rng: &mut impl Rng, scale: f32) -> Self {
let scale = scale * (1.0 + rng.gen::<f32>().powi(4));
let log_scale = 1.0 + scale.log2().max(0.0);
Self {
trunk_len: 9.0 * scale,
trunk_radius: 4.0 * scale,
branch_child_len: 0.9,
branch_child_radius: 0.7,
leaf_radius: 2.25 * log_scale..3.5 * log_scale,
straightness: 0.5,
max_depth: (7.0 + log_scale) as usize,
splits: 1.5..2.75,
split_range: 1.0..1.1,
2021-02-07 23:50:05 +00:00
branch_len_bias: 0.0,
leaf_vertical_scale: 1.0,
proportionality: 0.0,
}
}
}
2021-01-03 20:43:07 +00:00
// TODO: Rename this to `Tree` when the name conflict is gone
2021-01-05 19:04:16 +00:00
pub struct ProceduralTree {
2021-01-03 20:43:07 +00:00
branches: Vec<Branch>,
trunk_idx: usize,
2021-01-03 20:43:07 +00:00
}
impl ProceduralTree {
/// Generate a new tree using the given configuration and seed.
2021-02-06 23:53:25 +00:00
pub fn generate(config: TreeConfig, rng: &mut impl Rng) -> Self {
let mut this = Self {
branches: Vec::new(),
trunk_idx: 0, // Gets replaced later
};
// Add the tree trunk (and sub-branches) recursively
let (trunk_idx, _) = this.add_branch(
&config,
// Our trunk starts at the origin...
Vec3::zero(),
// ...and has a roughly upward direction
2021-02-06 23:53:25 +00:00
Vec3::new(rng.gen_range(-1.0..1.0), rng.gen_range(-1.0..1.0), 10.0).normalized(),
2021-01-06 12:59:16 +00:00
config.trunk_len,
config.trunk_radius,
0,
None,
2021-02-06 23:53:25 +00:00
rng,
);
this.trunk_idx = trunk_idx;
2021-01-03 20:43:07 +00:00
this
}
2021-01-03 20:43:07 +00:00
2021-02-06 23:53:25 +00:00
// Recursively add a branch (with sub-branches) to the tree's branch graph,
// returning the index and AABB of the branch. This AABB gets propagated
// down to the parent and is used later during sampling to cull the branches to
// be sampled.
fn add_branch(
&mut self,
config: &TreeConfig,
start: Vec3<f32>,
dir: Vec3<f32>,
2021-01-06 12:59:16 +00:00
branch_len: f32,
branch_radius: f32,
depth: usize,
sibling_idx: Option<usize>,
rng: &mut impl Rng,
) -> (usize, Aabb<f32>) {
2021-01-06 12:59:16 +00:00
let end = start + dir * branch_len;
let line = LineSegment3 { start, end };
2021-01-06 12:59:16 +00:00
let wood_radius = branch_radius;
let leaf_radius = if depth == config.max_depth {
2021-01-31 14:09:50 +00:00
rng.gen_range(config.leaf_radius.clone())
} else {
0.0
};
2021-02-06 23:53:25 +00:00
// The AABB that covers this branch, along with wood and leaves that eminate
// from it
let mut aabb = Aabb {
min: Vec3::partial_min(start, end) - wood_radius.max(leaf_radius),
max: Vec3::partial_max(start, end) + wood_radius.max(leaf_radius),
};
let mut child_idx = None;
// Don't add child branches if we're already enough layers into the tree
if depth < config.max_depth {
2021-02-06 23:53:25 +00:00
let x_axis = dir
.cross(Vec3::<f32>::zero().map(|_| rng.gen_range(-1.0..1.0)))
.normalized();
let y_axis = dir.cross(x_axis).normalized();
2021-01-31 14:09:50 +00:00
let screw_shift = rng.gen_range(0.0..f32::consts::TAU);
2021-02-07 19:55:13 +00:00
let splits = rng.gen_range(config.splits.clone()).round() as usize;
for i in 0..splits {
2021-02-06 23:53:25 +00:00
let dist = Lerp::lerp(
2021-02-07 19:55:13 +00:00
i as f32 / (splits - 1) as f32,
2021-02-06 23:53:25 +00:00
rng.gen_range(0.0..1.0),
config.proportionality,
);
const PHI: f32 = 0.618;
const RAD_PER_BRANCH: f32 = f32::consts::TAU * PHI;
2021-02-07 19:55:13 +00:00
let screw = (screw_shift + dist * splits as f32 * RAD_PER_BRANCH).sin()
2021-02-06 23:53:25 +00:00
* x_axis
2021-02-07 19:55:13 +00:00
+ (screw_shift + dist * splits as f32 * RAD_PER_BRANCH).cos() * y_axis;
2021-02-06 23:53:25 +00:00
// Choose a point close to the branch to act as the target direction for the
// branch to grow in let split_factor =
// rng.gen_range(config.split_range.start, config.split_range.end).clamped(0.0,
// 1.0);
let split_factor =
Lerp::lerp(config.split_range.start, config.split_range.end, dist);
let tgt = Lerp::lerp_unclamped(start, end, split_factor)
+ Lerp::lerp(
Vec3::<f32>::zero().map(|_| rng.gen_range(-1.0..1.0)),
screw,
config.proportionality,
);
// Start the branch at the closest point to the target
let branch_start = line.projected_point(tgt);
2021-02-06 23:53:25 +00:00
// Now, interpolate between the target direction and the parent branch's
// direction to find a direction
let branch_dir =
Lerp::lerp(tgt - branch_start, dir, config.straightness).normalized();
2021-01-06 12:59:16 +00:00
let (branch_idx, branch_aabb) = self.add_branch(
config,
branch_start,
branch_dir,
branch_len
* config.branch_child_len
2021-02-06 23:53:25 +00:00
* (1.0
- (split_factor - 0.5)
* 2.0
* config.branch_len_bias.clamped(-1.0, 1.0)),
2021-01-06 12:59:16 +00:00
branch_radius * config.branch_child_radius,
depth + 1,
child_idx,
rng,
);
child_idx = Some(branch_idx);
2021-02-06 23:53:25 +00:00
// Parent branches AABBs include the AABBs of child branches to allow for
// culling during sampling
aabb.expand_to_contain(branch_aabb);
2021-01-04 21:19:43 +00:00
}
}
2021-01-03 20:43:07 +00:00
let idx = self.branches.len(); // Compute the index that this branch is going to have
self.branches.push(Branch {
line,
wood_radius,
leaf_radius,
leaf_vertical_scale: config.leaf_vertical_scale,
aabb,
sibling_idx,
child_idx,
});
(idx, aabb)
2021-01-03 20:43:07 +00:00
}
/// Get the bounding box that covers the tree (all branches and leaves)
2021-02-06 23:53:25 +00:00
pub fn get_bounds(&self) -> Aabb<f32> { self.branches[self.trunk_idx].aabb }
2021-01-03 20:43:07 +00:00
// Recursively search for branches or leaves by walking the tree's branch graph.
fn is_branch_or_leaves_at_inner(&self, pos: Vec3<f32>, branch_idx: usize) -> (bool, bool) {
let branch = &self.branches[branch_idx];
2021-02-06 23:53:25 +00:00
// Always probe the sibling branch, since our AABB doesn't include its bounds
// (it's not one of our children)
let branch_or_leaves = branch
.sibling_idx
.map(|idx| Vec2::from(self.is_branch_or_leaves_at_inner(pos, idx)))
.unwrap_or_default();
2021-02-06 23:53:25 +00:00
// Only continue probing this sub-graph of the tree if the sample position falls
// within its AABB
if branch.aabb.contains_point(pos) {
(branch_or_leaves
// Probe this branch
| Vec2::from(branch.is_branch_or_leaves_at(pos))
// Probe the children of this branch
| branch.child_idx
.map(|idx| Vec2::from(self.is_branch_or_leaves_at_inner(pos, idx)))
.unwrap_or_default())
2021-02-06 23:53:25 +00:00
.into_tuple()
} else {
branch_or_leaves.into_tuple()
}
}
2021-02-06 23:53:25 +00:00
/// Determine whether there are either branches or leaves at the given
/// position in the tree.
#[inline(always)]
pub fn is_branch_or_leaves_at(&self, pos: Vec3<f32>) -> (bool, bool) {
self.is_branch_or_leaves_at_inner(pos, self.trunk_idx)
2021-01-03 20:43:07 +00:00
}
}
2021-02-06 23:53:25 +00:00
// Branches are arranged in a graph shape. Each branch points to both its first
// child (if any) and also to the next branch in the list of child branches
// associated with the parent. This means that the entire tree is laid out in a
// walkable graph where each branch refers only to two other branches. As a
// result, walking the tree is simply a case of performing double recursion.
2021-01-03 20:43:07 +00:00
struct Branch {
line: LineSegment3<f32>,
wood_radius: f32,
leaf_radius: f32,
leaf_vertical_scale: f32,
aabb: Aabb<f32>,
sibling_idx: Option<usize>,
child_idx: Option<usize>,
2021-01-03 20:43:07 +00:00
}
2021-01-05 16:21:15 +00:00
impl Branch {
2021-02-06 23:53:25 +00:00
/// Determine whether there are either branches or leaves at the given
/// position in the branch.
pub fn is_branch_or_leaves_at(&self, pos: Vec3<f32>) -> (bool, bool) {
// fn finvsqrt(x: f32) -> f32 {
// let y = f32::from_bits(0x5f375a86 - (x.to_bits() >> 1));
// y * (1.5 - ( x * 0.5 * y * y ))
// }
let p = self.line.projected_point(pos);
let p_d2 = p.distance_squared(pos);
if p_d2 < self.wood_radius.powi(2) {
(true, false)
} else {
let diff = (p - pos) / Vec3::new(1.0, 1.0, self.leaf_vertical_scale);
(false, diff.magnitude_squared() < self.leaf_radius.powi(2))
2021-01-05 16:21:15 +00:00
}
}
}