veloren/world/src/lib.rs

161 lines
4.4 KiB
Rust
Raw Normal View History

2019-06-22 21:44:27 +00:00
#![feature(euclidean_division, bind_by_move_pattern_guards, option_flattening)]
2019-06-11 18:39:25 +00:00
mod all;
2019-06-09 10:24:18 +00:00
mod block;
mod column;
2019-06-15 12:34:28 +00:00
pub mod config;
2019-06-18 21:22:31 +00:00
pub mod sim;
2019-06-15 12:34:28 +00:00
pub mod util;
2019-06-09 10:24:18 +00:00
// Reexports
pub use crate::config::CONFIG;
2019-06-15 10:36:26 +00:00
use crate::{
block::BlockGen,
2019-06-15 12:34:28 +00:00
column::{ColumnGen, ColumnSample},
2019-07-01 18:40:41 +00:00
util::{Sampler, SamplerMut},
2019-06-15 10:36:26 +00:00
};
2019-01-15 15:13:11 +00:00
use common::{
terrain::{Block, TerrainChunk, TerrainChunkMeta, TerrainChunkSize},
2019-08-02 14:37:26 +00:00
vol::{ReadVol, VolSize, Vox, WriteVol},
2019-01-15 15:13:11 +00:00
};
2019-08-02 14:37:26 +00:00
use rand::Rng;
2019-06-09 10:24:18 +00:00
use std::time::Duration;
use vek::*;
2019-01-15 15:13:11 +00:00
#[derive(Debug)]
pub enum Error {
Other(String),
}
pub struct World {
sim: sim::WorldSim,
}
2019-01-15 15:13:11 +00:00
impl World {
pub fn generate(seed: u32) -> Self {
Self {
sim: sim::WorldSim::generate(seed),
}
}
pub fn sim(&self) -> &sim::WorldSim {
&self.sim
}
2019-07-01 18:40:41 +00:00
pub fn tick(&self, _dt: Duration) {
// TODO
2019-01-15 15:13:11 +00:00
}
2019-06-15 12:34:28 +00:00
pub fn sample_columns(
&self,
) -> impl Sampler<Index = Vec2<i32>, Sample = Option<ColumnSample>> + '_ {
ColumnGen::new(self)
}
2019-07-04 23:14:55 +00:00
pub fn sample_blocks(&self) -> BlockGen {
2019-06-09 10:24:18 +00:00
BlockGen::new(self, ColumnGen::new(self))
}
2019-08-02 14:37:26 +00:00
pub fn generate_chunk(&self, chunk_pos: Vec2<i32>) -> (TerrainChunk, ChunkSupplement) {
2019-01-23 20:01:58 +00:00
let air = Block::empty();
2019-06-29 21:09:36 +00:00
let stone = Block::new(2, Rgb::new(200, 220, 255));
let water = Block::new(5, Rgb::new(100, 150, 255));
2019-01-23 20:01:58 +00:00
2019-06-04 17:19:40 +00:00
let chunk_size2d = Vec2::from(TerrainChunkSize::SIZE);
2019-06-19 14:55:26 +00:00
let (base_z, sim_chunk) = match self
.sim
.get_interpolated(
chunk_pos.map2(chunk_size2d, |e, sz: u32| e * sz as i32 + sz as i32 / 2),
|chunk| chunk.get_base_z(),
)
.and_then(|base_z| self.sim.get(chunk_pos).map(|sim_chunk| (base_z, sim_chunk)))
{
2019-06-18 21:22:31 +00:00
Some((base_z, sim_chunk)) => (base_z as i32, sim_chunk),
2019-07-09 20:42:27 +00:00
None => {
2019-08-02 14:37:26 +00:00
return (
TerrainChunk::new(
CONFIG.sea_level as i32,
water,
air,
TerrainChunkMeta::void(),
),
ChunkSupplement::default(),
2019-07-09 20:42:27 +00:00
)
}
};
2019-06-25 15:59:09 +00:00
let meta = TerrainChunkMeta::new(sim_chunk.get_name(&self.sim), sim_chunk.get_biome());
2019-06-15 12:34:28 +00:00
let mut sampler = self.sample_blocks();
2019-08-02 14:37:26 +00:00
let chunk_block_pos = Vec3::from(chunk_pos) * TerrainChunkSize::SIZE.map(|e| e as i32);
let mut chunk = TerrainChunk::new(base_z, stone, air, meta);
for x in 0..TerrainChunkSize::SIZE.x as i32 {
for y in 0..TerrainChunkSize::SIZE.y as i32 {
let wpos2d = Vec2::new(x, y)
+ Vec3::from(chunk_pos) * TerrainChunkSize::SIZE.map(|e| e as i32);
let z_cache = match sampler.get_z_cache(wpos2d) {
Some(z_cache) => z_cache,
None => continue,
};
2019-07-04 23:14:55 +00:00
let (min_z, max_z) = z_cache.get_z_limits();
for z in base_z..min_z as i32 {
let _ = chunk.set(Vec3::new(x, y, z), stone);
}
for z in min_z as i32..max_z as i32 {
let lpos = Vec3::new(x, y, z);
2019-08-02 14:37:26 +00:00
let wpos = chunk_block_pos + lpos;
if let Some(block) = sampler.get_with_z_cache(wpos, Some(&z_cache)) {
2019-06-09 10:24:18 +00:00
let _ = chunk.set(lpos, block);
}
}
}
2019-01-23 20:01:58 +00:00
}
2019-08-02 14:37:26 +00:00
let gen_entity_pos = || {
let lpos2d = Vec2::from(TerrainChunkSize::SIZE)
.map(|sz| rand::thread_rng().gen::<u32>().rem_euclid(sz));
let mut lpos = Vec3::new(lpos2d.x as i32, lpos2d.y as i32, 0);
while chunk.get(lpos).map(|vox| !vox.is_empty()).unwrap_or(false) {
lpos.z += 1;
}
(chunk_block_pos + lpos).map(|e| e as f32) + 0.5
};
const SPAWN_RATE: f32 = 0.1;
let supplement = ChunkSupplement {
npcs: if rand::thread_rng().gen::<f32>() < SPAWN_RATE {
vec![NpcInfo {
pos: gen_entity_pos(),
}]
} else {
Vec::new()
},
};
(chunk, supplement)
}
}
pub struct NpcInfo {
pub pos: Vec3<f32>,
}
pub struct ChunkSupplement {
pub npcs: Vec<NpcInfo>,
}
impl Default for ChunkSupplement {
fn default() -> Self {
Self { npcs: Vec::new() }
}
}