From 1456548dbd11a58fd3ded1486e9350d0195539ef Mon Sep 17 00:00:00 2001 From: jshipsey Date: Tue, 22 Oct 2019 22:56:45 -0400 Subject: [PATCH] fish_medium and bird_medium skelingtons --- .../voxygen/voxel/npc/duck_m/duck_m_body.vox | Bin 0 -> 56063 bytes .../voxygen/voxel/npc/duck_m/duck_m_head.vox | Bin 0 -> 55703 bytes .../voxygen/voxel/npc/duck_m/duck_m_leg_l.vox | Bin 0 -> 55615 bytes .../voxygen/voxel/npc/duck_m/duck_m_leg_r.vox | Bin 0 -> 55615 bytes .../voxygen/voxel/npc/duck_m/duck_m_tail.vox | Bin 0 -> 55643 bytes .../voxygen/voxel/npc/duck_m/duck_m_wing.vox | Bin 0 -> 55631 bytes .../voxygen/voxel/npc/marlin/marlin_fin_l.vox | Bin 0 -> 1152 bytes .../voxygen/voxel/npc/marlin/marlin_fin_r.vox | Bin 0 -> 1152 bytes .../voxygen/voxel/npc/marlin/marlin_head.vox | Bin 0 -> 1352 bytes .../voxygen/voxel/npc/marlin/marlin_rear.vox | Bin 0 -> 1360 bytes .../voxygen/voxel/npc/marlin/marlin_tail.vox | Bin 0 -> 1308 bytes .../voxygen/voxel/npc/marlin/marlin_torso.vox | Bin 0 -> 1476 bytes common/src/comp/body.rs | 4 + common/src/comp/body/bird_medium.rs | 68 ++++++++ common/src/comp/body/fish_medium.rs | 61 +++++++ common/src/comp/mod.rs | 2 +- common/src/npc.rs | 1 + voxygen/src/anim/birdmedium/idle.rs | 74 +++++++++ voxygen/src/anim/birdmedium/jump.rs | 62 +++++++ voxygen/src/anim/birdmedium/mod.rs | 80 ++++++++++ voxygen/src/anim/birdmedium/run.rs | 74 +++++++++ voxygen/src/anim/character/run - Copy.rs | 147 +++++++++++++++++ voxygen/src/anim/character/sneak.rs | 151 ++++++++++++++++++ voxygen/src/anim/fishmedium/idle.rs | 70 ++++++++ voxygen/src/anim/fishmedium/jump.rs | 58 +++++++ voxygen/src/anim/fishmedium/mod.rs | 77 +++++++++ voxygen/src/anim/fishmedium/run.rs | 58 +++++++ voxygen/src/anim/mod.rs | 2 + voxygen/src/scene/figure/cache.rs | 36 +++++ voxygen/src/scene/figure/load.rs | 119 +++++++++++++- voxygen/src/scene/figure/mod.rs | 138 +++++++++++++++- 31 files changed, 1279 insertions(+), 3 deletions(-) create mode 100644 assets/voxygen/voxel/npc/duck_m/duck_m_body.vox create mode 100644 assets/voxygen/voxel/npc/duck_m/duck_m_head.vox create mode 100644 assets/voxygen/voxel/npc/duck_m/duck_m_leg_l.vox create mode 100644 assets/voxygen/voxel/npc/duck_m/duck_m_leg_r.vox create mode 100644 assets/voxygen/voxel/npc/duck_m/duck_m_tail.vox create mode 100644 assets/voxygen/voxel/npc/duck_m/duck_m_wing.vox create mode 100644 assets/voxygen/voxel/npc/marlin/marlin_fin_l.vox create mode 100644 assets/voxygen/voxel/npc/marlin/marlin_fin_r.vox create mode 100644 assets/voxygen/voxel/npc/marlin/marlin_head.vox create mode 100644 assets/voxygen/voxel/npc/marlin/marlin_rear.vox create mode 100644 assets/voxygen/voxel/npc/marlin/marlin_tail.vox create mode 100644 assets/voxygen/voxel/npc/marlin/marlin_torso.vox create mode 100644 common/src/comp/body/bird_medium.rs create mode 100644 common/src/comp/body/fish_medium.rs create mode 100644 voxygen/src/anim/birdmedium/idle.rs create mode 100644 voxygen/src/anim/birdmedium/jump.rs create mode 100644 voxygen/src/anim/birdmedium/mod.rs create mode 100644 voxygen/src/anim/birdmedium/run.rs create mode 100644 voxygen/src/anim/character/run - Copy.rs create mode 100644 voxygen/src/anim/character/sneak.rs create mode 100644 voxygen/src/anim/fishmedium/idle.rs create mode 100644 voxygen/src/anim/fishmedium/jump.rs create mode 100644 voxygen/src/anim/fishmedium/mod.rs create mode 100644 voxygen/src/anim/fishmedium/run.rs diff --git a/assets/voxygen/voxel/npc/duck_m/duck_m_body.vox b/assets/voxygen/voxel/npc/duck_m/duck_m_body.vox new file mode 100644 index 0000000000000000000000000000000000000000..027b928462caa5bbd83607071da2177694919341 GIT binary patch literal 56063 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 QAFrFgbmuBN8kGR=#~>tEOaK4? literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/duck_m/duck_m_head.vox b/assets/voxygen/voxel/npc/duck_m/duck_m_head.vox new file mode 100644 index 0000000000000000000000000000000000000000..d7b05f86f5933a29062441c3c7151bfc64ebe994 GIT binary patch literal 55703 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 literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/duck_m/duck_m_leg_r.vox b/assets/voxygen/voxel/npc/duck_m/duck_m_leg_r.vox new file mode 100644 index 0000000000000000000000000000000000000000..c1719336d70a3e0c6305e30400e6bf24f72938d1 GIT binary patch literal 55615 zcmdtqdsGzH9S87>#>9v*8WAHVu!0Xj#o1j}5MTHpf+!*yRHAMcc34LC)n$2EnpCkS z#+n#H(Y6{jsnyhK)PjvoTT869wy8D7rb$z4O`4|frZ!E}=dbs7XIXijn#NILy@f>HAQII>NF2C}#JPAHzEseig@T3^ z3z~9V(BNu8cQgwcxl>U55kZ+tV`;H(IF)T0Nspbnmlie&THY$C{FtD|OM*r(PoP92 znZ`Gz)AaUS8XJnEPirp0COC9eK7k2s-wZpw2^rE}W2YuAl>Xf?g;Qbk!rMOvVkR zf_A!Pd7YpIX@b^g3#wZvXtyk1o|!;KUNUW5o=R=HjhgPyqPD69)cNQtdiS6~lT+ep zmMw`2=cbZgnnl%%1vRe|v@a~^%oBp9Cniuq`gmGnpF-=ircr5nJQ(5CVOYWIA3N4(`!{5Gs8rki3dNOMtO|u@LncFtg znC3Bb{ux169uRaSHI>rR(x@t>j+W$~B+fMogP1V zmTug*L5|=ynzMNy6+V52Rvf=fxw*M?UG89FusrDl8hBaIwbSPKHlKa#Bky06&);s_ z3_6@Omrl>lqgV2Zcpp^nUG%fiF5jtXB#Ip*awhh?Cv&5Dz3W9{uJ92#@_WU~B`?VY zK6p>z?-gqA3^xXK{>I>R8pk86d z4po6;XE0L2TaMERgv`*cLRVNdr*WCnqjA|(EyH1kRx>l(jOxuOOLbJ~t!?0Os2WbM zJEVf{8s5EeG;GJw@G|3Q_<_XH@QaP3;avzv!y7V==5{E}5GYM7l!o6KocHDqD9tb^ z&7Dvh{z(qbdouz`a~G6mB$S4KT8Z=CjDpg{Lup1sY3_#7jDgbdZ&+~lw6Rbc{>>cD z+T07JNrciQL1`>dnsHE?WGGDvlx94XW&)IEB9vwllqMBQlLnnhLTU1#H1|Pi7C~wFpUC6ErCAN7Sp%ghgVH^Ei~|2`J4@D9w{lnieR{Q&5^#D9z`f zG;L6tT~M0cP@2y}X}$oZ`685N50qvvl;&wD%|0m2ekjcWD9tlanrERjUxLycgwlK& zN^=NGa~MkV9F*oOP?{r9ny*4>jzVdUL1~_c(j1444Ilg3^2) zO7jAg<}{S%8&H}Tp)}uw(tHa_^KB^287R$npfsIOnwOw7XQ4FTh0>gZ(tHm}a~?|b zeJITZD9sO`G(UvWybPuJ5tQa4l;+1!noCfcSD-YPp){{TX|6zNegdU=4NCJ{KZDY|38nctl;$dw<`+2Y5oPJ`8Sm2KTw+gLTLy}BVvRYDn*0j=?pgp^}&4MHYzJ?YxLXrx?VRtRbiPX zthaI;3eVf41oa9tcBl#*JA;vuTeyl52$`W>g|4t_PUAACN8_@oT86_6 zt!8Gn8P%Inmg=a|Tid|nP&J%hcSr@HW%C!O$WlHA$TKQYed+!>Rp1M_^=P`Gl7(PT z&ve6I$!$1|NWUtmj+I3=hr<#%)ik*L9eYS^p$Yt%(B6xg8h)x=c>>bQwa$L;b@rz@z3 z!nGmYgxDNvB;{V0TX*_Zoo)p;AIPEzf}4*b)J3J?^{ETJAsFBW%+;zh%n_?wt_r>8 z=6F0VA1{n8!^YFIy1$my%9MKp0iP21j0STl%_Xa;aCyy#U%j6$enHJ=gV)y?t_tZj zRROP>QY-h78|kCRqpIq9U!H%TazMf@svhqf*W?0dv3PF zcQlLq;_h=IReE%9&Y9bz_fts~)f?_`WM5JfwJy>MSB!S7E<7Ql8-a^d*J>eag02d~ zuhzoc?o_kp;%hdQBH!wu*VU*Nh=;p5(h=W`c4RiDhr_b5ZZoGAJUR%~go<x boE1GHmx>fqEhg8oTI83iT%mitoM8PQf;amL literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/duck_m/duck_m_tail.vox b/assets/voxygen/voxel/npc/duck_m/duck_m_tail.vox new file mode 100644 index 0000000000000000000000000000000000000000..7799a36beb3928dc719362f4f4830af168d85de6 GIT binary patch literal 55643 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 literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/duck_m/duck_m_wing.vox b/assets/voxygen/voxel/npc/duck_m/duck_m_wing.vox new file mode 100644 index 0000000000000000000000000000000000000000..08c1dc009bda68e354b1ce164311719613f6ef77 GIT binary patch literal 55631 zcmdtrdsI}%9S86cG%=DGjffEwSiuLN;yzdx!3REwB8rFxm8drhds(jRtIL95X;Q(O z7-M1xLECE7q*ha_Q42OUZ7s3Z+NRbRnL-0MLrfIGmkNm8)V`-Cte#5!DWuS7OO>Wk~OQvYOz}bOXnBIbAU`7&*|EHTwXY9 zWl16zx)L8_w-#{wb*@->h8v@}u~(F>XB4loSCqYH6tAdPl%r=<|GuKKdPec8dyVDn z8O3L&SJafAQ8)J$<>IK4d9!B;UXV=L+1b?C*hp7e1kK-dno{FGr1#_Fsk3tthti>f z4k{|Nl6!S7HP#hU$L0d+XmL}=_7J%`f;1#AfrjRfpybBUl(f!5B7Xph>Omy--a+DY z0u5Ot==MTEgNp@CJR)ddwV>OY1r6IKDB+->tVMCOz(0h_HVmUjj^9ml8wD+C6;ys$ zP{VmaBbFpml957V8Z&50dmfDn$5YDIM9SEgOtv%QX~Zf)V>ZaK+XPKJDM&1n^RFLB zqIEEdWAU_k{7~9&9Znagk0M$ynp)-Mc3kGyWPV5H)_OsQpA^)&U(ng3vYsbsZ@!=x zN(5c<3M!NJ`cgsLJhHt`(42HZYqAB^%@wpmwlB#_BqKkC)-Oq;Hr-B*_qwR9Y7TWi zyqw} z7L{%3DKsT>9Oc;4Y1_uT=OO)KppOf;x`4b1<`icIW3fKSrW?Jq-^Jr{mkrJ$xWhUOP!i z^x)W5nvvT=o-Lc`&NU~5$`JbPbO2wJusnjun9$KDH8{LcPV4I&B zw(qCrXWQwqW2flq)vM$TZKfGbyQ%Q$leF~61uWmo z*2f;dEU&*E_NjEhHIq(E&!<=Ni}*RH?z`w~p+nwNlSvdiN#u;{eoV$j<9hb%s&j{r ziIIOhmMwfq25{nM3jb|T&(26gNFOLGn#ZWDtgX@caiXo$4R2M1<81s8(!#-7ZxufP zt(i8}t0tsZn02S>z;!Lu)xwV)%?O6gszX)X5jC8Z+cdA0+a_5voo3Z$MrNC}b!N?_ zCaUz+*7G`456$NZtICZUetP3**p8#&ZN|~?1&O2K8yiQ%Pazx)Kag=Ww?JtIL22Tk zG66d}d4y8$e(u{!8+zF)_38mpr zSa6J(T9tP?|<4&1axA4?t-i zgwlK#O0xk<(*&hyhSF?=(tHj|vk6M`5R~R&D9vUl%_C5nEl`?Ap)^~eG><`P9*5Fw zgVHgrTHP0=4B|&kDxT?pfo>*(wv9VyaJ`U0Ht{q zN^=oP^Ajk|Yfze>LTO%y(!2qs`5BbvO(@OJp){AEG{1n-T!zy85=wIgO7j+!=4~j= zub?#VKxuvrrFj=h^BXA5Z=p25gVOvSO7jOO%^#sOe}dBd8A|gWl;$r`nyXNnzd~u= zhtm8FO7j7f=I>CN51};wfYSUEO7kx$&A*{E|AEr{7fM4=8qx3n{r_GKFP^<1RqhF= z7?}sitVJ3^dVenHfjZUT4|;S-6;=*6WP#I7GItL*0+l?5X2ishjlkkYR$B|RD;K;sg6$Sl1Di4$lWKc4H%z1DX)8y8lbNs zjB=kj1W#vE@^CN`oqJDL$AR*-``qq`Uva|j2F+ulv!V^vgJgV`(_*t`v0b(M4ZV9H zH}rXi_<+|uAK6YdYtW&2)S9l5JhL{N+MKyvYvrENwerHX%Ai-fcCO5E`rIDX z=#AR@+-xT0rP+MQ}8OEI9(RVCy($e6j`Wz{`i)2b$qc2eW4Gdz(hb7FO(R(FlUQK~baC{^%=jT;EjQPf7(gxnQIK<$O? z`mpShceP!G$h$h^b2miyx~Xd-J{X8CA96evc`-S9bUh#N zx<*F-bNK3LazB^6ZjiBr-M9+7QB~~%8L?kgVN+GrwhGloLba6;Hi15HTm{*LAlv7U zAM9~*d9BaS-Ba?9`}_WDI`zMom;U?Y#Qp57^q!2*NR)^>hi-|mWAsLE^z3@1kMxm) zNFV9>aHUszrB@22=g);+=q37vXE{%HwM!jteMp~r)+NQ`&GV|ZSF_;xh{JO;qHn6a?2 zNEr(rW5@6q^D#VSRCo-J;W0dhr;HDe;W0dh$MEEl8Xm)Acnpu>DPt~^$(p<=p=Ul3 zPs9`PL_85sxgB^So(Ljx!N}s}7Uh#Z-)5g4PsCGh8=it@06p!L5j{zRVqj(gL;wg^?9tEs;6o}#} zj|+<_Cn{f33S~a|5mM%(%tx7zG9P6=)cI*4M)G8`h7FJ45j=uN@Ccsr_~8*ef=BQO z9+o8Zcm$8&5jzeP1 F`~z8Tat8na literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/marlin/marlin_fin_r.vox b/assets/voxygen/voxel/npc/marlin/marlin_fin_r.vox new file mode 100644 index 0000000000000000000000000000000000000000..ed3a43f38a2b924a79907531d62f216c69a21a7a GIT binary patch literal 1152 zcmXZaJ&O}j6vpv0N$fLd(+3P0;-ma=$RgEr$j;v$F{Ic)!0NKWDS}dww3jPEI0DPfH(etP@cp5$uLJVZ-Q+-sstkM4#vrJBdEg zvkIkGdZkwirROh%Ug#yocgdXu4atjcc+eGz@ojhv&&zjTE=PdJ02ntjE^J(+j0KOe zVR(%B7@jgJJch^c7#_n@#)rr77#_o8c=AXMkKr*qhR5)fF_+0?O@%!tm znB9>$xFMI1w>+ghnXJhRJ(pHIibwG%9>r5$13ZdH@hBd}Q(g@`3Rv+d5XDno7cQop zs2rpe%6#%Gq|8T|k1`)+KFWNk^V>p<W4&I-a=DDvY874AMccN~G)>fX&9Nf?04Sev A`~Uy| literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/marlin/marlin_head.vox b/assets/voxygen/voxel/npc/marlin/marlin_head.vox new file mode 100644 index 0000000000000000000000000000000000000000..830d13093434be0e6f25aff636d1339d60775928 GIT binary patch literal 1352 zcmXAnv1=4T6o=o;?(ObHoBI#sT5V*svN3uhDbz%{7;S81KoHBM5lm+qF+~tw5Ueb6 zwUxnAh%jZ?tFE@YGAV>SznR?Lw>$IO-I;IS%;uA=YhOg<(dyb`E^=!i@@(zpgR8m2 z?+Oq}16wa%uGNkPzq`T4#$RX z@Vxu<;k{LUf$(5X@ zMe;}<=|u8K&cyC*hNX93Sk8iE>ta6rWcgzFn38 zA4Nc!Oc_`iq?838rJ?vJ^-+9Esqj&J6d%P$@hRoQNAXd76d%PWmsIgld=wwWNAW3T zE|tkP`3OKxeIh;)pNLPyC*o6<1D}XbL?SX_Wbm>?`K8Y{Xp`d;@hQuOPsAtU^X4WZ zJ7Qno-Z%U4&fY)2ZGXSNw(UD&o7d%XdCMy0%4C~-kTZ0^C*TwC3HStj$~M3!;1lo( z_yl~)*1#tq9PkN<1boVNVX(l9frkP?pgwsE1?m&1PoO@5`UL7joOcT-()wpc9e^z_tbvzhJh Y@7uw_fla4Vn@lD)9*=D_8u6^if0Wdmd;kCd literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/marlin/marlin_rear.vox b/assets/voxygen/voxel/npc/marlin/marlin_rear.vox new file mode 100644 index 0000000000000000000000000000000000000000..2afa3c2035d98e1fa77038ef5ca1757ccc3eb6d1 GIT binary patch literal 1360 zcmXAnv1=4T6o=o=%$`k}`w!$=ZRBC4P2!29P!r{1Ok*Phf>%?DT45VU}cf3 ztqhhzgei}`>T0|D2P~f7OtSfIW`8^T=G(Ws_jLc}7ZG{9x&4Ge?)M_kw+|j})D!-0 z08tzFUmk34bTt*(f)qwW?D}Qj#bspP$25d445JvAedrm(kRYbSwJ-$6uFgZ+NwHr> z5Y8pVAuta?hMxO_unvUUk^4r}Pvq=42J#<5o z>JxVo^+%yPunuA!xjv$H$J#{wgilJW?-I`z@Q%yCep2glf2!+4-GgVX&(Ha@M_Zec zT3cjsap7Kyf85>m--n0(@9N6`93Q)%pSRqLF=kK@cTU|3Ffno?H*%()$uoK8B$H=y z23K+=S8^p#a{gV&g20jI!f=|Jx;8P#1;8XA^_!J?>2fJos{2D%n&#MpLu3LbQAz*A~6gG;q zw%}t-3?HLEhEHo1K8BCsWB3?8t$p|yK8BCsWBAmT8a{@P;bZt1KCR8IGu5UZ3gq-B zLRGWH`Q(Ey+d=wwWNAYQ&0X~Y4;-mN|KJC-MM-f(h z6cNRzeJ&JJZd4X2fzqG)2`T+i`lIwm>5tMM;`+6KLG4Vn;lM}m5qtz6!AJ0EpC3Mg zkKiNt2tJeqa(o0I!AJ068z9F=@DY3j8MX#;3JNz13)h4zxP&>Jz!A(~3KJNEtYHP0 zFozSM=Q4vSOkfPMh80}G98TZ}W-x^bjD5Xc`)alF#bV*}`P@%WPyOud%xANiPp4C# MOeQ`ak6A17AEkDkyZ`_I literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/marlin/marlin_tail.vox b/assets/voxygen/voxel/npc/marlin/marlin_tail.vox new file mode 100644 index 0000000000000000000000000000000000000000..27b677a98ecb3e8b68192726d758a959283b9b95 GIT binary patch literal 1308 zcmXAnF>ljA7=~Xvwvz%&{sGDg3p#aTDg>bpq$((-bYVd!AtA#liL;uKCymhOQduNo%Qw9yBZNtSVLOS{3&y##B2kbgGUk)vH?dqz8>^@|vu- zQ$r6L_2@T~8hUEzH;ie>TIOpR)AC$9X=!(AkNZ%mW6qAXI@apet-47EeRVqn`i=Tz z{dK4Qs;~ZJKzpO~2lNN40c^kxc>4IkmUyXr%r7q7Q}Xw_yZ-C&(EnXs`R|hx_s^e8 z?#Z}D2q1U5ZV6}@xse+=O^@V}Jkp8ek(`SM@<1NQ13@6?=OP!mqW@6NBsb=2Dq#lg|L;)FPjaRX z_yl|cJ^`PAPuT|e1bhNM0iS?R*&6r+gabYSk$_LxE=(49Ffb?(1nQHwP@q16`UL6| zs866i#Cf-XL9R@;;lxMy2p{1ie1uQge)tF<;Uj#64^sj;KEg-%2p^UKa(sl3@DVaB z4dhHH9u@OcDNu7wVGJV}f|js=OPIka9KjUEFoK~k lmrGwP7CxWPeKwo<+1Z(&pP&15I`#2*?4!}hhr=ObrT>&HiU|M! literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/marlin/marlin_torso.vox b/assets/voxygen/voxel/npc/marlin/marlin_torso.vox new file mode 100644 index 0000000000000000000000000000000000000000..a0c220bd89bbd9663dac240fd91a892317ac5832 GIT binary patch literal 1476 zcmXAny>AmS7>D2U`96}CA^!o&3JbEBmZU6ANT7et2&$rTb4 zLo+cjQ86MF$zyakWoFAi!0__e&E<*hU;Odhd$*YF-~KEjPj_~wH1fR?dA@t_csmXF z41q{)!T!sG-IIb2pF^m{cD0mU1r|y$P#2~zi>hzU)~v}4r?nYQTQguhnAK33!PG)y zTK;RNm1$=dTGKG6VNS!GhB?i&q;J`;f*KlAaF*>V&S{M>uDcp4uu!HpJD7f8nnh)X zok?TUG9D~i(=G;PYtfi()^GQ<*<$ThZ<$vVrfbb`Vc5SZOayyMwVkotBW+1* zX{{~$indB^@?oBZGPS8T%&Yld@HOmHByUmzSzB>W95O{e5(#|E{m~&&i4E#YN6F23ijeM3te22y_B@ zAP?kp)suU2&%l#=avF8yj@*$u0!PkQgrY9UC<&x1fCmTU{1N$y_(XgnJ`ta+ zhfl;O;`8fc%7CLHJ`taYPsAtUlWzl`h)={P;uG;nw-)h<_(Xi7kmG}0HVOO&d;&g) z@4w!Z0H1(B;IP2PflYE*@CkGRK7slKd~&Jq3HStj0zLtsTt0jPJ^`PAPrxUYG~g5P z3HStj0zSFSxiZNnT@J{pkH^R3{f9;yC#tM6Xi*T3Ih>Cpqx#T~hMy!k1m$|Rd~k+ZeK$Km7fariiV z@@s&P!^h#{@NxL$R|6l1u*1h8;_%6@3!6C>Iwm;+M}5*Of@*nae7-I zK&ni#;lfAo5qtz6!AJ1PuOB{wkKiNt2tI5H^g=3h*1jaA|*}xjEU Self { + let mut rng = thread_rng(); + Self { + head: *(&ALL_HEADS).choose(&mut rng).unwrap(), + torso: *(&ALL_TORSOS).choose(&mut rng).unwrap(), + tail: *(&ALL_TAILS).choose(&mut rng).unwrap(), + wing_l: *(&ALL_WING_LS).choose(&mut rng).unwrap(), + wing_r: *(&ALL_WING_RS).choose(&mut rng).unwrap(), + leg_l: *(&ALL_LEG_LS).choose(&mut rng).unwrap(), + leg_r: *(&ALL_LEG_RS).choose(&mut rng).unwrap(), + } + } +} + +#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +pub enum Head { + Default, +} +const ALL_HEADS: [Head; 1] = [Head::Default]; + +#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +pub enum Torso { + Default, +} +const ALL_TORSOS: [Torso; 1] = [Torso::Default]; + +#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +pub enum Tail { + Default, +} +const ALL_TAILS: [Tail; 1] = [Tail::Default]; + +#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +pub enum WingL { + Default, +} +const ALL_WING_LS: [WingL; 1] = [WingL::Default]; + +#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +pub enum WingR { + Default, +} +const ALL_WING_RS: [WingR; 1] = [WingR::Default]; + +#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +pub enum LegL { + Default, +} +const ALL_LEG_LS: [LegL; 1] = [LegL::Default]; + +#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +pub enum LegR { + Default, +} +const ALL_LEG_RS: [LegR; 1] = [LegR::Default]; \ No newline at end of file diff --git a/common/src/comp/body/fish_medium.rs b/common/src/comp/body/fish_medium.rs new file mode 100644 index 0000000000..7da9bce91f --- /dev/null +++ b/common/src/comp/body/fish_medium.rs @@ -0,0 +1,61 @@ +use rand::{seq::SliceRandom, thread_rng}; + +#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +pub struct Body { + pub head: Head, + pub torso: Torso, + pub rear: Rear, + pub tail: Tail, + pub fin_l: FinL, + pub fin_r: FinR, +} +impl Body { + pub fn random() -> Self { + let mut rng = thread_rng(); + Self { + head: *(&ALL_HEADS).choose(&mut rng).unwrap(), + torso: *(&ALL_TORSOS).choose(&mut rng).unwrap(), + rear: *(&ALL_REARS).choose(&mut rng).unwrap(), + tail: *(&ALL_TAILS).choose(&mut rng).unwrap(), + fin_l: *(&ALL_FIN_LS).choose(&mut rng).unwrap(), + fin_r: *(&ALL_FIN_RS).choose(&mut rng).unwrap(), + + } + } +} + +#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +pub enum Head { + Default, +} +const ALL_HEADS: [Head; 1] = [Head::Default]; + +#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +pub enum Torso { + Default, +} +const ALL_TORSOS: [Torso; 1] = [Torso::Default]; + +#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +pub enum Rear { + Default, +} +const ALL_REARS: [Rear; 1] = [Rear::Default]; + +#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +pub enum Tail { + Default, +} +const ALL_TAILS: [Tail; 1] = [Tail::Default]; + +#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +pub enum FinL { + Default, +} +const ALL_FIN_LS: [FinL; 1] = [FinL::Default]; + +#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +pub enum FinR { + Default, +} +const ALL_FIN_RS: [FinR; 1] = [FinR::Default]; diff --git a/common/src/comp/mod.rs b/common/src/comp/mod.rs index d79fb41ae7..2a0c67e3cd 100644 --- a/common/src/comp/mod.rs +++ b/common/src/comp/mod.rs @@ -16,7 +16,7 @@ mod visual; // Reexports pub use admin::Admin; pub use agent::Agent; -pub use body::{humanoid, object, quadruped, quadruped_medium, Body}; +pub use body::{humanoid, object, quadruped, quadruped_medium, bird_medium, fish_medium, Body}; pub use character_state::{ActionState, CharacterState, MovementState}; pub use controller::{ ControlEvent, Controller, ControllerInputs, InventoryManip, MountState, Mounting, diff --git a/common/src/npc.rs b/common/src/npc.rs index 3f95c17cca..0c0ebe25d5 100644 --- a/common/src/npc.rs +++ b/common/src/npc.rs @@ -30,6 +30,7 @@ impl FromStr for NpcKind { "humanoid" => Ok(NpcKind::Humanoid), "wolf" => Ok(NpcKind::Wolf), "pig" => Ok(NpcKind::Pig), + _ => Err(()), } } diff --git a/voxygen/src/anim/birdmedium/idle.rs b/voxygen/src/anim/birdmedium/idle.rs new file mode 100644 index 0000000000..b3770c6142 --- /dev/null +++ b/voxygen/src/anim/birdmedium/idle.rs @@ -0,0 +1,74 @@ +use super::{ + super::{Animation, SkeletonAttr}, + BirdMediumSkeleton, +}; +use std::{f32::consts::PI, ops::Mul}; +use vek::*; + +pub struct IdleAnimation; + +impl Animation for IdleAnimation { + type Skeleton = BirdMediumSkeleton; + type Dependency = (f64); + + fn update_skeleton( + skeleton: &Self::Skeleton, + global_time: Self::Dependency, + anim_time: f64, + _rate: &mut f32, + _skeleton_attr: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let wave_ultra_slow = (anim_time as f32 * 1.0 + PI).sin(); + let wave_ultra_slow_cos = (anim_time as f32 * 1.0 + PI).cos(); + let wave_slow = (anim_time as f32 * 3.5 + PI).sin(); + let wave_slow_cos = (anim_time as f32 * 3.5 + PI).cos(); + + let duck_m_look = Vec2::new( + ((global_time + anim_time) as f32 / 8.0) + .floor() + .mul(7331.0) + .sin() + * 0.5, + ((global_time + anim_time) as f32 / 8.0) + .floor() + .mul(1337.0) + .sin() + * 0.25, + ); + + + next.duck_m_head.offset = Vec3::new(0.0, 7.5, 15.0) / 11.0; + next.duck_m_head.ori = + Quaternion::rotation_z(duck_m_look.x) * Quaternion::rotation_x(duck_m_look.y); + next.duck_m_head.scale = Vec3::one() / 10.88; + + next.duck_m_torso.offset = + Vec3::new(0.0, 4.5 - wave_ultra_slow_cos * 0.12, 2.0); + next.duck_m_torso.ori = Quaternion::rotation_x(0.0); + next.duck_m_torso.scale = Vec3::one() * 1.01; + + next.duck_m_tail.offset = Vec3::new(0.0, 3.1, -4.5); + next.duck_m_tail.ori = Quaternion::rotation_z(0.0); + next.duck_m_tail.scale = Vec3::one() * 0.98; + + next.duck_m_wing_l.offset = Vec3::new(0.0, -13.0, 8.0) / 11.0; + next.duck_m_wing_l.ori = Quaternion::rotation_z(0.0) + * Quaternion::rotation_x(0.0); + next.duck_m_wing_l.scale = Vec3::one() / 11.0; + + next.duck_m_wing_r.offset = Vec3::new(0.0, -11.7, 11.0) / 11.0; + next.duck_m_wing_r.ori = Quaternion::rotation_y(0.0); + next.duck_m_wing_r.scale = Vec3::one() / 11.0; + + next.duck_m_leg_l.offset = Vec3::new(0.0, 0.0, 12.0) / 11.0; + next.duck_m_leg_l.ori = Quaternion::rotation_y(0.0); + next.duck_m_leg_l.scale = Vec3::one() / 10.5; + + next.duck_m_leg_r.offset = Vec3::new(0.0, 0.75, 5.25); + next.duck_m_leg_r.ori = Quaternion::rotation_x(0.0); + next.duck_m_leg_r.scale = Vec3::one() * 1.00; + next + } +} diff --git a/voxygen/src/anim/birdmedium/jump.rs b/voxygen/src/anim/birdmedium/jump.rs new file mode 100644 index 0000000000..191412c2d5 --- /dev/null +++ b/voxygen/src/anim/birdmedium/jump.rs @@ -0,0 +1,62 @@ +use super::{ + super::{Animation, SkeletonAttr}, + BirdMediumSkeleton, +}; +use std::f32::consts::PI; +use vek::*; + +pub struct JumpAnimation; + +impl Animation for JumpAnimation { + type Skeleton = BirdMediumSkeleton; + type Dependency = (f32, f64); + + fn update_skeleton( + skeleton: &Self::Skeleton, + _global_time: Self::Dependency, + anim_time: f64, + _rate: &mut f32, + _skeleton_attr: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let wave_ultra_slow = (anim_time as f32 * 1.0 + PI).sin(); + let wave_ultra_slow_cos = (anim_time as f32 * 1.0 + PI).cos(); + let wave_slow = (anim_time as f32 * 3.5 + PI).sin(); + let wave_slow_cos = (anim_time as f32 * 3.5 + PI).cos(); + + + + next.duck_m_head.offset = Vec3::new(0.0, 7.5, 15.0) / 11.0; + next.duck_m_head.ori = + Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); + next.duck_m_head.scale = Vec3::one() / 10.88; + + next.duck_m_torso.offset = + Vec3::new(0.0, 4.5 - wave_ultra_slow_cos * 0.12, 2.0); + next.duck_m_torso.ori = Quaternion::rotation_x(0.0); + next.duck_m_torso.scale = Vec3::one() * 1.01; + + next.duck_m_tail.offset = Vec3::new(0.0, 3.1, -4.5); + next.duck_m_tail.ori = Quaternion::rotation_z(0.0); + next.duck_m_tail.scale = Vec3::one() * 0.98; + + next.duck_m_wing_l.offset = Vec3::new(0.0, -13.0, 8.0) / 11.0; + next.duck_m_wing_l.ori = Quaternion::rotation_z(0.0) + * Quaternion::rotation_x(0.0); + next.duck_m_wing_l.scale = Vec3::one() / 11.0; + + next.duck_m_wing_r.offset = Vec3::new(0.0, -11.7, 11.0) / 11.0; + next.duck_m_wing_r.ori = Quaternion::rotation_y(0.0); + next.duck_m_wing_r.scale = Vec3::one() / 11.0; + + next.duck_m_leg_l.offset = Vec3::new(0.0, 0.0, 12.0) / 11.0; + next.duck_m_leg_l.ori = Quaternion::rotation_y(0.0); + next.duck_m_leg_l.scale = Vec3::one() / 10.5; + + next.duck_m_leg_r.offset = Vec3::new(0.0, 0.75, 5.25); + next.duck_m_leg_r.ori = Quaternion::rotation_x(0.0); + next.duck_m_leg_r.scale = Vec3::one() * 1.00; + next + } +} \ No newline at end of file diff --git a/voxygen/src/anim/birdmedium/mod.rs b/voxygen/src/anim/birdmedium/mod.rs new file mode 100644 index 0000000000..53292603ba --- /dev/null +++ b/voxygen/src/anim/birdmedium/mod.rs @@ -0,0 +1,80 @@ +pub mod idle; +pub mod jump; +pub mod run; + +// Reexports +pub use self::idle::IdleAnimation; +pub use self::jump::JumpAnimation; +pub use self::run::RunAnimation; + +use super::{Bone, Skeleton}; +use crate::render::FigureBoneData; + +#[derive(Clone)] +pub struct BirdMediumSkeleton { + duck_m_head: Bone, + duck_m_torso: Bone, + duck_m_tail: Bone, + duck_m_wing_l: Bone, + duck_m_wing_r: Bone, + duck_m_leg_l: Bone, + duck_m_leg_r: Bone, + +} + +impl BirdMediumSkeleton { + pub fn new() -> Self { + Self { + duck_m_head: Bone::default(), + duck_m_torso: Bone::default(), + duck_m_tail: Bone::default(), + duck_m_wing_l: Bone::default(), + duck_m_wing_r: Bone::default(), + duck_m_leg_l: Bone::default(), + duck_m_leg_r: Bone::default(), + + + } + } +} + +impl Skeleton for BirdMediumSkeleton { + fn compute_matrices(&self) -> [FigureBoneData; 16] { + let torso_mat = self.duck_m_torso.compute_base_matrix(); + + + [ + FigureBoneData::new(self.duck_m_head.compute_base_matrix() * torso_mat), + FigureBoneData::new( + torso_mat, + ), + FigureBoneData::new(self.duck_m_tail.compute_base_matrix() * torso_mat), + FigureBoneData::new(self.duck_m_wing_l.compute_base_matrix() * torso_mat), + FigureBoneData::new(self.duck_m_wing_r.compute_base_matrix() * torso_mat), + FigureBoneData::new(self.duck_m_leg_l.compute_base_matrix()), + FigureBoneData::new(self.duck_m_leg_r.compute_base_matrix()), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + ] + } + + fn interpolate(&mut self, target: &Self, dt: f32) { + self.duck_m_head + .interpolate(&target.duck_m_head, dt); + self.duck_m_torso.interpolate(&target.duck_m_torso, dt); + self.duck_m_tail + .interpolate(&target.duck_m_tail, dt); + self.duck_m_wing_l.interpolate(&target.duck_m_wing_l, dt); + self.duck_m_wing_r + .interpolate(&target.duck_m_wing_r, dt); + self.duck_m_leg_l.interpolate(&target.duck_m_leg_l, dt); + self.duck_m_leg_r.interpolate(&target.duck_m_leg_r, dt); + } +} diff --git a/voxygen/src/anim/birdmedium/run.rs b/voxygen/src/anim/birdmedium/run.rs new file mode 100644 index 0000000000..ac940f6ae7 --- /dev/null +++ b/voxygen/src/anim/birdmedium/run.rs @@ -0,0 +1,74 @@ +use super::{ + super::{Animation, SkeletonAttr}, + BirdMediumSkeleton, +}; +use std::{f32::consts::PI, ops::Mul}; +use vek::*; + +pub struct RunAnimation; + +impl Animation for RunAnimation { + type Skeleton = BirdMediumSkeleton; + type Dependency = (f32, f64); + + fn update_skeleton( + skeleton: &Self::Skeleton, + (_velocity, global_time): Self::Dependency, + anim_time: f64, + _rate: &mut f32, + _skeleton_attr: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let wave_ultra_slow = (anim_time as f32 * 1.0 + PI).sin(); + let wave_ultra_slow_cos = (anim_time as f32 * 1.0 + PI).cos(); + let wave_slow = (anim_time as f32 * 3.5 + PI).sin(); + let wave_slow_cos = (anim_time as f32 * 3.5 + PI).cos(); + + let duck_look = Vec2::new( + ((global_time + anim_time) as f32 / 8.0) + .floor() + .mul(7331.0) + .sin() + * 0.5, + ((global_time + anim_time) as f32 / 8.0) + .floor() + .mul(1337.0) + .sin() + * 0.25, + ); + + + next.duck_m_head.offset = Vec3::new(0.0, 7.5, 15.0) / 11.0; + next.duck_m_head.ori = + Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); + next.duck_m_head.scale = Vec3::one() / 10.88; + + next.duck_m_torso.offset = + Vec3::new(0.0, 4.5 - wave_ultra_slow_cos * 0.12, 2.0); + next.duck_m_torso.ori = Quaternion::rotation_x(0.0); + next.duck_m_torso.scale = Vec3::one() * 1.01; + + next.duck_m_tail.offset = Vec3::new(0.0, 3.1, -4.5); + next.duck_m_tail.ori = Quaternion::rotation_z(0.0); + next.duck_m_tail.scale = Vec3::one() * 0.98; + + next.duck_m_wing_l.offset = Vec3::new(0.0, -13.0, 8.0) / 11.0; + next.duck_m_wing_l.ori = Quaternion::rotation_z(0.0) + * Quaternion::rotation_x(0.0); + next.duck_m_wing_l.scale = Vec3::one() / 11.0; + + next.duck_m_wing_r.offset = Vec3::new(0.0, -11.7, 11.0) / 11.0; + next.duck_m_wing_r.ori = Quaternion::rotation_y(0.0); + next.duck_m_wing_r.scale = Vec3::one() / 11.0; + + next.duck_m_leg_l.offset = Vec3::new(0.0, 0.0, 12.0) / 11.0; + next.duck_m_leg_l.ori = Quaternion::rotation_y(0.0); + next.duck_m_leg_l.scale = Vec3::one() / 10.5; + + next.duck_m_leg_r.offset = Vec3::new(0.0, 0.75, 5.25); + next.duck_m_leg_r.ori = Quaternion::rotation_x(0.0); + next.duck_m_leg_r.scale = Vec3::one() * 1.00; + next + } +} \ No newline at end of file diff --git a/voxygen/src/anim/character/run - Copy.rs b/voxygen/src/anim/character/run - Copy.rs new file mode 100644 index 0000000000..6e0a6aff0b --- /dev/null +++ b/voxygen/src/anim/character/run - Copy.rs @@ -0,0 +1,147 @@ +use super::{ + super::{Animation, SkeletonAttr}, + CharacterSkeleton, +}; +use std::f32::consts::PI; +use std::ops::Mul; +use vek::*; + +pub struct SneakAnimation; + +impl Animation for SneakAnimation { + type Skeleton = CharacterSkeleton; + type Dependency = (Vec3, Vec3, Vec3, f64); + + fn update_skeleton( + skeleton: &Self::Skeleton, + (velocity, orientation, last_ori, global_time): Self::Dependency, + anim_time: f64, + rate: &mut f32, + skeleton_attr: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let speed = Vec2::::from(velocity).magnitude(); + *rate = speed; + + let constant = 1.0; + let wave = (((5.0) + / (1.1 + 3.9 * ((anim_time as f32 * constant as f32 * 1.2).sin()).powf(2.0 as f32))) + .sqrt()) + * ((anim_time as f32 * constant as f32 * 1.2).sin()); + let wave_cos = (((5.0) + / (1.1 + 3.9 * ((anim_time as f32 * constant as f32 * 2.4).sin()).powf(2.0 as f32))) + .sqrt()) + * ((anim_time as f32 * constant as f32 * 1.5).sin()); + let wave_cos_dub = (((5.0) + / (1.1 + 3.9 * ((anim_time as f32 * constant as f32 * 4.8).sin()).powf(2.0 as f32))) + .sqrt()) + * ((anim_time as f32 * constant as f32 * 1.5).sin()); + + let wave_diff = (anim_time as f32 * 0.6).sin(); + let wave_stop = (anim_time as f32 * 2.6).min(PI / 2.0).sin(); + let head_look = Vec2::new( + ((global_time + anim_time) as f32 / 4.0) + .floor() + .mul(7331.0) + .sin() + * 0.2, + ((global_time + anim_time) as f32 / 4.0) + .floor() + .mul(1337.0) + .sin() + * 0.1, + ); + + let ori = Vec2::from(orientation); + let last_ori = Vec2::from(last_ori); + + let tilt = if Vec2::new(ori, last_ori) + .map(|o| Vec2::::from(o).magnitude_squared()) + .map(|m| m > 0.001 && m.is_finite()) + .reduce_and() + && ori.angle_between(last_ori).is_finite() + { + ori.angle_between(last_ori).min(0.5) + * last_ori.determine_side(Vec2::zero(), ori).signum() + } else { + 0.0 + } * 1.3; + + next.head.offset = Vec3::new( + 0.0, + -3.0 + skeleton_attr.neck_forward, + skeleton_attr.neck_height + 20.0 + wave_cos * 1.3, + ); + next.head.ori = Quaternion::rotation_z(head_look.x + wave * 0.1) + * Quaternion::rotation_x(head_look.y + 0.35); + next.head.scale = Vec3::one() * skeleton_attr.head_scale; + + next.chest.offset = Vec3::new(0.0, 0.0, 7.0 + wave_cos * 1.1); + next.chest.ori = Quaternion::rotation_z(wave * 0.2); + next.chest.scale = Vec3::one(); + + next.belt.offset = Vec3::new(0.0, 0.0, 5.0 + wave_cos * 1.1); + next.belt.ori = Quaternion::rotation_z(wave * 0.35); + next.belt.scale = Vec3::one(); + + next.shorts.offset = Vec3::new(0.0, 0.0, 2.0 + wave_cos * 1.1); + next.shorts.ori = Quaternion::rotation_z(wave * 0.6); + next.shorts.scale = Vec3::one(); + + next.l_hand.offset = Vec3::new( + -6.0 + wave_stop * -1.0, + -0.25 + wave_cos * 2.0, + 5.0 - wave * 1.5, + ); + next.l_hand.ori = + Quaternion::rotation_x(0.8 + wave_cos * 1.2) * Quaternion::rotation_y(wave_stop * 0.1); + next.l_hand.scale = Vec3::one(); + + next.r_hand.offset = Vec3::new( + 6.0 + wave_stop * 1.0, + -0.25 - wave_cos * 2.0, + 5.0 + wave * 1.5, + ); + next.r_hand.ori = Quaternion::rotation_x(0.8 + wave_cos * -1.2) + * Quaternion::rotation_y(wave_stop * -0.1); + next.r_hand.scale = Vec3::one(); + + next.l_foot.offset = Vec3::new(-3.4, 0.0 + wave_cos * 1.0, 6.0 - wave_cos_dub * 0.7); + next.l_foot.ori = Quaternion::rotation_x(-0.0 - wave_cos * 1.2); + next.l_foot.scale = Vec3::one(); + + next.r_foot.offset = Vec3::new(3.4, 0.0 - wave_cos * 1.0, 6.0 - wave_cos_dub * 0.7); + next.r_foot.ori = Quaternion::rotation_x(-0.0 + wave_cos * 1.2); + next.r_foot.scale = Vec3::one(); + + next.weapon.offset = Vec3::new( + -7.0 + skeleton_attr.weapon_x, + -5.0 + skeleton_attr.weapon_y, + 15.0, + ); + next.weapon.ori = + Quaternion::rotation_y(2.5) * Quaternion::rotation_z(1.57 + wave_cos * 0.25); + next.weapon.scale = Vec3::one(); + + next.l_shoulder.offset = Vec3::new(-5.0, 0.0, 4.7); + next.l_shoulder.ori = Quaternion::rotation_x(wave_cos * 0.15); + next.l_shoulder.scale = Vec3::one() * 1.1; + + next.r_shoulder.offset = Vec3::new(5.0, 0.0, 4.7); + next.r_shoulder.ori = Quaternion::rotation_x(wave * 0.15); + next.r_shoulder.scale = Vec3::one() * 1.1; + + next.draw.offset = Vec3::new(0.0, 5.0, 0.0); + next.draw.ori = Quaternion::rotation_y(0.0); + next.draw.scale = Vec3::one() * 0.0; + + next.torso.offset = Vec3::new(0.0, 0.3 + wave * -0.08, 0.4) * skeleton_attr.scaler; + next.torso.ori = + Quaternion::rotation_x(wave_stop * speed * -0.06 + wave_diff * speed * -0.005) + * Quaternion::rotation_y(tilt); + next.torso.scale = Vec3::one() / 11.0 * skeleton_attr.scaler; + + next + } +} diff --git a/voxygen/src/anim/character/sneak.rs b/voxygen/src/anim/character/sneak.rs new file mode 100644 index 0000000000..389c370bb4 --- /dev/null +++ b/voxygen/src/anim/character/sneak.rs @@ -0,0 +1,151 @@ +use super::{ + super::{Animation, SkeletonAttr}, + CharacterSkeleton, +}; +use std::f32::consts::PI; +use std::ops::Mul; +use vek::*; + +pub struct SneakAnimation; + +impl Animation for SneakAnimation { + type Skeleton = CharacterSkeleton; + type Dependency = (Vec3, Vec3, Vec3, f64); + + fn update_skeleton( + skeleton: &Self::Skeleton, + (velocity, orientation, last_ori, global_time): Self::Dependency, + anim_time: f64, + rate: &mut f32, + skeleton_attr: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let speed = Vec2::::from(velocity).magnitude(); + *rate = speed; + + let constant = 1.0; + let wave = (((5.0) + / (1.1 + 3.9 * ((anim_time as f32 * constant as f32 * 1.2).sin()).powf(2.0 as f32))) + .sqrt()) + * ((anim_time as f32 * constant as f32 * 1.2).sin()); + let wavecos = (((5.0) + / (1.1 + 3.9 * ((anim_time as f32 * constant as f32 * 1.2).sin()).powf(2.0 as f32))) + .sqrt()) + * ((anim_time as f32 * constant as f32 * 1.2).cos()); + let wave_cos = (((5.0) + / (1.1 + 3.9 * ((anim_time as f32 * constant as f32 * 2.4).sin()).powf(2.0 as f32))) + .sqrt()) + * ((anim_time as f32 * constant as f32 * 1.5).sin()); + let wave_cos_dub = (((5.0) + / (1.1 + 3.9 * ((anim_time as f32 * constant as f32 * 4.8).sin()).powf(2.0 as f32))) + .sqrt()) + * ((anim_time as f32 * constant as f32 * 1.5).sin()); + let wave_slow = (anim_time as f32 * 0.1).sin(); + let wave_diff = (anim_time as f32 * 0.6).sin(); + let wave_stop = (anim_time as f32 * 2.6).min(PI / 2.0).sin(); + let head_look = Vec2::new( + ((global_time + anim_time) as f32 *0.25) + .floor() + .mul(7331.0) + .sin() + * 0.4, + ((global_time + anim_time) as f32 *0.25) + .floor() + .mul(1337.0) + .sin() + * 0.2, + ); + + let ori = Vec2::from(orientation); + let last_ori = Vec2::from(last_ori); + + let tilt = if Vec2::new(ori, last_ori) + .map(|o| Vec2::::from(o).magnitude_squared()) + .map(|m| m > 0.001 && m.is_finite()) + .reduce_and() + && ori.angle_between(last_ori).is_finite() + { + ori.angle_between(last_ori).min(0.5) + * last_ori.determine_side(Vec2::zero(), ori).signum() + } else { + 0.0 + } * 1.3; + + next.head.offset = Vec3::new( + 0.0, + 0.0 + skeleton_attr.neck_forward, + skeleton_attr.neck_height + 16.0, + ); + next.head.ori = Quaternion::rotation_z(head_look.x + wave * 0.1) + * Quaternion::rotation_x(head_look.y + 0.05); + next.head.scale = Vec3::one() * skeleton_attr.head_scale; + + next.chest.offset = Vec3::new(0.0, -1.5, 3.0 +wave_slow * 2.0); + next.chest.ori = Quaternion::rotation_x(-0.5) * Quaternion::rotation_z(wave * 0.15); + next.chest.scale = Vec3::one(); + + next.belt.offset = Vec3::new(0.0, 0.0, 1.5 + wave_cos * 0.3); + next.belt.ori = Quaternion::rotation_x(-0.1) * Quaternion::rotation_z(wave * 0.25); + next.belt.scale = Vec3::one(); + + next.shorts.offset = Vec3::new(0.0, 1.0, -1.0 + wave_cos * 0.3); + next.shorts.ori = Quaternion::rotation_x(0.2) *Quaternion::rotation_z(wave * 0.4); + next.shorts.scale = Vec3::one(); + + next.l_hand.offset = Vec3::new( + -5.0 + wave_stop * -0.5, + 2.25, + 4.0 - wave * 1.0, + ); + next.l_hand.ori = + Quaternion::rotation_x(1.5 + wave_cos * 0.1) * Quaternion::rotation_y(wave_stop * 0.1); + next.l_hand.scale = Vec3::one(); + + next.r_hand.offset = Vec3::new( + 5.0 + wave_stop * 0.5, + 2.25, + 4.0 + wave * 1.0, + ); + next.r_hand.ori = Quaternion::rotation_x(1.5 + wave_cos * -0.1) + * Quaternion::rotation_y(wave_stop * -0.1); + next.r_hand.scale = Vec3::one(); + + next.l_foot.offset = Vec3::new(-3.4, 5.0+ wave * -3.0, 4.0); + next.l_foot.ori = Quaternion::rotation_x(-0.8 + wavecos * 0.15); + next.l_foot.scale = Vec3::one(); + + next.r_foot.offset = Vec3::new(3.4, 5.0+ wave * 3.0, 4.0); + next.r_foot.ori = Quaternion::rotation_x(-0.8 - wavecos * 0.15); + next.r_foot.scale = Vec3::one(); + + next.weapon.offset = Vec3::new( + -7.0 + skeleton_attr.weapon_x, + -5.0 + skeleton_attr.weapon_y, + 15.0, + ); + next.weapon.ori = + Quaternion::rotation_y(2.5) * Quaternion::rotation_z(1.57 + wave_cos * 0.25); + next.weapon.scale = Vec3::one(); + + next.l_shoulder.offset = Vec3::new(-5.0, 0.0, 4.7); + next.l_shoulder.ori = Quaternion::rotation_x(wavecos * 0.05); + next.l_shoulder.scale = Vec3::one() * 1.1; + + next.r_shoulder.offset = Vec3::new(5.0, 0.0, 4.7); + next.r_shoulder.ori = Quaternion::rotation_x(wave * 0.05); + next.r_shoulder.scale = Vec3::one() * 1.1; + + next.draw.offset = Vec3::new(0.0, 5.0, 0.0); + next.draw.ori = Quaternion::rotation_y(0.0); + next.draw.scale = Vec3::one() * 0.0; + + next.torso.offset = Vec3::new(0.0, 0.3 + wave * -0.08, 0.4) * skeleton_attr.scaler; + next.torso.ori = + Quaternion::rotation_x(wave_stop * speed * -0.03 + wave_diff * speed * -0.005) + * Quaternion::rotation_y(tilt); + next.torso.scale = Vec3::one() / 11.0 * skeleton_attr.scaler; + + next + } +} diff --git a/voxygen/src/anim/fishmedium/idle.rs b/voxygen/src/anim/fishmedium/idle.rs new file mode 100644 index 0000000000..2b04526f34 --- /dev/null +++ b/voxygen/src/anim/fishmedium/idle.rs @@ -0,0 +1,70 @@ +use super::{ + super::{Animation, SkeletonAttr}, + FishMediumSkeleton, +}; +use std::{f32::consts::PI, ops::Mul}; +use vek::*; + +pub struct IdleAnimation; + +impl Animation for IdleAnimation { + type Skeleton = FishMediumSkeleton; + type Dependency = (f64); + + fn update_skeleton( + skeleton: &Self::Skeleton, + global_time: Self::Dependency, + anim_time: f64, + _rate: &mut f32, + _skeleton_attr: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let wave_ultra_slow = (anim_time as f32 * 1.0 + PI).sin(); + let wave_ultra_slow_cos = (anim_time as f32 * 1.0 + PI).cos(); + let wave_slow = (anim_time as f32 * 3.5 + PI).sin(); + let wave_slow_cos = (anim_time as f32 * 3.5 + PI).cos(); + + let duck_m_look = Vec2::new( + ((global_time + anim_time) as f32 / 8.0) + .floor() + .mul(7331.0) + .sin() + * 0.5, + ((global_time + anim_time) as f32 / 8.0) + .floor() + .mul(1337.0) + .sin() + * 0.25, + ); + + + next.marlin_head.offset = Vec3::new(0.0, 7.5, 15.0) / 11.0; + next.marlin_head.ori = + Quaternion::rotation_z(duck_m_look.x) * Quaternion::rotation_x(duck_m_look.y); + next.marlin_head.scale = Vec3::one() / 10.88; + + next.marlin_torso.offset = + Vec3::new(0.0, 4.5 - wave_ultra_slow_cos * 0.12, 2.0); + next.marlin_torso.ori = Quaternion::rotation_x(0.0); + next.marlin_torso.scale = Vec3::one() * 1.01; + + next.marlin_rear.offset = Vec3::new(0.0, 3.1, -4.5); + next.marlin_rear.ori = Quaternion::rotation_z(0.0); + next.marlin_rear.scale = Vec3::one() * 0.98; + + next.marlin_tail.offset = Vec3::new(0.0, -13.0, 8.0) / 11.0; + next.marlin_tail.ori = Quaternion::rotation_z(0.0) + * Quaternion::rotation_x(0.0); + next.marlin_tail.scale = Vec3::one() / 11.0; + + next.marlin_fin_l.offset = Vec3::new(0.0, -11.7, 11.0) / 11.0; + next.marlin_fin_l.ori = Quaternion::rotation_y(0.0); + next.marlin_fin_l.scale = Vec3::one() / 11.0; + + next.marlin_fin_r.offset = Vec3::new(0.0, 0.0, 12.0) / 11.0; + next.marlin_fin_r.ori = Quaternion::rotation_y(0.0); + next.marlin_fin_r.scale = Vec3::one() / 10.5; + next + } +} diff --git a/voxygen/src/anim/fishmedium/jump.rs b/voxygen/src/anim/fishmedium/jump.rs new file mode 100644 index 0000000000..cb817e502b --- /dev/null +++ b/voxygen/src/anim/fishmedium/jump.rs @@ -0,0 +1,58 @@ +use super::{ + super::{Animation, SkeletonAttr}, + FishMediumSkeleton, +}; +use std::f32::consts::PI; +use vek::*; + +pub struct JumpAnimation; + +impl Animation for JumpAnimation { + type Skeleton = FishMediumSkeleton; + type Dependency = (f32, f64); + + fn update_skeleton( + skeleton: &Self::Skeleton, + global_time: Self::Dependency, + anim_time: f64, + _rate: &mut f32, + _skeleton_attr: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let wave_ultra_slow = (anim_time as f32 * 1.0 + PI).sin(); + let wave_ultra_slow_cos = (anim_time as f32 * 1.0 + PI).cos(); + let wave_slow = (anim_time as f32 * 3.5 + PI).sin(); + let wave_slow_cos = (anim_time as f32 * 3.5 + PI).cos(); + + + + next.marlin_head.offset = Vec3::new(0.0, 7.5, 15.0) / 11.0; + next.marlin_head.ori = + Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); + next.marlin_head.scale = Vec3::one() / 10.88; + + next.marlin_torso.offset = + Vec3::new(0.0, 4.5 - wave_ultra_slow_cos * 0.12, 2.0); + next.marlin_torso.ori = Quaternion::rotation_x(0.0); + next.marlin_torso.scale = Vec3::one() * 1.01; + + next.marlin_rear.offset = Vec3::new(0.0, 3.1, -4.5); + next.marlin_rear.ori = Quaternion::rotation_z(0.0); + next.marlin_rear.scale = Vec3::one() * 0.98; + + next.marlin_tail.offset = Vec3::new(0.0, -13.0, 8.0) / 11.0; + next.marlin_tail.ori = Quaternion::rotation_z(0.0) + * Quaternion::rotation_x(0.0); + next.marlin_tail.scale = Vec3::one() / 11.0; + + next.marlin_fin_l.offset = Vec3::new(0.0, -11.7, 11.0) / 11.0; + next.marlin_fin_l.ori = Quaternion::rotation_y(0.0); + next.marlin_fin_l.scale = Vec3::one() / 11.0; + + next.marlin_fin_r.offset = Vec3::new(0.0, 0.0, 12.0) / 11.0; + next.marlin_fin_r.ori = Quaternion::rotation_y(0.0); + next.marlin_fin_r.scale = Vec3::one() / 10.5; + next + } +} diff --git a/voxygen/src/anim/fishmedium/mod.rs b/voxygen/src/anim/fishmedium/mod.rs new file mode 100644 index 0000000000..6689dbea3a --- /dev/null +++ b/voxygen/src/anim/fishmedium/mod.rs @@ -0,0 +1,77 @@ +pub mod idle; +pub mod jump; +pub mod run; + +// Reexports +pub use self::idle::IdleAnimation; +pub use self::jump::JumpAnimation; +pub use self::run::RunAnimation; + +use super::{Bone, Skeleton}; +use crate::render::FigureBoneData; + +#[derive(Clone)] +pub struct FishMediumSkeleton { + marlin_head: Bone, + marlin_torso: Bone, + marlin_rear: Bone, + marlin_tail: Bone, + marlin_fin_l: Bone, + marlin_fin_r: Bone, + +} + +impl FishMediumSkeleton { + pub fn new() -> Self { + Self { + marlin_head: Bone::default(), + marlin_torso: Bone::default(), + marlin_rear: Bone::default(), + marlin_tail: Bone::default(), + marlin_fin_l: Bone::default(), + marlin_fin_r: Bone::default(), + + } + } +} + +impl Skeleton for FishMediumSkeleton { + fn compute_matrices(&self) -> [FigureBoneData; 16] { + let torso_mat = self.marlin_torso.compute_base_matrix(); + let rear_mat = self.marlin_rear.compute_base_matrix(); + + + [ + FigureBoneData::new(self.marlin_head.compute_base_matrix() * torso_mat), + FigureBoneData::new( + torso_mat, + ), + FigureBoneData::new(rear_mat * torso_mat), + FigureBoneData::new(self.marlin_tail.compute_base_matrix() * rear_mat), + FigureBoneData::new(self.marlin_fin_l.compute_base_matrix() * rear_mat), + FigureBoneData::new(self.marlin_fin_r.compute_base_matrix() * rear_mat), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + FigureBoneData::default(), + ] + } + + fn interpolate(&mut self, target: &Self, dt: f32) { + self.marlin_head + .interpolate(&target.marlin_head, dt); + self.marlin_torso.interpolate(&target.marlin_torso, dt); + self.marlin_rear + .interpolate(&target.marlin_rear, dt); + self.marlin_tail.interpolate(&target.marlin_tail, dt); + self.marlin_fin_l + .interpolate(&target.marlin_fin_l, dt); + self.marlin_fin_r.interpolate(&target.marlin_fin_r, dt); + } +} diff --git a/voxygen/src/anim/fishmedium/run.rs b/voxygen/src/anim/fishmedium/run.rs new file mode 100644 index 0000000000..0f1e92743f --- /dev/null +++ b/voxygen/src/anim/fishmedium/run.rs @@ -0,0 +1,58 @@ +use super::{ + super::{Animation, SkeletonAttr}, + FishMediumSkeleton, +}; +use std::{f32::consts::PI}; +use vek::*; + +pub struct RunAnimation; + +impl Animation for RunAnimation { + type Skeleton = FishMediumSkeleton; + type Dependency = (f32, f64); + + fn update_skeleton( + skeleton: &Self::Skeleton, + global_time: Self::Dependency, + anim_time: f64, + _rate: &mut f32, + _skeleton_attr: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let wave_ultra_slow = (anim_time as f32 * 1.0 + PI).sin(); + let wave_ultra_slow_cos = (anim_time as f32 * 1.0 + PI).cos(); + let wave_slow = (anim_time as f32 * 3.5 + PI).sin(); + let wave_slow_cos = (anim_time as f32 * 3.5 + PI).cos(); + + + + next.marlin_head.offset = Vec3::new(0.0, 7.5, 15.0) / 11.0; + next.marlin_head.ori = + Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); + next.marlin_head.scale = Vec3::one() / 10.88; + + next.marlin_torso.offset = + Vec3::new(0.0, 4.5 - wave_ultra_slow_cos * 0.12, 2.0); + next.marlin_torso.ori = Quaternion::rotation_x(0.0); + next.marlin_torso.scale = Vec3::one() * 1.01; + + next.marlin_rear.offset = Vec3::new(0.0, 3.1, -4.5); + next.marlin_rear.ori = Quaternion::rotation_z(0.0); + next.marlin_rear.scale = Vec3::one() * 0.98; + + next.marlin_tail.offset = Vec3::new(0.0, -13.0, 8.0) / 11.0; + next.marlin_tail.ori = Quaternion::rotation_z(0.0) + * Quaternion::rotation_x(0.0); + next.marlin_tail.scale = Vec3::one() / 11.0; + + next.marlin_fin_l.offset = Vec3::new(0.0, -11.7, 11.0) / 11.0; + next.marlin_fin_l.ori = Quaternion::rotation_y(0.0); + next.marlin_fin_l.scale = Vec3::one() / 11.0; + + next.marlin_fin_r.offset = Vec3::new(0.0, 0.0, 12.0) / 11.0; + next.marlin_fin_r.ori = Quaternion::rotation_y(0.0); + next.marlin_fin_r.scale = Vec3::one() / 10.5; + next + } +} diff --git a/voxygen/src/anim/mod.rs b/voxygen/src/anim/mod.rs index a527d759ce..b9600a531a 100644 --- a/voxygen/src/anim/mod.rs +++ b/voxygen/src/anim/mod.rs @@ -3,6 +3,8 @@ pub mod fixture; pub mod object; pub mod quadruped; pub mod quadrupedmedium; +pub mod birdmedium; +pub mod fishmedium; use crate::render::FigureBoneData; use common::comp::{self, item::Tool}; diff --git a/voxygen/src/scene/figure/cache.rs b/voxygen/src/scene/figure/cache.rs index 5063b50a14..9c89226f2a 100644 --- a/voxygen/src/scene/figure/cache.rs +++ b/voxygen/src/scene/figure/cache.rs @@ -226,6 +226,42 @@ impl FigureModelCache { None, None, ], + Body::BirdMedium(body) => [ + Some(mesh_duck_m_head(body.head)), + Some(mesh_duck_m_torso(body.torso)), + Some(mesh_duck_m_tail(body.tail)), + Some(mesh_duck_m_wing_l(body.wing_l)), + Some(mesh_duck_m_wing_r(body.wing_r)), + Some(mesh_duck_m_leg_l(body.leg_l)), + Some(mesh_duck_m_leg_r(body.leg_r)), + None, + None, + None, + None, + None, + None, + None, + None, + None, + ], + Body::FishMedium(body) => [ + Some(mesh_marlin_head(body.head)), + Some(mesh_marlin_torso(body.torso)), + Some(mesh_marlin_rear(body.rear)), + Some(mesh_marlin_tail(body.tail)), + Some(mesh_marlin_fin_l(body.fin_l)), + Some(mesh_marlin_fin_r(body.fin_r)), + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + ], Body::Object(object) => [ Some(mesh_object(object)), None, diff --git a/voxygen/src/scene/figure/load.rs b/voxygen/src/scene/figure/load.rs index b5783de10c..20ed90280c 100644 --- a/voxygen/src/scene/figure/load.rs +++ b/voxygen/src/scene/figure/load.rs @@ -10,7 +10,7 @@ use common::{ Belt, BodyType, Chest, EyeColor, Eyebrows, Foot, Hand, Pants, Race, Shoulder, Skin, }, item::Tool, - object, quadruped, quadruped_medium, Item, ItemKind, + object, quadruped, quadruped_medium, bird_medium, fish_medium, Item, Itemkind }, figure::{DynaUnionizer, MatSegment, Material, Segment}, }; @@ -697,7 +697,124 @@ pub fn mesh_wolf_foot_rb(foot_rb: quadruped_medium::FootRB) -> Mesh Mesh { + load_mesh( + match head { + bird_medium::Head::Default => "npc.duck_m.duck_m_head", + }, + Vec3::new(-7.0, -6.0, -6.0), + ) +} +pub fn mesh_duck_m_torso(torso: bird_medium::Torso) -> Mesh { + load_mesh( + match torso { + bird_medium::Torso::Default => "npc.duck_m.duck_m_body", + }, + Vec3::new(-8.0, -5.5, -6.0), + ) +} + +pub fn mesh_duck_m_tail(tail: bird_medium::Tail) -> Mesh { + load_mesh( + match tail { + bird_medium::Tail::Default => "npc.duck_m.duck_m_tail", + }, + Vec3::new(-4.0, -1.0, -1.0), + ) +} + +pub fn mesh_duck_m_wing_l(wing_l: bird_medium::WingL) -> Mesh { + load_mesh( + match wing_l { + bird_medium::WingL::Default => "npc.duck_m.duck_m_wing", + }, + Vec3::new(-2.5, -4.0, -2.5), + ) +} + +pub fn mesh_duck_m_wing_r(wing_r: bird_medium::WingR) -> Mesh { + load_mesh( + match wing_r { + bird_medium::WingR::Default => "npc.duck_m.duck_m_wing", + }, + Vec3::new(-2.5, -4.0, -2.5), + ) +} + +pub fn mesh_duck_m_leg_l(leg_l: bird_medium::LegL) -> Mesh { + load_mesh( + match leg_l { + bird_medium::LegL::Default => "npc.duck_m.duck_m_leg_l", + }, + Vec3::new(-2.5, -4.0, -2.5), + ) +} + +pub fn mesh_duck_m_leg_r(leg_r: bird_medium::LegR) -> Mesh { + load_mesh( + match leg_r { + bird_medium::LegR::Default => "npc.duck_m.duck_m_leg_r", + }, + Vec3::new(-2.5, -4.0, -2.5), + ) +} +//// +pub fn mesh_marlin_head(head: fish_medium::Head) -> Mesh { + load_mesh( + match head { + fish_medium::Head::Default => "npc.marlin.marlin_head", + }, + Vec3::new(-7.0, -6.0, -6.0), + ) +} + +pub fn mesh_marlin_torso(torso: fish_medium::Torso) -> Mesh { + load_mesh( + match torso { + fish_medium::Torso::Default => "npc.marlin.marlin_torso", + }, + Vec3::new(-7.0, -6.0, -6.0), + ) +} + +pub fn mesh_marlin_rear(rear: fish_medium::Rear) -> Mesh { + load_mesh( + match rear { + fish_medium::Rear::Default => "npc.marlin.marlin_rear", + }, + Vec3::new(-7.0, -6.0, -6.0), + ) +} + +pub fn mesh_marlin_tail(tail: fish_medium::Tail) -> Mesh { + load_mesh( + match tail { + fish_medium::Tail::Default => "npc.marlin.marlin_tail", + }, + Vec3::new(-7.0, -6.0, -6.0), + ) +} + +pub fn mesh_marlin_fin_l(fin_l: fish_medium::FinL) -> Mesh { + load_mesh( + match fin_l { + fish_medium::FinL::Default => "npc.marlin.marlin_fin_l", + }, + Vec3::new(-7.0, -6.0, -6.0), + ) +} + +pub fn mesh_marlin_fin_r(fin_r: fish_medium::FinR) -> Mesh { + load_mesh( + match fin_r { + fish_medium::FinR::Default => "npc.marlin.marlin_fin_r", + }, + Vec3::new(-7.0, -6.0, -6.0), + ) +} +//// pub fn mesh_object(obj: object::Body) -> Mesh { use object::Body; diff --git a/voxygen/src/scene/figure/mod.rs b/voxygen/src/scene/figure/mod.rs index ef0b35f046..3c87e24ffd 100644 --- a/voxygen/src/scene/figure/mod.rs +++ b/voxygen/src/scene/figure/mod.rs @@ -7,7 +7,7 @@ pub use load::load_mesh; // TODO: Don't make this public. use crate::{ anim::{ self, character::CharacterSkeleton, object::ObjectSkeleton, quadruped::QuadrupedSkeleton, - quadrupedmedium::QuadrupedMediumSkeleton, Animation, Skeleton, + quadrupedmedium::QuadrupedMediumSkeleton, birdmedium::BirdMediumSkeleton, fishmedium::FishMediumSkeleton,Animation, Skeleton, }, render::{Consts, FigureBoneData, FigureLocals, Globals, Light, Renderer, Shadow}, scene::camera::{Camera, CameraMode}, @@ -32,6 +32,8 @@ pub struct FigureMgr { character_states: HashMap>, quadruped_states: HashMap>, quadruped_medium_states: HashMap>, + bird_medium_states: HashMap>, + fish_medium_states: HashMap>, object_states: HashMap>, } @@ -42,6 +44,8 @@ impl FigureMgr { character_states: HashMap::new(), quadruped_states: HashMap::new(), quadruped_medium_states: HashMap::new(), + bird_medium_states: HashMap::new(), + fish_medium_states: HashMap::new(), object_states: HashMap::new(), } } @@ -93,6 +97,12 @@ impl FigureMgr { Body::QuadrupedMedium(_) => { self.quadruped_medium_states.remove(&entity); } + Body::BirdMedium(_) => { + self.bird_medium_states.remove(&entity); + } + Body::FishMedium(_) => { + self.fish_medium_states.remove(&entity); + } Body::Object(_) => { self.object_states.remove(&entity); } @@ -378,6 +388,120 @@ impl FigureMgr { action_animation_rate, ); } + Body::BirdMedium(_) => { + let state = self + .bird_medium_states + .entry(entity) + .or_insert_with(|| { + FigureState::new(renderer, BirdMediumSkeleton::new()) + }); + + let (character, last_character) = match (character, last_character) { + (Some(c), Some(l)) => (c, l), + _ => continue, + }; + + if !character.is_same_movement(&last_character.0) { + state.movement_time = 0.0; + } + + let target_base = match character.movement { + Stand => anim::birdmedium::IdleAnimation::update_skeleton( + &BirdMediumSkeleton::new(), + time, + state.movement_time, + &mut movement_animation_rate, + skeleton_attr, + ), + Run => anim::birdmedium::RunAnimation::update_skeleton( + &BirdMediumSkeleton::new(), + (vel.0.magnitude(), time), + state.movement_time, + &mut movement_animation_rate, + skeleton_attr, + ), + Jump => anim::birdmedium::JumpAnimation::update_skeleton( + &BirdMediumSkeleton::new(), + (vel.0.magnitude(), time), + state.movement_time, + &mut movement_animation_rate, + skeleton_attr, + ), + + // TODO! + _ => state.skeleton_mut().clone(), + }; + + state.skeleton.interpolate(&target_base, dt); + state.update( + renderer, + pos.0, + vel.0, + ori.0, + scale, + col, + dt, + movement_animation_rate, + action_animation_rate, + ); + } + Body::FishMedium(_) => { + let state = self + .fish_medium_states + .entry(entity) + .or_insert_with(|| { + FigureState::new(renderer, FishMediumSkeleton::new()) + }); + + let (character, last_character) = match (character, last_character) { + (Some(c), Some(l)) => (c, l), + _ => continue, + }; + + if !character.is_same_movement(&last_character.0) { + state.movement_time = 0.0; + } + + let target_base = match character.movement { + Stand => anim::fishmedium::IdleAnimation::update_skeleton( + &FishMediumSkeleton::new(), + time, + state.movement_time, + &mut movement_animation_rate, + skeleton_attr, + ), + Run => anim::fishmedium::RunAnimation::update_skeleton( + &FishMediumSkeleton::new(), + (vel.0.magnitude(), time), + state.movement_time, + &mut movement_animation_rate, + skeleton_attr, + ), + Jump => anim::fishmedium::JumpAnimation::update_skeleton( + &FishMediumSkeleton::new(), + (vel.0.magnitude(), time), + state.movement_time, + &mut movement_animation_rate, + skeleton_attr, + ), + + // TODO! + _ => state.skeleton_mut().clone(), + }; + + state.skeleton.interpolate(&target_base, dt); + state.update( + renderer, + pos.0, + vel.0, + ori.0, + scale, + col, + dt, + movement_animation_rate, + action_animation_rate, + ); + } Body::Object(_) => { let state = self .object_states @@ -407,6 +531,10 @@ impl FigureMgr { .retain(|entity, _| ecs.entities().is_alive(*entity)); self.quadruped_medium_states .retain(|entity, _| ecs.entities().is_alive(*entity)); + self.bird_medium_states + .retain(|entity, _| ecs.entities().is_alive(*entity)); + self.fish_medium_states + .retain(|entity, _| ecs.entities().is_alive(*entity)); self.object_states .retain(|entity, _| ecs.entities().is_alive(*entity)); } @@ -464,6 +592,14 @@ impl FigureMgr { .quadruped_medium_states .get(&entity) .map(|state| (state.locals(), state.bone_consts())), + Body::BirdMedium(_) => self + .bird_medium_states + .get(&entity) + .map(|state| (state.locals(), state.bone_consts())), + Body::FishMedium(_) => self + .fish_medium_states + .get(&entity) + .map(|state| (state.locals(), state.bone_consts())), Body::Object(_) => self .object_states .get(&entity)