elseware/src/ship/drops/rare_drop_table.rs

212 lines
7.3 KiB
Rust
Raw Normal View History

2020-03-26 00:06:28 -07:00
use std::collections::HashMap;
use rand::Rng;
use serde::{Serialize, Deserialize};
use crate::entity::item::weapon::{Weapon, WeaponType};
use crate::entity::item::armor::{Armor, ArmorType};
use crate::entity::item::shield::{Shield, ShieldType};
use crate::entity::item::unit::{Unit, UnitType};
use crate::entity::item::tool::{Tool, ToolType};
2020-03-28 12:06:24 -07:00
use crate::entity::item::mag::{Mag, MagType};
2020-03-26 00:06:28 -07:00
use crate::entity::character::SectionID;
use crate::ship::monster::MonsterType;
use crate::ship::room::{Difficulty, Episode};
2020-04-03 22:51:55 -07:00
use crate::ship::map::MapArea;
2020-03-29 11:54:58 -07:00
use crate::ship::drops::{ItemDropType, load_data_file};
2020-03-26 00:06:28 -07:00
use crate::ship::drops::generic_weapon::AttributeTable;
use crate::ship::drops::generic_armor::GenericArmorTable;
use crate::ship::drops::generic_shield::GenericShieldTable;
2022-08-02 18:37:06 -06:00
type ItemParseFn = Box<dyn Fn(&String) -> Option<RareDropItem>>;
2020-03-26 00:06:28 -07:00
#[derive(Debug, Copy, Clone)]
2020-03-28 12:06:24 -07:00
pub enum RareDropItem {
2020-03-26 00:06:28 -07:00
Weapon(WeaponType),
Armor(ArmorType),
Shield(ShieldType),
Unit(UnitType),
Tool(ToolType),
2020-03-28 12:06:24 -07:00
Mag(MagType)
2020-03-26 00:06:28 -07:00
}
impl RareDropItem {
2020-03-28 12:06:24 -07:00
pub fn from_string(name: String) -> RareDropItem {
2022-08-02 18:37:06 -06:00
let parse_funcs: [ItemParseFn; 6] = [
2021-06-18 12:45:00 -06:00
Box::new(|i| Some(RareDropItem::Weapon(str::parse::<WeaponType>(i).ok()?))),
Box::new(|i| Some(RareDropItem::Armor(str::parse::<ArmorType>(i).ok()?))),
Box::new(|i| Some(RareDropItem::Shield(str::parse::<ShieldType>(i).ok()?))),
Box::new(|i| Some(RareDropItem::Unit(str::parse::<UnitType>(i).ok()?))),
Box::new(|i| Some(RareDropItem::Tool(str::parse::<ToolType>(i).ok()?))),
Box::new(|i| Some(RareDropItem::Mag(str::parse::<MagType>(i).ok()?))),
2020-03-28 12:06:24 -07:00
];
2020-03-26 00:06:28 -07:00
for parse in parse_funcs.iter() {
2021-06-18 12:45:00 -06:00
if let Some (k) = parse(&name) {
return k
}
2020-03-26 00:06:28 -07:00
}
panic!()
}
}
2023-01-29 15:31:49 -07:00
pub struct RareDropRate {
pub rate: f32,
pub item: RareDropItem
2020-03-26 00:06:28 -07:00
}
#[derive(Debug, Serialize, Deserialize)]
pub struct RareDropConfigEntity {
pub rate: f32,
pub item: String,
}
pub struct RareDropTable {
rates: HashMap<MonsterType, Vec<RareDropRate>>,
attribute_table: AttributeTable,
armor_stats: GenericArmorTable,
shield_stats: GenericShieldTable,
}
2023-01-29 15:31:49 -07:00
fn load_default_monster_rates(episode: Episode, difficulty: Difficulty, section_id: SectionID) -> HashMap<MonsterType, Vec<RareDropRate>> {
let cfg: HashMap<String, Vec<RareDropConfigEntity>> = load_data_file(episode, difficulty, section_id, "rare_rate.toml");
cfg.into_iter()
.map(|(monster, drops)| {
let monster = monster.parse().unwrap();
let drops = drops.into_iter().map(|drop| {
RareDropRate {
rate: drop.rate,
item: RareDropItem::from_string(drop.item),
}
2020-03-26 00:06:28 -07:00
}).collect();
2023-01-29 15:31:49 -07:00
(monster, drops)
}).collect()
}
2020-03-26 00:06:28 -07:00
2023-01-29 15:31:49 -07:00
impl RareDropTable {
pub fn new(episode: Episode, difficulty: Difficulty, section_id: SectionID) -> RareDropTable {
2020-03-26 00:06:28 -07:00
RareDropTable {
2023-01-29 15:31:49 -07:00
rates: load_default_monster_rates(episode, difficulty, section_id),
2020-03-26 00:06:28 -07:00
attribute_table: AttributeTable::new(episode, difficulty, section_id),
armor_stats: GenericArmorTable::new(episode, difficulty, section_id),
shield_stats: GenericShieldTable::new(episode, difficulty, section_id),
}
}
2023-01-29 15:31:49 -07:00
pub fn builder() -> RareDropTableBuilder {
RareDropTableBuilder {
rates: None,
attribute_table: None,
armor_stats: None,
shield_stats: None,
}
}
2020-04-03 22:51:55 -07:00
pub fn apply_item_stats<R: Rng>(&self, map_area: &MapArea, item: RareDropItem, rng: &mut R) -> ItemDropType {
2020-03-26 00:06:28 -07:00
match item {
RareDropItem::Weapon(weapon) => {
2020-03-29 11:54:58 -07:00
ItemDropType::Weapon(Weapon {
2021-06-18 20:38:29 -06:00
weapon,
2020-03-26 00:06:28 -07:00
special: None,
grind: 0,
attrs: self.attribute_table.generate_rare_attributes(map_area, rng),
tekked: false,
})
},
RareDropItem::Armor(armor) => {
2020-03-29 11:54:58 -07:00
ItemDropType::Armor(Armor {
2021-06-18 20:38:29 -06:00
armor,
2020-03-26 00:06:28 -07:00
dfp: self.armor_stats.dfp_modifier(&armor, rng) as u8,
evp: self.armor_stats.evp_modifier(&armor, rng) as u8,
slots: self.armor_stats.slots(map_area, rng) as u8,
})
},
RareDropItem::Shield(shield) => {
2020-03-29 11:54:58 -07:00
ItemDropType::Shield(Shield {
2021-06-18 20:38:29 -06:00
shield,
2020-03-26 00:06:28 -07:00
dfp: self.shield_stats.dfp_modifier(&shield, rng) as u8,
evp: self.shield_stats.evp_modifier(&shield, rng) as u8,
})
},
RareDropItem::Unit(unit) => {
2020-03-29 11:54:58 -07:00
ItemDropType::Unit(Unit {
2021-06-18 20:38:29 -06:00
unit,
2020-03-26 00:06:28 -07:00
modifier: None,
})
},
RareDropItem::Tool(tool) => {
2020-03-29 11:54:58 -07:00
ItemDropType::Tool(Tool {
2021-06-18 20:38:29 -06:00
tool,
2020-03-26 00:06:28 -07:00
})
2020-03-28 12:06:24 -07:00
},
2020-10-26 00:02:48 -06:00
RareDropItem::Mag(_mag) => {
2020-08-20 22:59:40 -06:00
ItemDropType::Mag(Mag::baby_mag(rng.gen_range(0, 18)))
2020-03-26 00:06:28 -07:00
}
}
}
2020-04-03 22:51:55 -07:00
pub fn get_drop<R: Rng>(&self, map_area: &MapArea, monster: &MonsterType, rng: &mut R) -> Option<ItemDropType> {
2020-03-26 00:06:28 -07:00
self.rates.get(monster)
.and_then(|drop_rates| {
drop_rates.iter()
.filter_map(|drop_rate| {
let rand: f32 = rng.gen();
if rand < drop_rate.rate {
Some(self.apply_item_stats(map_area, drop_rate.item, rng))
}
else {
None
}
2021-06-18 12:45:00 -06:00
}).next()
2020-03-26 00:06:28 -07:00
})
}
}
2023-01-29 15:31:49 -07:00
pub struct RareDropTableBuilder {
rates: Option<HashMap<MonsterType, Vec<RareDropRate>>>,
attribute_table: Option<AttributeTable>,
armor_stats: Option<GenericArmorTable>,
shield_stats: Option<GenericShieldTable>,
}
// TODO: add the rest of these later I just need these ones right now
impl RareDropTableBuilder {
pub fn rates(mut self, rates: HashMap<MonsterType, Vec<RareDropRate>>) -> RareDropTableBuilder {
self.rates = Some(rates);
self
}
#[must_use]
pub fn rate(mut self, monster_type: MonsterType, drop_rate: RareDropRate) -> RareDropTableBuilder {
match &mut self.rates {
Some(rates) => {
rates.entry(monster_type)
.or_insert(Vec::new())
.push(drop_rate);
},
None => {
let mut rates = HashMap::default();
rates.insert(monster_type, vec![drop_rate]);
self.rates = Some(rates);
}
}
self
}
pub fn build(self, episode: Episode, difficulty: Difficulty, section_id: SectionID) -> RareDropTable {
RareDropTable {
rates: self.rates.unwrap_or_else(|| load_default_monster_rates(episode, difficulty, section_id)),
attribute_table: self.attribute_table.unwrap_or_else(|| AttributeTable::new(episode, difficulty, section_id)),
armor_stats: self.armor_stats.unwrap_or_else(|| GenericArmorTable::new(episode, difficulty, section_id)),
shield_stats: self.shield_stats.unwrap_or_else(|| GenericShieldTable::new(episode, difficulty, section_id)),
}
}
}