veloren/common/src/lottery.rs

110 lines
3.3 KiB
Rust
Raw Normal View History

// Example for calculating a drop rate:
//
// On every roll an f32 between 0 and 1 is created.
// For every loot table a total range is created by the sum of the individual
// ranges per item.
//
// This range is the sum of all single ranges defined per item in a table.
// // Individual Range
// (3, "common.items.food.cheese"), // 0.0..3.0
// (3, "common.items.food.apple"), // 3.0..6.0
// (3, "common.items.food.mushroom"), // 6.0..9.0
// (1, "common.items.food.coconut"), // 9.0..10.0
// (0.05, "common.items.food.apple_mushroom_curry"), // 10.0..10.05
// (0.10, "common.items.food.apple_stick"), // 10.05..10.15
// (0.10, "common.items.food.mushroom_stick"), // 10.15..10.25
//
// The f32 is multiplied by the max. value needed to drop an item in this
// particular table. X = max. value needed = 10.15
//
// Example roll
// [Random Value 0..1] * X = Number inside the table's total range
// 0.45777 * X = 4.65
// 4.65 is in the range of 3.0..6.0 => Apple drops
//
// Example drop chance calculation
// Cheese drop rate = 3/X = 29.6%
// Coconut drop rate = 1/X = 9.85%
use crate::{assets::{self, AssetExt}, comp::Item};
2020-07-03 13:49:17 +00:00
use rand::prelude::*;
use serde::{de::DeserializeOwned, Deserialize, Serialize};
2020-07-03 13:49:17 +00:00
#[derive(Clone, Debug, PartialEq, Deserialize)]
2020-07-03 13:49:17 +00:00
pub struct Lottery<T> {
items: Vec<(f32, T)>,
2020-07-03 13:49:17 +00:00
total: f32,
}
2020-12-12 22:14:24 +00:00
impl<T: DeserializeOwned + Send + Sync + 'static> assets::Asset for Lottery<T> {
type Loader = assets::LoadFrom<Vec<(f32, T)>, assets::RonLoader>;
2020-12-13 01:09:57 +00:00
const EXTENSION: &'static str = "ron";
2020-07-03 13:49:17 +00:00
}
2020-12-12 22:14:24 +00:00
impl<T> From<Vec<(f32, T)>> for Lottery<T> {
fn from(mut items: Vec<(f32, T)>) -> Lottery<T> {
2020-07-03 13:49:17 +00:00
let mut total = 0.0;
2020-12-12 22:14:24 +00:00
for (rate, _) in &mut items {
total += *rate;
*rate = total - *rate;
}
2020-07-03 13:49:17 +00:00
Self { items, total }
}
2020-12-12 22:14:24 +00:00
}
2020-07-03 13:49:17 +00:00
2020-12-12 22:14:24 +00:00
impl<T> Lottery<T> {
pub fn choose_seeded(&self, seed: u32) -> &T {
let x = ((seed % 65536) as f32 / 65536.0) * self.total;
&self.items[self
.items
.binary_search_by(|(y, _)| y.partial_cmp(&x).unwrap())
.unwrap_or_else(|i| i.saturating_sub(1))]
.1
2020-07-03 13:49:17 +00:00
}
2020-08-12 14:10:12 +00:00
pub fn choose(&self) -> &T { self.choose_seeded(thread_rng().gen()) }
pub fn iter(&self) -> impl Iterator<Item = &(f32, T)> { self.items.iter() }
2021-03-22 09:39:35 +00:00
pub fn total(&self) -> f32 { self.total }
}
#[derive(Clone, Debug, PartialEq, Deserialize, Serialize)]
pub enum LootSpec {
Item(String),
LootTable(String),
}
impl LootSpec {
pub fn to_item(&self) -> Item {
match self {
Self::Item(item) => Item::new_from_asset_expect(&item),
Self::LootTable(table) => {
Lottery::<LootSpec>::load_expect(&table).read().choose().to_item()
},
}
}
}
#[cfg(test)]
mod tests {
use super::*;
2020-12-12 22:14:24 +00:00
use crate::{assets::AssetExt, comp::Item};
#[test]
fn test_loot_table() {
let test = Lottery::<LootSpec>::load_expect("common.loot_tables.fallback");
for (_, to_itemifier) in test.read().iter() {
assert!(
Item::new_from_asset(to_itemifier).is_ok(),
"Invalid loot table item '{}'",
to_itemifier
);
}
}
}