use common::{ comp::{ inventory::trade_pricing::TradePricing, item::{ armor::{Armor, ArmorKind, Protection}, tool::{Hands, StatKind, Stats, Tool, ToolKind}, Item, ItemDesc, ItemKind, MaterialStatManifest, ModularComponent, }, BuffKind, }, effect::Effect, trade::{Good, SitePrices}, }; use std::{borrow::Cow, fmt::Write}; pub fn loadout_slot_text<'a>( item: Option<&'a impl ItemDesc>, mut empty: impl FnMut() -> (&'a str, &'a str), msm: &'a MaterialStatManifest, ) -> (&'a str, Cow<'a, str>) { item.map_or_else( || { let (title, desc) = empty(); (title, Cow::Borrowed(desc)) }, |item| item_text(item, msm), ) } pub fn item_text<'a>( item: &'a impl ItemDesc, msm: &'a MaterialStatManifest, ) -> (&'a str, Cow<'a, str>) { let desc: Cow = match item.kind() { ItemKind::Armor(armor) => { Cow::Owned(armor_desc(armor, item.description(), item.num_slots())) }, ItemKind::Tool(tool) => Cow::Owned(tool_desc( &tool, item.components(), &msm, item.description(), )), ItemKind::ModularComponent(mc) => Cow::Owned(modular_component_desc( mc, item.components(), &msm, item.description(), )), ItemKind::Glider(_glider) => Cow::Owned(glider_desc(item.description())), ItemKind::Consumable { effect, .. } => { Cow::Owned(consumable_desc(effect, item.description())) }, ItemKind::Throwable { .. } => Cow::Owned(throwable_desc(item.description())), ItemKind::Utility { .. } => Cow::Owned(utility_desc(item.description())), ItemKind::Ingredient { .. } => Cow::Owned(ingredient_desc( item.description(), item.item_definition_id(), msm, )), ItemKind::Lantern { .. } => Cow::Owned(lantern_desc(item.description())), ItemKind::TagExamples { .. } => Cow::Borrowed(item.description()), //_ => Cow::Borrowed(item.description()), }; (item.name(), desc) } pub fn append_price_desc(desc: &mut String, prices: &Option, item_definition_id: &str) { if let Some(prices) = prices { let (material, factor) = TradePricing::get_material(item_definition_id); let coinprice = prices.values.get(&Good::Coin).cloned().unwrap_or(1.0); let buyprice = prices.values.get(&material).cloned().unwrap_or_default() * factor; let sellprice = buyprice * material.trade_margin(); *desc += &format!( "\n\nBuy price: {:0.1} coins\nSell price: {:0.1} coins", buyprice / coinprice, sellprice / coinprice ); } } // TODO: localization fn modular_component_desc( mc: &ModularComponent, components: &[Item], msm: &MaterialStatManifest, description: &str, ) -> String { let stats = StatKind::Direct(mc.stats).resolve_stats(msm, components); let statblock = statblock_desc(&stats); let mut result = format!("Modular Component\n\n{}\n\n{}", statblock, description); if !components.is_empty() { result += "\n\nMade from:\n"; for component in components { result += component.name(); result += "\n" } result += "\n"; } result } fn glider_desc(desc: &str) -> String { format!("Glider\n\n{}\n\n", desc) } fn consumable_desc(effects: &[Effect], desc: &str) -> String { // TODO: localization let mut description = "Consumable".to_string(); for effect in effects { if let Effect::Buff(buff) = effect { let strength = buff.data.strength * 0.1; let dur_secs = buff.data.duration.map(|d| d.as_secs_f32()); let str_total = dur_secs.map_or(strength, |secs| strength * secs); let buff_desc = match buff.kind { BuffKind::Saturation | BuffKind::Regeneration | BuffKind::Potion => { format!("Restores {} Health", str_total) }, BuffKind::IncreaseMaxEnergy => { format!("Raises Maximum Stamina by {}", strength) }, BuffKind::IncreaseMaxHealth => { format!("Raises Maximum Health by {}", strength) }, BuffKind::Invulnerability => "Grants invulnerability".to_string(), BuffKind::Bleeding | BuffKind::CampfireHeal | BuffKind::Cursed | BuffKind::ProtectingWard => continue, }; write!(&mut description, "\n\n{}", buff_desc).unwrap(); let dur_desc = if dur_secs.is_some() { match buff.kind { BuffKind::Saturation | BuffKind::Regeneration => { format!("over {} seconds", dur_secs.unwrap()) }, BuffKind::IncreaseMaxEnergy | BuffKind::IncreaseMaxHealth | BuffKind::Invulnerability => { format!("for {} seconds", dur_secs.unwrap()) }, BuffKind::Bleeding | BuffKind::Potion | BuffKind::CampfireHeal | BuffKind::Cursed | BuffKind::ProtectingWard => continue, } } else if let BuffKind::Saturation | BuffKind::Regeneration = buff.kind { "every second".to_string() } else { continue; }; write!(&mut description, " {}", dur_desc).unwrap(); } } if !desc.is_empty() { write!(&mut description, "\n\n{}", desc).unwrap(); } write!(&mut description, "\n\n").unwrap(); description } fn throwable_desc(desc: &str) -> String { format!("Can be thrown\n\n{}\n\n", desc) } fn utility_desc(desc: &str) -> String { format!("{}\n\n", desc) } fn ingredient_desc(desc: &str, item_id: &str, msm: &MaterialStatManifest) -> String { let mut result = format!("Crafting Ingredient\n\n{}", desc); if let Some(stats) = msm.0.get(item_id) { result += "\n\nStat multipliers:\n"; result += &statblock_desc(stats); } result } fn lantern_desc(desc: &str) -> String { format!("Lantern\n\n{}\n\n", desc) } fn armor_desc(armor: &Armor, desc: &str, slots: u16) -> String { // TODO: localization let kind = match armor.kind { ArmorKind::Shoulder(_) => "Shoulders", ArmorKind::Chest(_) => "Chest", ArmorKind::Belt(_) => "Belt", ArmorKind::Hand(_) => "Hands", ArmorKind::Pants(_) => "Legs", ArmorKind::Foot(_) => "Feet", ArmorKind::Back(_) => "Back", ArmorKind::Ring(_) => "Ring", ArmorKind::Neck(_) => "Neck", ArmorKind::Head(_) => "Head", ArmorKind::Tabard(_) => "Tabard", ArmorKind::Bag(_) => "Bag", }; let armor_protection = match armor.get_protection() { Protection::Normal(a) => a.to_string(), Protection::Invincible => "Inf".to_string(), }; let armor_poise_resilience = match armor.get_poise_resilience() { Protection::Normal(a) => a.to_string(), Protection::Invincible => "Inf".to_string(), }; let mut description = format!( "{}\n\nArmor: {}\n\nPoise Resilience: {}", kind, armor_protection, armor_poise_resilience ); if !desc.is_empty() { write!(&mut description, "\n\n{}", desc).unwrap(); } if slots > 0 { write!(&mut description, "\n\nSlots: {}", slots).unwrap(); } write!(&mut description, "\n\n").unwrap(); description } fn tool_desc(tool: &Tool, components: &[Item], msm: &MaterialStatManifest, desc: &str) -> String { let kind = match tool.kind { ToolKind::Sword => "Sword", ToolKind::Axe => "Axe", ToolKind::Hammer => "Hammer", ToolKind::Bow => "Bow", ToolKind::Dagger => "Dagger", ToolKind::Staff => "Staff", ToolKind::Sceptre => "Sceptre", ToolKind::Shield => "Shield", ToolKind::Spear => "Spear", ToolKind::HammerSimple => "HammerSimple", ToolKind::SwordSimple => "SwordSimple", ToolKind::StaffSimple => "StaffSimple", ToolKind::AxeSimple => "AxeSimple", ToolKind::BowSimple => "BowSimple", ToolKind::Unique(_) => "Unique", ToolKind::Debug => "Debug", ToolKind::Farming => "Farming Tool", ToolKind::Pick => "Pickaxe", ToolKind::Empty => "Empty", }; // Get tool stats let stats = tool.stats.resolve_stats(msm, components).clamp_speed(); let hands = match tool.hands { Hands::One => "One", Hands::Two => "Two", }; let mut result = format!("{}-Handed {}\n\n", hands, kind); result += &statblock_desc(&stats); if !components.is_empty() { result += "Made from:\n"; for component in components { result += component.name(); result += "\n" } result += "\n"; } if !desc.is_empty() { result += &format!("{}\n\n", desc); } result += ""; result } fn statblock_desc(stats: &Stats) -> String { format!( "Power: {:0.1}\n\nPoise Strength: {:0.1}\n\nSpeed: {:0.1}\n\n", stats.power * 10.0, stats.poise_strength * 10.0, stats.speed, ) + &format!( "Crit chance: {:0.1}%\n\nCrit damage: x{:0.1}\n\n", stats.crit_chance * 100.0, stats.crit_mult, ) } #[cfg(test)] mod tests { use super::*; #[test] fn test_glider_desc() { let item_description = "mushrooms"; assert_eq!( "Glider\n\nmushrooms\n\n", glider_desc(item_description) ); } #[test] fn test_consumable_desc() { let item_description = "mushrooms"; assert_eq!( "Consumable\n\nmushrooms\n\n", consumable_desc(&[], item_description) ); } #[test] fn test_throwable_desc() { let item_description = "mushrooms"; assert_eq!( "Can be thrown\n\nmushrooms\n\n", throwable_desc(item_description) ); } #[test] fn test_utility_desc() { let item_description = "mushrooms"; assert_eq!( "mushrooms\n\n", utility_desc(item_description) ); } #[test] fn test_ingredient_desc() { let mut testmsm = MaterialStatManifest(hashbrown::HashMap::new()); testmsm.0.insert( "common.items.crafting_ing.bronze_ingot".to_string(), Stats { equip_time_secs: 0.0, power: 3.0, poise_strength: 5.0, speed: 7.0, crit_chance: 0.5, crit_mult: 2.0, }, ); assert_eq!( "Crafting Ingredient\n\nmushrooms", ingredient_desc("mushrooms", "common.items.food.mushroom", &testmsm) ); assert_eq!( "Crafting Ingredient\n\nA bronze ingot.\n\nStat multipliers:\nPower: 30.0\n\nPoise \ Strength: 50.0\n\nSpeed: 7.0\n\nCrit chance: 50.0%\n\nCrit damage: x2.0\n\n", ingredient_desc( "A bronze ingot.", "common.items.crafting_ing.bronze_ingot", &testmsm ) ); } #[test] fn test_lantern_desc() { let item_description = "mushrooms"; assert_eq!( "Lantern\n\nmushrooms\n\n", lantern_desc(item_description) ); } }