mirror of
https://gitlab.com/veloren/veloren.git
synced 2024-08-30 18:12:32 +00:00
337 lines
11 KiB
Rust
337 lines
11 KiB
Rust
use crate::{
|
|
assets::{AssetExt, AssetHandle},
|
|
comp::{self, body, AllBodies, Body},
|
|
};
|
|
use lazy_static::lazy_static;
|
|
use rand::seq::SliceRandom;
|
|
use serde::Deserialize;
|
|
use std::str::FromStr;
|
|
|
|
#[derive(Clone, Copy, PartialEq, Eq)]
|
|
pub enum NpcKind {
|
|
Humanoid,
|
|
Wolf,
|
|
Pig,
|
|
Duck,
|
|
Phoenix,
|
|
Clownfish,
|
|
Marlin,
|
|
Ogre,
|
|
Gnome,
|
|
Archaeos,
|
|
StoneGolem,
|
|
Reddragon,
|
|
Crocodile,
|
|
Tarantula,
|
|
}
|
|
|
|
pub const ALL_NPCS: [NpcKind; 14] = [
|
|
NpcKind::Humanoid,
|
|
NpcKind::Wolf,
|
|
NpcKind::Pig,
|
|
NpcKind::Duck,
|
|
NpcKind::Phoenix,
|
|
NpcKind::Clownfish,
|
|
NpcKind::Marlin,
|
|
NpcKind::Ogre,
|
|
NpcKind::Gnome,
|
|
NpcKind::Archaeos,
|
|
NpcKind::StoneGolem,
|
|
NpcKind::Reddragon,
|
|
NpcKind::Crocodile,
|
|
NpcKind::Tarantula,
|
|
];
|
|
|
|
/// Body-specific NPC name metadata.
|
|
///
|
|
/// NOTE: Deliberately don't (yet?) implement serialize.
|
|
#[derive(Clone, Debug, Deserialize)]
|
|
pub struct BodyNames {
|
|
/// The keyword used to refer to this body type (e.g. via the command
|
|
/// console). Should be unique per body type.
|
|
pub keyword: String,
|
|
/// A list of canonical names for NPCs with this body types (currently used
|
|
/// when spawning this kind of NPC from the console). Going forward,
|
|
/// these names will likely be split up by species.
|
|
pub names_0: Vec<String>,
|
|
pub names_1: Option<Vec<String>>,
|
|
}
|
|
|
|
/// Species-specific NPC name metadata.
|
|
///
|
|
/// NOTE: Deliberately don't (yet?) implement serialize.
|
|
#[derive(Clone, Debug, Deserialize)]
|
|
pub struct SpeciesNames {
|
|
/// The keyword used to refer to this species (e.g. via the command
|
|
/// console). Should be unique per species and distinct from all body
|
|
/// types (maybe in the future, it will just be unique per body type).
|
|
pub keyword: String,
|
|
/// The generic name for NPCs of this species.
|
|
pub generic: String,
|
|
}
|
|
|
|
/// Type holding configuration data for NPC names.
|
|
pub type NpcNames = AllBodies<BodyNames, SpeciesNames>;
|
|
|
|
lazy_static! {
|
|
pub static ref NPC_NAMES: AssetHandle<NpcNames> = NpcNames::load_expect("common.npc_names");
|
|
}
|
|
|
|
impl FromStr for NpcKind {
|
|
type Err = ();
|
|
|
|
fn from_str(s: &str) -> Result<Self, ()> {
|
|
let npc_names = &*NPC_NAMES.read();
|
|
ALL_NPCS
|
|
.iter()
|
|
.copied()
|
|
.find(|&npc| npc_names[npc].keyword == s)
|
|
.ok_or(())
|
|
}
|
|
}
|
|
|
|
pub fn get_npc_name(npc_type: NpcKind, body_type: Option<BodyType>) -> String {
|
|
let npc_names = NPC_NAMES.read();
|
|
let BodyNames {
|
|
keyword,
|
|
names_0,
|
|
names_1,
|
|
} = &npc_names[npc_type];
|
|
|
|
// If no pretty name is found, fall back to the keyword.
|
|
match body_type {
|
|
Some(BodyType::Male) => names_0
|
|
.choose(&mut rand::thread_rng())
|
|
.unwrap_or(keyword)
|
|
.clone(),
|
|
Some(BodyType::Female) if names_1.is_some() => {
|
|
names_1
|
|
.as_ref()
|
|
.unwrap() // Unwrap safe since is_some is true
|
|
.choose(&mut rand::thread_rng())
|
|
.unwrap_or(keyword)
|
|
.clone()
|
|
},
|
|
_ => names_0
|
|
.choose(&mut rand::thread_rng())
|
|
.unwrap_or(keyword)
|
|
.clone(),
|
|
}
|
|
}
|
|
|
|
/// Randomly generates a body associated with this NPC kind.
|
|
pub fn kind_to_body(kind: NpcKind) -> Body {
|
|
match kind {
|
|
NpcKind::Humanoid => comp::humanoid::Body::random().into(),
|
|
NpcKind::Pig => comp::quadruped_small::Body::random().into(),
|
|
NpcKind::Wolf => comp::quadruped_medium::Body::random().into(),
|
|
NpcKind::Duck => comp::bird_medium::Body::random().into(),
|
|
NpcKind::Phoenix => comp::bird_large::Body::random().into(),
|
|
NpcKind::Clownfish => comp::fish_small::Body::random().into(),
|
|
NpcKind::Marlin => comp::fish_medium::Body::random().into(),
|
|
NpcKind::Ogre => comp::biped_large::Body::random().into(),
|
|
NpcKind::Gnome => comp::biped_small::Body::random().into(),
|
|
NpcKind::Archaeos => comp::theropod::Body::random().into(),
|
|
NpcKind::StoneGolem => comp::golem::Body::random().into(),
|
|
NpcKind::Reddragon => comp::dragon::Body::random().into(),
|
|
NpcKind::Crocodile => comp::quadruped_low::Body::random().into(),
|
|
NpcKind::Tarantula => comp::arthropod::Body::random().into(),
|
|
}
|
|
}
|
|
|
|
/// A combination of an NpcKind (representing an outer species to generate), and
|
|
/// a function that generates a fresh Body of a species that is part of that
|
|
/// NpcKind each time it's called. The reason things are done this way is that
|
|
/// when parsing spawn strings, we'd like to be able to randomize features that
|
|
/// haven't already been specified; for instance, if no species is specified we
|
|
/// should randomize species, while if a species is specified we can still
|
|
/// randomize other attributes like gender or clothing.
|
|
///
|
|
/// TODO: Now that we return a closure, consider having the closure accept a
|
|
/// source of randomness explicitly, rather than always using ThreadRng.
|
|
pub struct NpcBody(pub NpcKind, pub Box<dyn FnMut() -> Body>);
|
|
|
|
impl FromStr for NpcBody {
|
|
type Err = ();
|
|
|
|
/// Get an NPC kind from a string. If a body kind is matched without an
|
|
/// associated species, generate the species randomly within it; if an
|
|
/// explicit species is found, generate a random member of the species;
|
|
/// otherwise, return Err(()).
|
|
fn from_str(s: &str) -> Result<Self, Self::Err> { Self::from_str_with(s, kind_to_body) }
|
|
}
|
|
|
|
impl NpcBody {
|
|
/// If there is an exact name match for a body kind, call kind_to_body on
|
|
/// it. Otherwise, if an explicit species is found, generate a random
|
|
/// member of the species; otherwise, return Err(()).
|
|
#[allow(clippy::result_unit_err)]
|
|
pub fn from_str_with(s: &str, kind_to_body: fn(NpcKind) -> Body) -> Result<Self, ()> {
|
|
fn parse<
|
|
'a,
|
|
B: Into<Body> + 'static,
|
|
// NOTE: Should be cheap in all cases, but if it weren't we should revamp the indexing
|
|
// method to take references instead of owned values.
|
|
Species: 'static,
|
|
BodyMeta,
|
|
SpeciesData: for<'b> core::ops::Index<&'b Species, Output = SpeciesNames>,
|
|
>(
|
|
s: &str,
|
|
npc_kind: NpcKind,
|
|
body_data: &'a comp::BodyData<BodyMeta, SpeciesData>,
|
|
conv_func: for<'d> fn(&mut rand::rngs::ThreadRng, &'d Species) -> B,
|
|
) -> Option<NpcBody>
|
|
where
|
|
&'a SpeciesData: IntoIterator<Item = Species>,
|
|
{
|
|
let npc_names = &body_data.species;
|
|
body_data
|
|
.species
|
|
.into_iter()
|
|
.find(|species| npc_names[species].keyword == s)
|
|
.map(|species| {
|
|
NpcBody(
|
|
npc_kind,
|
|
Box::new(move || conv_func(&mut rand::thread_rng(), &species).into()),
|
|
)
|
|
})
|
|
}
|
|
let npc_names = &*NPC_NAMES.read();
|
|
// First, parse npc kind names.
|
|
NpcKind::from_str(s)
|
|
.map(|kind| NpcBody(kind, Box::new(move || kind_to_body(kind))))
|
|
.ok()
|
|
// Otherwise, npc kind names aren't sufficient; we parse species names instead.
|
|
.or_else(|| {
|
|
parse(
|
|
s,
|
|
NpcKind::Humanoid,
|
|
&npc_names.humanoid,
|
|
comp::humanoid::Body::random_with,
|
|
)
|
|
})
|
|
.or_else(|| {
|
|
parse(
|
|
s,
|
|
NpcKind::Pig,
|
|
&npc_names.quadruped_small,
|
|
comp::quadruped_small::Body::random_with,
|
|
)
|
|
})
|
|
.or_else(|| {
|
|
parse(
|
|
s,
|
|
NpcKind::Wolf,
|
|
&npc_names.quadruped_medium,
|
|
comp::quadruped_medium::Body::random_with,
|
|
)
|
|
})
|
|
.or_else(|| {
|
|
parse(
|
|
s,
|
|
NpcKind::Duck,
|
|
&npc_names.bird_medium,
|
|
comp::bird_medium::Body::random_with,
|
|
)
|
|
})
|
|
.or_else(|| {
|
|
parse(
|
|
s,
|
|
NpcKind::Phoenix,
|
|
&npc_names.bird_large,
|
|
comp::bird_large::Body::random_with,
|
|
)
|
|
})
|
|
.or_else(|| {
|
|
parse(
|
|
s,
|
|
NpcKind::Clownfish,
|
|
&npc_names.fish_small,
|
|
comp::fish_small::Body::random_with,
|
|
)
|
|
})
|
|
.or_else(|| {
|
|
parse(
|
|
s,
|
|
NpcKind::Marlin,
|
|
&npc_names.fish_medium,
|
|
comp::fish_medium::Body::random_with,
|
|
)
|
|
})
|
|
.or_else(|| {
|
|
parse(
|
|
s,
|
|
NpcKind::Ogre,
|
|
&npc_names.biped_large,
|
|
comp::biped_large::Body::random_with,
|
|
)
|
|
})
|
|
.or_else(|| {
|
|
parse(
|
|
s,
|
|
NpcKind::Gnome,
|
|
&npc_names.biped_small,
|
|
comp::biped_small::Body::random_with,
|
|
)
|
|
})
|
|
.or_else(|| {
|
|
parse(
|
|
s,
|
|
NpcKind::Archaeos,
|
|
&npc_names.theropod,
|
|
comp::theropod::Body::random_with,
|
|
)
|
|
})
|
|
.or_else(|| {
|
|
parse(
|
|
s,
|
|
NpcKind::StoneGolem,
|
|
&npc_names.golem,
|
|
comp::golem::Body::random_with,
|
|
)
|
|
})
|
|
.or_else(|| {
|
|
parse(
|
|
s,
|
|
NpcKind::Reddragon,
|
|
&npc_names.dragon,
|
|
comp::dragon::Body::random_with,
|
|
)
|
|
})
|
|
.or_else(|| {
|
|
parse(
|
|
s,
|
|
NpcKind::Crocodile,
|
|
&npc_names.quadruped_low,
|
|
comp::quadruped_low::Body::random_with,
|
|
)
|
|
})
|
|
.or_else(|| {
|
|
parse(
|
|
s,
|
|
NpcKind::Tarantula,
|
|
&npc_names.arthropod,
|
|
comp::arthropod::Body::random_with,
|
|
)
|
|
})
|
|
.ok_or(())
|
|
}
|
|
}
|
|
|
|
pub enum BodyType {
|
|
Male,
|
|
Female,
|
|
}
|
|
|
|
impl BodyType {
|
|
pub fn from_body(body: Body) -> Option<BodyType> {
|
|
match body {
|
|
Body::Humanoid(humanoid) => match humanoid.body_type {
|
|
body::humanoid::BodyType::Male => Some(BodyType::Male),
|
|
body::humanoid::BodyType::Female => Some(BodyType::Female),
|
|
},
|
|
_ => None,
|
|
}
|
|
}
|
|
}
|