From 7bf9182768202a29cb233f52b6ce991004efe2b7 Mon Sep 17 00:00:00 2001 From: Ben Wallis Date: Mon, 29 Jun 2020 17:22:19 +0100 Subject: [PATCH] Moved skills stuff to new skills.rs file --- common/src/comp/mod.rs | 1 + common/src/comp/skills.rs | 136 ++++++++++++++++++++++++++++++++++++++ common/src/comp/stats.rs | 122 +--------------------------------- 3 files changed, 139 insertions(+), 120 deletions(-) create mode 100644 common/src/comp/skills.rs diff --git a/common/src/comp/mod.rs b/common/src/comp/mod.rs index 24dc6b3a8f..9027cc50bc 100644 --- a/common/src/comp/mod.rs +++ b/common/src/comp/mod.rs @@ -13,6 +13,7 @@ mod location; mod phys; mod player; pub mod projectile; +mod skills; mod stats; mod visual; diff --git a/common/src/comp/skills.rs b/common/src/comp/skills.rs new file mode 100644 index 0000000000..6b9f3cd0a3 --- /dev/null +++ b/common/src/comp/skills.rs @@ -0,0 +1,136 @@ +use lazy_static::lazy_static; +use std::collections::HashMap; +use tracing::warn; + +/// Represents a skill that a player can unlock, that either grants them some +/// kind of active ability, or a passive effect etc. Obviously because this is +/// an enum it doesn't describe what the skill actually -does-, this will be +/// handled by dedicated ECS systems. +#[derive(Clone, Copy, Debug, Hash, PartialEq, Eq, Serialize, Deserialize)] +pub enum Skill { + TestT1Skill1, + TestT1Skill2, + TestT1Skill3, + TestT1Skill4, + TestT1Skill5, + TestSwordSkill1, + TestSwordSkill2, + TestSwordSkill3, + TestAxeSkill1, + TestAxeSkill2, + TestAxeSkill3, +} + +#[derive(Clone, Copy, Debug, Hash, PartialEq, Eq, Serialize, Deserialize)] +pub enum SkillGroupType { + T1, + Swords, + Axes, +} + +/// A group of skills that have been unlocked by a player. Each skill group has +/// independent exp and skill points which are used to unlock skills in that +/// skill group. +#[derive(Clone, Debug, Serialize, Deserialize)] +pub struct SkillGroup { + pub skills: Vec, + pub exp: u32, + pub available_sp: u8, +} + +impl Default for SkillGroup { + fn default() -> Self { + Self { + skills: Vec::new(), + exp: 0, + available_sp: 0, + } + } +} + +lazy_static! { + // Determines the skills that comprise each skill group - this data is used to determine + // which of a player's skill groups a particular skill should be added to when a skill unlock + // is requested. TODO: Externalise this data in a RON file for ease of modification + static ref SKILL_GROUP_DEFS: HashMap> = { + let mut defs = HashMap::new(); + defs.insert(SkillGroupType::T1, vec![ + Skill::TestT1Skill1, + Skill::TestT1Skill2, + Skill::TestT1Skill3, + Skill::TestT1Skill4, + Skill::TestT1Skill5]); + + defs.insert(SkillGroupType::Swords, vec![ + Skill::TestSwordSkill1, + Skill::TestSwordSkill2, + Skill::TestSwordSkill3]); + + defs.insert(SkillGroupType::Axes, vec![ + Skill::TestAxeSkill1, + Skill::TestAxeSkill2, + Skill::TestAxeSkill3]); + + defs + }; +} + +/// Contains all of a player's skill groups and provides methods for +/// manipulating assigned skills including unlocking skills, refunding skills +/// etc. +#[derive(Clone, Debug, Serialize, Deserialize)] +pub struct SkillSet { + pub skill_groups: HashMap, +} + +impl Default for SkillSet { + /// Instantiate a new skill set with the default skill groups with no + /// unlocked skills in them - used when adding a skill set to a new + /// player + fn default() -> Self { + let mut skill_groups = HashMap::new(); + skill_groups.insert(SkillGroupType::T1, SkillGroup::default()); + skill_groups.insert(SkillGroupType::Swords, SkillGroup::default()); + skill_groups.insert(SkillGroupType::Axes, SkillGroup::default()); + Self { skill_groups } + } +} + +impl SkillSet { + pub fn refund_skill(&mut self, _skill: Skill) { + // TODO: check player has skill, remove skill and increase SP in skill group by 1 + } + + pub fn unlock_skill(&mut self, skill: Skill) { + // Find the skill group type for the skill from the static skill definitions + let skill_group_type = SKILL_GROUP_DEFS.iter().find_map(|(key, val)| { + if val.contains(&skill) { + Some(*key) + } else { + None + } + }); + + // Find the skill group for the skill on the player, check that the skill is not + // already unlocked and that they have available SP in that group, and then + // allocate the skill and reduce the player's SP in that skill group by 1. + if let Some(skill_group_type) = skill_group_type { + if let Some(skill_group) = self.skill_groups.get_mut(&skill_group_type) { + if !skill_group.skills.contains(&skill) { + if skill_group.available_sp > 0 { + skill_group.skills.push(skill); + skill_group.available_sp -= 1; + } else { + warn!("Tried to unlock skill for skill group with no available SP"); + } + } else { + warn!("Tried to unlock already unlocked skill"); + } + } else { + warn!("Tried to unlock skill for a skill group that player does not have"); + } + } else { + warn!("Tried to unlock skill that does not exist in any skill group!"); + } + } +} diff --git a/common/src/comp/stats.rs b/common/src/comp/stats.rs index 5ae518714a..bed12234f1 100644 --- a/common/src/comp/stats.rs +++ b/common/src/comp/stats.rs @@ -1,10 +1,11 @@ use crate::{ comp, - comp::{body::humanoid::Species, Body}, + comp::{body::humanoid::Species, skills::SkillSet, Body}, sync::Uid, }; use specs::{Component, FlaggedStorage}; use specs_idvs::IDVStorage; +use std::{error::Error, fmt}; #[derive(Clone, Copy, Debug, Hash, PartialEq, Eq, Serialize, Deserialize)] pub struct HealthChange { @@ -73,10 +74,6 @@ pub enum StatChangeError { Underflow, Overflow, } -use std::{error::Error, fmt}; -use std::collections::HashMap; -use crate::comp::stats::Skill::{TestT1Skill2, TestT1Skill1}; -use crate::comp::stats; impl fmt::Display for StatChangeError { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { @@ -122,88 +119,6 @@ impl Level { pub fn change_by(&mut self, level: u32) { self.amount += level; } } -#[derive(Clone, Copy, Debug, Hash, PartialEq, Eq, Serialize, Deserialize)] -pub enum Skill { - TestT1Skill1, - TestT1Skill2, - TestT1Skill3, - TestT1Skill4, - TestT1Skill5, - TestSwordSkill1, - TestSwordSkill2, - TestSwordSkill3, - TestAxeSkill1, - TestAxeSkill2, - TestAxeSkill3, -} - -#[derive(Clone, Copy, Debug, Hash, PartialEq, Eq, Serialize, Deserialize)] -pub enum SkillGroupType { - T1, - Swords, - Axes -} - -#[derive(Clone, Debug, Serialize, Deserialize)] -pub struct SkillGroup { - pub skills: Vec, - pub exp: u32, - pub available_sp: u8 -} - -impl Default for SkillGroup { - fn default() -> Self { - Self { - skills: Vec::new(), - exp: 0, - available_sp: 0 - } - } -} - -// TODO: Better way to store this static data that doesn't create a new HashMap each time -pub fn skill_group_definitions() -> HashMap> { - let mut skill_group_definitions = HashMap::new(); - skill_group_definitions.insert(SkillGroupType::T1, vec![ - Skill::TestT1Skill1, - Skill::TestT1Skill2, - Skill::TestT1Skill3, - Skill::TestT1Skill4, - Skill::TestT1Skill5]); - - skill_group_definitions.insert(SkillGroupType::Swords, vec![ - Skill::TestSwordSkill1, - Skill::TestSwordSkill2, - Skill::TestSwordSkill3]); - - skill_group_definitions.insert(SkillGroupType::Axes, vec![ - Skill::TestAxeSkill1, - Skill::TestAxeSkill2, - Skill::TestAxeSkill3]); - - skill_group_definitions -} - -#[derive(Clone, Debug, Serialize, Deserialize)] -pub struct SkillSet -{ - pub skill_groups: HashMap -} - -impl SkillSet { - /// Instantiate a new skill set with the default skill groups with no unlocked skills in them - - /// used when adding a skill set to a new player - fn new() -> Self { - let mut skill_groups = HashMap::new(); - skill_groups.insert(SkillGroupType::T1, SkillGroup::default()); - skill_groups.insert(SkillGroupType::Swords, SkillGroup::default()); - skill_groups.insert(SkillGroupType::Axes, SkillGroup::default()); - Self { - skill_groups - } - } -} - #[derive(Clone, Debug, Serialize, Deserialize)] pub struct Stats { pub name: String, @@ -228,39 +143,6 @@ impl Stats { // TODO: Delete this once stat points will be a thing pub fn update_max_hp(&mut self) { self.health.set_maximum(52 + 3 * self.level.amount); } - - pub fn refund_skill(&mut self, skill: Skill) { - // TODO: check player has skill, remove skill and increase SP in skill group by 1 - } - - pub fn unlock_skill(&mut self, skill: Skill) { - // Find the skill group type for the skill from the static skill definitions - let skill_group_type = skill_group_definitions() - .iter() - .find_map(|(key, val)| if val.contains(&skill) {Some(key) } else { None } ); - - // Find the skill group for the skill on the player, check that the skill is not - // already unlocked and that they have available SP in that group, and then allocate the - // skill and reduce the player's SP in that skill group by 1. - if let Some(skill_group_type) = skill_group_type { - if let Some(skill_group) = self.skill_set.skill_groups.get_mut(skill_group_type) { - if !skill_group.skills.contains(&skill) { - if skill_group.available_sp > 0 { - skill_group.skills.push(skill); - skill_group.available_sp -= 1; - } else { - warn!("Tried to unlock skill for skill group with no available SP"); - } - } else { - warn!("Tried to unlock already unlocked skill"); - } - } else { - warn!("Tried to unlock skill for a skill group that player does not have"); - } - } else { - warn!("Tried to unlock skill that does not exist in any skill group!"); - } - } } impl Stats {