From 7a03c689f65000c72114075663b80d7f82c37d62 Mon Sep 17 00:00:00 2001 From: juliancoffee Date: Sat, 19 Jun 2021 23:19:43 +0300 Subject: [PATCH 1/3] Add config file for dungeon distribution - Now you can specify probability of "spawn" of dungeon with specific difficulty --- .../world/dungeon/difficulty_distribution.ron | 20 +++++++++++ world/src/site/dungeon/mod.rs | 35 ++++++++++++++----- 2 files changed, 47 insertions(+), 8 deletions(-) create mode 100644 assets/world/dungeon/difficulty_distribution.ron diff --git a/assets/world/dungeon/difficulty_distribution.ron b/assets/world/dungeon/difficulty_distribution.ron new file mode 100644 index 0000000000..09cd410377 --- /dev/null +++ b/assets/world/dungeon/difficulty_distribution.ron @@ -0,0 +1,20 @@ +/// Distribution of different dungeon levels. +/// +/// first number is dungeon level, integer +/// second number is probability, any normal positive float (not a NaN, for example) +/// +/// Values are relative to each other, +/// lesser probability means there will be less dungeons. +/// +/// General rules: +/// 1) Probability should not be less or equal to zero +/// 2) Keep it synced with number of dungeon levels +/// 3) Keep these pairs sorted from lowest to highest tier +([ + (0, 1.0), + (1, 1.0), + (2, 1.0), + (3, 1.0), + (4, 1.0), + (5, 1.0), +]) diff --git a/world/src/site/dungeon/mod.rs b/world/src/site/dungeon/mod.rs index 72f9763664..981e3f5650 100644 --- a/world/src/site/dungeon/mod.rs +++ b/world/src/site/dungeon/mod.rs @@ -9,7 +9,7 @@ use crate::{ }; use common::{ - assets::AssetHandle, + assets::{self, AssetExt, AssetHandle}, astar::Astar, comp::{self}, generation::{ChunkSupplement, EntityInfo}, @@ -20,7 +20,7 @@ use common::{ use core::{f32, hash::BuildHasherDefault}; use fxhash::FxHasher64; use lazy_static::lazy_static; -use rand::prelude::*; +use rand::{prelude::*, seq::SliceRandom}; use serde::Deserialize; use vek::*; @@ -47,13 +47,34 @@ pub struct Colors { const ALT_OFFSET: i32 = -2; +#[derive(Deserialize)] +struct DungeonDistribution(Vec<(u32, f32)>); +impl assets::Asset for DungeonDistribution { + type Loader = assets::RonLoader; + + const EXTENSION: &'static str = "ron"; +} + +lazy_static! { + static ref DUNGEON_DISTRIBUTION: Vec<(u32, f32)> = + DungeonDistribution::load_expect("world.dungeon.difficulty_distribution") + .read() + .0 + .clone(); +} + impl Dungeon { - #[allow(clippy::let_and_return)] // TODO: Pending review in #587 pub fn generate(wpos: Vec2, sim: Option<&WorldSim>, rng: &mut impl Rng) -> Self { let mut ctx = GenCtx { sim, rng }; - let difficulty = ctx.rng.gen_range(0..6); + + let difficulty = DUNGEON_DISTRIBUTION + .choose_weighted(&mut ctx.rng, |pair| pair.1) + .map(|(difficulty, _)| *difficulty) + .expect("this can never fail"); + let floors = 3 + difficulty / 2; - let this = Self { + + Self { name: { let name = NameGen::location(ctx.rng).generate(); match ctx.rng.gen_range(0..5) { @@ -81,9 +102,7 @@ impl Dungeon { }) .collect(), difficulty, - }; - - this + } } pub fn name(&self) -> &str { &self.name } From 9cf83dc6e01d9a5148a1c695c9aa8fee7bcf49b0 Mon Sep 17 00:00:00 2001 From: juliancoffee Date: Sat, 19 Jun 2021 23:32:12 +0300 Subject: [PATCH 2/3] Tip how to make map with dungeons of same level --- assets/world/dungeon/difficulty_distribution.ron | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/assets/world/dungeon/difficulty_distribution.ron b/assets/world/dungeon/difficulty_distribution.ron index 09cd410377..50a5ce163a 100644 --- a/assets/world/dungeon/difficulty_distribution.ron +++ b/assets/world/dungeon/difficulty_distribution.ron @@ -10,6 +10,10 @@ /// 1) Probability should not be less or equal to zero /// 2) Keep it synced with number of dungeon levels /// 3) Keep these pairs sorted from lowest to highest tier +/// +/// Tips: +/// 1) Set every probability to 0.0 and left one with any other number +/// and you will have map full of dungeons of same level ([ (0, 1.0), (1, 1.0), From 5491902a1c3aef7f2e6d5691dea3803b375efe17 Mon Sep 17 00:00:00 2001 From: juliancoffee Date: Sun, 20 Jun 2021 00:14:54 +0300 Subject: [PATCH 3/3] Create distribution map for dungeons * tl;dr T0 - 27%, T1/T2 - 22%, T3/T4 - 11%, T5 - 5%. Before every dungeon had 16% chance to be created. --- .../world/dungeon/difficulty_distribution.ron | 21 ++++++++++--------- world/src/site/dungeon/mod.rs | 7 ++++++- 2 files changed, 17 insertions(+), 11 deletions(-) diff --git a/assets/world/dungeon/difficulty_distribution.ron b/assets/world/dungeon/difficulty_distribution.ron index 50a5ce163a..1460bfc6e6 100644 --- a/assets/world/dungeon/difficulty_distribution.ron +++ b/assets/world/dungeon/difficulty_distribution.ron @@ -1,24 +1,25 @@ /// Distribution of different dungeon levels. /// /// first number is dungeon level, integer -/// second number is probability, any normal positive float (not a NaN, for example) +/// second number is weight, any normal positive float (not a NaN, for example) /// /// Values are relative to each other, -/// lesser probability means there will be less dungeons. +/// lesser weight means there will be less dungeons of that tier. /// /// General rules: -/// 1) Probability should not be less or equal to zero -/// 2) Keep it synced with number of dungeon levels -/// 3) Keep these pairs sorted from lowest to highest tier +/// 1) Weight should not be less then zero +/// 2) At least some of weights shouldn't be a zero +/// 3) Keep it synced with number of dungeon levels +/// 4) Keep these pairs sorted from lowest to highest tier /// /// Tips: /// 1) Set every probability to 0.0 and left one with any other number /// and you will have map full of dungeons of same level ([ - (0, 1.0), - (1, 1.0), - (2, 1.0), - (3, 1.0), - (4, 1.0), + (0, 5.0), + (1, 4.0), + (2, 4.0), + (3, 2.0), + (4, 2.0), (5, 1.0), ]) diff --git a/world/src/site/dungeon/mod.rs b/world/src/site/dungeon/mod.rs index 981e3f5650..bfbfe39a9d 100644 --- a/world/src/site/dungeon/mod.rs +++ b/world/src/site/dungeon/mod.rs @@ -70,7 +70,12 @@ impl Dungeon { let difficulty = DUNGEON_DISTRIBUTION .choose_weighted(&mut ctx.rng, |pair| pair.1) .map(|(difficulty, _)| *difficulty) - .expect("this can never fail"); + .unwrap_or_else(|err| { + panic!( + "Failed to choose difficulty (check instruction in config). Error: {}", + err + ) + }); let floors = 3 + difficulty / 2;