pub mod alpha; pub mod beam; pub mod beta; pub mod block; pub mod chargeswing; pub mod climb; pub mod consume; pub mod dance; pub mod dash; pub mod equip; pub mod glidewield; pub mod gliding; pub mod idle; pub mod jump; pub mod leapmelee; pub mod mount; pub mod repeater; pub mod roll; pub mod run; pub mod shockwave; pub mod shoot; pub mod sit; pub mod sneak; pub mod spin; pub mod spinmelee; pub mod staggered; pub mod stand; pub mod stunned; pub mod swim; pub mod swimwield; pub mod talk; pub mod wield; // Reexports pub use self::{ alpha::AlphaAnimation, beam::BeamAnimation, beta::BetaAnimation, block::BlockAnimation, chargeswing::ChargeswingAnimation, climb::ClimbAnimation, consume::ConsumeAnimation, dance::DanceAnimation, dash::DashAnimation, equip::EquipAnimation, glidewield::GlideWieldAnimation, gliding::GlidingAnimation, idle::IdleAnimation, jump::JumpAnimation, leapmelee::LeapAnimation, mount::MountAnimation, repeater::RepeaterAnimation, roll::RollAnimation, run::RunAnimation, shockwave::ShockwaveAnimation, shoot::ShootAnimation, sit::SitAnimation, sneak::SneakAnimation, spin::SpinAnimation, spinmelee::SpinMeleeAnimation, staggered::StaggeredAnimation, stand::StandAnimation, stunned::StunnedAnimation, swim::SwimAnimation, swimwield::SwimWieldAnimation, talk::TalkAnimation, wield::WieldAnimation, }; use super::{make_bone, vek::*, FigureBoneData, Offsets, Skeleton}; use common::comp; use core::{convert::TryFrom, f32::consts::PI}; pub type Body = comp::humanoid::Body; skeleton_impls!(struct CharacterSkeleton { + head, + chest, + belt, + back, + shorts, + hand_l, + hand_r, + foot_l, + foot_r, + shoulder_l, + shoulder_r, + glider, + main, + second, + lantern, + hold, torso, control, control_l, control_r, :: // Begin non-bone fields holding_lantern: bool, offsets: Option>, mountee_body: Option, }); impl CharacterSkeleton { pub fn new( holding_lantern: bool, offsets: Option>, mountee_body: Option, ) -> Self { Self { holding_lantern, offsets, mountee_body, ..Self::default() } } } impl Skeleton for CharacterSkeleton { type Attr = SkeletonAttr; type Body = Body; const BONE_COUNT: usize = 16; #[cfg(feature = "use-dyn-lib")] const COMPUTE_FN: &'static [u8] = b"character_compute_mats\0"; #[cfg_attr(feature = "be-dyn-lib", export_name = "character_compute_mats")] fn compute_matrices_inner( &self, base_mat: Mat4, buf: &mut [FigureBoneData; super::MAX_BONE_COUNT], ) -> Offsets { let torso_mat = base_mat * if let Some((offset, body)) = self.offsets.zip(self.mountee_body) { let hitbox_offsets = body.mounting_offset(); let visual_offset = mounting_offset(body); Mat4::::from(Transform { position: offset.position - Vec3::from([0.0, hitbox_offsets.y, hitbox_offsets.z]), orientation: offset.orientation, scale: Vec3::::one(), }) * Mat4::::from(self.torso).translated_3d(visual_offset) } else { Mat4::::from(self.torso) }; let chest_mat = torso_mat * Mat4::::from(self.chest); let head_mat = chest_mat * Mat4::::from(self.head); let shorts_mat = chest_mat * Mat4::::from(self.shorts); let control_mat = chest_mat * Mat4::::from(self.control); let control_l_mat = control_mat * Mat4::::from(self.control_l); let control_r_mat = control_mat * Mat4::::from(self.control_r); let hand_r_mat = control_r_mat * Mat4::::from(self.hand_r); let hand_l_mat = Mat4::::from(self.hand_l); let lantern_mat = if self.holding_lantern { hand_r_mat } else { shorts_mat } * Mat4::::from(self.lantern); *(<&mut [_; Self::BONE_COUNT]>::try_from(&mut buf[0..Self::BONE_COUNT]).unwrap()) = [ make_bone(head_mat), make_bone(chest_mat), make_bone(chest_mat * Mat4::::from(self.belt)), make_bone(chest_mat * Mat4::::from(self.back)), make_bone(shorts_mat), make_bone(control_l_mat * hand_l_mat), make_bone(hand_r_mat), make_bone(torso_mat * Mat4::::from(self.foot_l)), make_bone(torso_mat * Mat4::::from(self.foot_r)), make_bone(chest_mat * Mat4::::from(self.shoulder_l)), make_bone(chest_mat * Mat4::::from(self.shoulder_r)), make_bone(chest_mat * Mat4::::from(self.glider)), make_bone(control_l_mat * Mat4::::from(self.main)), make_bone(control_r_mat * Mat4::::from(self.second)), make_bone(lantern_mat), // FIXME: Should this be control_l_mat? make_bone(control_mat * hand_l_mat * Mat4::::from(self.hold)), ]; Offsets { lantern: (lantern_mat * Vec4::new(0.0, 0.0, -4.0, 1.0)).xyz(), mount_bone: self.chest, } } } pub struct SkeletonAttr { scaler: f32, head_scale: f32, head: (f32, f32), chest: (f32, f32), belt: (f32, f32), back: (f32, f32), shorts: (f32, f32), hand: (f32, f32, f32), foot: (f32, f32, f32), shoulder: (f32, f32, f32), lantern: (f32, f32, f32), shl: (f32, f32, f32, f32, f32, f32), shr: (f32, f32, f32, f32, f32, f32), sc: (f32, f32, f32, f32, f32, f32), hhl: (f32, f32, f32, f32, f32, f32), hhr: (f32, f32, f32, f32, f32, f32), hc: (f32, f32, f32, f32, f32, f32), sthl: (f32, f32, f32, f32, f32, f32), sthr: (f32, f32, f32, f32, f32, f32), stc: (f32, f32, f32, f32, f32, f32), ahl: (f32, f32, f32, f32, f32, f32), ahr: (f32, f32, f32, f32, f32, f32), ac: (f32, f32, f32, f32, f32, f32), bhl: (f32, f32, f32, f32, f32, f32), bhr: (f32, f32, f32, f32, f32, f32), bc: (f32, f32, f32, f32, f32, f32), } impl Default for SkeletonAttr { fn default() -> Self { Self { scaler: 0.0, head_scale: 0.0, head: (0.0, 0.0), chest: (0.0, 0.0), belt: (0.0, 0.0), back: (0.0, 0.0), shorts: (0.0, 0.0), hand: (0.0, 0.0, 0.0), foot: (0.0, 0.0, 0.0), shoulder: (0.0, 0.0, 0.0), lantern: (0.0, 0.0, 0.0), shl: (0.0, 0.0, 0.0, 0.0, 0.0, 0.0), shr: (0.0, 0.0, 0.0, 0.0, 0.0, 0.0), sc: (0.0, 0.0, 0.0, 0.0, 0.0, 0.0), hhl: (0.0, 0.0, 10.0, 0.0, 0.0, 0.0), hhr: (0.0, 0.0, 0.0, 0.0, 0.0, 0.0), hc: (0.0, 0.0, 0.0, 0.0, 0.0, 0.0), sthl: (0.0, 0.0, 10.0, 0.0, 0.0, 0.0), sthr: (0.0, 0.0, 0.0, 0.0, 0.0, 0.0), stc: (0.0, 0.0, 0.0, 0.0, 0.0, 0.0), ahl: (0.0, 0.0, 10.0, 0.0, 0.0, 0.0), ahr: (0.0, 0.0, 0.0, 0.0, 0.0, 0.0), ac: (0.0, 0.0, 0.0, 0.0, 0.0, 0.0), bhl: (0.0, 0.0, 10.0, 0.0, 0.0, 0.0), bhr: (0.0, 0.0, 0.0, 0.0, 0.0, 0.0), bc: (0.0, 0.0, 0.0, 0.0, 0.0, 0.0), } } } impl<'a> std::convert::TryFrom<&'a comp::Body> for SkeletonAttr { type Error = (); fn try_from(body: &'a comp::Body) -> Result { match body { comp::Body::Humanoid(body) => Ok(SkeletonAttr::from(body)), _ => Err(()), } } } impl<'a> From<&'a Body> for SkeletonAttr { #[allow(clippy::match_single_binding)] // TODO: Pending review in #587 fn from(body: &'a Body) -> Self { use comp::humanoid::{BodyType::*, Species::*}; Self { scaler: match (body.species, body.body_type) { // TODO : Derive scale from body proportions (Orc, Male) => 0.91, (Orc, Female) => 0.81, (Human, Male) => 0.81, (Human, Female) => 0.76, (Elf, Male) => 0.82, (Elf, Female) => 0.76, (Dwarf, Male) => 0.67, (Dwarf, Female) => 0.62, (Undead, Male) => 0.78, (Undead, Female) => 0.72, (Danari, Male) => 0.56, (Danari, Female) => 0.56, }, head_scale: match (body.species, body.body_type) { (Orc, Male) => 0.9, (Orc, Female) => 0.9, (Human, Male) => 0.9, (Human, Female) => 0.9, (Elf, Male) => 0.9, (Elf, Female) => 0.9, (Dwarf, Male) => 1.0, (Dwarf, Female) => 1.0, (Undead, Male) => 0.9, (Undead, Female) => 0.9, (Danari, Male) => 1.15, (Danari, Female) => 1.15, }, head: match (body.species, body.body_type) { (Orc, Male) => (-2.0, 9.0), (Orc, Female) => (-2.0, 9.5), (Human, Male) => (-2.3, 9.5), (Human, Female) => (-2.0, 9.5), (Elf, Male) => (-2.5, 9.5), (Elf, Female) => (-1.0, 9.5), (Dwarf, Male) => (-2.0, 10.0), (Dwarf, Female) => (-2.0, 9.5), (Undead, Male) => (-1.5, 8.5), (Undead, Female) => (-1.5, 9.5), (Danari, Male) => (-1.5, 7.0), (Danari, Female) => (-1.5, 7.0), }, chest: match (body.species, body.body_type) { _ => (0.0, 8.0), }, belt: match (body.species, body.body_type) { _ => (0.0, -2.0), }, back: match (body.species, body.body_type) { _ => (-3.1, 7.25), }, shorts: match (body.species, body.body_type) { _ => (0.0, -5.0), }, hand: match (body.species, body.body_type) { _ => (7.0, -0.25, 0.5), }, foot: match (body.species, body.body_type) { _ => (3.4, 0.5, 2.0), }, shoulder: match (body.species, body.body_type) { _ => (5.0, 0.0, 5.0), }, lantern: match (body.species, body.body_type) { _ => (5.0, 2.5, 5.5), }, shl: match (body.species, body.body_type) { _ => (-0.75, -1.0, 0.5, 1.47, -0.2, 0.0), }, shr: match (body.species, body.body_type) { _ => (0.75, -1.5, -2.5, 1.47, 0.3, 0.0), }, sc: match (body.species, body.body_type) { _ => (-7.0, 7.0, 2.0, -0.1, 0.0, 0.0), }, hhl: match (body.species, body.body_type) { _ => (0.1, 0.0, 11.0, 4.71, 0.0, PI), }, hhr: match (body.species, body.body_type) { _ => (0.0, 0.0, 0.0, 4.71, 0.0, PI), }, hc: match (body.species, body.body_type) { _ => (6.0, 7.0, 1.0, -0.3, -PI / 2.0, 3.64), }, sthl: match (body.species, body.body_type) { _ => (0.0, 0.0, 6.0, 1.97, 0.0, 0.0), }, sthr: match (body.species, body.body_type) { _ => (0.0, 0.0, 0.0, 1.27, 0.2, 0.0), }, stc: match (body.species, body.body_type) { _ => (-5.0, 7.0, -2.0, -0.3, 0.15, 0.0), }, ahl: match (body.species, body.body_type) { _ => (-0.5, -1.0, 7.0, 1.17, PI, 0.0), }, ahr: match (body.species, body.body_type) { _ => (0.0, -1.0, 1.0, -2.0, 0.0, PI), }, ac: match (body.species, body.body_type) { _ => (-8.0, 11.0, 3.0, 2.0, 0.0, 0.0), }, bhl: match (body.species, body.body_type) { _ => (0.0, -4.0, 1.0, PI / 2.0, 0.0, 0.0), }, bhr: match (body.species, body.body_type) { _ => (1.0, 2.0, -2.0, PI / 2.0, 0.0, 0.0), }, bc: match (body.species, body.body_type) { _ => (-5.0, 9.0, 1.0, 0.0, 1.2, -0.6), }, } } } pub fn mounting_offset(body: comp::Body) -> Vec3 { match body { comp::Body::QuadrupedMedium(quadruped_medium) => { match (quadruped_medium.species, quadruped_medium.body_type) { (comp::quadruped_medium::Species::Grolgar, _) => Vec3::from([0.0, -0.6, 0.5]), (comp::quadruped_medium::Species::Saber, _) => Vec3::from([0.0, -0.75, 0.23]), (comp::quadruped_medium::Species::Tiger, _) => Vec3::from([0.0, -0.75, 0.23]), (comp::quadruped_medium::Species::Tuskram, _) => Vec3::from([0.0, -0.75, 0.25]), (comp::quadruped_medium::Species::Lion, _) => Vec3::from([0.0, -0.9, 0.25]), (comp::quadruped_medium::Species::Tarasque, _) => Vec3::from([0.0, -1.1, 0.3]), (comp::quadruped_medium::Species::Wolf, _) => Vec3::from([0.0, -0.7, 0.15]), (comp::quadruped_medium::Species::Frostfang, _) => Vec3::from([0.0, -0.8, 0.1]), (comp::quadruped_medium::Species::Mouflon, _) => Vec3::from([0.0, -0.8, 0.1]), (comp::quadruped_medium::Species::Catoblepas, _) => Vec3::from([0.0, -0.4, 0.4]), (comp::quadruped_medium::Species::Bonerattler, _) => Vec3::from([0.0, -0.3, 0.2]), (comp::quadruped_medium::Species::Deer, _) => Vec3::from([0.0, -0.9, 0.1]), (comp::quadruped_medium::Species::Hirdrasil, _) => Vec3::from([0.0, -1.0, 0.0]), (comp::quadruped_medium::Species::Roshwalr, _) => Vec3::from([0.0, -0.2, 0.7]), (comp::quadruped_medium::Species::Donkey, _) => Vec3::from([0.0, -0.5, 0.15]), (comp::quadruped_medium::Species::Camel, _) => Vec3::from([0.0, -1.4, 0.3]), (comp::quadruped_medium::Species::Zebra, _) => Vec3::from([0.0, -0.6, 0.1]), (comp::quadruped_medium::Species::Antelope, _) => Vec3::from([0.0, -0.8, 0.1]), (comp::quadruped_medium::Species::Kelpie, _) => Vec3::from([0.0, -0.8, 0.05]), (comp::quadruped_medium::Species::Horse, _) => Vec3::from([0.0, -0.8, 0.1]), (comp::quadruped_medium::Species::Barghest, _) => Vec3::from([0.0, -0.8, 0.6]), ( comp::quadruped_medium::Species::Cattle, comp::quadruped_medium::BodyType::Male, ) => Vec3::from([0.0, -0.4, 0.7]), ( comp::quadruped_medium::Species::Cattle, comp::quadruped_medium::BodyType::Female, ) => Vec3::from([0.0, -0.3, 0.5]), (comp::quadruped_medium::Species::Darkhound, _) => Vec3::from([0.0, -0.5, 0.2]), (comp::quadruped_medium::Species::Highland, _) => Vec3::from([0.0, -0.2, 0.8]), (comp::quadruped_medium::Species::Yak, _) => Vec3::from([0.0, -0.2, 0.8]), (comp::quadruped_medium::Species::Panda, _) => Vec3::from([0.0, -0.8, 0.2]), (comp::quadruped_medium::Species::Bear, _) => Vec3::from([0.0, -1.5, 0.6]), (comp::quadruped_medium::Species::Dreadhorn, _) => Vec3::from([0.0, -1.5, 1.6]), (comp::quadruped_medium::Species::Moose, _) => Vec3::from([0.0, -0.9, 0.3]), (comp::quadruped_medium::Species::Snowleopard, _) => Vec3::from([0.0, -0.9, 0.2]), } }, comp::Body::Ship(comp::ship::Body::DefaultAirship) => Vec3::from([0.0, 0.0, 10.0]), comp::Body::Dragon(_) => Vec3::from([0.0, -0.7, 6.4]), _ => Vec3::unit_z(), } }