veloren/server/src/character_creator.rs

116 lines
3.7 KiB
Rust
Raw Normal View History

use crate::persistence::character_updater::CharacterUpdater;
2021-12-13 00:13:33 +00:00
use common::{
character::CharacterId,
comp::{inventory::loadout_builder::LoadoutBuilder, Body, Inventory, Item, SkillSet, Stats},
};
use specs::{Entity, WriteExpect};
2021-07-29 22:38:35 +00:00
const VALID_STARTER_ITEMS: &[[Option<&str>; 2]] = &[
[None, None], // Not used with an unmodified client but should still be allowed (zesterer)
[Some("common.items.weapons.hammer.starter_hammer"), None],
[Some("common.items.weapons.bow.starter"), None],
[Some("common.items.weapons.axe.starter_axe"), None],
[Some("common.items.weapons.staff.starter_staff"), None],
[Some("common.items.weapons.sword.starter"), None],
[
Some("common.items.weapons.sword_1h.starter"),
Some("common.items.weapons.sword_1h.starter"),
],
];
2021-07-29 22:38:35 +00:00
#[derive(Debug)]
pub enum CreationError {
InvalidWeapon,
InvalidBody,
}
pub fn create_character(
entity: Entity,
player_uuid: String,
character_alias: String,
2021-07-29 22:38:35 +00:00
character_mainhand: Option<String>,
character_offhand: Option<String>,
body: Body,
character_updater: &mut WriteExpect<'_, CharacterUpdater>,
2021-07-29 22:38:35 +00:00
) -> Result<(), CreationError> {
// quick fix whitelist validation for now; eventually replace the
// `Option<String>` with an index into a server-provided list of starter
// items, and replace `comp::body::Body` with `comp::body::humanoid::Body`
// throughout the messages involved
if !matches!(body, Body::Humanoid(_)) {
2021-07-29 22:38:35 +00:00
return Err(CreationError::InvalidBody);
}
2021-07-29 22:38:35 +00:00
if !VALID_STARTER_ITEMS.contains(&[character_mainhand.as_deref(), character_offhand.as_deref()])
{
return Err(CreationError::InvalidWeapon);
};
// The client sends None if a weapon hand is empty
let loadout = LoadoutBuilder::empty()
.defaults()
2021-07-29 22:38:35 +00:00
.active_mainhand(character_mainhand.map(|x| Item::new_from_asset_expect(&x)))
.active_offhand(character_offhand.map(|x| Item::new_from_asset_expect(&x)))
.build();
let mut inventory = Inventory::new_with_loadout(loadout);
2021-07-29 22:38:35 +00:00
let stats = Stats::new(character_alias.to_string());
let skill_set = SkillSet::default();
// Default items for new characters
inventory
.push(Item::new_from_asset_expect(
"common.items.consumable.potion_minor",
))
.expect("Inventory has at least 2 slots left!");
inventory
.push(Item::new_from_asset_expect("common.items.food.cheese"))
.expect("Inventory has at least 1 slot left!");
2020-11-03 00:12:49 +00:00
let waypoint = None;
character_updater.create_character(
entity,
player_uuid,
character_alias,
2021-11-27 04:19:46 +00:00
(
body,
stats,
skill_set,
inventory,
waypoint,
Vec::new(),
Default::default(),
),
);
2021-07-29 22:38:35 +00:00
Ok(())
}
2021-12-13 00:13:33 +00:00
pub fn edit_character(
entity: Entity,
player_uuid: String,
id: CharacterId,
character_alias: String,
body: Body,
character_updater: &mut WriteExpect<'_, CharacterUpdater>,
) -> Result<(), CreationError> {
if !matches!(body, Body::Humanoid(_)) {
return Err(CreationError::InvalidBody);
}
character_updater.edit_character(entity, player_uuid, id, character_alias, (body,));
Ok(())
}
2021-07-29 22:38:35 +00:00
// Error handling
impl core::fmt::Display for CreationError {
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
match self {
CreationError::InvalidWeapon => write!(
f,
"Invalid weapon.\nServer and client might be partially incompatible."
),
CreationError::InvalidBody => write!(
f,
"Invalid Body.\nServer and client might be partially incompatible"
),
}
}
}