From 5b8fbe9a23c5942da0c1422f4aa6a5777806ddb6 Mon Sep 17 00:00:00 2001 From: Snowram Date: Fri, 2 Apr 2021 02:04:32 +0200 Subject: [PATCH 01/10] Swaps bird_small skeleton to a new bird_large --- assets/common/npc_names.ron | 15 +- .../voxel/bird_large_central_manifest.ron | 54 +++ .../voxel/bird_large_lateral_manifest.ron | 86 ++++ .../voxygen/voxel/npc/phoenix/male/beak.vox | Bin 0 -> 26739 bytes .../voxygen/voxel/npc/phoenix/male/chest.vox | Bin 0 -> 2120 bytes .../voxygen/voxel/npc/phoenix/male/foot_l.vox | Bin 0 -> 1156 bytes .../voxygen/voxel/npc/phoenix/male/foot_r.vox | Bin 0 -> 1156 bytes .../voxygen/voxel/npc/phoenix/male/head.vox | Bin 0 -> 1408 bytes .../voxygen/voxel/npc/phoenix/male/leg_l.vox | Bin 0 -> 1172 bytes .../voxygen/voxel/npc/phoenix/male/leg_r.vox | Bin 0 -> 1172 bytes .../voxygen/voxel/npc/phoenix/male/neck.vox | Bin 0 -> 1240 bytes .../voxel/npc/phoenix/male/tail_front.vox | Bin 0 -> 1136 bytes .../voxel/npc/phoenix/male/tail_rear.vox | Bin 0 -> 1120 bytes .../voxel/npc/phoenix/male/wing_in_l.vox | Bin 0 -> 1248 bytes .../voxel/npc/phoenix/male/wing_in_r.vox | Bin 0 -> 1248 bytes .../voxel/npc/phoenix/male/wing_mid_l.vox | Bin 0 -> 1204 bytes .../voxel/npc/phoenix/male/wing_mid_r.vox | Bin 0 -> 1204 bytes .../voxel/npc/phoenix/male/wing_out_l.vox | Bin 0 -> 1272 bytes .../voxel/npc/phoenix/male/wing_out_r.vox | Bin 0 -> 1272 bytes common/src/comp/agent.rs | 2 +- common/src/comp/body.rs | 22 +- common/src/comp/body/bird_large.rs | 79 ++++ common/src/comp/body/bird_small.rs | 66 --- common/src/comp/fluid_dynamics.rs | 4 +- common/src/comp/mod.rs | 2 +- common/src/generation.rs | 1 + common/src/npc.rs | 13 +- common/src/states/utils.rs | 10 +- server/src/rtsim/entity.rs | 2 +- voxygen/anim/src/bird_large/fly.rs | 146 ++++++ voxygen/anim/src/bird_large/idle.rs | 89 ++++ voxygen/anim/src/bird_large/mod.rs | 174 ++++++++ voxygen/anim/src/bird_large/run.rs | 140 ++++++ voxygen/anim/src/bird_small/idle.rs | 42 -- voxygen/anim/src/bird_small/jump.rs | 42 -- voxygen/anim/src/bird_small/mod.rs | 67 --- voxygen/anim/src/bird_small/run.rs | 42 -- voxygen/anim/src/lib.rs | 2 +- .../audio/sfx/event_mapper/movement/mod.rs | 4 +- .../audio/sfx/event_mapper/movement/tests.rs | 8 +- voxygen/src/scene/figure/load.rs | 416 +++++++++++++++--- voxygen/src/scene/figure/mod.rs | 78 ++-- 42 files changed, 1233 insertions(+), 373 deletions(-) create mode 100644 assets/voxygen/voxel/bird_large_central_manifest.ron create mode 100644 assets/voxygen/voxel/bird_large_lateral_manifest.ron create mode 100644 assets/voxygen/voxel/npc/phoenix/male/beak.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/chest.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/foot_l.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/foot_r.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/head.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/leg_l.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/leg_r.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/neck.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/tail_front.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/tail_rear.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/wing_in_l.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/wing_in_r.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/wing_mid_l.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/wing_mid_r.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/wing_out_l.vox create mode 100644 assets/voxygen/voxel/npc/phoenix/male/wing_out_r.vox create mode 100644 common/src/comp/body/bird_large.rs delete mode 100644 common/src/comp/body/bird_small.rs create mode 100644 voxygen/anim/src/bird_large/fly.rs create mode 100644 voxygen/anim/src/bird_large/idle.rs create mode 100644 voxygen/anim/src/bird_large/mod.rs create mode 100644 voxygen/anim/src/bird_large/run.rs delete mode 100644 voxygen/anim/src/bird_small/idle.rs delete mode 100644 voxygen/anim/src/bird_small/jump.rs delete mode 100644 voxygen/anim/src/bird_small/mod.rs delete mode 100644 voxygen/anim/src/bird_small/run.rs diff --git a/assets/common/npc_names.ron b/assets/common/npc_names.ron index 5f273b5593..71d0a83499 100644 --- a/assets/common/npc_names.ron +++ b/assets/common/npc_names.ron @@ -1068,12 +1068,19 @@ ) ) ), - bird_small: ( + bird_large: ( body: ( - keyword: "bird_small", - names_0: [] + keyword: "bird_large", + names_0: [ + "Aitvaras" + ] ), - species: () + species: ( + phoenix: ( + keyword: "phoenix", + generic: "Phoenix" + ), + ) ), quadruped_low: ( body: ( diff --git a/assets/voxygen/voxel/bird_large_central_manifest.ron b/assets/voxygen/voxel/bird_large_central_manifest.ron new file mode 100644 index 0000000000..2671ef6b84 --- /dev/null +++ b/assets/voxygen/voxel/bird_large_central_manifest.ron @@ -0,0 +1,54 @@ +({ + (Phoenix, Male): ( + head: ( + offset: (-2.0, -1.0, -0.0), + central: ("npc.phoenix.male.head"), + ), + beak: ( + offset: (-2.0, 0.0, -1.0), + central: ("npc.phoenix.male.beak"), + ), + neck: ( + offset: (-2.0, -0.0, -0.0), + central: ("npc.phoenix.male.neck"), + ), + chest: ( + offset: (-3.0, -5.5, -4.0), + central: ("npc.phoenix.male.chest"), + ), + tail_front: ( + offset: (-2.0, -3.0, -3.0), + central: ("npc.phoenix.male.tail_front"), + ), + tail_rear: ( + offset: (-1.0, -3.0, -3.0), + central: ("npc.phoenix.male.tail_rear"), + ) + ), + (Phoenix, Female): ( + head: ( + offset: (-2.0, -1.0, -0.0), + central: ("npc.phoenix.male.head"), + ), + beak: ( + offset: (-2.0, 0.0, -1.0), + central: ("npc.phoenix.male.beak"), + ), + neck: ( + offset: (-2.0, -0.0, -0.0), + central: ("npc.phoenix.male.neck"), + ), + chest: ( + offset: (-3.0, -5.5, -4.0), + central: ("npc.phoenix.male.chest"), + ), + tail_front: ( + offset: (-2.0, -3.0, -3.0), + central: ("npc.phoenix.male.tail_front"), + ), + tail_rear: ( + offset: (-1.0, -3.0, -3.0), + central: ("npc.phoenix.male.tail_rear"), + ) + ), +}) diff --git a/assets/voxygen/voxel/bird_large_lateral_manifest.ron b/assets/voxygen/voxel/bird_large_lateral_manifest.ron new file mode 100644 index 0000000000..1f8e4ab12d --- /dev/null +++ b/assets/voxygen/voxel/bird_large_lateral_manifest.ron @@ -0,0 +1,86 @@ +({ + (Phoenix, Male): ( + wing_in_l: ( + offset: (-6.0, -5.0, -2.0), + lateral: ("npc.phoenix.male.wing_in_l"), + ), + wing_in_r: ( + offset: (0.0, -5.0, -2.0), + lateral: ("npc.phoenix.male.wing_in_r"), + ), + wing_mid_l: ( + offset: (-2.5, -7.0, -2.0), + lateral: ("npc.phoenix.male.wing_mid_l"), + ), + wing_mid_r: ( + offset: (-2.5, -7.0, -2.0), + lateral: ("npc.phoenix.male.wing_mid_r"), + ), + wing_out_l: ( + offset: (-9.0, -8.0, -2.0), + lateral: ("npc.phoenix.male.wing_out_l"), + ), + wing_out_r: ( + offset: (0.0, -8.0, -2.0), + lateral: ("npc.phoenix.male.wing_out_r"), + ), + leg_l: ( + offset: (-1.5, -1.5, -1.5), + lateral: ("npc.phoenix.male.leg_l"), + ), + leg_r: ( + offset: (-1.5, -1.5, -1.5), + lateral: ("npc.phoenix.male.leg_r"), + ), + foot_l: ( + offset: (-1.5, -2.0, -4.0), + lateral: ("npc.phoenix.male.foot_l"), + ), + foot_r: ( + offset: (-1.5, -2.0, -4.0), + lateral: ("npc.phoenix.male.foot_r"), + ) + ), + (Phoenix, Female): ( + wing_in_l: ( + offset: (-6.0, -5.0, -2.0), + lateral: ("npc.phoenix.male.wing_in_l"), + ), + wing_in_r: ( + offset: (0.0, -5.0, -2.0), + lateral: ("npc.phoenix.male.wing_in_r"), + ), + wing_mid_l: ( + offset: (-2.5, -7.0, -2.0), + lateral: ("npc.phoenix.male.wing_mid_l"), + ), + wing_mid_r: ( + offset: (-2.5, -7.0, -2.0), + lateral: ("npc.phoenix.male.wing_mid_r"), + ), + wing_out_l: ( + offset: (-9.0, -8.0, -2.0), + lateral: ("npc.phoenix.male.wing_out_l"), + ), + wing_out_r: ( + offset: (0.0, -8.0, -2.0), + lateral: ("npc.phoenix.male.wing_out_r"), + ), + leg_l: ( + offset: (-1.5, -1.5, -1.5), + lateral: ("npc.phoenix.male.leg_l"), + ), + leg_r: ( + offset: (-1.5, -1.5, -1.5), + lateral: ("npc.phoenix.male.leg_r"), + ), + foot_l: ( + offset: (-1.5, -2.0, -4.0), + lateral: ("npc.phoenix.male.foot_l"), + ), + foot_r: ( + offset: (-1.5, -2.0, -4.0), + lateral: ("npc.phoenix.male.foot_r"), + ) + ), +}) \ No newline at end of file diff --git a/assets/voxygen/voxel/npc/phoenix/male/beak.vox b/assets/voxygen/voxel/npc/phoenix/male/beak.vox new file mode 100644 index 0000000000000000000000000000000000000000..d3fc53871879bb0d9bbc1ac27f062039eb6aca5d GIT binary patch literal 26739 zcmc(ncYGUX8OJYi>MoXP*-N;EGR;XR)ge%*Q-vgLRvo(}q2)MBck(roPCZF>!lo=k z2eiD& z_hqg%3}ffm_Fece>}kW;yZykHTS)=i2E=iQb%=fY4{R^uQ?}csAA`u>vK}#D7=x%Z zxNaTFhym;y>rqCmM-0gR0QM2~|S^x9vGq64>OpY|F=Ff5+JV@w>`SXOm;nZ@cc)Tq&3$wX&8ZkExYi zT}zyfsg+w@ON@@GWv#9yb8$?q{OVfEv(;l?SY2yyE!LT5pzIF4ME)LX? za+1W!k(+PAr-!hTBk?7YLf|cq5Xi^qQjF>-CK2)xb|T~>(PE8`bNAQW@6TwG^ z06q}}A~Y5O908)kf`buuBKSn`iQuC{1)m5$5qu)}Y&#u!bUT7i1fK{#5qu)}$dE?x ziQp5#CxTA|A01{L8L^RXq9}7Mfd@T4_ z@X=+ZD@JBXMn-J#f{y_o13m_P4EPxE!RZL(;bXwZfR6zmAQ8yJ$AFIk9|JyU27x?$ z4EPxEF(8Ac5y%5*;A8_8myldUEFk6)bBI~QBm(#1WEe4oFqRODhy}zvVh%Bjm_!s2 z!-ye-v4mJeEFk6)bBI~QB%+8IMhqoOOH0Y(;$pI}u#n8p&nI(pbII)NY%)1HnG}n~ zWO#Tu85$ZwUHREDvEw}1!6x^o<)0aL>u|$80SUY8dtSTg%1y@(gLY+FZg}QMR%+UQ z5T->-ZkBdglIDn6Sc#P*1gx}6zzULpl{-2CE3-nt%Fhl0R@M^%dm;yW5(j%S2P}@$%`2tHATip8@*xPfkr*N?H#hx&>ci>>}$id!;gS|5cdlwE?euX2f zgu8LD^2;y5_8uJUJvrEWaj+Q<_TC&UzQ!|F!f71r=^X5RIN1AguxD_v_v2vi&%r)` zgMA!B#lfX%5!sU@zui4|1?o4z|X@);ZXK zgKcoImvFEl2ixReTO4eggPq}E@xK-s-^LGdu!lL=$8)ex;9#G~!9IzDeKH686b|;O z9PHCL*h@Lsr*p8+;9#H0!9I(FeKrR>$H6{_gMBUsdl?6NIR|?M2YV$4`#cWz`5f$3 z9PHH`>`OS}xpK8#vh4aHwXJ34)(nq?E5&_1rGN89P9@;*bj2BAL3v?%)x$ygZ(H6`!Npo z;~eZKIM`2eu%F^!Kh42@hJ*bq2m3h=Hs)YI&%u6ygZ&~0`y~$c%N*=iIM}apuwUa~ zzs|vagM7;q?{ly};9zg&V1LNL{)mJ9F$en-4)&)U z?9VvZpL4Lk;9!5r!TySa{WS;s8xHoj9PAPY`#TQy_Z;jWIM_dOuz%uUk8rSm=3xKA z!QR5b{*{CM8wdM$4)z}$>_0i!e{rz?=3o;JR-RP50qElp|Ko#d!`++Dy$~4L1lL6# zkAgiFwCkmw$Kax6BX^=4Yc||cyXu6JDV(60lGAFrX+lqT>uy*%EEOzi=QQldwjR-K zNBUvA^%+gnf^z!6a;ff1U2KDh-s*nc6Nc>%;N2V<)xyD(QOFevnNG-Qwo|WET{*d6 zW(uZKoX~ZoQ9P!1R1;;tc~BfpX>=4jvI)7j(RSVc?~Mv#mjYJJkap9bsw%VLw5Cr* zt!>3|+stR}Jt*5->N6wLk8Z$(M0IfQbck`CC$g0>o1c3GX4OL|_`Oou+3)qyN_T;gKTTG;eP44mD7fm@o4V5ogNwDk4z?G}2qIt5)#^0du+pBtH zqu6xj&74-jHQ%T@dTUbMOFIq8!EQq-%x$#*TQO`prCkYKJX(^D9W6~&>{3t3(USV; zOv*Xg3|@M2$*oqULN+{SY-hV;BOh{t){uLn^Z^UGsT40LwRPd^plhmoyVI09hfdjV z>oPR8KVH~ww%Rq>H%Ic>d@h^U8Sh>$D+R=;uL5!f_f)ikV literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/phoenix/male/chest.vox b/assets/voxygen/voxel/npc/phoenix/male/chest.vox new file mode 100644 index 0000000000000000000000000000000000000000..86e1d808ba6ae9883cc5eb1954cd2ca999d6f156 GIT binary patch literal 2120 zcmXZcL2l$k7(n6QRe!lVf=zCKB5q(7h!ul^h7kfJAP8o|&U;q5LX@60QXPR@Ay=p? zMB)Tlu{WO$-SfI!Z>sIrWh=k@@ardkM#Oh--hFR~&(0A)y?g)l$NR)<#z&yz*Wcd1 z%XF~6Ce(Orh-rFkh%!A+NHj#9E>oQz3n~(gi{wzz5N*1YW{&1~igGDY9xKvuOyzMx zK}Dh=3Xg({L_<^_1r>>g$a0yIM@6C`n)N8CNVI)Eo1>hJh^byCl&R{dmozqG6uwj) zJQ9tI%2ys0iN;0N#|Z@$iH2y_Hy#yrqgVXlHA$ns&~Z#!_az^QBZHBIgfNnx{F^apR)R?)R*VN1|~t^D~c%L_?fr`^lps z(GX|r&pav;4bv5OHQQ6pSn_nKd9oKL*F3rA>5|5nITTbR8m80fQcfJw@pQHZ6^S3; ze)T5)|9h|rULA*;#Bo@1Mjo!M*l_X2+3 zdOfS2RnMwt)pOhqJ*%Eo#Oi{plOMOpmsSf_lU2`Ze~#OxXVtUnS@rz<_KWS?FF)UY z|LN8(<@Kz3Rz1hkaHqG!>w=vnj}kA|K_&!T4$ zvgkP;7bjbMaj|G27W=b5LW})b^elQ7J&T^>u{xfZ{Vwf0v)6dTp74Ns+~F2CxZYl0U$>W+m+krad3$<#+8!Prw)^}0?e6YwyS=^L OZf+5T4BmM;4x2`VU;UHbdU6oK(3H0%oQTh z3uM*(@Ok3=vHiy$e;)hW=ew7GBI4uK%_pn)c^+|lbN}x7yyJ5w7ov{4@Ao&C*7N6u zOc9UM6c^LNA!pXI5C=a_=eFndv*^=e@}Ddg57Tt^FfGo0f4P2pH3l_KAD`Br{{6L8 zrSzxvtM_KJ`DrQoR>a|Ud+O6h>QkTk`P8bf`f8`@tKO;?ebE^%sXUdZ@>HJjICv^gg{UrA4L%-`ld1!%oyt?q&vj3(eo6Z!c%w(PvIG_0Z-v6JcXz5 zj8}uF@D!c`Qh3JeVzA=G;-I1^=4XC|iuox#g{SZop7C0ZH)eiH^UTZ|3s3L_Pw)g! z@Hk!cJi!w@!DC3#^8`=u1dq#zo+o&MCtzG!^aczkhXYT_qa0)}JK4%c*0PeN#7T~F zkiG0=D;rtMN|q8QIm$uyvXiZBWGyRM_S5OqkH=#_91eZI-}l{a*SFhk-)uI0y4x2`VU;UHbdU6oK(3H0%oQTh z3uM*(@Ok3=vHiy$e;)hW=ew7GBI4uK%_pn)c^+|lbN}x7yyJ5w7ov{4@Ao&C*7N6u zOc9UM6c^LNA!pXI5C=a_=eFndv*^=e@}Ddg57Tt^FfGo0f4P2pH3l_KAD`Br{{6L8 zrSzxvtM_KJ`DrQoR>a|Ud+O6h>QkTk`P8bf`f8`@tKO;?ebE^%sXUdZ@>HJjICv^gg{UrA4L%-`ld1!%oyt?q&vj3(eo6Z!c%w(PvIG_0Z-v6JcXz5 zj8}uF@D!c`Qh3JeVzA=G;-I1^=4XC|iuox#g{SZop7C0ZH)eiH^UTZ|3s3L_Pw)g! z@Hk!cJi!w@!DC3#^8`=u1dq#zo+o&MCtzG!^aczkhXYT_qa0)}JK4%c*0PeN#7T~F zkiG0=D;rtMN|q8QIm$uyvXiZBWGyRM_S5OqkH=#_91eZI-}l{a*SFhk-)uI0yTf9Zr}~7SWpyHULioDq$=I8^NJ0tTp=ew@)5`tas^)@ z5+}&2;?2%{#%d-{rNEH6C~OR`R~PI_d|Nb00U){bn2tVY(3O&ZHajjSEp z^t^17b&_?ot+HN`GT;12FpO`0EIFGenV-$Ij7kLpBx$g*h$U`1- zIufi`Lq&ny58bWv>3lk$&S&cJ>3lk$&ZqO4%6vMX&ZqO~eCFTa)A@8holobp{XQPvg`0%q`&42sDc|6E~C0ZSiB%XnY!<#%FF7pT?*0X?%Y8O73qr zK8;V~)A%$#>y|b?jZfp#_%uFqo9E7~HtSU&w?CCnC!=4-&G@F{!>pTcLp8hi?$!lw`^eCF$7vSMJ-sR)YwSzn=Ie+r+%r|>C!=4&^g#RX@aaKr(7?8g0mAGh0WT(8%0xm?Ei Zd>*IMX&jHoaX1{te!m~P-A><#{{bIzp^E?j literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/phoenix/male/leg_l.vox b/assets/voxygen/voxel/npc/phoenix/male/leg_l.vox new file mode 100644 index 0000000000000000000000000000000000000000..2390b7c75752cd1c539c2b0407e09b95bc739cf7 GIT binary patch literal 1172 zcmYk)J+2cm7zNNT10_ww22jKf+#o6*3W6uZkANVcp>t1_mNE)G(zgV%g>1oFh{Os~ z(VJt_kjc%=-1$4P=i{4C4}M0(tDD=`R`LCA#QWPX&+d&KpF98M)8{X@Z)Y2E|KIw! zia5q9jw@NldWj#aSREIM)k<$$VzDjDSjXbn;@IMt^|Bn_J%4%=o;^N&ep>(Y`|msWk%S36Z-^;W&;i@xZKqUe2l=tCd+(Bo)i zEGbor=pAT&8c*YCJdJ1acp6XRX*`W*l6e|W<7qsNXZ{VI#?yEjPvaSXt?@LT#?yEj z58afc-{eW2Bi|Mb66Zq8_APA$uozFCwY=5d0sr!``gKrJjs(h$uovD zd6Fl2k|%lQFwc=0nz0no8=uNkc`8rksXTK!cq&hYs2*6|d`^*zsuQZ6%2SQcoHm}y zQ+X=Shv$#`%O{We*Y{gfs^_UZm1j=xoTV``Lo*gVPvI#%g{SZop7|Q^6rRFUcnZ&a zHFye7;VB@6XTC0OR$N$|R20Sdj8~`_pTbjk3Qysguho2G#w?AI85#>u@B~lr1W)j| zT=YD_6FkA=mZIkgp5O@{j}bjj@B~l5c(mxt1_mNE)G(zgV%g>1oFh{Os~ z(VJt_kjc%=-1$4P=i{4C4}M0(tDD=`R`LCA#QWPX&+d&KpF98M)8{X@Z)Y2E|KIw! zia5q9jw@NldWj#aSREIM)k<$$VzDjDSjXbn;@IMt^|Bn_J%4%=o;^N&ep>(Y`|msWk%S36Z-^;W&;i@xZKqUe2l=tCd+(Bo)i zEGbor=pAT&8c*YCJdJ1acp6XRX*`W*l6e|W<7qsNXZ{VI#?yEjPvaSXt?@LT#?yEj z58afc-{eW2Bi|Mb66Zq8_APA$uozFCwY=5d0sr!``gKrJjs(h$uovD zd6Fl2k|%lQFwc=0nz0no8=uNkc`8rksXTK!cq&hYs2*6|d`^*zsuQZ6%2SQcoHm}y zQ+X=Shv$#`%O{We*Y{gfs^_UZm1j=xoTV``Lo*gVPvI#%g{SZop7|Q^6rRFUcnZ&a zHFye7;VB@6XTC0OR$N$|R20Sdj8~`_pTbjk3Qysguho2G#w?AI85#>u@B~lr1W)j| zT=YD_6FkA=mZIkgp5O@{j}bjj@B~l5c(mxyzIR@$SX#dqezv6!Gcy`>V%e;$=FMj9c^@JR~MPf;>&8M z@>HJ6Gna~|@>HJ6^X5kHZ&#kmQ+X;+EC}(MV?P`YeZSxL-EP;n+il-$Hja(>4_^U^D*ylh literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/phoenix/male/tail_front.vox b/assets/voxygen/voxel/npc/phoenix/male/tail_front.vox new file mode 100644 index 0000000000000000000000000000000000000000..7247b3cdf671ee6b7b56fdf093b48d503df63372 GIT binary patch literal 1136 zcmXZaF>Vt<6b9gDhYHd72FRkMn+8!4WE3I$`W)#)*&9xcHyj?|0WPN9pra z77_Q$h=*7#9u~4(EbcGtW4U}-ESL97`wROoS8p%pKQg^%t@^63PO84@?RwD{ebE<1(fjM64}IuEkE4;XrBo@R_mAeQ z@id;s(|9J2r|~qN#?yEvnWynIp2pL7=Fi}1JdLOEG@kL(8c*YCJdLOE&`nADPM+jR zo;d|P36LI^22O*_Y4Oi=kUYtgJaejek|%kR=lu)4pPf9(lRU|jJY!0eCwY=5d6H*N z^PHKX8Cwy(`KdgWr}9*u$}^XPr}9*Y>VehZbBWwkT~M7=o@##Pvhh@&%2Rp1UcK%g z-n{BRZ#I@x&r^9S&s^TQN@He*W^8(%!c%w(PvI#%^EKcpJcXz56rTBN@D!fHQ$PyO zd|eDy+*n*x6vh0ESE!hu!c%w(PvM!b)qG>dDvg;L8XHgW1W)h;Pw=>1^gO{6Ji%i~ z(engP@C1*?h@K~Sf+t`+TJ#1CH-`()%99-BAbZ)#RyMMhmBd+2a+HJYWhYzN$XZqs zXF17H4zib>Y-J;BS@rYz+)t-dKOT?$a5(h+e&2VyUEglEeY4s0^?Kb`tCedb{sCA& BYGwcc literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/phoenix/male/tail_rear.vox b/assets/voxygen/voxel/npc/phoenix/male/tail_rear.vox new file mode 100644 index 0000000000000000000000000000000000000000..580e90a07136b50c82c064056835a3eb7f36665b GIT binary patch literal 1120 zcmXZaJ#G_07zW_a22s(tr$81rm$LpME@uV5uT(su-W1z%yVV2KmB zNZ?B$>()Q%v z*WDuGJ{F7n#bR+6%jNxIxxD*w`Q{>wGqygr)*t@-c2uSGhxW7g=CJu_Df(8#*M>)t|cCwX?tYsx}mXjRiAbZ)# sRyMMhRX?B4{d79@410GRe}=0R!n$@iq$ZJBhrrV0V zXcT#o7kPg*%dWpldEtpL{w#ZA_fUJ}GWc3-_#wYX1d@`Sp zFQ2s6Pan6hAGWqs&L{K9e8%>UUFwzTHvK2>Th1r(NqiEY#Ah4>K8a7_llUY)<7n_n zd=j5TB=H%?#bn8g$)F@i>eG)p@kx9VpK+|liRru4E7NWK`cdN(e1cE#2|iwj zoKNrxKEcP7kn;&X!6*1w268^ZC-?*zOG9qL@NyV<6<%=02?y-4!xkH?5m#Jr#t8@P zvBMS{tPxjSaK;G-?6JcZ8?4*)dTp1>rJc{`b~>Hf;c#gC{l4vXySCkK+h((A>-E~$ Fh=1v{dxQW0 literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/phoenix/male/wing_in_r.vox b/assets/voxygen/voxel/npc/phoenix/male/wing_in_r.vox new file mode 100644 index 0000000000000000000000000000000000000000..3273e5237fb520007916c932483d22b9aa6528cd GIT binary patch literal 1248 zcmW;KG0xL47>DsUY5K9S^aiLxT)+!r0#pQ_5Fi0Xz{2L8saL4@n2{`T0eXdA!B?on z2|BTxpPlrH?SB*J$!mXl_xZuEhx=t+;JCp(%=rA}>+NrCKOSMd zio0cryES4RcN1o;u=M_oekZ1%(=`2@u|mw#-Gq5I#)4SMFt22*>BpkK*2W@VrXRC* zrCqSrkF^{#R#ld|o}2`|QG}@F{!>pTeh?wD2i>3ZKHK@EK(umFYJ9 zOpvQj=9BqkKABJEGnRu-=95WeFU%$%OXQ*KhHQ}eWc3-##wYX1d@`SpFQ2s6Pan5$ zA9j{h&L{K9e8%#QRqB=LHvN?INqiEY#3%7de8x85llUY)iBIA)wg#WXC-F%{5}&bM zOqM*D+>``Kefkzk>XY~+K8a7_Gq%;(F@2SKWx9=%Pw)vo!6*0xACE)MC-?-P;A2Y2 z`2?Tf6MVb|az4Q)_yifRhTMeV;c(+sc)=Mb9C5%NJ8ZE*Tyen}CmeCW9y@HYL0oac x87CZZz#cnnv1!-qwOuZkc0Qlm>2zwx7{ literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/phoenix/male/wing_mid_l.vox b/assets/voxygen/voxel/npc/phoenix/male/wing_mid_l.vox new file mode 100644 index 0000000000000000000000000000000000000000..d21b85cdc2ddba8de1ec9eab92b0d5f2f022169d GIT binary patch literal 1204 zcmW;KJ#G_07zW^PZ6`D|u4$0P4W>a<1Q~@00TRI|q@kM{Dqq3Esgb6m#aHkZ<_eZL zfvfbxGqdY=e_rp;v$Om4)7|sm5%J;b=A%{oSw!64+`oO=cT5-ZL}qgL?f&M6b)TN~ z#YODnGWL_qdScqors-iW(`@=Rllg3VSjaT@-`u_|Bo@r)$3ZIA!{+RBpJWBu7$wOK7{5q(YKaJ#MOi;bc$`l9crW_{LYJ6WIg zRxRn1KIxMp>HS*hLm&Fk<7lMMMY1H(`=eS^p2|~sD$n5YRG!LHc`DB!^HiS7Q+X=S zcnqG(Q+X;+<>`-Bc`8rksXUd3Zb*t{;VC?YXAA*P0Vr-(T)4Q%7#4pP8-=Iv6rM3u zJcXz56rOi4^d7tL6rRFUcnVJ+(!x`C3QyrFJY$&0$aGDgljx04=E*#nC-Y>U@j7@i zPlm{Dm|c9lA}3`BWILHB8=vvocrs7s$vj`KU$ytIU$*bJYp+z#lX)`Fc)jB-^^xhC zKI?fBPvS{Di6`-l&wwZKB%Z{Rc*du}lXwzO0!cjMb8)fc#N?o)NXDmsLdp0fp2U-Q z63_Uo#uwAyQXiSFG4ljZ@B~lr1dr22&l5bs6Fe>{dY<43p5SpC(engP@C1xoi{1sp z$>G4W@+3z&$WFGhk+rO3DRGvQ9OWQ8*~&)NvXZ65Sx$14gY0B08(GUrmhF5#x6|p= gj>ls=91d-_+qLa>+cuj`Td&t`wOY01a_QKJ{{R4V!~g&Q literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/phoenix/male/wing_mid_r.vox b/assets/voxygen/voxel/npc/phoenix/male/wing_mid_r.vox new file mode 100644 index 0000000000000000000000000000000000000000..94585e73ca9a30533c6940811aa19f5ba6b67ed8 GIT binary patch literal 1204 zcmW;KJ&w~r7zW_aju#r5)HEpK2Gbxa76mOU1V}7F%QkdVLzOE;o2`+iqa|0!73K<& zI6HZAFsKIxMp>3v%0Lm&Fk<7i}Fi)2Zn_e1qnc`8rksXU#>Q+X;+<*7WK%u{(P zPvxmR{crG8p2|~sD$o4aDo^F9Je8;N&~-`iU3dyl;ps!bQviyG6*n$!(uc*5#ZKWV zJcXwZ6;I(QJcZ}|OTE8ccnVM9DLjQ|9@4^7cnVM9DLj3c`^e0ic}=1>KA9);WS-2E zdHQtlWS$I>Jutg@pCT7!Cu9eiCmWwWZ9JJL^JJc{SFhWLH?P`{o3$y`^JJdP)2Fx3 z(mXP=W?uC?i6`+Sp2U-Q`fI?GcoI+INj&}4;7L4*CxIlM{<^qXa$#~(QY7Ouze36Q zB%Z{RcoI*4t@;}?&(b_Hv&PC3Ji!w@!4o_#7d=n#1W)j|rRaHrCwPL#V?@srJi!w% z9xZw|3>SwJ&&rb=|`q&S<6b65@$KdQ4X?`t!!j1D_OSl m`P@#YQ#&4y?Ql4>-EP;m+ilxyHf_CLx7BLZmdmAcBmM&_P;|}! literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/phoenix/male/wing_out_l.vox b/assets/voxygen/voxel/npc/phoenix/male/wing_out_l.vox new file mode 100644 index 0000000000000000000000000000000000000000..7ff492352a305838a4671679c919fbd5d9947160 GIT binary patch literal 1272 zcmW;Kv92RE7>410lF8((M3Xx}5ha})M8%@4mLmj6y96!U(7E3#w-B9k>Xm#6q^8R) z_!c5jP^a47JoZf9nei9f^UiPo@!N;5KS#tbuOEMPi~r_`_m3aneAOF{S9rh*AO8IK z_>X%(zQw$a=M5IDV|tzu)AT%H#s&+vm{!yC3Tw=(>1BguH9c*$A9}u)ucxOCmi6?q z)qcP<8;A8gz0BBP(cbpvSQ+5l2I0a77v9_;ZyjGQt>H#3ZKH~r*GswyYMM| z3ZKHK@aZKjdC!Mwv%tx=lY5ls=91d;2-?!avr*FjnKpTIC literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/phoenix/male/wing_out_r.vox b/assets/voxygen/voxel/npc/phoenix/male/wing_out_r.vox new file mode 100644 index 0000000000000000000000000000000000000000..e45c808905cbf23f58bb7022711bf9b31ba8d31d GIT binary patch literal 1272 zcmW;Kv98-T7>40rvTUb77T-Z(bnM2Z$kYT5(u1Nckkmzzr5pE7z6Il)GY8-$)N?oA z0&l@Uht8g|vkxixwZu;(_39`8`0eAjUn1g{*AKtC#D7b~hlkH^U-pLM8@#}bkDoq2 z{Nvh>?=f%VX@#W|)6*K$G(F8&VZjzJ5VL2P=jm}ZPtOb1+8b;UE6*{nrsvgadR(yX z*)l!Pi}_p^tg+GFVp>m+GgfQwTI;uQzj2Q>Hkh}@V!``&Z(fJ*>qi?$ZD0TW+g(O# zU+Z73Rkzh&Es)m)-8Ea07dJ&-C!MhW;70>#UUiHk`_Sv)Lm3ZKHK@EN7zQ}`4yR}v zpR7J(+4y8WnNQ~P`@2`|=O2G)e||VvQaPW@C-WK0J65SzrrY#W&L{Cnd=j6;C-E8E zfKTF+_#{4w&)6D#5}(8;5lMW;b}?BpFzJ*8NqzbjO6rsNBtD5x;xo3@*fD*TdS$we zlTYvoKEWsW1Ruj8=M#K_Pw+7%XFgSF)3vamMf-_Dy z;($GNh&yh$;({|yIO2dkc8EJ}xZ;8{PB`L#J$CJWzqi}%)~?rUyId~qd_K3+>C}$L RV>=uUZNJ~Q-EOCE#Q%5wj literal 0 HcmV?d00001 diff --git a/common/src/comp/agent.rs b/common/src/comp/agent.rs index a777055891..68b30b314f 100644 --- a/common/src/comp/agent.rs +++ b/common/src/comp/agent.rs @@ -247,7 +247,7 @@ impl<'a> From<&'a Body> for Psyche { }, Body::BipedSmall(_) => 0.5, Body::BirdMedium(_) => 0.5, - Body::BirdSmall(_) => 0.4, + Body::BirdLarge(_) => 0.4, Body::FishMedium(_) => 0.15, Body::FishSmall(_) => 0.0, Body::BipedLarge(_) => 1.0, diff --git a/common/src/comp/body.rs b/common/src/comp/body.rs index 4da8d22f83..cca0db27de 100644 --- a/common/src/comp/body.rs +++ b/common/src/comp/body.rs @@ -1,7 +1,7 @@ pub mod biped_large; pub mod biped_small; +pub mod bird_large; pub mod bird_medium; -pub mod bird_small; pub mod dragon; pub mod fish_medium; pub mod fish_small; @@ -38,7 +38,7 @@ make_case_elim!( BirdMedium(body: bird_medium::Body) = 3, FishMedium(body: fish_medium::Body) = 4, Dragon(body: dragon::Body) = 5, - BirdSmall(body: bird_small::Body) = 6, + BirdLarge(body: bird_large::Body) = 6, FishSmall(body: fish_small::Body) = 7, BipedLarge(body: biped_large::Body)= 8, BipedSmall(body: biped_small::Body)= 9, @@ -73,7 +73,7 @@ pub struct AllBodies { pub bird_medium: BodyData>, pub fish_medium: BodyData>, pub dragon: BodyData>, - pub bird_small: BodyData, + pub bird_large: BodyData>, pub fish_small: BodyData>, pub biped_large: BodyData>, pub biped_small: BodyData>, @@ -95,6 +95,7 @@ impl core::ops::Index for AllBodies &self.quadruped_small.body, NpcKind::Wolf => &self.quadruped_medium.body, NpcKind::Duck => &self.bird_medium.body, + NpcKind::Phoenix => &self.bird_large.body, NpcKind::Marlin => &self.fish_medium.body, NpcKind::Clownfish => &self.fish_small.body, NpcKind::Ogre => &self.biped_large.body, @@ -118,9 +119,10 @@ impl<'a, BodyMeta, SpeciesMeta> core::ops::Index<&'a Body> for AllBodies &self.quadruped_small.body, Body::QuadrupedMedium(_) => &self.quadruped_medium.body, Body::BirdMedium(_) => &self.bird_medium.body, + Body::BirdLarge(_) => &self.bird_large.body, Body::FishMedium(_) => &self.fish_medium.body, Body::Dragon(_) => &self.dragon.body, - Body::BirdSmall(_) => &self.bird_small.body, + Body::BirdLarge(_) => &self.bird_large.body, Body::FishSmall(_) => &self.fish_small.body, Body::BipedLarge(_) => &self.biped_large.body, Body::BipedSmall(_) => &self.biped_small.body, @@ -152,7 +154,7 @@ impl Body { let d = match self { // based on a house sparrow (Passer domesticus) Body::BirdMedium(_) => 700.0, - Body::BirdSmall(_) => 700.0, + Body::BirdLarge(_) => 700.0, // based on its mass divided by the volume of a bird scaled up to the size of the dragon Body::Dragon(_) => 3_700.0, @@ -184,7 +186,7 @@ impl Body { // ravens are 0.69-2 kg, crows are 0.51 kg on average Body::BirdMedium(_) => 1.0, // australian magpies are around 0.22-0.35 kg - Body::BirdSmall(_) => 0.3, + Body::BirdLarge(_) => 0.3, Body::Dragon(_) => 20_000.0, Body::FishMedium(_) => 2.5, @@ -285,7 +287,7 @@ impl Body { bird_medium::Species::Cockatrice => Vec3::new(2.0, 1.0, 1.8), _ => Vec3::new(2.0, 1.0, 1.1), }, - Body::BirdSmall(_) => Vec3::new(1.2, 0.6, 1.1), + Body::BirdLarge(_) => Vec3::new(1.2, 0.6, 1.1), Body::Dragon(_) => Vec3::new(16.0, 10.0, 16.0), Body::FishMedium(_) => Vec3::new(0.5, 2.0, 0.8), Body::FishSmall(_) => Vec3::new(0.3, 1.2, 0.6), @@ -435,7 +437,7 @@ impl Body { }, Body::FishMedium(_) => 50, Body::Dragon(_) => 5000, - Body::BirdSmall(_) => 50, + Body::BirdLarge(_) => 50, Body::FishSmall(_) => 20, Body::BipedLarge(biped_large) => match biped_large.species { biped_large::Species::Ogre => 2500, @@ -548,7 +550,7 @@ impl Body { }, Body::FishMedium(_) => 10, Body::Dragon(_) => 500, - Body::BirdSmall(_) => 10, + Body::BirdLarge(_) => 10, Body::FishSmall(_) => 10, Body::BipedLarge(biped_large) => match biped_large.species { biped_large::Species::Ogre => 70, @@ -587,7 +589,7 @@ impl Body { pub fn flying_height(&self) -> f32 { match self { - Body::BirdSmall(_) => 30.0, + Body::BirdLarge(_) => 30.0, Body::BirdMedium(_) => 40.0, Body::Dragon(_) => 60.0, Body::Ship(ship::Body::DefaultAirship) => 60.0, diff --git a/common/src/comp/body/bird_large.rs b/common/src/comp/body/bird_large.rs new file mode 100644 index 0000000000..2c12eee089 --- /dev/null +++ b/common/src/comp/body/bird_large.rs @@ -0,0 +1,79 @@ +use crate::{make_case_elim, make_proj_elim}; +use rand::{seq::SliceRandom, thread_rng}; +use serde::{Deserialize, Serialize}; + +make_proj_elim!( + body, + #[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] + pub struct Body { + pub species: Species, + pub body_type: BodyType, + } +); + +impl Body { + pub fn random() -> Self { + let mut rng = thread_rng(); + let species = *(&ALL_SPECIES).choose(&mut rng).unwrap(); + Self::random_with(&mut rng, &species) + } + + #[inline] + pub fn random_with(rng: &mut impl rand::Rng, &species: &Species) -> Self { + let body_type = *(&ALL_BODY_TYPES).choose(rng).unwrap(); + Self { species, body_type } + } +} + +impl From for super::Body { + fn from(body: Body) -> Self { super::Body::BirdLarge(body) } +} + +make_case_elim!( + species, + #[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] + #[repr(u32)] + pub enum Species { + Phoenix = 0, + } +); + +/// Data representing per-species generic data. +/// +/// NOTE: Deliberately don't (yet?) implement serialize. +#[derive(Clone, Debug, Deserialize)] +pub struct AllSpecies { + pub phoenix: SpeciesMeta, +} + +impl<'a, SpeciesMeta> core::ops::Index<&'a Species> for AllSpecies { + type Output = SpeciesMeta; + + #[inline] + fn index(&self, &index: &'a Species) -> &Self::Output { + match index { + Species::Phoenix => &self.phoenix, + } + } +} + +pub const ALL_SPECIES: [Species; 1] = [Species::Phoenix]; + +impl<'a, SpeciesMeta: 'a> IntoIterator for &'a AllSpecies { + type IntoIter = std::iter::Copied>; + type Item = Species; + + fn into_iter(self) -> Self::IntoIter { ALL_SPECIES.iter().copied() } +} + +make_case_elim!( + body_type, + #[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] + #[repr(u32)] + pub enum BodyType { + Female = 0, + Male = 1, + } +); + +pub const ALL_BODY_TYPES: [BodyType; 2] = [BodyType::Female, BodyType::Male]; diff --git a/common/src/comp/body/bird_small.rs b/common/src/comp/body/bird_small.rs deleted file mode 100644 index a3be8c3087..0000000000 --- a/common/src/comp/body/bird_small.rs +++ /dev/null @@ -1,66 +0,0 @@ -use crate::{make_case_elim, make_proj_elim}; -use rand::{seq::SliceRandom, thread_rng}; -use serde::{Deserialize, Serialize}; - -make_proj_elim!( - body, - #[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] - pub struct Body { - pub head: Head, - pub torso: Torso, - pub wing_l: WingL, - pub wing_r: WingR, - } -); - -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(), - wing_l: *(&ALL_WING_LS).choose(&mut rng).unwrap(), - wing_r: *(&ALL_WING_RS).choose(&mut rng).unwrap(), - } - } -} - -make_case_elim!( - head, - #[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] - #[repr(u32)] - pub enum Head { - Default = 0, - } -); -const ALL_HEADS: [Head; 1] = [Head::Default]; - -make_case_elim!( - torso, - #[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] - #[repr(u32)] - pub enum Torso { - Default = 0, - } -); -const ALL_TORSOS: [Torso; 1] = [Torso::Default]; - -make_case_elim!( - wing_l, - #[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] - #[repr(u32)] - pub enum WingL { - Default = 0, - } -); -const ALL_WING_LS: [WingL; 1] = [WingL::Default]; - -make_case_elim!( - wing_r, - #[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] - #[repr(u32)] - pub enum WingR { - Default = 0, - } -); -const ALL_WING_RS: [WingR; 1] = [WingR::Default]; diff --git a/common/src/comp/fluid_dynamics.rs b/common/src/comp/fluid_dynamics.rs index 79b60eaf2e..6508160168 100644 --- a/common/src/comp/fluid_dynamics.rs +++ b/common/src/comp/fluid_dynamics.rs @@ -129,11 +129,11 @@ impl Body { }, // Cross-section, zero-lift angle; exclude the wings (width * 0.2) - Body::BirdMedium(_) | Body::BirdSmall(_) | Body::Dragon(_) => { + Body::BirdMedium(_) | Body::BirdLarge(_) | Body::Dragon(_) => { let dim = self.dimensions().map(|a| a * 0.5); let cd = match self { Body::BirdMedium(_) => 0.2, - Body::BirdSmall(_) => 0.4, + Body::BirdLarge(_) => 0.4, _ => 0.7, }; cd * std::f32::consts::PI * dim.x * 0.2 * dim.z diff --git a/common/src/comp/mod.rs b/common/src/comp/mod.rs index 2c3664a2c7..5f563a3836 100644 --- a/common/src/comp/mod.rs +++ b/common/src/comp/mod.rs @@ -50,7 +50,7 @@ pub use self::{ aura::{Aura, AuraChange, AuraKind, Auras}, beam::{Beam, BeamSegment}, body::{ - biped_large, biped_small, bird_medium, bird_small, dragon, fish_medium, fish_small, golem, + biped_large, biped_small, bird_large, bird_medium, dragon, fish_medium, fish_small, golem, humanoid, object, quadruped_low, quadruped_medium, quadruped_small, ship, theropod, AllBodies, Body, BodyData, }, diff --git a/common/src/generation.rs b/common/src/generation.rs index 9b187ecc05..e922f604c7 100644 --- a/common/src/generation.rs +++ b/common/src/generation.rs @@ -135,6 +135,7 @@ impl EntityInfo { Some(get_npc_name(&npc_names.quadruped_medium, body.species)) }, Body::BirdMedium(body) => Some(get_npc_name(&npc_names.bird_medium, body.species)), + Body::BirdLarge(body) => Some(get_npc_name(&npc_names.bird_large, body.species)), Body::FishSmall(body) => Some(get_npc_name(&npc_names.fish_small, body.species)), Body::FishMedium(body) => Some(get_npc_name(&npc_names.fish_medium, body.species)), Body::Theropod(body) => Some(get_npc_name(&npc_names.theropod, body.species)), diff --git a/common/src/npc.rs b/common/src/npc.rs index d7eebe5f17..5712c33696 100644 --- a/common/src/npc.rs +++ b/common/src/npc.rs @@ -13,6 +13,7 @@ pub enum NpcKind { Wolf, Pig, Duck, + Phoenix, Clownfish, Marlin, Ogre, @@ -23,11 +24,12 @@ pub enum NpcKind { Crocodile, } -pub const ALL_NPCS: [NpcKind; 12] = [ +pub const ALL_NPCS: [NpcKind; 13] = [ NpcKind::Humanoid, NpcKind::Wolf, NpcKind::Pig, NpcKind::Duck, + NpcKind::Phoenix, NpcKind::Clownfish, NpcKind::Marlin, NpcKind::Ogre, @@ -122,6 +124,7 @@ pub fn kind_to_body(kind: NpcKind) -> Body { 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(), @@ -228,6 +231,14 @@ impl NpcBody { 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, diff --git a/common/src/states/utils.rs b/common/src/states/utils.rs index 654fa44ade..d5a5acdf23 100644 --- a/common/src/states/utils.rs +++ b/common/src/states/utils.rs @@ -77,7 +77,7 @@ impl Body { Body::BirdMedium(_) => 80.0, Body::FishMedium(_) => 80.0, Body::Dragon(_) => 250.0, - Body::BirdSmall(_) => 75.0, + Body::BirdLarge(_) => 75.0, Body::FishSmall(_) => 60.0, Body::BipedSmall(biped_small) => match biped_small.species { biped_small::Species::Haniwa => 65.0, @@ -133,7 +133,7 @@ impl Body { Body::BirdMedium(_) => 6.0, Body::FishMedium(_) => 6.0, Body::Dragon(_) => 1.0, - Body::BirdSmall(_) => 7.0, + Body::BirdLarge(_) => 7.0, Body::FishSmall(_) => 7.0, Body::BipedLarge(_) => 1.6, Body::BipedSmall(_) => 2.4, @@ -165,7 +165,7 @@ impl Body { Body::BipedLarge(_) | Body::Golem(_) => Some(200.0 * self.mass().0), Body::BipedSmall(_) => Some(100.0 * self.mass().0), Body::BirdMedium(_) => Some(50.0 * self.mass().0), - Body::BirdSmall(_) => Some(50.0 * self.mass().0), + Body::BirdLarge(_) => Some(50.0 * self.mass().0), Body::FishMedium(_) => Some(50.0 * self.mass().0), Body::FishSmall(_) => Some(50.0 * self.mass().0), Body::Dragon(_) => Some(200.0 * self.mass().0), @@ -188,7 +188,7 @@ impl Body { pub fn fly_thrust(&self) -> Option { match self { Body::BirdMedium(_) => Some(GRAVITY * self.mass().0 * 2.0), - Body::BirdSmall(_) => Some(GRAVITY * self.mass().0 * 2.0), + Body::BirdLarge(_) => Some(GRAVITY * self.mass().0 * 2.0), Body::Dragon(_) => Some(200_000.0), Body::Ship(ship::Body::DefaultAirship) => Some(300_000.0), _ => None, @@ -377,7 +377,7 @@ pub fn fly_move(data: &JoinData, update: &mut StateUpdate, efficiency: f32) -> b // Elevation control match data.body { // flappy flappy - Body::Dragon(_) | Body::BirdMedium(_) | Body::BirdSmall(_) => { + Body::Dragon(_) | Body::BirdMedium(_) | Body::BirdLarge(_) => { let anti_grav = GRAVITY * (1.0 + data.inputs.move_z.min(0.0)); update.vel.0.z += data.dt.0 * (anti_grav + accel * data.inputs.move_z.max(0.0)); }, diff --git a/server/src/rtsim/entity.rs b/server/src/rtsim/entity.rs index 8d265b929d..c79a4e3b43 100644 --- a/server/src/rtsim/entity.rs +++ b/server/src/rtsim/entity.rs @@ -60,7 +60,7 @@ impl Entity { comp::Body::BirdMedium(b) => { get_npc_name(&npc_names.bird_medium, b.species).to_string() }, - comp::Body::BirdSmall(_) => "Warbler".to_string(), + comp::Body::BirdLarge(_) => "Warbler".to_string(), comp::Body::Dragon(b) => get_npc_name(&npc_names.dragon, b.species).to_string(), comp::Body::Humanoid(b) => get_npc_name(&npc_names.humanoid, b.species).to_string(), comp::Body::Ship(_) => "Veloren Air".to_string(), diff --git a/voxygen/anim/src/bird_large/fly.rs b/voxygen/anim/src/bird_large/fly.rs new file mode 100644 index 0000000000..e5b1ec551c --- /dev/null +++ b/voxygen/anim/src/bird_large/fly.rs @@ -0,0 +1,146 @@ +use super::{ + super::{vek::*, Animation}, + BirdLargeSkeleton, SkeletonAttr, +}; + +pub struct FlyAnimation; + +impl Animation for FlyAnimation { + type Dependency = (Vec3, Vec3, Vec3, f32, Vec3, f32); + type Skeleton = BirdLargeSkeleton; + + #[cfg(feature = "use-dyn-lib")] + const UPDATE_FN: &'static [u8] = b"bird_large_fly\0"; + + #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_fly")] + fn update_skeleton_inner( + skeleton: &Self::Skeleton, + (velocity, orientation, last_ori, _global_time, avg_vel, _acc_vel): Self::Dependency, + anim_time: f32, + _rate: &mut f32, + s_a: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let slow = (anim_time * 2.0).sin(); + let fast = (anim_time * 4.0).sin(); + + let freq = 8.0; + let off1 = 0.0; + let off2 = -1.7; + let off3 = -2.0; + let off4 = -2.4; + let flap1 = 7.0 / 16.0 * (freq * anim_time + off1).sin() + + 7.0 / 64.0 * (freq * 2.0 * anim_time + off1).sin() + + 1.0 / 48.0 * (freq * 3.0 * anim_time + off1).sin(); + let flap2 = 7.0 / 16.0 * (freq * anim_time + off2).sin() + + 7.0 / 64.0 * (freq * 2.0 * anim_time + off2).sin() + + 1.0 / 48.0 * (freq * 3.0 * anim_time + off2).sin(); + let flap3 = 7.0 / 16.0 * (freq * anim_time + off3).sin() + + 7.0 / 64.0 * (freq * 2.0 * anim_time + off3).sin() + + 1.0 / 48.0 * (freq * 3.0 * anim_time + off3).sin(); + let flap4 = 7.0 / 16.0 * (freq * anim_time + off4).sin() + + 7.0 / 64.0 * (freq * 2.0 * anim_time + off4).sin() + + 1.0 / 48.0 * (freq * 3.0 * anim_time + off4).sin(); + + let ori: Vec2 = Vec2::from(orientation); + let last_ori = Vec2::from(last_ori); + let tilt = if ::vek::Vec2::new(ori, last_ori) + .map(|o| 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.2) + * last_ori.determine_side(Vec2::zero(), ori).signum() + } else { + 0.0 + } * 1.3; + let x_tilt = avg_vel.z.atan2(avg_vel.xy().magnitude()); + + next.chest.scale = Vec3::one() * s_a.scaler / 4.0; + + next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); + next.neck.orientation = Quaternion::rotation_x(0.2); + + next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1 - 3.0); + next.head.orientation = Quaternion::rotation_x(0.2 + fast * 0.05); + + next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); + + dbg!(velocity); + if velocity.z > 2.0 { + next.chest.position = + Vec3::new(0.0, s_a.chest.0, s_a.chest.1 - flap4 * 1.5) * s_a.scaler / 4.0; + next.chest.orientation = + Quaternion::rotation_x(-0.5 - flap1 * 0.2) * Quaternion::rotation_y(tilt * 1.8); + + next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + + next.wing_in_l.orientation = + Quaternion::rotation_y(-flap1 * 1.9 + 0.2) * Quaternion::rotation_x(0.4); + next.wing_in_r.orientation = + Quaternion::rotation_y(flap1 * 1.9 - 0.2) * Quaternion::rotation_x(0.4); + + next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_l.orientation = Quaternion::rotation_y(-flap2 * 1.4 - 0.2); + next.wing_mid_r.orientation = Quaternion::rotation_y(flap2 * 1.4 + 0.2); + + next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_l.orientation = Quaternion::rotation_y(-flap3 * 1.2 - 0.3); + next.wing_out_r.orientation = Quaternion::rotation_y(flap3 * 1.2 + 0.3); + + next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); + next.tail_front.orientation = + Quaternion::rotation_x(-flap2 * 0.2) * Quaternion::rotation_z(-tilt * 1.0); + next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); + next.tail_rear.orientation = + Quaternion::rotation_x(-flap3 * 0.3) * Quaternion::rotation_z(-tilt * 1.0); + } else { + next.chest.position = + Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + slow * 0.05) * s_a.scaler / 4.0; + next.chest.orientation = + Quaternion::rotation_x(-0.5 + slow * 0.05) * Quaternion::rotation_y(tilt * 1.8); + + next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + + next.wing_in_l.orientation = + Quaternion::rotation_y(0.1 + slow * 0.04) * Quaternion::rotation_x(0.6); + next.wing_in_r.orientation = + Quaternion::rotation_y(-0.1 + slow * -0.04) * Quaternion::rotation_x(0.6); + + next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_l.orientation = Quaternion::rotation_y(0.1 + slow * 0.04); + next.wing_mid_r.orientation = Quaternion::rotation_y(-0.1 + slow * -0.04); + + next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_l.orientation = Quaternion::rotation_y(0.1 + slow * 0.04); + next.wing_out_r.orientation = Quaternion::rotation_y(-0.1 + slow * -0.04); + + next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); + next.tail_front.orientation = + Quaternion::rotation_x(slow * 0.04) * Quaternion::rotation_z(-tilt * 1.0); + next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); + next.tail_rear.orientation = + Quaternion::rotation_x(-0.2 + slow * 0.08) * Quaternion::rotation_z(-tilt * 1.0); + } + + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2 + 3.0); + next.leg_l.orientation = Quaternion::rotation_x(-1.0 - flap1 * 0.1); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2 + 3.0); + next.leg_r.orientation = Quaternion::rotation_x(-1.0 - flap1 * 0.1); + + next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_l.orientation = Quaternion::rotation_x(-1.0 - flap1 * 0.1); + next.foot_r.position = Vec3::new(s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_r.orientation = Quaternion::rotation_x(-1.0 - flap1 * 0.1); + + next + } +} diff --git a/voxygen/anim/src/bird_large/idle.rs b/voxygen/anim/src/bird_large/idle.rs new file mode 100644 index 0000000000..f25cfa7664 --- /dev/null +++ b/voxygen/anim/src/bird_large/idle.rs @@ -0,0 +1,89 @@ +use super::{ + super::{vek::*, Animation}, + BirdLargeSkeleton, SkeletonAttr, +}; + +pub struct IdleAnimation; + +impl Animation for IdleAnimation { + type Dependency = f32; + type Skeleton = BirdLargeSkeleton; + + #[cfg(feature = "use-dyn-lib")] + const UPDATE_FN: &'static [u8] = b"bird_large_idle\0"; + + #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_idle")] + fn update_skeleton_inner( + skeleton: &Self::Skeleton, + _global_time: Self::Dependency, + anim_time: f32, + _rate: &mut f32, + s_a: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let fast = anim_time * 4.0; + + let freq = 8.0; + let off2 = -1.7; + let off3 = -2.0; + let off4 = -2.4; + let flap1 = 7.0 / 16.0 * (freq * anim_time).sin() + + 7.0 / 64.0 * (freq * 2.0 * anim_time).sin() + + 1.0 / 48.0 * (freq * 3.0 * anim_time).sin(); + let flap2 = 7.0 / 16.0 * (freq * anim_time + off2).sin() + + 7.0 / 64.0 * (freq * 2.0 * anim_time + off2).sin() + + 1.0 / 48.0 * (freq * 3.0 * anim_time + off2).sin(); + let flap3 = 7.0 / 16.0 * (freq * anim_time + off3).sin() + + 7.0 / 64.0 * (freq * 2.0 * anim_time + off3).sin() + + 1.0 / 48.0 * (freq * 3.0 * anim_time + off3).sin(); + let flap4 = 7.0 / 16.0 * (freq * anim_time + off4).sin() + + 7.0 / 64.0 * (freq * 2.0 * anim_time + off4).sin() + + 1.0 / 48.0 * (freq * 3.0 * anim_time + off4).sin(); + + next.chest.scale = Vec3::one() * s_a.scaler / 4.0; + next.chest.position = Vec3::new(0.0, s_a.chest.0, s_a.chest.1) * s_a.scaler / 4.0; + next.chest.orientation = Quaternion::rotation_x(0.0); + + next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); + next.neck.orientation = Quaternion::rotation_x(0.0); + + next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); + next.head.orientation = Quaternion::rotation_x(0.0); + + next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); + + next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); + next.tail_front.orientation = Quaternion::rotation_x(0.0); + next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); + next.tail_rear.orientation = Quaternion::rotation_x(0.0); + + next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + + next.wing_in_l.orientation = Quaternion::rotation_y(0.0); + next.wing_in_r.orientation = Quaternion::rotation_y(0.0) * Quaternion::rotation_x(0.0); + + next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_l.orientation = Quaternion::rotation_y(0.0); + next.wing_mid_r.orientation = Quaternion::rotation_y(0.0); + + next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_l.orientation = Quaternion::rotation_y(0.0); + next.wing_out_r.orientation = Quaternion::rotation_y(0.0); + + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2 + 3.0); + next.leg_l.orientation = Quaternion::rotation_x(0.0); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2 + 3.0); + next.leg_r.orientation = Quaternion::rotation_x(0.0); + + next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_l.orientation = Quaternion::rotation_x(0.0); + next.foot_r.position = Vec3::new(s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_r.orientation = Quaternion::rotation_x(0.0); + + next + } +} diff --git a/voxygen/anim/src/bird_large/mod.rs b/voxygen/anim/src/bird_large/mod.rs new file mode 100644 index 0000000000..a4c938bff3 --- /dev/null +++ b/voxygen/anim/src/bird_large/mod.rs @@ -0,0 +1,174 @@ +pub mod fly; +pub mod idle; +pub mod run; + +// Reexports +pub use self::{fly::FlyAnimation, idle::IdleAnimation, run::RunAnimation}; + +use super::{make_bone, vek::*, FigureBoneData, Skeleton}; +use common::comp::{self}; +use core::convert::TryFrom; + +pub type Body = comp::bird_large::Body; + +skeleton_impls!(struct BirdLargeSkeleton { + + head, + + beak, + + neck, + + chest, + + tail_front, + + tail_rear, + + wing_in_l, + + wing_in_r, + + wing_mid_l, + + wing_mid_r, + + wing_out_l, + + wing_out_r, + + leg_l, + + leg_r, + + foot_l, + + foot_r, +}); + +impl Skeleton for BirdLargeSkeleton { + type Attr = SkeletonAttr; + type Body = Body; + + const BONE_COUNT: usize = 16; + #[cfg(feature = "use-dyn-lib")] + const COMPUTE_FN: &'static [u8] = b"bird_large_compute_mats\0"; + + #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_compute_mats")] + + fn compute_matrices_inner( + &self, + base_mat: Mat4, + buf: &mut [FigureBoneData; super::MAX_BONE_COUNT], + ) -> Vec3 { + let chest_mat = base_mat * Mat4::::from(self.chest); + let neck_mat = chest_mat * Mat4::::from(self.neck); + let head_mat = neck_mat * Mat4::::from(self.head); + let beak_mat = head_mat * Mat4::::from(self.beak); + let tail_front_mat = chest_mat * Mat4::::from(self.tail_front); + let tail_rear_mat = tail_front_mat * Mat4::::from(self.tail_rear); + let wing_in_l_mat = chest_mat * Mat4::::from(self.wing_in_l); + let wing_in_r_mat = chest_mat * Mat4::::from(self.wing_in_r); + let wing_mid_l_mat = wing_in_l_mat * Mat4::::from(self.wing_mid_l); + let wing_mid_r_mat = wing_in_r_mat * Mat4::::from(self.wing_mid_r); + let wing_out_l_mat = wing_mid_l_mat * Mat4::::from(self.wing_out_l); + let wing_out_r_mat = wing_mid_r_mat * Mat4::::from(self.wing_out_r); + let leg_l_mat = chest_mat * Mat4::::from(self.leg_l); + let leg_r_mat = chest_mat * Mat4::::from(self.leg_r); + let foot_l_mat = chest_mat * Mat4::::from(self.leg_l); + let foot_r_mat = chest_mat * Mat4::::from(self.leg_r); + + *(<&mut [_; Self::BONE_COUNT]>::try_from(&mut buf[0..Self::BONE_COUNT]).unwrap()) = [ + make_bone(head_mat), + make_bone(beak_mat), + make_bone(neck_mat), + make_bone(chest_mat), + make_bone(tail_front_mat), + make_bone(tail_rear_mat), + make_bone(wing_in_l_mat), + make_bone(wing_in_r_mat), + make_bone(wing_mid_l_mat), + make_bone(wing_mid_r_mat), + make_bone(wing_out_l_mat), + make_bone(wing_out_r_mat), + make_bone(leg_l_mat), + make_bone(leg_r_mat), + make_bone(foot_l_mat), + make_bone(foot_r_mat), + ]; + Vec3::default() + } +} + +pub struct SkeletonAttr { + chest: (f32, f32), + neck: (f32, f32), + head: (f32, f32), + beak: (f32, f32), + tail_front: (f32, f32), + tail_rear: (f32, f32), + wing_in: (f32, f32, f32), + wing_mid: (f32, f32, f32), + wing_out: (f32, f32, f32), + leg: (f32, f32, f32), + foot: (f32, f32, f32), + scaler: f32, +} + +impl<'a> std::convert::TryFrom<&'a comp::Body> for SkeletonAttr { + type Error = (); + + fn try_from(body: &'a comp::Body) -> Result { + match body { + comp::Body::BirdLarge(body) => Ok(SkeletonAttr::from(body)), + _ => Err(()), + } + } +} + +impl Default for SkeletonAttr { + fn default() -> Self { + Self { + chest: (0.0, 0.0), + neck: (0.0, 0.0), + head: (0.0, 0.0), + beak: (0.0, 0.0), + tail_front: (0.0, 0.0), + tail_rear: (0.0, 0.0), + wing_in: (0.0, 0.0, 0.0), + wing_mid: (0.0, 0.0, 0.0), + wing_out: (0.0, 0.0, 0.0), + leg: (0.0, 0.0, 0.0), + foot: (0.0, 0.0, 0.0), + scaler: 0.0, + } + } +} + +impl<'a> From<&'a Body> for SkeletonAttr { + fn from(body: &'a Body) -> Self { + use comp::bird_large::Species::*; + Self { + chest: match (body.species, body.body_type) { + (Phoenix, _) => (2.5, 8.5), + }, + neck: match (body.species, body.body_type) { + (Phoenix, _) => (0.5, 3.0), + }, + head: match (body.species, body.body_type) { + (Phoenix, _) => (2.0, 3.0), + }, + beak: match (body.species, body.body_type) { + (Phoenix, _) => (2.0, 1.0), + }, + tail_front: match (body.species, body.body_type) { + (Phoenix, _) => (-5.5, -2.0), + }, + tail_rear: match (body.species, body.body_type) { + (Phoenix, _) => (-3.0, -3.0), + }, + wing_in: match (body.species, body.body_type) { + (Phoenix, _) => (3.0, 2.5, 4.0), + }, + wing_mid: match (body.species, body.body_type) { + (Phoenix, _) => (6.5, -1.0, 0.0), + }, + wing_out: match (body.species, body.body_type) { + (Phoenix, _) => (0.5, -1.0, 0.0), + }, + leg: match (body.species, body.body_type) { + (Phoenix, _) => (2.5, -2.5, -6.5), + }, + foot: match (body.species, body.body_type) { + (Phoenix, _) => (2.5, -16.5, -6.5), + }, + scaler: match (body.species, body.body_type) { + (Phoenix, _) => (1.0), + }, + } + } +} diff --git a/voxygen/anim/src/bird_large/run.rs b/voxygen/anim/src/bird_large/run.rs new file mode 100644 index 0000000000..a308ce2e4f --- /dev/null +++ b/voxygen/anim/src/bird_large/run.rs @@ -0,0 +1,140 @@ +use super::{ + super::{vek::*, Animation}, + BirdLargeSkeleton, SkeletonAttr, +}; +use std::f32::consts::PI; + +pub struct RunAnimation; + +impl Animation for RunAnimation { + type Dependency = (Vec3, Vec3, Vec3, f32, Vec3, f32); + type Skeleton = BirdLargeSkeleton; + + #[cfg(feature = "use-dyn-lib")] + const UPDATE_FN: &'static [u8] = b"bird_large_run\0"; + + #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_run")] + fn update_skeleton_inner( + skeleton: &Self::Skeleton, + (velocity, orientation, last_ori, _global_time, avg_vel, acc_vel): Self::Dependency, + _anim_time: f32, + rate: &mut f32, + s_a: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + let speed = (Vec2::::from(velocity).magnitude()).min(22.0); + *rate = 1.0; + + //let speednorm = speed / 13.0; + let speednorm = (speed / 13.0).powf(0.25); + + let speedmult = 1.0; + let lab: f32 = 0.6; //6 + + let short = ((1.0 + / (0.72 + + 0.28 * ((acc_vel * 1.0 * lab * speedmult + PI * -0.15 - 0.5).sin()).powi(2))) + .sqrt()) + * ((acc_vel * 1.0 * lab * speedmult + PI * -0.15 - 0.5).sin()) + * speednorm; + + // + let shortalt = (acc_vel * 1.0 * lab * speedmult + PI * 3.0 / 8.0 - 0.5).sin() * speednorm; + + //FL + let foot1a = (acc_vel * 1.0 * lab * speedmult + 0.0 + PI).sin() * speednorm; //1.5 + let foot1b = (acc_vel * 1.0 * lab * speedmult + 1.57 + PI).sin() * speednorm; //1.9 + //FR + let foot2a = (acc_vel * 1.0 * lab * speedmult).sin() * speednorm; //1.2 + let foot2b = (acc_vel * 1.0 * lab * speedmult + 1.57).sin() * speednorm; //1.6 + let ori: Vec2 = Vec2::from(orientation); + let last_ori = Vec2::from(last_ori); + let tilt = if ::vek::Vec2::new(ori, last_ori) + .map(|o| 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.2) + * last_ori.determine_side(Vec2::zero(), ori).signum() + } else { + 0.0 + } * 1.3; + let x_tilt = avg_vel.z.atan2(avg_vel.xy().magnitude()) * speednorm; + + next.head.scale = Vec3::one() * 1.02; + next.neck.scale = Vec3::one() * 0.98; + next.beak.scale = Vec3::one() * 0.98; + next.foot_l.scale = Vec3::one() * 0.96; + next.foot_r.scale = Vec3::one() * 0.96; + next.chest.scale = Vec3::one() * s_a.scaler / 4.0; + + next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); + next.head.orientation = Quaternion::rotation_x(-0.1 * speednorm + short * -0.05) + * Quaternion::rotation_y(tilt * 0.8) + * Quaternion::rotation_z(shortalt * -0.2 - tilt * 2.5); + + next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); + next.beak.orientation = Quaternion::rotation_x(short * -0.03); + + next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); + next.neck.orientation = Quaternion::rotation_x(-0.1 * speednorm + short * -0.04) + * Quaternion::rotation_y(tilt * 0.3) + * Quaternion::rotation_z(shortalt * -0.1 - tilt * 3.2); + + next.chest.position = Vec3::new( + 0.0, + s_a.chest.0, + s_a.chest.1 + short * 0.5 + x_tilt * 10.0 + 0.5 * speednorm, + ) * s_a.scaler + / 4.0; + next.chest.orientation = Quaternion::rotation_x(short * 0.07 + x_tilt) + * Quaternion::rotation_y(tilt * 0.8) + * Quaternion::rotation_z(shortalt * 0.15 + tilt * -1.5); + + next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); + next.tail_front.orientation = Quaternion::rotation_x(0.1 + short * -0.02) + * Quaternion::rotation_z(shortalt * -0.1 + tilt * 1.0); + + next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); + next.tail_rear.orientation = Quaternion::rotation_x(0.2 + short * -0.1) + * Quaternion::rotation_z(shortalt * -0.2 + tilt * 1.4); + + next.leg_l.position = Vec3::new( + -s_a.leg.0 + speednorm * 1.5, + s_a.leg.1 + foot1b * -1.3, + s_a.leg.2 + foot1a * 1.0, + ); + next.leg_l.orientation = Quaternion::rotation_x(-0.2 * speednorm + foot1a * 0.15) + * Quaternion::rotation_y(tilt * 0.5) + * Quaternion::rotation_z(foot1a * -0.3 + tilt * -0.5); + + next.leg_r.position = Vec3::new( + s_a.leg.0 + speednorm * -1.5, + s_a.leg.1 + foot2b * -1.3, + s_a.leg.2 + foot2a * 1.0, + ); + next.leg_r.orientation = Quaternion::rotation_x(-0.2 * speednorm + foot2a * 0.15) + * Quaternion::rotation_y(tilt * 0.5) + * Quaternion::rotation_z(foot2a * 0.3 + tilt * -0.5); + + next.foot_l.position = Vec3::new( + -s_a.foot.0, + s_a.foot.1 + foot1b * -2.0, + s_a.foot.2 + speednorm * 0.5 + (foot1a * 1.5).max(0.0), + ); + next.foot_l.orientation = Quaternion::rotation_x(-0.2 * speednorm + foot1b * -0.35) + * Quaternion::rotation_y(tilt * -1.0) + * Quaternion::rotation_z(tilt * -0.5); + + next.foot_r.position = Vec3::new( + s_a.foot.0, + s_a.foot.1 + foot2b * -2.0, + s_a.foot.2 + speednorm * 0.5 + (foot2a * 1.5).max(0.0), + ); + next.foot_r.orientation = Quaternion::rotation_x(-0.2 * speednorm + foot2b * -0.35) + * Quaternion::rotation_y(tilt * -1.0); + + next + } +} diff --git a/voxygen/anim/src/bird_small/idle.rs b/voxygen/anim/src/bird_small/idle.rs deleted file mode 100644 index c87c59a1a8..0000000000 --- a/voxygen/anim/src/bird_small/idle.rs +++ /dev/null @@ -1,42 +0,0 @@ -use super::{super::Animation, BirdSmallSkeleton, SkeletonAttr}; -//use std::{f32::consts::PI, ops::Mul}; -use super::super::vek::*; - -pub struct IdleAnimation; - -impl Animation for IdleAnimation { - type Dependency = f32; - type Skeleton = BirdSmallSkeleton; - - #[cfg(feature = "use-dyn-lib")] - const UPDATE_FN: &'static [u8] = b"bird_small_idle\0"; - - #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_small_idle")] - fn update_skeleton_inner( - skeleton: &Self::Skeleton, - _global_time: Self::Dependency, - _anim_time: f32, - _rate: &mut f32, - _skeleton_attr: &SkeletonAttr, - ) -> Self::Skeleton { - let mut next = (*skeleton).clone(); - - next.head.position = Vec3::new(0.0, 7.5, 15.0) / 11.0; - next.head.orientation = Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); - next.head.scale = Vec3::one() / 10.88; - - next.torso.position = Vec3::new(0.0, 7.5, 15.0) / 11.0; - next.torso.orientation = Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); - next.torso.scale = Vec3::one() / 10.88; - - next.wing_l.position = Vec3::new(0.0, 7.5, 15.0) / 11.0; - next.wing_l.orientation = Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); - next.wing_l.scale = Vec3::one() / 10.88; - - next.wing_r.position = Vec3::new(0.0, 7.5, 15.0) / 11.0; - next.wing_r.orientation = Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); - next.wing_r.scale = Vec3::one() / 10.88; - - next - } -} diff --git a/voxygen/anim/src/bird_small/jump.rs b/voxygen/anim/src/bird_small/jump.rs deleted file mode 100644 index 5a940935ce..0000000000 --- a/voxygen/anim/src/bird_small/jump.rs +++ /dev/null @@ -1,42 +0,0 @@ -use super::{super::Animation, BirdSmallSkeleton, SkeletonAttr}; -//use std::f32::consts::PI; -use super::super::vek::*; - -pub struct JumpAnimation; - -impl Animation for JumpAnimation { - type Dependency = (f32, f32); - type Skeleton = BirdSmallSkeleton; - - #[cfg(feature = "use-dyn-lib")] - const UPDATE_FN: &'static [u8] = b"bird_small_jump\0"; - - #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_small_jump")] - fn update_skeleton_inner( - skeleton: &Self::Skeleton, - _global_time: Self::Dependency, - _anim_time: f32, - _rate: &mut f32, - _skeleton_attr: &SkeletonAttr, - ) -> Self::Skeleton { - let mut next = (*skeleton).clone(); - - next.head.position = Vec3::new(0.0, 7.5, 15.0) / 11.0; - next.head.orientation = Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); - next.head.scale = Vec3::one() / 10.88; - - next.torso.position = Vec3::new(0.0, 7.5, 15.0) / 11.0; - next.torso.orientation = Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); - next.torso.scale = Vec3::one() / 10.88; - - next.wing_l.position = Vec3::new(0.0, 7.5, 15.0) / 11.0; - next.wing_l.orientation = Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); - next.wing_l.scale = Vec3::one() / 10.88; - - next.wing_r.position = Vec3::new(0.0, 7.5, 15.0) / 11.0; - next.wing_r.orientation = Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); - next.wing_r.scale = Vec3::one() / 10.88; - - next - } -} diff --git a/voxygen/anim/src/bird_small/mod.rs b/voxygen/anim/src/bird_small/mod.rs deleted file mode 100644 index b5ffa053a5..0000000000 --- a/voxygen/anim/src/bird_small/mod.rs +++ /dev/null @@ -1,67 +0,0 @@ -pub mod idle; -pub mod jump; -pub mod run; - -// Reexports -pub use self::{idle::IdleAnimation, jump::JumpAnimation, run::RunAnimation}; - -use super::{make_bone, vek::*, FigureBoneData, Skeleton}; -use common::comp::{self}; -use core::convert::TryFrom; - -pub type Body = comp::bird_small::Body; - -skeleton_impls!(struct BirdSmallSkeleton { - + head, - + torso, - + wing_l, - + wing_r, -}); - -impl Skeleton for BirdSmallSkeleton { - type Attr = SkeletonAttr; - type Body = Body; - - const BONE_COUNT: usize = 4; - #[cfg(feature = "use-dyn-lib")] - const COMPUTE_FN: &'static [u8] = b"bird_small_compute_mats\0"; - - #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_small_compute_mats")] - - fn compute_matrices_inner( - &self, - base_mat: Mat4, - buf: &mut [FigureBoneData; super::MAX_BONE_COUNT], - ) -> Vec3 { - let torso_mat = base_mat * Mat4::::from(self.torso); - - *(<&mut [_; Self::BONE_COUNT]>::try_from(&mut buf[0..Self::BONE_COUNT]).unwrap()) = [ - make_bone(torso_mat * Mat4::::from(self.head)), - make_bone(torso_mat), - make_bone(torso_mat * Mat4::::from(self.wing_l)), - make_bone(torso_mat * Mat4::::from(self.wing_r)), - ]; - Vec3::default() - } -} - -pub struct SkeletonAttr; - -impl<'a> std::convert::TryFrom<&'a comp::Body> for SkeletonAttr { - type Error = (); - - fn try_from(body: &'a comp::Body) -> Result { - match body { - comp::Body::BirdSmall(body) => Ok(SkeletonAttr::from(body)), - _ => Err(()), - } - } -} - -impl Default for SkeletonAttr { - fn default() -> Self { Self } -} - -impl<'a> From<&'a Body> for SkeletonAttr { - fn from(_body: &'a Body) -> Self { Self } -} diff --git a/voxygen/anim/src/bird_small/run.rs b/voxygen/anim/src/bird_small/run.rs deleted file mode 100644 index 67b01ad7af..0000000000 --- a/voxygen/anim/src/bird_small/run.rs +++ /dev/null @@ -1,42 +0,0 @@ -use super::{super::Animation, BirdSmallSkeleton, SkeletonAttr}; -//use std::{f32::consts::PI, ops::Mul}; -use super::super::vek::*; - -pub struct RunAnimation; - -impl Animation for RunAnimation { - type Dependency = (f32, f32); - type Skeleton = BirdSmallSkeleton; - - #[cfg(feature = "use-dyn-lib")] - const UPDATE_FN: &'static [u8] = b"bird_small_run\0"; - - #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_small_run")] - fn update_skeleton_inner( - skeleton: &Self::Skeleton, - (_velocity, _global_time): Self::Dependency, - _anim_time: f32, - _rate: &mut f32, - _skeleton_attr: &SkeletonAttr, - ) -> Self::Skeleton { - let mut next = (*skeleton).clone(); - - next.head.position = Vec3::new(0.0, 7.5, 15.0) / 11.0; - next.head.orientation = Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); - next.head.scale = Vec3::one() / 10.88; - - next.torso.position = Vec3::new(0.0, 7.5, 15.0) / 11.0; - next.torso.orientation = Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); - next.torso.scale = Vec3::one() / 10.88; - - next.wing_l.position = Vec3::new(0.0, 7.5, 15.0) / 11.0; - next.wing_l.orientation = Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); - next.wing_l.scale = Vec3::one() / 10.88; - - next.wing_r.position = Vec3::new(0.0, 7.5, 15.0) / 11.0; - next.wing_r.orientation = Quaternion::rotation_z(0.0) * Quaternion::rotation_x(0.0); - next.wing_r.scale = Vec3::one() / 10.88; - - next - } -} diff --git a/voxygen/anim/src/lib.rs b/voxygen/anim/src/lib.rs index 8895e92841..45d7e65e87 100644 --- a/voxygen/anim/src/lib.rs +++ b/voxygen/anim/src/lib.rs @@ -50,8 +50,8 @@ macro_rules! skeleton_impls { pub mod biped_large; pub mod biped_small; +pub mod bird_large; pub mod bird_medium; -pub mod bird_small; pub mod character; pub mod dragon; #[cfg(feature = "use-dyn-lib")] pub mod dyn_lib; diff --git a/voxygen/src/audio/sfx/event_mapper/movement/mod.rs b/voxygen/src/audio/sfx/event_mapper/movement/mod.rs index 1816f057c8..d4300e1610 100644 --- a/voxygen/src/audio/sfx/event_mapper/movement/mod.rs +++ b/voxygen/src/audio/sfx/event_mapper/movement/mod.rs @@ -93,7 +93,7 @@ impl EventMapper for MovementEventMapper { Body::QuadrupedMedium(_) | Body::QuadrupedSmall(_) | Body::QuadrupedLow(_) => { Self::map_quadruped_movement_event(physics, vel.0, underfoot_block_kind) }, - Body::BirdMedium(_) | Body::BirdSmall(_) | Body::BipedLarge(_) => { + Body::BirdMedium(_) | Body::BirdLarge(_) | Body::BipedLarge(_) => { Self::map_non_humanoid_movement_event(physics, vel.0, underfoot_block_kind) }, _ => SfxEvent::Idle, // Ignore fish, etc... @@ -281,7 +281,7 @@ impl MovementEventMapper { Body::QuadrupedMedium(_) => 0.7, Body::QuadrupedLow(_) => 0.7, Body::BirdMedium(_) => 0.3, - Body::BirdSmall(_) => 0.2, + Body::BirdLarge(_) => 0.2, Body::BipedLarge(_) => 1.0, _ => 0.9, } diff --git a/voxygen/src/audio/sfx/event_mapper/movement/tests.rs b/voxygen/src/audio/sfx/event_mapper/movement/tests.rs index 0fa422308a..0c89e29f2e 100644 --- a/voxygen/src/audio/sfx/event_mapper/movement/tests.rs +++ b/voxygen/src/audio/sfx/event_mapper/movement/tests.rs @@ -2,7 +2,7 @@ use super::*; use crate::audio::sfx::SfxEvent; use common::{ comp::{ - bird_small, humanoid, quadruped_medium, quadruped_small, Body, CharacterState, InputKind, + bird_large, humanoid, quadruped_medium, quadruped_small, Body, CharacterState, InputKind, PhysicsState, }, states, @@ -340,12 +340,12 @@ fn determines_relative_volumes() { quadruped_small::Body::random(), )); - let bird_small = - MovementEventMapper::get_volume_for_body_type(&Body::BirdSmall(bird_small::Body::random())); + let bird_large = + MovementEventMapper::get_volume_for_body_type(&Body::BirdLarge(bird_large::Body::random())); assert!(quadruped_medium < human); assert!(quadruped_small < quadruped_medium); - assert!(bird_small < quadruped_small); + assert!(bird_large < quadruped_small); } fn empty_ability_info() -> states::utils::AbilityInfo { diff --git a/voxygen/src/scene/figure/load.rs b/voxygen/src/scene/figure/load.rs index ae677b6127..bb612c8ef2 100644 --- a/voxygen/src/scene/figure/load.rs +++ b/voxygen/src/scene/figure/load.rs @@ -4,8 +4,8 @@ use common::{ comp::{ biped_large::{self, BodyType as BLBodyType, Species as BLSpecies}, biped_small, + bird_large::{self, BodyType as BLABodyType, Species as BLASpecies}, bird_medium::{self, BodyType as BMBodyType, Species as BMSpecies}, - bird_small, dragon::{self, BodyType as DBodyType, Species as DSpecies}, fish_medium::{self, BodyType as FMBodyType, Species as FMSpecies}, fish_small::{self, BodyType as FSBodyType, Species as FSSpecies}, @@ -3109,65 +3109,379 @@ impl DragonLateralSpec { } //// +#[derive(Deserialize)] +struct BirdLargeCentralSpec(HashMap<(BLASpecies, BLABodyType), SidedBLACentralVoxSpec>); + +#[derive(Deserialize)] +struct SidedBLACentralVoxSpec { + head: BirdLargeCentralSubSpec, + beak: BirdLargeCentralSubSpec, + neck: BirdLargeCentralSubSpec, + chest: BirdLargeCentralSubSpec, + tail_front: BirdLargeCentralSubSpec, + tail_rear: BirdLargeCentralSubSpec, +} +#[derive(Deserialize)] +struct BirdLargeCentralSubSpec { + offset: [f32; 3], // Should be relative to initial origin + central: VoxSimple, +} + +#[derive(Deserialize)] +struct BirdLargeLateralSpec(HashMap<(BLASpecies, BLABodyType), SidedBLALateralVoxSpec>); + +#[derive(Deserialize)] +struct SidedBLALateralVoxSpec { + wing_in_l: BirdLargeLateralSubSpec, + wing_in_r: BirdLargeLateralSubSpec, + wing_mid_l: BirdLargeLateralSubSpec, + wing_mid_r: BirdLargeLateralSubSpec, + wing_out_l: BirdLargeLateralSubSpec, + wing_out_r: BirdLargeLateralSubSpec, + leg_l: BirdLargeLateralSubSpec, + leg_r: BirdLargeLateralSubSpec, + foot_l: BirdLargeLateralSubSpec, + foot_r: BirdLargeLateralSubSpec, +} +#[derive(Deserialize)] +struct BirdLargeLateralSubSpec { + offset: [f32; 3], // Should be relative to initial origin + lateral: VoxSimple, +} + make_vox_spec!( - bird_small::Body, - struct BirdSmallSpec {}, - |FigureKey { body, .. }, _spec| { + bird_large::Body, + struct BirdLargeSpec { + central: BirdLargeCentralSpec = "voxygen.voxel.bird_large_central_manifest", + lateral: BirdLargeLateralSpec = "voxygen.voxel.bird_large_lateral_manifest", + }, + |FigureKey { body, .. }, spec| { [ - Some(mesh_bird_small_head(body.head)), - Some(mesh_bird_small_torso(body.torso)), - Some(mesh_bird_small_wing_l(body.wing_l)), - Some(mesh_bird_small_wing_r(body.wing_r)), - None, - None, - None, - None, - None, - None, - None, - None, - None, - None, - None, - None, + Some(spec.central.read().0.mesh_head( + body.species, + body.body_type, + )), + Some(spec.central.read().0.mesh_beak( + body.species, + body.body_type, + )), + Some(spec.central.read().0.mesh_neck( + body.species, + body.body_type, + )), + Some(spec.central.read().0.mesh_chest( + body.species, + body.body_type, + )), + Some(spec.central.read().0.mesh_tail_front( + body.species, + body.body_type, + )), + Some(spec.central.read().0.mesh_tail_rear( + body.species, + body.body_type, + )), + Some(spec.lateral.read().0.mesh_wing_in_l( + body.species, + body.body_type, + )), + Some(spec.lateral.read().0.mesh_wing_in_r( + body.species, + body.body_type, + )), + Some(spec.lateral.read().0.mesh_wing_mid_l( + body.species, + body.body_type, + )), + Some(spec.lateral.read().0.mesh_wing_mid_r( + body.species, + body.body_type, + )), + Some(spec.lateral.read().0.mesh_wing_out_l( + body.species, + body.body_type, + )), + Some(spec.lateral.read().0.mesh_wing_out_r( + body.species, + body.body_type, + )), + Some(spec.lateral.read().0.mesh_leg_l( + body.species, + body.body_type, + )), + Some(spec.lateral.read().0.mesh_leg_r( + body.species, + body.body_type, + )), + Some(spec.lateral.read().0.mesh_foot_l( + body.species, + body.body_type, + )), + Some(spec.lateral.read().0.mesh_foot_r( + body.species, + body.body_type, + )), ] }, ); -fn mesh_bird_small_head(head: bird_small::Head) -> BoneMeshes { - load_mesh( - match head { - bird_small::Head::Default => "npc.crow.head", - }, - Vec3::new(-7.0, -6.0, -6.0), - ) -} +impl BirdLargeCentralSpec { + fn mesh_head(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No head specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let central = graceful_load_segment(&spec.head.central.0); -fn mesh_bird_small_torso(torso: bird_small::Torso) -> BoneMeshes { - load_mesh( - match torso { - bird_small::Torso::Default => "npc.crow.torso", - }, - Vec3::new(-7.0, -6.0, -6.0), - ) -} + (central, Vec3::from(spec.head.offset)) + } -fn mesh_bird_small_wing_l(wing_l: bird_small::WingL) -> BoneMeshes { - load_mesh( - match wing_l { - bird_small::WingL::Default => "npc.crow.wing_l", - }, - Vec3::new(-7.0, -6.0, -6.0), - ) -} + fn mesh_beak(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No beak specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let central = graceful_load_segment(&spec.beak.central.0); -fn mesh_bird_small_wing_r(wing_r: bird_small::WingR) -> BoneMeshes { - load_mesh( - match wing_r { - bird_small::WingR::Default => "npc.crow.wing_r", - }, - Vec3::new(-7.0, -6.0, -6.0), - ) + (central, Vec3::from(spec.beak.offset)) + } + + fn mesh_neck(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No neck specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let central = graceful_load_segment(&spec.neck.central.0); + + (central, Vec3::from(spec.neck.offset)) + } + + fn mesh_chest(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No chest specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let central = graceful_load_segment(&spec.chest.central.0); + + (central, Vec3::from(spec.chest.offset)) + } + + fn mesh_tail_front(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No tail front specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let central = graceful_load_segment(&spec.tail_front.central.0); + + (central, Vec3::from(spec.tail_front.offset)) + } + + fn mesh_tail_rear(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No tail rear specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let central = graceful_load_segment(&spec.tail_rear.central.0); + + (central, Vec3::from(spec.tail_rear.offset)) + } +} +impl BirdLargeLateralSpec { + fn mesh_wing_in_l(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No wing in in left specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let lateral = graceful_load_segment(&spec.wing_in_l.lateral.0); + + (lateral, Vec3::from(spec.wing_in_l.offset)) + } + + fn mesh_wing_in_r(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No wing in right specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let lateral = graceful_load_segment(&spec.wing_in_r.lateral.0); + + (lateral, Vec3::from(spec.wing_in_r.offset)) + } + + fn mesh_wing_mid_l(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No wing mid specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let lateral = graceful_load_segment(&spec.wing_mid_l.lateral.0); + + (lateral, Vec3::from(spec.wing_mid_l.offset)) + } + + fn mesh_wing_mid_r(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No wing mid specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let lateral = graceful_load_segment(&spec.wing_in_l.lateral.0); + + (lateral, Vec3::from(spec.wing_mid_r.offset)) + } + + fn mesh_wing_out_l(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No wing out specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let lateral = graceful_load_segment(&spec.wing_out_l.lateral.0); + + (lateral, Vec3::from(spec.wing_out_l.offset)) + } + + fn mesh_wing_out_r(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No wing out specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let lateral = graceful_load_segment(&spec.wing_out_r.lateral.0); + + (lateral, Vec3::from(spec.wing_out_r.offset)) + } + + fn mesh_leg_l(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No leg specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let lateral = graceful_load_segment(&spec.leg_l.lateral.0); + + (lateral, Vec3::from(spec.leg_l.offset)) + } + + fn mesh_leg_r(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No leg specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let lateral = graceful_load_segment(&spec.leg_r.lateral.0); + + (lateral, Vec3::from(spec.leg_r.offset)) + } + + fn mesh_foot_l(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No foot specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let lateral = graceful_load_segment(&spec.foot_l.lateral.0); + + (lateral, Vec3::from(spec.foot_l.offset)) + } + + fn mesh_foot_r(&self, species: BLASpecies, body_type: BLABodyType) -> BoneMeshes { + let spec = match self.0.get(&(species, body_type)) { + Some(spec) => spec, + None => { + error!( + "No foot specification exists for the combination of {:?} and {:?}", + species, body_type + ); + return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); + }, + }; + let lateral = graceful_load_segment(&spec.foot_r.lateral.0); + + (lateral, Vec3::from(spec.foot_r.offset)) + } } //// diff --git a/voxygen/src/scene/figure/mod.rs b/voxygen/src/scene/figure/mod.rs index e583942cfa..a74a4e3b5a 100644 --- a/voxygen/src/scene/figure/mod.rs +++ b/voxygen/src/scene/figure/mod.rs @@ -19,7 +19,7 @@ use crate::{ }; use anim::{ biped_large::BipedLargeSkeleton, biped_small::BipedSmallSkeleton, - bird_medium::BirdMediumSkeleton, bird_small::BirdSmallSkeleton, character::CharacterSkeleton, + bird_large::BirdLargeSkeleton, bird_medium::BirdMediumSkeleton, character::CharacterSkeleton, dragon::DragonSkeleton, fish_medium::FishMediumSkeleton, fish_small::FishSmallSkeleton, golem::GolemSkeleton, object::ObjectSkeleton, quadruped_low::QuadrupedLowSkeleton, quadruped_medium::QuadrupedMediumSkeleton, quadruped_small::QuadrupedSmallSkeleton, @@ -95,7 +95,7 @@ struct FigureMgrStates { fish_medium_states: HashMap>, theropod_states: HashMap>, dragon_states: HashMap>, - bird_small_states: HashMap>, + bird_large_states: HashMap>, fish_small_states: HashMap>, biped_large_states: HashMap>, biped_small_states: HashMap>, @@ -115,7 +115,7 @@ impl FigureMgrStates { fish_medium_states: HashMap::new(), theropod_states: HashMap::new(), dragon_states: HashMap::new(), - bird_small_states: HashMap::new(), + bird_large_states: HashMap::new(), fish_small_states: HashMap::new(), biped_large_states: HashMap::new(), biped_small_states: HashMap::new(), @@ -164,8 +164,8 @@ impl FigureMgrStates { .get_mut(&entity) .map(DerefMut::deref_mut), Body::Dragon(_) => self.dragon_states.get_mut(&entity).map(DerefMut::deref_mut), - Body::BirdSmall(_) => self - .bird_small_states + Body::BirdLarge(_) => self + .bird_large_states .get_mut(&entity) .map(DerefMut::deref_mut), Body::FishSmall(_) => self @@ -202,7 +202,7 @@ impl FigureMgrStates { Body::FishMedium(_) => self.fish_medium_states.remove(&entity).map(|e| e.meta), Body::Theropod(_) => self.theropod_states.remove(&entity).map(|e| e.meta), Body::Dragon(_) => self.dragon_states.remove(&entity).map(|e| e.meta), - Body::BirdSmall(_) => self.bird_small_states.remove(&entity).map(|e| e.meta), + Body::BirdLarge(_) => self.bird_large_states.remove(&entity).map(|e| e.meta), Body::FishSmall(_) => self.fish_small_states.remove(&entity).map(|e| e.meta), Body::BipedLarge(_) => self.biped_large_states.remove(&entity).map(|e| e.meta), Body::BipedSmall(_) => self.biped_small_states.remove(&entity).map(|e| e.meta), @@ -222,7 +222,7 @@ impl FigureMgrStates { self.fish_medium_states.retain(|k, v| f(k, &mut *v)); self.theropod_states.retain(|k, v| f(k, &mut *v)); self.dragon_states.retain(|k, v| f(k, &mut *v)); - self.bird_small_states.retain(|k, v| f(k, &mut *v)); + self.bird_large_states.retain(|k, v| f(k, &mut *v)); self.fish_small_states.retain(|k, v| f(k, &mut *v)); self.biped_large_states.retain(|k, v| f(k, &mut *v)); self.biped_small_states.retain(|k, v| f(k, &mut *v)); @@ -241,7 +241,7 @@ impl FigureMgrStates { + self.fish_medium_states.len() + self.theropod_states.len() + self.dragon_states.len() - + self.bird_small_states.len() + + self.bird_large_states.len() + self.fish_small_states.len() + self.biped_large_states.len() + self.biped_small_states.len() @@ -291,7 +291,7 @@ impl FigureMgrStates { .filter(|(_, c)| c.visible()) .count() + self - .bird_small_states + .bird_large_states .iter() .filter(|(_, c)| c.visible()) .count() @@ -332,7 +332,7 @@ pub struct FigureMgr { quadruped_medium_model_cache: FigureModelCache, quadruped_low_model_cache: FigureModelCache, bird_medium_model_cache: FigureModelCache, - bird_small_model_cache: FigureModelCache, + bird_large_model_cache: FigureModelCache, dragon_model_cache: FigureModelCache, fish_medium_model_cache: FigureModelCache, fish_small_model_cache: FigureModelCache, @@ -354,7 +354,7 @@ impl FigureMgr { quadruped_medium_model_cache: FigureModelCache::new(), quadruped_low_model_cache: FigureModelCache::new(), bird_medium_model_cache: FigureModelCache::new(), - bird_small_model_cache: FigureModelCache::new(), + bird_large_model_cache: FigureModelCache::new(), dragon_model_cache: FigureModelCache::new(), fish_medium_model_cache: FigureModelCache::new(), fish_small_model_cache: FigureModelCache::new(), @@ -381,7 +381,7 @@ impl FigureMgr { .clean(&mut self.col_lights, tick); self.bird_medium_model_cache .clean(&mut self.col_lights, tick); - self.bird_small_model_cache + self.bird_large_model_cache .clean(&mut self.col_lights, tick); self.dragon_model_cache.clean(&mut self.col_lights, tick); self.fish_medium_model_cache @@ -3263,8 +3263,8 @@ impl FigureMgr { physics.ground_vel, ); }, - Body::BirdSmall(body) => { - let (model, skeleton_attr) = self.bird_small_model_cache.get_or_create_model( + Body::BirdLarge(body) => { + let (model, skeleton_attr) = self.bird_large_model_cache.get_or_create_model( renderer, &mut self.col_lights, *body, @@ -3277,14 +3277,14 @@ impl FigureMgr { let state = self .states - .bird_small_states + .bird_large_states .entry(entity) .or_insert_with(|| { - FigureState::new(renderer, BirdSmallSkeleton::default()) + FigureState::new(renderer, BirdLargeSkeleton::default()) }); // Average velocity relative to the current ground - let _rel_avg_vel = state.avg_vel - physics.ground_vel; + let rel_avg_vel = state.avg_vel - physics.ground_vel; let (character, last_character) = match (character, last_character) { (Some(c), Some(l)) => (c, l), @@ -3301,33 +3301,49 @@ impl FigureMgr { physics.in_liquid().is_some(), // In water ) { // Standing - (true, false, false) => anim::bird_small::IdleAnimation::update_skeleton( - &BirdSmallSkeleton::default(), + (true, false, false) => anim::bird_large::IdleAnimation::update_skeleton( + &BirdLargeSkeleton::default(), time, state.state_time, &mut state_animation_rate, skeleton_attr, ), // Running - (true, true, false) => anim::bird_small::RunAnimation::update_skeleton( - &BirdSmallSkeleton::default(), - (rel_vel.magnitude(), time), + (true, true, false) => anim::bird_large::RunAnimation::update_skeleton( + &BirdLargeSkeleton::default(), + ( + rel_vel, + // TODO: Update to use the quaternion. + ori * anim::vek::Vec3::::unit_y(), + state.last_ori * anim::vek::Vec3::::unit_y(), + time, + rel_avg_vel, + state.acc_vel, + ), state.state_time, &mut state_animation_rate, skeleton_attr, ), // In air - (false, _, false) => anim::bird_small::JumpAnimation::update_skeleton( - &BirdSmallSkeleton::default(), - (rel_vel.magnitude(), time), + (false, _, false) => anim::bird_large::FlyAnimation::update_skeleton( + &BirdLargeSkeleton::default(), + ( + rel_vel, + // TODO: Update to use the quaternion. + ori * anim::vek::Vec3::::unit_y(), + state.last_ori * anim::vek::Vec3::::unit_y(), + time, + rel_avg_vel, + state.acc_vel, + ), state.state_time, &mut state_animation_rate, skeleton_attr, ), // TODO! - _ => anim::bird_small::IdleAnimation::update_skeleton( - &BirdSmallSkeleton::default(), + _ => anim::bird_large::IdleAnimation::update_skeleton( + &BirdLargeSkeleton::default(), time, state.state_time, &mut state_animation_rate, @@ -4572,7 +4588,7 @@ impl FigureMgr { quadruped_medium_model_cache, quadruped_low_model_cache, bird_medium_model_cache, - bird_small_model_cache, + bird_large_model_cache, dragon_model_cache, fish_medium_model_cache, fish_small_model_cache, @@ -4591,7 +4607,7 @@ impl FigureMgr { fish_medium_states, theropod_states, dragon_states, - bird_small_states, + bird_large_states, fish_small_states, biped_large_states, biped_small_states, @@ -4738,14 +4754,14 @@ impl FigureMgr { ), ) }), - Body::BirdSmall(body) => bird_small_states + Body::BirdLarge(body) => bird_large_states .get(&entity) .filter(|state| filter_state(&*state)) .map(move |state| { ( state.locals(), state.bone_consts(), - bird_small_model_cache.get_model( + bird_large_model_cache.get_model( col_lights, *body, inventory, From e930b5ee18d1b86b585cddf01d9e7f58866f7c79 Mon Sep 17 00:00:00 2001 From: Snowram Date: Thu, 8 Apr 2021 01:38:03 +0200 Subject: [PATCH 02/10] Idle, run, vel/ori aware fly anims --- voxygen/anim/src/bird_large/fly.rs | 20 ++++++---- voxygen/anim/src/bird_large/idle.rs | 59 +++++++++++++++-------------- voxygen/anim/src/bird_large/mod.rs | 8 ++-- voxygen/anim/src/bird_large/run.rs | 57 +++++++++++++++++----------- voxygen/src/scene/figure/load.rs | 2 +- 5 files changed, 82 insertions(+), 64 deletions(-) diff --git a/voxygen/anim/src/bird_large/fly.rs b/voxygen/anim/src/bird_large/fly.rs index e5b1ec551c..37fa3b7068 100644 --- a/voxygen/anim/src/bird_large/fly.rs +++ b/voxygen/anim/src/bird_large/fly.rs @@ -56,7 +56,6 @@ impl Animation for FlyAnimation { } else { 0.0 } * 1.3; - let x_tilt = avg_vel.z.atan2(avg_vel.xy().magnitude()); next.chest.scale = Vec3::one() * s_a.scaler / 4.0; @@ -103,15 +102,18 @@ impl Animation for FlyAnimation { next.chest.position = Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + slow * 0.05) * s_a.scaler / 4.0; next.chest.orientation = - Quaternion::rotation_x(-0.5 + slow * 0.05) * Quaternion::rotation_y(tilt * 1.8); + Quaternion::rotation_x(-0.5 + slow * 0.05 + (0.8 * velocity.z / 80.0).min(0.8)) + * Quaternion::rotation_y(tilt * 1.8); next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_l.orientation = - Quaternion::rotation_y(0.1 + slow * 0.04) * Quaternion::rotation_x(0.6); + Quaternion::rotation_y(0.1 + slow * 0.04 + (0.8 * velocity.z / 80.0).min(0.8)) + * Quaternion::rotation_x(0.6); next.wing_in_r.orientation = - Quaternion::rotation_y(-0.1 + slow * -0.04) * Quaternion::rotation_x(0.6); + Quaternion::rotation_y(-0.1 + slow * -0.04 - (0.8 * velocity.z / 80.0).min(0.8)) + * Quaternion::rotation_x(0.6); next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); @@ -120,8 +122,10 @@ impl Animation for FlyAnimation { next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); - next.wing_out_l.orientation = Quaternion::rotation_y(0.1 + slow * 0.04); - next.wing_out_r.orientation = Quaternion::rotation_y(-0.1 + slow * -0.04); + next.wing_out_l.orientation = + Quaternion::rotation_y(0.1 + slow * 0.04 + (0.4 * velocity.z / 80.0).min(0.2)); + next.wing_out_r.orientation = + Quaternion::rotation_y(-0.1 + slow * -0.04 - (0.4 * velocity.z / 80.0).min(0.2)); next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); next.tail_front.orientation = @@ -131,9 +135,9 @@ impl Animation for FlyAnimation { Quaternion::rotation_x(-0.2 + slow * 0.08) * Quaternion::rotation_z(-tilt * 1.0); } - next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2 + 3.0); + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2); next.leg_l.orientation = Quaternion::rotation_x(-1.0 - flap1 * 0.1); - next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2 + 3.0); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2); next.leg_r.orientation = Quaternion::rotation_x(-1.0 - flap1 * 0.1); next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); diff --git a/voxygen/anim/src/bird_large/idle.rs b/voxygen/anim/src/bird_large/idle.rs index f25cfa7664..d7e96d74f2 100644 --- a/voxygen/anim/src/bird_large/idle.rs +++ b/voxygen/anim/src/bird_large/idle.rs @@ -2,6 +2,7 @@ use super::{ super::{vek::*, Animation}, BirdLargeSkeleton, SkeletonAttr, }; +use std::ops::Mul; pub struct IdleAnimation; @@ -15,43 +16,41 @@ impl Animation for IdleAnimation { #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_idle")] fn update_skeleton_inner( skeleton: &Self::Skeleton, - _global_time: Self::Dependency, + global_time: Self::Dependency, anim_time: f32, _rate: &mut f32, s_a: &SkeletonAttr, ) -> Self::Skeleton { let mut next = (*skeleton).clone(); - let fast = anim_time * 4.0; - - let freq = 8.0; - let off2 = -1.7; - let off3 = -2.0; - let off4 = -2.4; - let flap1 = 7.0 / 16.0 * (freq * anim_time).sin() - + 7.0 / 64.0 * (freq * 2.0 * anim_time).sin() - + 1.0 / 48.0 * (freq * 3.0 * anim_time).sin(); - let flap2 = 7.0 / 16.0 * (freq * anim_time + off2).sin() - + 7.0 / 64.0 * (freq * 2.0 * anim_time + off2).sin() - + 1.0 / 48.0 * (freq * 3.0 * anim_time + off2).sin(); - let flap3 = 7.0 / 16.0 * (freq * anim_time + off3).sin() - + 7.0 / 64.0 * (freq * 2.0 * anim_time + off3).sin() - + 1.0 / 48.0 * (freq * 3.0 * anim_time + off3).sin(); - let flap4 = 7.0 / 16.0 * (freq * anim_time + off4).sin() - + 7.0 / 64.0 * (freq * 2.0 * anim_time + off4).sin() - + 1.0 / 48.0 * (freq * 3.0 * anim_time + off4).sin(); + let duck_head_look = Vec2::new( + (global_time / 2.0 + anim_time / 8.0) + .floor() + .mul(7331.0) + .sin() + * 0.5, + (global_time / 2.0 + anim_time / 8.0) + .floor() + .mul(1337.0) + .sin() + * 0.25, + ); + let wave_slow_cos = (anim_time * 4.5).cos(); next.chest.scale = Vec3::one() * s_a.scaler / 4.0; - next.chest.position = Vec3::new(0.0, s_a.chest.0, s_a.chest.1) * s_a.scaler / 4.0; + next.chest.position = + Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + wave_slow_cos * 0.06) * s_a.scaler / 4.0; next.chest.orientation = Quaternion::rotation_x(0.0); next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); next.neck.orientation = Quaternion::rotation_x(0.0); next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); - next.head.orientation = Quaternion::rotation_x(0.0); + next.head.orientation = Quaternion::rotation_z(duck_head_look.x) + * Quaternion::rotation_x(-duck_head_look.y.abs() + wave_slow_cos * 0.01); next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); + next.beak.orientation = Quaternion::rotation_x(wave_slow_cos * -0.02 - 0.02); next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); next.tail_front.orientation = Quaternion::rotation_x(0.0); @@ -61,22 +60,24 @@ impl Animation for IdleAnimation { next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); - next.wing_in_l.orientation = Quaternion::rotation_y(0.0); - next.wing_in_r.orientation = Quaternion::rotation_y(0.0) * Quaternion::rotation_x(0.0); + next.wing_in_l.orientation = + Quaternion::rotation_y(-1.3 + wave_slow_cos * 0.06) * Quaternion::rotation_z(0.2); + next.wing_in_r.orientation = + Quaternion::rotation_y(1.3 - wave_slow_cos * 0.06) * Quaternion::rotation_z(-0.2); next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); - next.wing_mid_l.orientation = Quaternion::rotation_y(0.0); - next.wing_mid_r.orientation = Quaternion::rotation_y(0.0); + next.wing_mid_l.orientation = Quaternion::rotation_y(-0.1) * Quaternion::rotation_z(0.7); + next.wing_mid_r.orientation = Quaternion::rotation_y(0.1) * Quaternion::rotation_z(-0.7); next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); - next.wing_out_l.orientation = Quaternion::rotation_y(0.0); - next.wing_out_r.orientation = Quaternion::rotation_y(0.0); + next.wing_out_l.orientation = Quaternion::rotation_y(-0.2) * Quaternion::rotation_z(0.2); + next.wing_out_r.orientation = Quaternion::rotation_y(0.2) * Quaternion::rotation_z(-0.2); - next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2 + 3.0); + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2 - wave_slow_cos * 0.06); next.leg_l.orientation = Quaternion::rotation_x(0.0); - next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2 + 3.0); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2 - wave_slow_cos * 0.06); next.leg_r.orientation = Quaternion::rotation_x(0.0); next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); diff --git a/voxygen/anim/src/bird_large/mod.rs b/voxygen/anim/src/bird_large/mod.rs index a4c938bff3..d0a086bca4 100644 --- a/voxygen/anim/src/bird_large/mod.rs +++ b/voxygen/anim/src/bird_large/mod.rs @@ -134,13 +134,13 @@ impl<'a> From<&'a Body> for SkeletonAttr { use comp::bird_large::Species::*; Self { chest: match (body.species, body.body_type) { - (Phoenix, _) => (2.5, 8.5), + (Phoenix, _) => (2.5, 7.5), }, neck: match (body.species, body.body_type) { (Phoenix, _) => (0.5, 3.0), }, head: match (body.species, body.body_type) { - (Phoenix, _) => (2.0, 3.0), + (Phoenix, _) => (2.0, 2.0), }, beak: match (body.species, body.body_type) { (Phoenix, _) => (2.0, 1.0), @@ -152,7 +152,7 @@ impl<'a> From<&'a Body> for SkeletonAttr { (Phoenix, _) => (-3.0, -3.0), }, wing_in: match (body.species, body.body_type) { - (Phoenix, _) => (3.0, 2.5, 4.0), + (Phoenix, _) => (3.0, 2.5, 3.0), }, wing_mid: match (body.species, body.body_type) { (Phoenix, _) => (6.5, -1.0, 0.0), @@ -161,7 +161,7 @@ impl<'a> From<&'a Body> for SkeletonAttr { (Phoenix, _) => (0.5, -1.0, 0.0), }, leg: match (body.species, body.body_type) { - (Phoenix, _) => (2.5, -2.5, -6.5), + (Phoenix, _) => (2.5, -2.5, -3.5), }, foot: match (body.species, body.body_type) { (Phoenix, _) => (2.5, -16.5, -6.5), diff --git a/voxygen/anim/src/bird_large/run.rs b/voxygen/anim/src/bird_large/run.rs index a308ce2e4f..ab5fd3a4d9 100644 --- a/voxygen/anim/src/bird_large/run.rs +++ b/voxygen/anim/src/bird_large/run.rs @@ -28,7 +28,7 @@ impl Animation for RunAnimation { //let speednorm = speed / 13.0; let speednorm = (speed / 13.0).powf(0.25); - let speedmult = 1.0; + let speedmult = 2.0; let lab: f32 = 0.6; //6 let short = ((1.0 @@ -60,7 +60,6 @@ impl Animation for RunAnimation { } else { 0.0 } * 1.3; - let x_tilt = avg_vel.z.atan2(avg_vel.xy().magnitude()) * speednorm; next.head.scale = Vec3::one() * 1.02; next.neck.scale = Vec3::one() * 0.98; @@ -71,52 +70,66 @@ impl Animation for RunAnimation { next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); next.head.orientation = Quaternion::rotation_x(-0.1 * speednorm + short * -0.05) - * Quaternion::rotation_y(tilt * 0.8) - * Quaternion::rotation_z(shortalt * -0.2 - tilt * 2.5); + * Quaternion::rotation_y(tilt * 0.2) + * Quaternion::rotation_z(shortalt * -0.05 - tilt * 1.5); next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); - next.beak.orientation = Quaternion::rotation_x(short * -0.03); + next.beak.orientation = Quaternion::rotation_x(short * -0.02 - 0.02); next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); next.neck.orientation = Quaternion::rotation_x(-0.1 * speednorm + short * -0.04) - * Quaternion::rotation_y(tilt * 0.3) - * Quaternion::rotation_z(shortalt * -0.1 - tilt * 3.2); + * Quaternion::rotation_y(tilt * 0.1) + * Quaternion::rotation_z(shortalt * -0.1 - tilt * 0.5); next.chest.position = Vec3::new( 0.0, s_a.chest.0, - s_a.chest.1 + short * 0.5 + x_tilt * 10.0 + 0.5 * speednorm, + s_a.chest.1 + short * 0.5 + 0.5 * speednorm, ) * s_a.scaler / 4.0; - next.chest.orientation = Quaternion::rotation_x(short * 0.07 + x_tilt) + next.chest.orientation = Quaternion::rotation_x(short * 0.07) * Quaternion::rotation_y(tilt * 0.8) - * Quaternion::rotation_z(shortalt * 0.15 + tilt * -1.5); + * Quaternion::rotation_z(shortalt * 0.10); next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); - next.tail_front.orientation = Quaternion::rotation_x(0.1 + short * -0.02) - * Quaternion::rotation_z(shortalt * -0.1 + tilt * 1.0); + next.tail_front.orientation = Quaternion::rotation_x(short * -0.02); next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); - next.tail_rear.orientation = Quaternion::rotation_x(0.2 + short * -0.1) - * Quaternion::rotation_z(shortalt * -0.2 + tilt * 1.4); + next.tail_rear.orientation = Quaternion::rotation_x(short * -0.1); + + next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + + next.wing_in_l.orientation = Quaternion::rotation_y(-1.3) * Quaternion::rotation_z(0.2); + next.wing_in_r.orientation = Quaternion::rotation_y(1.3) * Quaternion::rotation_z(-0.2); + + next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_l.orientation = Quaternion::rotation_y(-0.1) * Quaternion::rotation_z(0.7); + next.wing_mid_r.orientation = Quaternion::rotation_y(0.1) * Quaternion::rotation_z(-0.7); + + next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_l.orientation = + Quaternion::rotation_y(-0.2 + short * 0.05) * Quaternion::rotation_z(0.2); + next.wing_out_r.orientation = + Quaternion::rotation_y(0.2 + short * -0.05) * Quaternion::rotation_z(-0.2); next.leg_l.position = Vec3::new( -s_a.leg.0 + speednorm * 1.5, - s_a.leg.1 + foot1b * -1.3, - s_a.leg.2 + foot1a * 1.0, + s_a.leg.1 + foot1b * -2.3, + s_a.leg.2, ); next.leg_l.orientation = Quaternion::rotation_x(-0.2 * speednorm + foot1a * 0.15) - * Quaternion::rotation_y(tilt * 0.5) - * Quaternion::rotation_z(foot1a * -0.3 + tilt * -0.5); + * Quaternion::rotation_y(tilt * 0.5); next.leg_r.position = Vec3::new( s_a.leg.0 + speednorm * -1.5, - s_a.leg.1 + foot2b * -1.3, - s_a.leg.2 + foot2a * 1.0, + s_a.leg.1 + foot2b * -2.3, + s_a.leg.2, ); next.leg_r.orientation = Quaternion::rotation_x(-0.2 * speednorm + foot2a * 0.15) - * Quaternion::rotation_y(tilt * 0.5) - * Quaternion::rotation_z(foot2a * 0.3 + tilt * -0.5); + * Quaternion::rotation_y(tilt * 0.5); next.foot_l.position = Vec3::new( -s_a.foot.0, diff --git a/voxygen/src/scene/figure/load.rs b/voxygen/src/scene/figure/load.rs index bb612c8ef2..e2a4ddeb10 100644 --- a/voxygen/src/scene/figure/load.rs +++ b/voxygen/src/scene/figure/load.rs @@ -3382,7 +3382,7 @@ impl BirdLargeLateralSpec { return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); }, }; - let lateral = graceful_load_segment(&spec.wing_in_l.lateral.0); + let lateral = graceful_load_segment(&spec.wing_mid_r.lateral.0); (lateral, Vec3::from(spec.wing_mid_r.offset)) } From e58e56cb0791bab9f3f7e2e68a874f5424ab108d Mon Sep 17 00:00:00 2001 From: Snowram Date: Sun, 11 Apr 2021 12:54:26 +0200 Subject: [PATCH 03/10] Cockatrice bird_large conversion --- assets/common/npc_names.ron | 8 +- .../voxel/bird_large_central_manifest.ron | 52 +++++++++ .../voxel/bird_large_lateral_manifest.ron | 84 +++++++++++++++ .../voxel/bird_medium_central_manifest.ron | 28 ----- .../voxel/bird_medium_lateral_manifest.ron | 36 ------- .../voxel/npc/cockatrice/male/beak.vox | Bin 0 -> 1232 bytes .../voxel/npc/cockatrice/male/chest.vox | Bin 0 -> 6840 bytes .../voxel/npc/cockatrice/male/foot_l.vox | Bin 0 -> 1588 bytes .../voxel/npc/cockatrice/male/foot_r.vox | Bin 0 -> 1588 bytes .../voxel/npc/cockatrice/male/head.vox | Bin 1952 -> 3112 bytes .../voxel/npc/cockatrice/male/leg_l.vox | Bin 0 -> 2224 bytes .../voxel/npc/cockatrice/male/leg_r.vox | Bin 1364 -> 2224 bytes .../voxel/npc/cockatrice/male/neck.vox | Bin 0 -> 3424 bytes .../voxel/npc/cockatrice/male/tail.vox | Bin 1216 -> 0 bytes .../voxel/npc/cockatrice/male/tail_front.vox | Bin 0 -> 2056 bytes .../voxel/npc/cockatrice/male/tail_rear.vox | Bin 0 -> 1772 bytes .../voxel/npc/cockatrice/male/torso.vox | Bin 3048 -> 0 bytes .../voxel/npc/cockatrice/male/wing_in_l.vox | Bin 0 -> 1412 bytes .../voxel/npc/cockatrice/male/wing_in_r.vox | Bin 0 -> 1412 bytes .../voxel/npc/cockatrice/male/wing_mid_l.vox | Bin 0 -> 1408 bytes .../voxel/npc/cockatrice/male/wing_mid_r.vox | Bin 0 -> 1408 bytes .../voxel/npc/cockatrice/male/wing_out_l.vox | Bin 0 -> 1624 bytes .../voxel/npc/cockatrice/male/wing_out_r.vox | Bin 0 -> 1624 bytes .../voxel/npc/cockatrice/male/wing_r.vox | Bin 1212 -> 0 bytes common/src/comp/body.rs | 9 +- common/src/comp/body/bird_large.rs | 8 +- common/src/comp/body/bird_medium.rs | 6 +- voxygen/anim/src/bird_large/feed.rs | 101 ++++++++++++++++++ voxygen/anim/src/bird_large/fly.rs | 65 +++++++---- voxygen/anim/src/bird_large/idle.rs | 12 ++- voxygen/anim/src/bird_large/mod.rs | 35 +++++- voxygen/anim/src/bird_large/run.rs | 29 ++--- voxygen/anim/src/bird_medium/mod.rs | 6 -- voxygen/src/scene/figure/mod.rs | 17 ++- world/src/layer/wildlife.rs | 4 +- 35 files changed, 368 insertions(+), 132 deletions(-) create mode 100644 assets/voxygen/voxel/npc/cockatrice/male/beak.vox create mode 100644 assets/voxygen/voxel/npc/cockatrice/male/chest.vox create mode 100644 assets/voxygen/voxel/npc/cockatrice/male/foot_l.vox create mode 100644 assets/voxygen/voxel/npc/cockatrice/male/foot_r.vox create mode 100644 assets/voxygen/voxel/npc/cockatrice/male/leg_l.vox create mode 100644 assets/voxygen/voxel/npc/cockatrice/male/neck.vox delete mode 100644 assets/voxygen/voxel/npc/cockatrice/male/tail.vox create mode 100644 assets/voxygen/voxel/npc/cockatrice/male/tail_front.vox create mode 100644 assets/voxygen/voxel/npc/cockatrice/male/tail_rear.vox delete mode 100644 assets/voxygen/voxel/npc/cockatrice/male/torso.vox create mode 100644 assets/voxygen/voxel/npc/cockatrice/male/wing_in_l.vox create mode 100644 assets/voxygen/voxel/npc/cockatrice/male/wing_in_r.vox create mode 100644 assets/voxygen/voxel/npc/cockatrice/male/wing_mid_l.vox create mode 100644 assets/voxygen/voxel/npc/cockatrice/male/wing_mid_r.vox create mode 100644 assets/voxygen/voxel/npc/cockatrice/male/wing_out_l.vox create mode 100644 assets/voxygen/voxel/npc/cockatrice/male/wing_out_r.vox delete mode 100644 assets/voxygen/voxel/npc/cockatrice/male/wing_r.vox create mode 100644 voxygen/anim/src/bird_large/feed.rs diff --git a/assets/common/npc_names.ron b/assets/common/npc_names.ron index 71d0a83499..2729cacb40 100644 --- a/assets/common/npc_names.ron +++ b/assets/common/npc_names.ron @@ -804,10 +804,6 @@ keyword: "parrot", generic: "Parrot" ), - cockatrice: ( - keyword: "cockatrice", - generic: "Cockatrice" - ) ) ), biped_large: ( @@ -1080,6 +1076,10 @@ keyword: "phoenix", generic: "Phoenix" ), + cockatrice: ( + keyword: "cockatrice", + generic: "Cockatrice" + ), ) ), quadruped_low: ( diff --git a/assets/voxygen/voxel/bird_large_central_manifest.ron b/assets/voxygen/voxel/bird_large_central_manifest.ron index 2671ef6b84..c09bbae0b3 100644 --- a/assets/voxygen/voxel/bird_large_central_manifest.ron +++ b/assets/voxygen/voxel/bird_large_central_manifest.ron @@ -51,4 +51,56 @@ central: ("npc.phoenix.male.tail_rear"), ) ), + (Cockatrice, Male): ( + head: ( + offset: (-4.5, -6.0, -3.5), + central: ("npc.cockatrice.male.head"), + ), + beak: ( + offset: (-1.5, 0.0, -1.5), + central: ("npc.cockatrice.male.beak"), + ), + neck: ( + offset: (-3.5, -0.0, -6.0), + central: ("npc.cockatrice.male.neck"), + ), + chest: ( + offset: (-5.5, -10.0, -9.5), + central: ("npc.cockatrice.male.chest"), + ), + tail_front: ( + offset: (-2.5, -9.0, -6.0), + central: ("npc.cockatrice.male.tail_front"), + ), + tail_rear: ( + offset: (-2.5, -13.0, -3.0), + central: ("npc.cockatrice.male.tail_rear"), + ) + ), + (Cockatrice, Female): ( + head: ( + offset: (-4.5, -6.0, -3.5), + central: ("npc.cockatrice.male.head"), + ), + beak: ( + offset: (-1.5, 0.0, -1.5), + central: ("npc.cockatrice.male.beak"), + ), + neck: ( + offset: (-3.5, -0.0, -6.0), + central: ("npc.cockatrice.male.neck"), + ), + chest: ( + offset: (-5.5, -10.0, -9.5), + central: ("npc.cockatrice.male.chest"), + ), + tail_front: ( + offset: (-2.5, -9.0, -6.0), + central: ("npc.cockatrice.male.tail_front"), + ), + tail_rear: ( + offset: (-2.5, -13.0, -3.0), + central: ("npc.cockatrice.male.tail_rear"), + ) + ), }) diff --git a/assets/voxygen/voxel/bird_large_lateral_manifest.ron b/assets/voxygen/voxel/bird_large_lateral_manifest.ron index 1f8e4ab12d..b470d45287 100644 --- a/assets/voxygen/voxel/bird_large_lateral_manifest.ron +++ b/assets/voxygen/voxel/bird_large_lateral_manifest.ron @@ -83,4 +83,88 @@ lateral: ("npc.phoenix.male.foot_r"), ) ), + (Cockatrice, Male): ( + wing_in_l: ( + offset: (-7.0, -8.0, -2.0), + lateral: ("npc.cockatrice.male.wing_in_l"), + ), + wing_in_r: ( + offset: (0.0, -8.0, -2.0), + lateral: ("npc.cockatrice.male.wing_in_r"), + ), + wing_mid_l: ( + offset: (-5.0, -9.0, -2.0), + lateral: ("npc.cockatrice.male.wing_mid_l"), + ), + wing_mid_r: ( + offset: (0.0, -9.0, -2.0), + lateral: ("npc.cockatrice.male.wing_mid_r"), + ), + wing_out_l: ( + offset: (-10.0, -11.0, -2.0), + lateral: ("npc.cockatrice.male.wing_out_l"), + ), + wing_out_r: ( + offset: (0.0, -11.0, -2.0), + lateral: ("npc.cockatrice.male.wing_out_r"), + ), + leg_l: ( + offset: (-3.0, -4.5, -4.0), + lateral: ("npc.cockatrice.male.leg_l"), + ), + leg_r: ( + offset: (-3.0, -4.5, -4.0), + lateral: ("npc.cockatrice.male.leg_r"), + ), + foot_l: ( + offset: (-3.5, -5.5, -10.0), + lateral: ("npc.cockatrice.male.foot_l"), + ), + foot_r: ( + offset: (-3.5, -5.5, -10.0), + lateral: ("npc.cockatrice.male.foot_r"), + ) + ), + (Cockatrice, Female): ( + wing_in_l: ( + offset: (-7.0, -8.0, -2.0), + lateral: ("npc.cockatrice.male.wing_in_l"), + ), + wing_in_r: ( + offset: (0.0, -8.0, -2.0), + lateral: ("npc.cockatrice.male.wing_in_r"), + ), + wing_mid_l: ( + offset: (-5.0, -9.0, -2.0), + lateral: ("npc.cockatrice.male.wing_mid_l"), + ), + wing_mid_r: ( + offset: (0.0, -9.0, -2.0), + lateral: ("npc.cockatrice.male.wing_mid_r"), + ), + wing_out_l: ( + offset: (-10.0, -11.0, -2.0), + lateral: ("npc.cockatrice.male.wing_out_l"), + ), + wing_out_r: ( + offset: (0.0, -11.0, -2.0), + lateral: ("npc.cockatrice.male.wing_out_r"), + ), + leg_l: ( + offset: (-3.0, -4.5, -4.0), + lateral: ("npc.cockatrice.male.leg_l"), + ), + leg_r: ( + offset: (-3.0, -4.5, -4.0), + lateral: ("npc.cockatrice.male.leg_r"), + ), + foot_l: ( + offset: (-3.5, -5.5, -10.0), + lateral: ("npc.cockatrice.male.foot_l"), + ), + foot_r: ( + offset: (-3.5, -5.5, -10.0), + lateral: ("npc.cockatrice.male.foot_r"), + ) + ), }) \ No newline at end of file diff --git a/assets/voxygen/voxel/bird_medium_central_manifest.ron b/assets/voxygen/voxel/bird_medium_central_manifest.ron index 57d03ecfe1..d24a0be621 100644 --- a/assets/voxygen/voxel/bird_medium_central_manifest.ron +++ b/assets/voxygen/voxel/bird_medium_central_manifest.ron @@ -195,32 +195,4 @@ central: ("npc.parrot.male.tail"), ) ), - (Cockatrice, Male): ( - head: ( - offset: (-2.5, 0.0, -8.0), - central: ("npc.cockatrice.male.head"), - ), - torso: ( - offset: (-3.5, -6.5, -7.5), - central: ("npc.cockatrice.male.torso"), - ), - tail: ( - offset: (-1.5, -3.5, -4.0), - central: ("npc.cockatrice.male.tail"), - ) - ), - (Cockatrice, Female): ( - head: ( - offset: (-2.5, 0.0, -8.0), - central: ("npc.cockatrice.male.head"), - ), - torso: ( - offset: (-3.5, -6.5, -7.5), - central: ("npc.cockatrice.male.torso"), - ), - tail: ( - offset: (-1.5, -3.5, -4.0), - central: ("npc.cockatrice.male.tail"), - ) - ), }) \ No newline at end of file diff --git a/assets/voxygen/voxel/bird_medium_lateral_manifest.ron b/assets/voxygen/voxel/bird_medium_lateral_manifest.ron index 2bcb349b14..95c4027cc7 100644 --- a/assets/voxygen/voxel/bird_medium_lateral_manifest.ron +++ b/assets/voxygen/voxel/bird_medium_lateral_manifest.ron @@ -251,40 +251,4 @@ lateral: ("npc.parrot.male.leg_r"), ) ), - (Cockatrice, Male): ( - wing_l: ( - offset: (-2.0, -3.0, -9.0), - lateral: ("npc.cockatrice.male.wing_r"), - ), - wing_r: ( - offset: (-2.0, -3.0, -9.0), - lateral: ("npc.cockatrice.male.wing_r"), - ), - foot_l: ( - offset: (-2.5, 0.0, -12.0), - lateral: ("npc.cockatrice.male.leg_r"), - ), - foot_r: ( - offset: (-2.5, 0.0, -12.0), - lateral: ("npc.cockatrice.male.leg_r"), - ) - ), - (Cockatrice, Female): ( - wing_l: ( - offset: (-2.0, -3.0, -9.0), - lateral: ("npc.cockatrice.male.wing_r"), - ), - wing_r: ( - offset: (-2.0, -3.0, -9.0), - lateral: ("npc.cockatrice.male.wing_r"), - ), - foot_l: ( - offset: (-2.5, 0.0, -12.0), - lateral: ("npc.cockatrice.male.leg_r"), - ), - foot_r: ( - offset: (-2.5, 0.0, -12.0), - lateral: ("npc.cockatrice.male.leg_r"), - ) - ), }) \ No newline at end of file diff --git a/assets/voxygen/voxel/npc/cockatrice/male/beak.vox b/assets/voxygen/voxel/npc/cockatrice/male/beak.vox new file mode 100644 index 0000000000000000000000000000000000000000..33b8eacaa0bd9aedcae0a7e4a171501de4affff2 GIT binary patch literal 1232 zcmeH`KTE?<6vgkoX>|!ALOKE`j5Gln<(pqX4cb9@VNt_&9TAUOK2u0}T zoJk5UPWcY~4w>Ap=ZSVH6(ZEh3%{I`yn7EX|Jp~Lje8L;GvXb z%)uMYbBjnC48$l)g#}n3FRkEUuS(drsd~MxS$VTy}3EBJL`Jmt?1+Ssx}=*FE)#xy$_e9 zPuDbe&g7Q8f|g5L>K2yOEf%J|x96na^Urok4)-S7WRKOFuwc!nqV?)Sd`?vLNb;NQ=6n$)M02ITal6>ZR`vkv?f zTj!J7d=j6}>hsA^iTOzb8!b6MNs>!^)bUFtR*6+&sN%S zf8e9CGBy~<#+V!SMt&zQ%Sm)Ot1c%)#g->EY_$0DEROH^sNiA0%&xQFf zoIi1J{2;zma+O>q2L_HG#Fs{{k!$3@!106l(#dslog5fAejM>Zd=Li%$1f{$Ux|T% z7CeE6-wu>s4-sI)3qT7cV#Uay4-LCCar#xk;3Z zqK5BL?twig`7D=`!dlDO@fVYO0WrC#32QDl#}1fXu8Wg* zxd<&Nc>7}YV>W)0WD3qR?Gq0P=_FvXoacMx65TgOJ@$CV4YfGft_fjW*&?` zs2g$;1H1H=UvXrP7Mes z7&rtZd3M0(f69Li{3Pe&PtW*gdD7#`i#|Ins`sLL5moLj<$CLrtKLHOI-A}W(`zyE zpkUw-ruT%&^*%dwu9xC^6;Ny==GcKIRO+VC%N)u*Nl$D6#WvK@AeT^KO2$w6q9v6* zpf?4>cl=0Y59p=BRG9-TmD;K|0mU|O>>!s=v8P7gMqjiv=FpV;RoMf2cL-uhzQWY$ z+m$`Qxq)>05;N5C13SC22bk~B+YHN~?_j>5gLOb10+L)s=aspxdMhBHD|2001MDD| z`18iK*{A{8Sf`C~!8g_klKi~u)3S31gMfm8LqHz+lRj~GvkS=cXW$S={LD_^37}x$ zhyR2R21SfLV(byS6t%l3Q8o66u}AD4YPZ?!zL>E`>?Ww1u}6$OLY`P+Pfnd1&j|WB zeNaUWoOnPItJF=C;xLC>HkD82=?Xh33Dn3CG0ZyL`D`hwycIJv-2w4zfro)gR! z3JAWUR+%dVb`q=T)acvj+vtlep!kM*iamAC4VKQib-M~EzTrCrB-BprpmuO> zP(TLrMGf`AI*?DO`18v6D{}?GR@7E{Lg430uZ?T7F~^N@Q3b>HjT-DEKkvpK;R(M} zvvVF`;Je*|SRVP4Js)p@<7sj1kmJuGj`-Q0c?+;dh=L#f6Z-&;zaoBfSk$jYo#%+N zNBqgaiG})oHfN7G?+@qw;p`E=8|vn_K+~K(;@8=!LyP-OP(=-OG$6@W{3-l42esjs z5`GnI#Wv!O9cU7-*i$k-Ea~hK=7+7=296(SLdBjcbEtk1IfG&wICelD`4jo$P4fBE zE z^27s8bn;-Y#AR{%cG`+(z6^<*C;`yhF*B;TIN z^CEd(9^ zRpyN!XhOxF8nunNq6MVU7d2uIft_e=$vcJGp8Q^54%h+;)|mrDXTE49KM#5idZIGu z35M;1^FkAUUg@<`1F|wlu$5kDkW1`&qtC{8sGtf4zA+xL+?dl&4HOViFmMRSopDaA z1vmtV)Hf=XYg^xlR_^zDH7HiEV)d>@&MNnJ<$kVw9}oxi%J)I#`=D|^SAIKG?qQCL z%6B7q)WCy0n&_O=WaYY5?p=C=f#XNkYi8woQB&ztE6;c4gKgLjQR#y}H)^0X#zM7G zgL4OIGMCVa?jx(RZd_b`cd{o}`m3xz#mGfR1Yh^yDqDIWI zW2F!Fyiv0;)<(@n4Yq+34@hG3Ua!l}`ho16`(C*=%o#uM6P6S6Ke2YDy{NPo(e_5Q zT?`d#+*|Ex*7zOQxCZ#%UJTp4vG$GUXXAHUW9>Nx+ad6mY<#10Oma*%*1Yj zzn>f5T;yxxw^?Jo8H1QZ5KEXFJz8VE>5nSDwZ?n2@g8lwM;qVa)O2b(HJx#=4cj5` z6Xwx)7E*)CXnb2yGnhYqK;q9U=dv=Nl^RsRR{B6d5}!BvZ1mX}Yoi7=Vh(|wuO6lq1_ZQn4C)ZrOY5vFxkkUnai~KOpF77w z>78$q&K_obaO{BWo#zWZ2K@(f8jOt^IPpLe=9QY28dN~B4fVD2ZlTZGd9RS)=&{iQ zY$J~!#Fw3O+dJQ<9J_NYwi64qoLC!BFz~~k#w(p+8Ly-a)_A-aDuON8hbq2}7X<_H z@kV{HzT=I=U~LC$J9y6zo)v@NmiXW~*sH|q;P=7ceNGI!w849u;~G7%y$$aF!Si`= z{}b;V--)AvzYLBUgEb=uHt2%}x~z=5l3N)U+fYYi9oJ=}=SI(sxr2e@$2Qn+%zY;g z27dS_&I|0&d1XzYfUK|8R?d5UQ818L-)LZ?lGwRk3@vqK?@(7)?#=bZ(bCqNG-8by z>S*cIbz*(xdxTggHdeknh@pl$T2|^-`mcZjL+N?^x{Uf6k}7xtkQ8p!z-|LcU~z#$;dum16G|N2{#{LlY?e($~a9$%LK?EP{4 z`>%K2d7R(>=HsmoULIfn&37LE^qYVALa&cL`sne=C!ah%{q)nv#~*+Eh5Uyfe)#yo z2X8&zfB*e2vn{FnF7Yyab)r^m1U zb^3$H9euy?&fAY~eEn;WufFY%uioGOeeKVFG(G$KvX>kuk!XwJdCdYY=z0HAGX{;=4JggK6 z2qIi(oibo!mH#3Cf~2xrzdi7pa6*K$a^c4}^UcoezV|{79vyCfAfmy3{~?67jVShy z?zeM-6+n(#;PCNL|A#|>^%4l$!~v#Slp61;Q6QcwhB{yd-zX|n3_jE$C#VIgHBgj5 z4MTBQZ3PXqz`Oz;l#p8q4@gNhLkU<5Il#pRas@btd$?M}HL7kX>MB+j)jQnB4H@bU zHFHr6In*Jy06)NkIw(P{z#KrpO3Ve6wFQKuDn|jF)dSI>8n12x2evj(6g`b*SH)DatEz*$4t3?Eecq-UR=>bg#w^1a3Ce7;tnPN(U~NheJvleOBW_saV}A0J;h z=iA*I=}qry`gHqRik(h+esk+=@5dLEzPzRj`|{kqUN`OS-by>&kao7Z=X>8hqx8G~ l>sv~*_kW)AH_!CHXY=3aWxTW1YIXkIFJu0{o|*pvegW|fA0Ge! literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/cockatrice/male/foot_r.vox b/assets/voxygen/voxel/npc/cockatrice/male/foot_r.vox new file mode 100644 index 0000000000000000000000000000000000000000..bfe29b9de51c2ca589ebce0a6034a1044b5331eb GIT binary patch literal 1588 zcmeH{v1?ON6vodz=iGDOt3gCarXr<67KK8=O@}@x5*|@rZc=I&cb9@VX`CEfEKZ69 z1SxcL&KPiU%KwmmK{B~rzgw_Ns1Tt}Uik5y^PPLYcas-#@c3}+BN6rYdXFHqWkj)e zwBO1NRsgx)0EbVGdOsWrtXDvAOdMdUMX7O~8V2I2Qm6-H@Qorv#o$9ddV?ao=0H&b zH3~KKR8XG_IR~&QgxBN!+qR9q0TTf z7sb#+J$ef813ajQ8k7p;00UGa7g*K}ARJja3ec<`7>^_XmdGMeM24DGftQgu9WrYa zpsYg?twa^RhwtHg_#VE8XOY>4!UoTVEQUNAX;!sHIN%updqY$xYiI;0V2>JTRs&LC zUf@{;=0(N6z#KJTLk%=*09xXFiFt`RYC{b)Ybcw+!)lN8{MA!k*J)k8w^}K$pWSY{ zIlh@6Q3 literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/cockatrice/male/head.vox b/assets/voxygen/voxel/npc/cockatrice/male/head.vox index 461f5c4ead6140a9da80161135b9e55a4a46bc00..721b81b168c7939333c8553764619d0f07fef0fc 100644 GIT binary patch literal 3112 zcmeH|zl$AJ6vyYDU-!z_uO;8-|snd_s&Om-~6Fznh&qv zI3zT$_f2!_#=Z9mo1d27Ti^n?`^miU=4oo0lx^)@Oj>+j+H@qPSxXnAX4}=J z9(%}ITG@|<90&+-#$gY*N>m^;$pjMV$XRow&rdfBqw}aH$R%9Q8Gn{=w zZnVB{wFdR0bLu_)2&heafgY$E-{V2ARI=hfL z&w2JDmpsxa3p)z8ac^{R5AAJOMA;l0p5YnhqsyB0nQ zeH8j_)U)h6UhCtbR}VAq3h&V9@H29)^&RVda_Snr7S49~S?BcFcLGh$xnAd#S#r)! zcr0iZJauYl??>J{a~fj+>G1#v94&(a+WK5Za}Z_B)=uUv&v|tADDa+Gv8eCz7|T2f zh-D#`DS0<|+j3?%d2X4Fj@c}ucX?D0>=B^Mk^{{|p!niIgBb!26h01wyZ{NV8EBuU z#92!LD6ndI2P-MOm5onKr8z#c(040A;-KM zGkVO)!Ha-@b~Fa(R?nhiKHvt&V(s7^I?q?)st0cIj<58%E+JBU>~ z2#8fjL_a-E734Q)Na!ox4oyYy3^F_t$Z$$^E~uA4b|T35VFi8gF#I;WHFz1?tp(27 z(vKT_HF|Fy>kD%GIP#j~{a^t$o{?+);Jz3f&y=XuJ7vA@*Cl!AuHFLE3TD87l);3* zr5!BsoQMsvsgZu9OfQ+U>f6RY=EFH(s4di$)!>pR*E}gvi6-0=8dH<=RDi<6LC`w^ zQF{b}{s;)>LF!D<&!C?|6k_JPO!b}*?s5>!w}2c5SR*?3RqASgW3BM_3PV50q2wsJ zf%km7@D~vDLx65{akro?sy;kYVCML-g>Nf-Lt}v4qhpDVCAHC0QNJG$%zrSaLKI>! z;{u#}KtV8rQs>b)6PUGp6MP?_Fb54N%uoZ=xx0YZxUyrd`q0{hvtD69r^a|TJoaGT zoc$F%^SO;}s+jI~^F&i3CH1l_+cV*BD=D|U#cRvvB=BWP{xeF5z{vQSS z_=AJ%js5li{^re_^ULD*zMt#Q)4jd$;oqTFUzau|9t=c3+wzgyfJ^Xx0-)?=fZqzdwc%kt;?I- zAHQtoU%qZ$xUbf|ws cJ=M~%EdfixqR_i?suOz^WXb_{kEA;zyIet|3=gQdtQD< WUxjB+o;-Qh_g7K>zphztfIk6YFpT#A literal 1952 zcmd6myQ>{l6o=Pqzvi5K4TKbPrih+G8^OjxgqcfD2y(e3f(a6%V4cQ7tZY*VAtDF~ zX>6GD6QDJ7kC!Y%JVF!@Fh2}DSWD;fA$kG=Q$t=W&yKfV6;JrTKl;qVGZ?i7)$ zhc_;*mw+$99-M*eU)(tS%5nIffJNR|eQUC`KSiWIJZ&t+Xk~x03bDr+tQOXjbqlM> z){J%JbUlToAvA3*6K5SQ8Tja0H!h_&DI1kCxPqVu3ZZ%9ntRtooCcHBJ6NHPqZ5$U zqh%;iiJdw0D3*Ff*DP%uU(i=&Ew!vFebJ+KqiGwAK2$5V^a>t%YiKLlg0)6nW4=SI zMo(DKgrD&t*eA^^`T);^JN-NBEpOFwZ{U`PAhuR^n#;~XQaMntRH$2qz?Udnjk2*| zKOl(3$vHW22x74^sIh-W4r1|J-C3w%mFv@wV~B6T4`vti;~+Pn%PtE-1^AvH%aZ8riNBStD)5% zOEk1vZ{QHb;_V&v#y^t7#u6K~jCEkOo3wP3frIAmz;d^7tS0=_^^VoR4}7e8M?HKn z>jrC_SLN)MGcMf23?=WZ*<{Tbi%B>?t3G)wcXqk!!nxyDY)YdnZz88jxoei4d&kX=(4T^>+-PS`yZiGm@1L2kp8Hqx{`WHfdP`nOV|(_+c|UzC z^FQk!+>!aw-51ASx4$=K{=+}_=D+6iue2v`O3$a!$IV+m&)05#H-B^c{(N$B^1toJ S@$vDze5dhGOHE$?f;~&m zkj{F`1skjV4*MOpvKxQ1XqqL$3bFDJhq*IzUY_%u=U)EzKDz(bDfm)yeeSNtSeYHVhe)tgB+uXU4RoRucUo2GTcS5h}g_S2s;4J;M&P27l$CgJ$S zDusHbb_bUgYYsM>v2NX?t{T2+Q7|R>ql?+*D&!$oM03$KsNrF2l+r!f*X%}oL|J3@ zWI~@#ek_Ij(bT}?%DFZTsEXJ$lm4xVc^9WvBz=@E)3&(GPtGJJy}N@Wbjz4L82o_+ z4$`J+H?vi}cD2$iWHYcT+*sAks&^}OwGv>-RqKQ8dgN?*2s zvyuyrEK84q9%{fFdRV;0f>vuD)7KbZXQy=AiDY z7_n&Fr*_etqd6na98Q^o(X+bXJKu$}Cb^OtNxq;?grha$(2-Yqh7UuN#`~hPZxisq zLsqzjGoQMrCY(BRbZ2hN$LZIEdUK{G%#(fAHHk@j%w(qU9O&HfbMiWzxT6aJug>?L z+16lw_%7V>y>QNb>e2aDNc4~2zjIsu`tBSa9**nU?>jqV{{GnCAFm$0J$4_xJ8nL` zH$Hp%^_5&FCnw|V>};H$pO4ei(<}GK$H(K-N4w+b=;+Gbb#YDqUmiYuZl2%Xd2M{V z|MK|xtyjj~-QDr!8{4Jt7a literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/cockatrice/male/leg_r.vox b/assets/voxygen/voxel/npc/cockatrice/male/leg_r.vox index 22e834e26761d0f6fd1dd2bc1906e87e91161bae..f494e9eecb50a75488a488cb1248a6fac1b1b753 100644 GIT binary patch literal 2224 zcmeH}y=#_d6vof@{RlxyA%W)u5er!?2tqeID2Rlhf`V9=?nMy9Nz$o)zG1)qzieQvg}+{&a8aeh~1PRK!#kSDDphHX}9}gIHY_a~U0ka}Jp-WOg`0MorGi z8IF`ii~6D-nCRr9qpD7(I=ku;sw2%tHKh+4ahNU%i|e9Jt}Fd?3Xra|1bXObu)_fo zNb0PqivdscBp*oOU-$qodID|KH~PUU#%wmb*eGhVs*MJ_8uQsCW|NvRADW?oG;*C> zH|Dd&i322%AV2Vg9RzU#N%X`!@qq&*nv`)*=)jA7IMJk$-|&GQ1hEd}4fF#lKAHI9 z;&BYy; zu{gR@>(mklNX`=Ifky;$qr(~aBzmGJIvgO0rDD>;r|4PeoD-HtenW5cZS*D9f%<`Z zppwj;+;>i@$z3ORopTVYCG#fpCi{_dTCyMLj_&Aia86hvd2gr(P8^ZUo7^+Lz!Dt} zI9FQHx02n(yYPm!Q4btMOXj3*pbs!DcDH!;#m?5~a9A-*F$-r9Ud&SLDH`w~C(vNc z;vFI{(Le$%v_h+5KRGwNThX8Tq1drTpT=)P<6UU@H+<1RMB^DWc83_YA_rJ&^Q~)F zF3W$vJKNja{iJxkw$@jlH=CRN+npc!`m-PVg%@}FFE4&Q=C!xC*AEU3`r+YW-{0Rq z_WkbeZvXk&df(aEIriO2=UV0e^WecpeO_HU)1Pdf?tg#(O~1Xq-XDFpyvzObvGjj` zlaJP|d@pWp^otkH_j4PqpIbIQY5S@9mcY84^0ZEa&LeD}Q!Nx)a!9+wth;q48q88RR7Gh;mNv&Kd{aUB7%Yz5eMGQ=gxb#u-lv6saqmy*P9#aXeA)Zo4aeHK2g(@tfH*w z?(8-ztby99{{++~LpeAojK-iafp7){8MDa*Q38}=NSH7M2}pF-hSzR2R&Q*LN$Xyl0u-e%r4%xf0tzmrEH-6QumU#itT*a< z7jM>TwcIeV&@Q{hV$?0~jeW`E`}>#GFMD3sz|7HI1dCa}I$sgUxf6+V)9S5h^t~fY$hvz*v W7!3aGKKlK>^EQls@;mkMd(bP>I{YdC diff --git a/assets/voxygen/voxel/npc/cockatrice/male/neck.vox b/assets/voxygen/voxel/npc/cockatrice/male/neck.vox new file mode 100644 index 0000000000000000000000000000000000000000..dcf9e3627ebfeca37564daedbe64e21e2324e8df GIT binary patch literal 3424 zcmeH}zpG_e7037bwf8 z=tyz~Ki(X^eAC>?A$snC-(eG;P3@pOYvz}6q>(>zGkD9pdGqSm-Wz=SK_aeYsGGNX zaO|SiYwZ$p5A_?xHq7C@t>J#H*hKhM;M=MlVgRE| z)DTQV=0;9n5x9%2Yisb)s3~_cZh}v{jZHooqqXZG2TwlW)ZyKzwT4}L*umjAY7h^O zz-u)QLLQ#=Cl9OmMz7XJj-4F(=87$Pp1ns;I5Hc+3-^&dW8A0Q?MmR-Gp`}2J-w8% z(|#B>9kU+br@!PDCq0R6blWt0ju;9?BX1*r!@qDBiKSLB5TAPiuQWIrd=u`Z$L8>@ z#5Z^|>GU4GgiO*e??j1uW5jn>V6XUd>Z0^aU@r~+sBI*E1=ASh3x=lHu#ijMb=iWr z#+{=tyaS3nhLIZ456BFTg=eDwx8m->@Su-izSDf=0&dADt%Zzl50nSd`#{G;G=;ys23dNkn7z#P;@=o8C) zaz02LC3DKYO8+{ZtxiYxX)zm4R)@3s%*>cht|Mz$gm=DjrU^}u z+2^CS`D8JnJ1-jZ*=#yH&z><_o@Z?4lehU8QO~NwIlLYPSwufpZTOmwA(dSEqS$C< zISW7L_%>!*8;@GYvv^yIHu4(r(@%oENL?&0jvDRgHXcPsbjBqD8?afM7t^*VSi;9s z)e*gEDHUAg;D!$Qh}Kgv_-Nx|6`dd_?Pm+{lAn4zOR$9a3WmfHb#eCK*&V*{U6`lU zIy{Zw15@qbfOqT1fXj#hF6JfRIq)A2C*Ua5q`+I~aID~~a2c#hJ|4rldPNsL8+EFz zK&K5S5$pv&lD^@ccma;EajEcFV_IsRz&bI@2QZA(7;eZ#lOt*^#CCEmHgX0(IaVD` zPTopi2wW9%E!0G)p@Xj^hqv{d;29ly$b2}Ktko@V)C(NN@JsGuu^3#rRr)OO8}v?H zl)GxwQDYJ2rwDhfYJ8XIU$x}6fwysPr;vtE&h%5^t|&dub3tFJL!pip-(dl-vBa$} zVrZpj;fX%3-j_=4h1x1S6)^K*h8u9hPo>vN_B_0cbo!tE%{~k`iL`C-G%Rid*!$6_ znNt4}$!Ui_d@CGLqmugQ8Zm`>Ri{UjbNZoFYMXiGdq95~ewA~uhtcccHn6$TcLrYi z&$9ol;J^vAeUD%zOs4Q!; z*~~sOduGm4*%Uv@98%_yr_Y>8oP)Wm#8P5O&SB1;XJSasDdvvw4GNwkW9A&n|2z|O zm^tS;>t^46_0CV;F8;;;qz4Zk>@U0iJ6${mswc-+%l0ColMV z{P^+y^y$<6*|TT+lP6DJxPSEM(f*sy?(7dAK78TsW%Ek?zkl@6*ZTR_kH5G7;ojx` zw>Mth-@9{X|NZOBt?pkxF8e<|EnjPwe&4!xcYo{6AMUT+ZToA>-LG{2^t-ZuY5&hZ lmi_UczuC|Kg6TV(m+#SUgR|$)pMO>NZ^Qrp+h9{%`M*Wf{9D;!duoe0>Lf1IhjJ^#A|> diff --git a/assets/voxygen/voxel/npc/cockatrice/male/tail_front.vox b/assets/voxygen/voxel/npc/cockatrice/male/tail_front.vox new file mode 100644 index 0000000000000000000000000000000000000000..52bced91501eee7486bd416f406e18769aa7e778 GIT binary patch literal 2056 zcmeH{J!=&~6o${&%>YYmq zZNZbk1RMmnZr@q`LshHuc`cR9S0jHV+D?aMpE44Bp@j&Kg&&w`^0( z3B=S}OmTR}$7ay^RGMYaN}2}etmJ7FaXhqO#0zr?7-r)sa;02xw9|YbUO)qn^a^C61snb2r6kl7-_)cGL za9G7!boADs)MVtfsf@FsPh%!ZEc0-n%*%l?18;MX#4Npch0`>1fyXK~lo_k{p>WLG z(Q(H|(}}TA^xG`4oG~;^n6V3c!n*APX}pn}`a%b#PRDnAZ#5IE=qTFS>Ofyxu9g>G z+?W?OSufNs;mpjre;Xo*sgOhIoBY)1W1Y%* z^jc1G3%v_7aK0rSzb73mwcaM@%{zd#$%@&56?33Kc+a}Y{QBjK7bM2dS=QFp>VENg zVWGC?+wyYl*U#0(=kt2z{mpvs{o_4ao12?;dwaX??CjL7t*t%#8yg$-?(@aEzP`R^ zZ@+c5^Kb9p|E12ym1Fhk@}c_X^x=AAaj`x;d2~Gc`jOOkPvnWQOEKhM5+AoWN5hi6hZU;bX_PkQXcqR9H{ z)}3ZHumxN~jm<|}tDlvI?F+P$R%_{LgVOa}l~fp{ z($-bXFnDIEmGjA9@Q{WI6&q==o<{3jvR0Gz46Y!971m10428j$)=5PMg~3B=YaJ8@ z4{4}S=a*G|U(I*w5q#^SZ#DX!Q6(`a&oeQ@GfP2Mkip=Yr6eoKVDQXRkyT_ccxI`6 z=b$inNJE8c7^{W8^P#Uk^o;64;GXEI5<}-=;GHnTGfN^%WH5MUDaZ;k7(BC-WF;94 zo>?lgifdJ5R6m<*<~ne1T)PQ2?a`oa-CdzdF?izYcD|2jJaL?;rlG z1AhNVYo@Zt9EY-6vpvH$pJSDrSJQsq%y#p>S(A&JtT}wuoY0MPQ(M(#i{4jlw&mm( zo7%-jJFJ6jhlZ?#?5DW+%=d!4)=3Tr;vN6oFJ0iZHpP~% ziY>d^^DTSqnsTu$xvzFgY>mUPl@srnJLgSWzR*`<$k~rf-otlO zUt{W#V9h|W`YxdlYSS2cFx_RiHnX7Xa^#;*R96Qb(;OqmKUk z&1JV-&Zgrm+!c7yW;@zshb(q!gFSg}@WeUKtQvEF;T&Ey>SgFUNOIQ_x!gqza&F|(HtXnjTQjgheIt=w!CmTVi5$e7oKL&Yxo6$6z6zYzD1Y$Z1CHar z;=p$jr#Nz8jt*;pgd>>_4@#HX6R#S6DsI+bZP<^wZw>#V$Q2v(jI$5+4yObTMdj3{ zv-xyZ?0L+)*|_(Z@HtkCL4BvS5nF>-4Sx+Dwc-stmAUUad-2?Ye!woi432B@eDcjF-h@u0=CL?=_f3HT2O0RFW9!!zI@-S?m` z8~eaUtNf(HYizK0knkcn3+^_0H>kS=I>ocV4{Lg^T0W>v5L2V3;>;cFo1+7wR_2k{ z{Hggg6C1oxL#Oy*=&*O@Pm=TIeHUtGr#NCB_#bP)ldRaQhnGA&1O5g)uK5>xW8dI; zf*^0vgMNiyrAti&E-CQ|xijPnr1Fa|Cx2$t1)dput?wzwan*!6R<7s?8PtNB8gLMU zoKTx!#x(LEHQ#{YS2$OAp!g2Bq4gdm4~nhp$_$U?Ug!&p8BWjSw)Ejq>BBWE_zc&U zvsl%l-osh6!?E9(J3_uSh$({n5mWV)GuN#hjruj_kT`jA;$n3Wa5DmKQ~#jP#E9Fl z0oNP^-_D#C<(^_s`DP8i8*+iuT41JFA(-bba0tGUJFnRTza8GkL_O%OR(u?AWhD7S z#y+l~|KN{wWWa!f=!q}54ex`QD!fCJ`XqNj)O@X6n9u|CIkQ;sp@9wH?Bn{~@y{%+ znGP4kb609*z(F{A_1{4K<2oCV8*<=qsSUj(Xr2E9V0pT?yaBi8=reIn)O}`tPWyLX zfBB~QfB)BykB?6;iO=6IPhb3SdbwrFKJuM zDuo2&Ovah43XsSchJYA}C1ZbMe`9}QNXe>2&Jj@pvvj91i`(Nz(^|!CY?9 zbJg>|9Uos?=ltQCf9T%w)BCr5zuELRoz;uEU*C#@hKJR*eoOABGYOS(Ae@8?+#nLW1I&nnR(&1Jn9iRm; z2?q8L4oe%14``M2;V5Jvj~)aN5OP4VfFsTebKZhujy$knPB=2?EqW&$dLYAjHXI2U z!=VKS9>^jN#BylBf&&la7-!&6z<_leJMcgrYan1e$JTSffd>k(1_CC)c!Fb|;DUe? zi98r1kp*W2=a5h==B%K;pg!h+oJ0)b1og4z1oZ{=1@#5>5eLR&9AF#>?ndIh?adD#z8?@E%>mE}cU$OXDE soBP^(b4vOd|9nl_x&1lMZ{F!YX7u0aIPPpP7<_&2l^W4fq^AW z0R{U9^>T?Z!3c^qPBkUY7+~g%0A)@aAb}^G7NEi@gOYNZfheaG7$D;m12NP=4NwRk zkPtkm0Sdzd0u)%7vlQVh1Bt+cT7>fydEkWwc?5X`dEkWwc?5X`dEh}QD##+pBFF-b3GOe| zoz3+%vX5h=N~PkX@*0nOFy0CUpKY#q;(FDK?L9wgpAYrwbUNPa^}OHjd$-#iy54HF z{O~&A&1Q4x+NimL`o~7&+deO)dA}@7`@_(9-pA4ZuX_f6fOk3*|DXT> literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/cockatrice/male/wing_mid_r.vox b/assets/voxygen/voxel/npc/cockatrice/male/wing_mid_r.vox new file mode 100644 index 0000000000000000000000000000000000000000..1cc432a662e4b3031cfaa841c2f42292065db526 GIT binary patch literal 1408 zcmeH{ze`(T6o$_^-*I9Vhh zl!9C5jDaql@;~H%$mDkVTtSy$A%dOy!ISg7?>*n)y*HP;^J#bfn~1d58=DmQnuu&S z_CJ(!!9vOe1lZl%Z>%XBR^F^-Y4JO&AgVJ8raBuy3~f4X)Bw@&f=(+Kgn$PU@xUT| z)CiK`fhBm<2vX3R7VscI1W7g1RBM*vQ6orEKWb4kD{3uJBS;AktfX0>#$3NbgTB_5 znvu-~P;2ZMq4dcBeBoleI;UsSx^ zZjW3WH&@>O{ovrqIltE4`dNL_Zyj1Gx@$5l5c+LYE$%O3b3-ks00)3<-JsA*@xXzGBU!)HfF49K^M9veL zNlj82*hCLJ0%D?v%rIGcHcRE;5i)foWY~uE96Ulps)&S)a7lU*2^ng%t5!S4l8%IR z+I3K`%+EC(Jc4IEXk~(~*}yvK$$%KBMQV{696Um#h6Iz8*{H!IAR;B+Co3~4GbuB% zj)aWRQ07qPQ0CC;$%DH)Jjc^xI2@K${b)8zef;eA%hl0#X&v1unWiwJC=?0_F3I Si^byXxnIQo|9a2*58xM1=mu8+ literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/cockatrice/male/wing_out_r.vox b/assets/voxygen/voxel/npc/cockatrice/male/wing_out_r.vox new file mode 100644 index 0000000000000000000000000000000000000000..fa531a0e9cff63870cd2cf7c9267ce83816ae5e2 GIT binary patch literal 1624 zcmeIyKZ_GV7{~GFng1Juh?v0c9%3$#M&aPVPKy&oLQuhj5y?tfK%Laa zNoC>Sos@x$WI_S)QmdyIte4V}o(zbL6IpoaNsvlmf|M2xp8G&XGLc1~E>ahzG^8aR zJmWw{BovXl#QVV{>PQEVfS9D0AO$rruyBR+JOW}NeGamxFtBhf>3amo1?ejcERxYW zTCI~+&Oz&d=(I|(UTf!hj(~{dwKPF%8?Gk1<| z=dI(t{OtJEO0Vg3n&prnv(NWiSM%G! z#r*lkrM%neqvvsE%jNR)cR!E*|GH=M5AX}wFa}=$ literal 0 HcmV?d00001 diff --git a/assets/voxygen/voxel/npc/cockatrice/male/wing_r.vox b/assets/voxygen/voxel/npc/cockatrice/male/wing_r.vox deleted file mode 100644 index c3bfe8b7eed7921c82a8ccdeab255c0eeac4ac39..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1212 zcmWIZk5HJ#z`)?^=;;Tf7*?<_Fa&!>x$=NGKmg=2Z~!qU5JyBtc~$^v5Re68W;Rw| zW(FW;Wzk}0W@~3=W!7S1V%KD5WzuA3Vbx-0Wd+iVn#{}$zRb)(HB8K!%#2LFOf2j` z^*}i$pc+Py8a7R4b{1a-Rt_a5W}to+mUd|C%0#|J$ab%9HB8!J>Zs72f}M|H~5 Vec3::new(4.6, 3.0, 6.0), }, Body::BipedSmall(_) => Vec3::new(1.0, 0.75, 1.4), - Body::BirdMedium(body) => match body.species { - bird_medium::Species::Cockatrice => Vec3::new(2.0, 1.0, 1.8), - _ => Vec3::new(2.0, 1.0, 1.1), - }, - Body::BirdLarge(_) => Vec3::new(1.2, 0.6, 1.1), + Body::BirdMedium(_) => Vec3::new(2.0, 1.0, 1.1), + Body::BirdLarge(_) => Vec3::new(2.0, 1.0, 1.8), Body::Dragon(_) => Vec3::new(16.0, 10.0, 16.0), Body::FishMedium(_) => Vec3::new(0.5, 2.0, 0.8), Body::FishSmall(_) => Vec3::new(0.3, 1.2, 0.6), @@ -431,7 +428,6 @@ impl Body { bird_medium::Species::Goose => 60, bird_medium::Species::Parrot => 60, bird_medium::Species::Peacock => 60, - bird_medium::Species::Cockatrice => 400, bird_medium::Species::Eagle => 400, _ => 100, }, @@ -544,7 +540,6 @@ impl Body { bird_medium::Species::Goose => 10, bird_medium::Species::Parrot => 10, bird_medium::Species::Peacock => 10, - bird_medium::Species::Cockatrice => 10, bird_medium::Species::Eagle => 10, _ => 20, }, diff --git a/common/src/comp/body/bird_large.rs b/common/src/comp/body/bird_large.rs index 2c12eee089..3d6ac3601d 100644 --- a/common/src/comp/body/bird_large.rs +++ b/common/src/comp/body/bird_large.rs @@ -35,6 +35,7 @@ make_case_elim!( #[repr(u32)] pub enum Species { Phoenix = 0, + Cockatrice = 1, } ); @@ -44,6 +45,7 @@ make_case_elim!( #[derive(Clone, Debug, Deserialize)] pub struct AllSpecies { pub phoenix: SpeciesMeta, + pub cockatrice: SpeciesMeta, } impl<'a, SpeciesMeta> core::ops::Index<&'a Species> for AllSpecies { @@ -53,11 +55,15 @@ impl<'a, SpeciesMeta> core::ops::Index<&'a Species> for AllSpecies fn index(&self, &index: &'a Species) -> &Self::Output { match index { Species::Phoenix => &self.phoenix, + Species::Cockatrice => &self.cockatrice, } } } -pub const ALL_SPECIES: [Species; 1] = [Species::Phoenix]; +pub const ALL_SPECIES: [Species; 2] = [ + Species::Phoenix, + Species::Cockatrice, +]; impl<'a, SpeciesMeta: 'a> IntoIterator for &'a AllSpecies { type IntoIter = std::iter::Copied>; diff --git a/common/src/comp/body/bird_medium.rs b/common/src/comp/body/bird_medium.rs index 264e6abf05..5eb7dbee65 100644 --- a/common/src/comp/body/bird_medium.rs +++ b/common/src/comp/body/bird_medium.rs @@ -41,7 +41,6 @@ make_case_elim!( Eagle = 4, Owl = 5, Parrot = 6, - Cockatrice = 7, } ); @@ -57,7 +56,6 @@ pub struct AllSpecies { pub eagle: SpeciesMeta, pub owl: SpeciesMeta, pub parrot: SpeciesMeta, - pub cockatrice: SpeciesMeta, } impl<'a, SpeciesMeta> core::ops::Index<&'a Species> for AllSpecies { @@ -73,12 +71,11 @@ impl<'a, SpeciesMeta> core::ops::Index<&'a Species> for AllSpecies Species::Eagle => &self.eagle, Species::Owl => &self.owl, Species::Parrot => &self.parrot, - Species::Cockatrice => &self.cockatrice, } } } -pub const ALL_SPECIES: [Species; 8] = [ +pub const ALL_SPECIES: [Species; 7] = [ Species::Duck, Species::Chicken, Species::Goose, @@ -86,7 +83,6 @@ pub const ALL_SPECIES: [Species; 8] = [ Species::Eagle, Species::Owl, Species::Parrot, - Species::Cockatrice, ]; impl<'a, SpeciesMeta: 'a> IntoIterator for &'a AllSpecies { diff --git a/voxygen/anim/src/bird_large/feed.rs b/voxygen/anim/src/bird_large/feed.rs new file mode 100644 index 0000000000..3626fac9f1 --- /dev/null +++ b/voxygen/anim/src/bird_large/feed.rs @@ -0,0 +1,101 @@ +use super::{ + super::{vek::*, Animation}, + BirdLargeSkeleton, SkeletonAttr, +}; +use std::ops::Mul; + +pub struct FeedAnimation; + +impl Animation for FeedAnimation { + type Dependency = f32; + type Skeleton = BirdLargeSkeleton; + + #[cfg(feature = "use-dyn-lib")] + const UPDATE_FN: &'static [u8] = b"bird_large_feed\0"; + + #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_feed")] + fn update_skeleton_inner( + skeleton: &Self::Skeleton, + global_time: Self::Dependency, + anim_time: f32, + _rate: &mut f32, + s_a: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let duck_head_look = Vec2::new( + (global_time / 2.0 + anim_time / 8.0) + .floor() + .mul(7331.0) + .sin() + * 0.5, + (global_time / 2.0 + anim_time / 8.0) + .floor() + .mul(1337.0) + .sin() + * 0.25, + ); + let wave_slow_cos = (anim_time * 4.5).cos(); + let wave_fast = (anim_time * 9.0).cos(); + let beak = (anim_time * 16.0).sin(); + + next.head.scale = Vec3::one() * 0.98; + next.neck.scale = Vec3::one() * 1.02; + next.beak.scale = Vec3::one() * 0.98; + next.leg_l.scale = Vec3::one() * 0.98; + next.leg_r.scale = Vec3::one() * 0.98; + next.foot_l.scale = Vec3::one() * 0.98; + next.foot_r.scale = Vec3::one() * 0.98; + next.chest.scale = Vec3::one() * s_a.scaler / 4.0; + + next.chest.position = Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + wave_slow_cos * 0.06 - 1.8) + * s_a.scaler + / 4.0; + next.chest.orientation = Quaternion::rotation_x(-0.5); + + next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); + next.neck.orientation = Quaternion::rotation_x(-0.8); + + next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); + next.head.orientation = Quaternion::rotation_z(duck_head_look.x) + * Quaternion::rotation_x(0.2 - duck_head_look.y.abs() + wave_slow_cos * 0.01); + + next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); + next.beak.orientation = Quaternion::rotation_x(beak * -0.1 - 0.1); + + next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); + next.tail_front.orientation = Quaternion::rotation_x(0.0); + next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); + next.tail_rear.orientation = Quaternion::rotation_x(0.0); + + next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + + next.wing_in_l.orientation = + Quaternion::rotation_y(-s_a.wings_angle + 0.3 + wave_fast * 0.08) * Quaternion::rotation_z(0.2); + next.wing_in_r.orientation = + Quaternion::rotation_y(s_a.wings_angle - 0.3 - wave_fast * 0.08) * Quaternion::rotation_z(-0.2); + + next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_l.orientation = Quaternion::rotation_y(-0.1) * Quaternion::rotation_z(0.7); + next.wing_mid_r.orientation = Quaternion::rotation_y(0.1) * Quaternion::rotation_z(-0.7); + + next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_l.orientation = Quaternion::rotation_y(-0.2) * Quaternion::rotation_z(0.2); + next.wing_out_r.orientation = Quaternion::rotation_y(0.2) * Quaternion::rotation_z(-0.2); + + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2 - wave_slow_cos * 0.06); + next.leg_l.orientation = Quaternion::rotation_x(0.5); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2 - wave_slow_cos * 0.06); + next.leg_r.orientation = Quaternion::rotation_x(0.5); + + next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_l.orientation = Quaternion::rotation_x(0.0); + next.foot_r.position = Vec3::new(s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_r.orientation = Quaternion::rotation_x(0.0); + + next + } +} diff --git a/voxygen/anim/src/bird_large/fly.rs b/voxygen/anim/src/bird_large/fly.rs index 37fa3b7068..eb30962272 100644 --- a/voxygen/anim/src/bird_large/fly.rs +++ b/voxygen/anim/src/bird_large/fly.rs @@ -57,22 +57,31 @@ impl Animation for FlyAnimation { 0.0 } * 1.3; + next.head.scale = Vec3::one() * 0.98; + next.neck.scale = Vec3::one() * 1.02; + next.beak.scale = Vec3::one() * 0.98; + next.leg_l.scale = Vec3::one() * 0.98; + next.leg_r.scale = Vec3::one() * 0.98; + next.foot_l.scale = Vec3::one() * 0.98; + next.foot_r.scale = Vec3::one() * 0.98; next.chest.scale = Vec3::one() * s_a.scaler / 4.0; next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); - next.neck.orientation = Quaternion::rotation_x(0.2); + next.neck.orientation = + Quaternion::rotation_x((-0.4 + 0.2 * velocity.xy().magnitude() / 5.0).max(-0.4)); - next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1 - 3.0); - next.head.orientation = Quaternion::rotation_x(0.2 + fast * 0.05); + next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); + next.head.orientation = + Quaternion::rotation_x((-0.6 + 0.2 * velocity.xy().magnitude() / 5.0).max(-0.6) + fast * 0.05); next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); - dbg!(velocity); - if velocity.z > 2.0 { + if velocity.z > 2.0 || velocity.xy().magnitude() < 1.8 { next.chest.position = Vec3::new(0.0, s_a.chest.0, s_a.chest.1 - flap4 * 1.5) * s_a.scaler / 4.0; - next.chest.orientation = - Quaternion::rotation_x(-0.5 - flap1 * 0.2) * Quaternion::rotation_y(tilt * 1.8); + next.chest.orientation = Quaternion::rotation_x( + (0.8 - 0.8 * velocity.xy().magnitude() / 5.0).max(-0.2) - flap1 * 0.2, + ) * Quaternion::rotation_y(tilt * 1.8 + fast * 0.01); next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); @@ -98,22 +107,36 @@ impl Animation for FlyAnimation { next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); next.tail_rear.orientation = Quaternion::rotation_x(-flap3 * 0.3) * Quaternion::rotation_z(-tilt * 1.0); + + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2); + next.leg_l.orientation = Quaternion::rotation_x( + (-1.0 - 0.8 * velocity.xy().magnitude() / 5.0).max(-0.8) + flap1 * -0.1, + ); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2); + next.leg_r.orientation = Quaternion::rotation_x( + (-1.0 - 0.8 * velocity.xy().magnitude() / 5.0).max(-0.8) + flap1 * -0.1, + ); + + next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_l.orientation = Quaternion::rotation_x(flap1 * -0.05); + next.foot_r.position = Vec3::new(s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_r.orientation = Quaternion::rotation_x(flap1 * -0.05); } else { next.chest.position = Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + slow * 0.05) * s_a.scaler / 4.0; next.chest.orientation = - Quaternion::rotation_x(-0.5 + slow * 0.05 + (0.8 * velocity.z / 80.0).min(0.8)) - * Quaternion::rotation_y(tilt * 1.8); + Quaternion::rotation_x(-0.2 + slow * 0.05 + (0.8 * velocity.z / 80.0).min(0.8)) + * Quaternion::rotation_y(tilt * 1.8 + fast * 0.01); next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_l.orientation = Quaternion::rotation_y(0.1 + slow * 0.04 + (0.8 * velocity.z / 80.0).min(0.8)) - * Quaternion::rotation_x(0.6); + * Quaternion::rotation_x(0.4); next.wing_in_r.orientation = Quaternion::rotation_y(-0.1 + slow * -0.04 - (0.8 * velocity.z / 80.0).min(0.8)) - * Quaternion::rotation_x(0.6); + * Quaternion::rotation_x(0.4); next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); @@ -133,18 +156,18 @@ impl Animation for FlyAnimation { next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); next.tail_rear.orientation = Quaternion::rotation_x(-0.2 + slow * 0.08) * Quaternion::rotation_z(-tilt * 1.0); + + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2); + next.leg_l.orientation = Quaternion::rotation_x(-1.0 + slow * -0.05); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2); + next.leg_r.orientation = Quaternion::rotation_x(-1.0 + slow * -0.05); + + next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_l.orientation = Quaternion::rotation_x(slow * -0.05); + next.foot_r.position = Vec3::new(s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_r.orientation = Quaternion::rotation_x(slow * -0.05); } - next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2); - next.leg_l.orientation = Quaternion::rotation_x(-1.0 - flap1 * 0.1); - next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2); - next.leg_r.orientation = Quaternion::rotation_x(-1.0 - flap1 * 0.1); - - next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); - next.foot_l.orientation = Quaternion::rotation_x(-1.0 - flap1 * 0.1); - next.foot_r.position = Vec3::new(s_a.foot.0, s_a.foot.1, s_a.foot.2); - next.foot_r.orientation = Quaternion::rotation_x(-1.0 - flap1 * 0.1); - next } } diff --git a/voxygen/anim/src/bird_large/idle.rs b/voxygen/anim/src/bird_large/idle.rs index d7e96d74f2..29947a21b2 100644 --- a/voxygen/anim/src/bird_large/idle.rs +++ b/voxygen/anim/src/bird_large/idle.rs @@ -37,7 +37,15 @@ impl Animation for IdleAnimation { ); let wave_slow_cos = (anim_time * 4.5).cos(); + next.head.scale = Vec3::one() * 0.98; + next.neck.scale = Vec3::one() * 1.02; + next.beak.scale = Vec3::one() * 0.98; + next.leg_l.scale = Vec3::one() * 0.98; + next.leg_r.scale = Vec3::one() * 0.98; + next.foot_l.scale = Vec3::one() * 0.98; + next.foot_r.scale = Vec3::one() * 0.98; next.chest.scale = Vec3::one() * s_a.scaler / 4.0; + next.chest.position = Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + wave_slow_cos * 0.06) * s_a.scaler / 4.0; next.chest.orientation = Quaternion::rotation_x(0.0); @@ -61,9 +69,9 @@ impl Animation for IdleAnimation { next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_l.orientation = - Quaternion::rotation_y(-1.3 + wave_slow_cos * 0.06) * Quaternion::rotation_z(0.2); + Quaternion::rotation_y(-1.0 + wave_slow_cos * 0.06) * Quaternion::rotation_z(0.2); next.wing_in_r.orientation = - Quaternion::rotation_y(1.3 - wave_slow_cos * 0.06) * Quaternion::rotation_z(-0.2); + Quaternion::rotation_y(1.0 - wave_slow_cos * 0.06) * Quaternion::rotation_z(-0.2); next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); diff --git a/voxygen/anim/src/bird_large/mod.rs b/voxygen/anim/src/bird_large/mod.rs index d0a086bca4..9c7e190c58 100644 --- a/voxygen/anim/src/bird_large/mod.rs +++ b/voxygen/anim/src/bird_large/mod.rs @@ -1,9 +1,10 @@ +pub mod feed; pub mod fly; pub mod idle; pub mod run; // Reexports -pub use self::{fly::FlyAnimation, idle::IdleAnimation, run::RunAnimation}; +pub use self::{feed::FeedAnimation, fly::FlyAnimation, idle::IdleAnimation, run::RunAnimation}; use super::{make_bone, vek::*, FigureBoneData, Skeleton}; use common::comp::{self}; @@ -59,8 +60,8 @@ impl Skeleton for BirdLargeSkeleton { let wing_out_r_mat = wing_mid_r_mat * Mat4::::from(self.wing_out_r); let leg_l_mat = chest_mat * Mat4::::from(self.leg_l); let leg_r_mat = chest_mat * Mat4::::from(self.leg_r); - let foot_l_mat = chest_mat * Mat4::::from(self.leg_l); - let foot_r_mat = chest_mat * Mat4::::from(self.leg_r); + let foot_l_mat = leg_l_mat * Mat4::::from(self.foot_l); + let foot_r_mat = leg_r_mat * Mat4::::from(self.foot_r); *(<&mut [_; Self::BONE_COUNT]>::try_from(&mut buf[0..Self::BONE_COUNT]).unwrap()) = [ make_bone(head_mat), @@ -97,6 +98,8 @@ pub struct SkeletonAttr { leg: (f32, f32, f32), foot: (f32, f32, f32), scaler: f32, + wings_angle: f32, + flight_angle: f32, } impl<'a> std::convert::TryFrom<&'a comp::Body> for SkeletonAttr { @@ -125,6 +128,8 @@ impl Default for SkeletonAttr { leg: (0.0, 0.0, 0.0), foot: (0.0, 0.0, 0.0), scaler: 0.0, + wings_angle: 0.0, + flight_angle: 0.0, } } } @@ -134,40 +139,60 @@ impl<'a> From<&'a Body> for SkeletonAttr { use comp::bird_large::Species::*; Self { chest: match (body.species, body.body_type) { - (Phoenix, _) => (2.5, 7.5), + (Phoenix, _) => (2.5, 8.0), + (Cockatrice, _) => (2.5, 16.0), }, neck: match (body.species, body.body_type) { (Phoenix, _) => (0.5, 3.0), + (Cockatrice, _) => (5.0, -1.5), }, head: match (body.species, body.body_type) { (Phoenix, _) => (2.0, 2.0), + (Cockatrice, _) => (8.0, 4.5), }, beak: match (body.species, body.body_type) { (Phoenix, _) => (2.0, 1.0), + (Cockatrice, _) => (2.0, -3.0), }, tail_front: match (body.species, body.body_type) { (Phoenix, _) => (-5.5, -2.0), + (Cockatrice, _) => (-5.0, -2.5), }, tail_rear: match (body.species, body.body_type) { (Phoenix, _) => (-3.0, -3.0), + (Cockatrice, _) => (-8.0, -3.0), }, wing_in: match (body.species, body.body_type) { (Phoenix, _) => (3.0, 2.5, 3.0), + (Cockatrice, _) => (3.5, 7.0, 3.5), }, wing_mid: match (body.species, body.body_type) { (Phoenix, _) => (6.5, -1.0, 0.0), + (Cockatrice, _) => (6.0, 0.0, 0.0), }, wing_out: match (body.species, body.body_type) { (Phoenix, _) => (0.5, -1.0, 0.0), + (Cockatrice, _) => (4.0, -1.0, 1.0), }, leg: match (body.species, body.body_type) { (Phoenix, _) => (2.5, -2.5, -3.5), + (Cockatrice, _) => (2.5, 2.5, -3.5), }, foot: match (body.species, body.body_type) { - (Phoenix, _) => (2.5, -16.5, -6.5), + (Phoenix, _) => (0.0, -0.5, -0.5), + (Cockatrice, _) => (1.5, -3.0, -3.0), }, scaler: match (body.species, body.body_type) { (Phoenix, _) => (1.0), + (Cockatrice, _) => (1.0), + }, + wings_angle: match (body.species, body.body_type) { + (Phoenix, _) => (1.3), + (Cockatrice, _) => (0.9), + }, + flight_angle: match (body.species, body.body_type) { + (Phoenix, _) => (-0.5), + (Cockatrice, _) => (1.0), }, } } diff --git a/voxygen/anim/src/bird_large/run.rs b/voxygen/anim/src/bird_large/run.rs index ab5fd3a4d9..0243871d25 100644 --- a/voxygen/anim/src/bird_large/run.rs +++ b/voxygen/anim/src/bird_large/run.rs @@ -61,11 +61,13 @@ impl Animation for RunAnimation { 0.0 } * 1.3; - next.head.scale = Vec3::one() * 1.02; - next.neck.scale = Vec3::one() * 0.98; + next.head.scale = Vec3::one() * 0.98; + next.neck.scale = Vec3::one() * 1.02; next.beak.scale = Vec3::one() * 0.98; - next.foot_l.scale = Vec3::one() * 0.96; - next.foot_r.scale = Vec3::one() * 0.96; + next.leg_l.scale = Vec3::one() * 0.98; + next.leg_r.scale = Vec3::one() * 0.98; + next.foot_l.scale = Vec3::one() * 0.98; + next.foot_r.scale = Vec3::one() * 0.98; next.chest.scale = Vec3::one() * s_a.scaler / 4.0; next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); @@ -100,8 +102,8 @@ impl Animation for RunAnimation { next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); - next.wing_in_l.orientation = Quaternion::rotation_y(-1.3) * Quaternion::rotation_z(0.2); - next.wing_in_r.orientation = Quaternion::rotation_y(1.3) * Quaternion::rotation_z(-0.2); + next.wing_in_l.orientation = Quaternion::rotation_y(-s_a.wings_angle) * Quaternion::rotation_z(0.2); + next.wing_in_r.orientation = Quaternion::rotation_y(s_a.wings_angle) * Quaternion::rotation_z(-0.2); next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); @@ -133,20 +135,21 @@ impl Animation for RunAnimation { next.foot_l.position = Vec3::new( -s_a.foot.0, - s_a.foot.1 + foot1b * -2.0, - s_a.foot.2 + speednorm * 0.5 + (foot1a * 1.5).max(0.0), + s_a.foot.1 + foot1b * -1.0, + s_a.foot.2 + (foot1a * 1.5).max(0.0), ); - next.foot_l.orientation = Quaternion::rotation_x(-0.2 * speednorm + foot1b * -0.35) + next.foot_l.orientation = Quaternion::rotation_x(0.2 * speednorm + foot1b * -0.5 + 0.1) * Quaternion::rotation_y(tilt * -1.0) * Quaternion::rotation_z(tilt * -0.5); next.foot_r.position = Vec3::new( s_a.foot.0, - s_a.foot.1 + foot2b * -2.0, - s_a.foot.2 + speednorm * 0.5 + (foot2a * 1.5).max(0.0), + s_a.foot.1 + foot2b * -1.0, + s_a.foot.2 + (foot2a * 1.5).max(0.0), ); - next.foot_r.orientation = Quaternion::rotation_x(-0.2 * speednorm + foot2b * -0.35) - * Quaternion::rotation_y(tilt * -1.0); + next.foot_r.orientation = Quaternion::rotation_x(0.2 * speednorm + foot2b * -0.5 + 0.1) + * Quaternion::rotation_y(tilt * -1.0) + * Quaternion::rotation_z(tilt * -0.5); next } diff --git a/voxygen/anim/src/bird_medium/mod.rs b/voxygen/anim/src/bird_medium/mod.rs index 8bffda2e5a..ab72ea8173 100644 --- a/voxygen/anim/src/bird_medium/mod.rs +++ b/voxygen/anim/src/bird_medium/mod.rs @@ -99,7 +99,6 @@ impl<'a> From<&'a Body> for SkeletonAttr { (Owl, Male) => (2.5, 5.0), (Owl, Female) => (2.5, 7.0), (Parrot, _) => (0.5, 4.5), - (Cockatrice, _) => (0.0, 4.0), }, chest: match (body.species, body.body_type) { (Duck, _) => (0.0, 5.0), @@ -111,7 +110,6 @@ impl<'a> From<&'a Body> for SkeletonAttr { (Owl, Male) => (0.0, 4.5), (Owl, Female) => (0.0, 4.5), (Parrot, _) => (0.0, 5.0), - (Cockatrice, _) => (0.0, 12.5), }, tail: match (body.species, body.body_type) { (Duck, _) => (-3.0, 1.5), @@ -123,7 +121,6 @@ impl<'a> From<&'a Body> for SkeletonAttr { (Owl, Male) => (-6.0, -2.0), (Owl, Female) => (-6.0, -2.5), (Parrot, _) => (-8.0, -2.0), - (Cockatrice, _) => (-10.0, -2.5), }, wing: match (body.species, body.body_type) { (Duck, _) => (2.75, 0.0, 1.0), @@ -135,7 +132,6 @@ impl<'a> From<&'a Body> for SkeletonAttr { (Owl, Male) => (3.5, -5.5, 4.0), (Owl, Female) => (3.5, -6.0, 3.5), (Parrot, _) => (2.0, -4.5, 3.0), - (Cockatrice, _) => (4.5, -2.5, 1.5), }, foot: match (body.species, body.body_type) { (Duck, _) => (2.0, -1.5, 4.0), @@ -147,7 +143,6 @@ impl<'a> From<&'a Body> for SkeletonAttr { (Owl, Male) => (1.5, -2.5, 7.0), (Owl, Female) => (1.5, -3.0, 6.5), (Parrot, _) => (1.5, -3.0, 3.0), - (Cockatrice, _) => (4.0, -3.5, 12.0), }, feed: match (body.species, body.body_type) { (Chicken, _) => 1.2, @@ -155,7 +150,6 @@ impl<'a> From<&'a Body> for SkeletonAttr { (Peacock, _) => 1.6, (Eagle, _) => 1.2, (Parrot, _) => 1.2, - (Cockatrice, _) => 1.3, _ => 1.0, }, } diff --git a/voxygen/src/scene/figure/mod.rs b/voxygen/src/scene/figure/mod.rs index a74a4e3b5a..b243d73c7b 100644 --- a/voxygen/src/scene/figure/mod.rs +++ b/voxygen/src/scene/figure/mod.rs @@ -3340,7 +3340,6 @@ impl FigureMgr { &mut state_animation_rate, skeleton_attr, ), - // TODO! _ => anim::bird_large::IdleAnimation::update_skeleton( &BirdLargeSkeleton::default(), @@ -3350,8 +3349,22 @@ impl FigureMgr { skeleton_attr, ), }; + let target_bones = match &character { + CharacterState::Sit { .. } => { + anim::bird_large::FeedAnimation::update_skeleton( + &target_base, + time, + state.state_time, + &mut state_animation_rate, + skeleton_attr, + ) + }, - state.skeleton = anim::vek::Lerp::lerp(&state.skeleton, &target_base, dt_lerp); + // TODO! + _ => target_base, + }; + + state.skeleton = anim::vek::Lerp::lerp(&state.skeleton, &target_bones, dt_lerp); state.update( renderer, pos.0, diff --git a/world/src/layer/wildlife.rs b/world/src/layer/wildlife.rs index abd7cda7d6..0bdef79828 100644 --- a/world/src/layer/wildlife.rs +++ b/world/src/layer/wildlife.rs @@ -1,7 +1,7 @@ use crate::{column::ColumnSample, sim::SimChunk, IndexRef, CONFIG}; use common::{ comp::{ - biped_large, bird_medium, fish_medium, fish_small, quadruped_low, quadruped_medium, + biped_large, bird_medium, bird_large, fish_medium, fish_small, quadruped_low, quadruped_medium, quadruped_small, theropod, Alignment, }, generation::{ChunkSupplement, EntityInfo}, @@ -603,7 +603,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( quadruped_low::Body::random_with(rng, &quadruped_low::Species::Monitor) .into() }, - 2 => bird_medium::Body::random_with(rng, &bird_medium::Species::Cockatrice) + 2 => bird_large::Body::random_with(rng, &bird_large::Species::Cockatrice) .into(), 3 => quadruped_small::Body::random_with( rng, From b0d617d91d450e37df07cdb465fec2faa533b35f Mon Sep 17 00:00:00 2001 From: Snowram Date: Mon, 12 Apr 2021 00:22:08 +0200 Subject: [PATCH 04/10] Breathe and stun anims --- .../unique/birdlargebreathe/flamethrower.ron | 14 ++ .../unique/birdlargebreathe/triplestrike.ron | 53 +++++++ .../abilities/weapon_ability_manifest.ron | 5 + .../npc_weapons/unique/birdlargebreathe.ron | 18 +++ .../voxel/npc/cockatrice/male/foot_l.vox | Bin 1588 -> 0 bytes .../voxel/npc/cockatrice/male/leg_l.vox | Bin 2224 -> 0 bytes .../voxel/npc/cockatrice/male/wing_in_l.vox | Bin 1412 -> 0 bytes .../voxel/npc/cockatrice/male/wing_mid_l.vox | Bin 1408 -> 0 bytes .../voxel/npc/cockatrice/male/wing_out_l.vox | Bin 1624 -> 0 bytes .../voxygen/voxel/npc/phoenix/male/foot_l.vox | Bin 1156 -> 0 bytes .../voxygen/voxel/npc/phoenix/male/leg_l.vox | Bin 1172 -> 0 bytes .../voxel/npc/phoenix/male/wing_in_l.vox | Bin 1248 -> 0 bytes .../voxel/npc/phoenix/male/wing_mid_l.vox | Bin 1204 -> 0 bytes .../voxel/npc/phoenix/male/wing_out_l.vox | Bin 1272 -> 0 bytes common/src/comp/body.rs | 3 +- common/src/comp/body/bird_large.rs | 5 +- common/src/comp/inventory/item/tool.rs | 1 + common/src/comp/inventory/loadout_builder.rs | 10 +- voxygen/anim/src/bird_large/alpha.rs | 87 +++++++++++ voxygen/anim/src/bird_large/breathe.rs | 137 ++++++++++++++++++ voxygen/anim/src/bird_large/feed.rs | 22 +-- voxygen/anim/src/bird_large/fly.rs | 46 +++--- voxygen/anim/src/bird_large/idle.rs | 19 +-- voxygen/anim/src/bird_large/mod.rs | 22 +-- voxygen/anim/src/bird_large/run.rs | 24 +-- voxygen/anim/src/bird_large/stunned.rs | 111 ++++++++++++++ voxygen/src/scene/figure/load.rs | 10 +- voxygen/src/scene/figure/mod.rs | 93 ++++++++++++ world/src/layer/mod.rs | 2 +- world/src/layer/wildlife.rs | 4 +- 30 files changed, 609 insertions(+), 77 deletions(-) create mode 100644 assets/common/abilities/unique/birdlargebreathe/flamethrower.ron create mode 100644 assets/common/abilities/unique/birdlargebreathe/triplestrike.ron create mode 100644 assets/common/items/npc_weapons/unique/birdlargebreathe.ron delete mode 100644 assets/voxygen/voxel/npc/cockatrice/male/foot_l.vox delete mode 100644 assets/voxygen/voxel/npc/cockatrice/male/leg_l.vox delete mode 100644 assets/voxygen/voxel/npc/cockatrice/male/wing_in_l.vox delete mode 100644 assets/voxygen/voxel/npc/cockatrice/male/wing_mid_l.vox delete mode 100644 assets/voxygen/voxel/npc/cockatrice/male/wing_out_l.vox delete mode 100644 assets/voxygen/voxel/npc/phoenix/male/foot_l.vox delete mode 100644 assets/voxygen/voxel/npc/phoenix/male/leg_l.vox delete mode 100644 assets/voxygen/voxel/npc/phoenix/male/wing_in_l.vox delete mode 100644 assets/voxygen/voxel/npc/phoenix/male/wing_mid_l.vox delete mode 100644 assets/voxygen/voxel/npc/phoenix/male/wing_out_l.vox create mode 100644 voxygen/anim/src/bird_large/alpha.rs create mode 100644 voxygen/anim/src/bird_large/breathe.rs create mode 100644 voxygen/anim/src/bird_large/stunned.rs diff --git a/assets/common/abilities/unique/birdlargebreathe/flamethrower.ron b/assets/common/abilities/unique/birdlargebreathe/flamethrower.ron new file mode 100644 index 0000000000..4bf507fa86 --- /dev/null +++ b/assets/common/abilities/unique/birdlargebreathe/flamethrower.ron @@ -0,0 +1,14 @@ +BasicBeam( + buildup_duration: 0.4, + recover_duration: 0.25, + beam_duration: 0.5, + damage: 50, + tick_rate: 3.0, + range: 15.0, + max_angle: 22.5, + damage_effect: None, + energy_regen: 0, + energy_drain: 0, + orientation_behavior: Normal, + specifier: Flamethrower, +) \ No newline at end of file diff --git a/assets/common/abilities/unique/birdlargebreathe/triplestrike.ron b/assets/common/abilities/unique/birdlargebreathe/triplestrike.ron new file mode 100644 index 0000000000..2eaa09c043 --- /dev/null +++ b/assets/common/abilities/unique/birdlargebreathe/triplestrike.ron @@ -0,0 +1,53 @@ +ComboMelee( + stage_data: [ + ( + stage: 1, + base_damage: 100, + damage_increase: 0, + base_poise_damage: 0, + poise_damage_increase: 0, + knockback: 5.0, + range: 4.5, + angle: 30.0, + base_buildup_duration: 0.4, + base_swing_duration: 0.1, + base_recover_duration: 0.3, + forward_movement: 2.0, + ), + ( + stage: 2, + base_damage: 80, + damage_increase: 0, + base_poise_damage: 0, + poise_damage_increase: 0, + knockback: 5.0, + range: 3.5, + angle: 30.0, + base_buildup_duration: 0.4, + base_swing_duration: 0.1, + base_recover_duration: 0.3, + forward_movement: 1.5, + ), + ( + stage: 3, + base_damage: 130, + damage_increase: 0, + base_poise_damage: 0, + poise_damage_increase: 0, + knockback: 10.0, + range: 3.5, + angle: 30.0, + base_buildup_duration: 0.65, + base_swing_duration: 0.1, + base_recover_duration: 0.3, + forward_movement: 1.5, + ), + ], + initial_energy_gain: 0, + max_energy_gain: 0, + energy_increase: 0, + speed_increase: 0.0, + max_speed_increase: 0.0, + scales_from_combo: 0, + is_interruptible: false, +) diff --git a/assets/common/abilities/weapon_ability_manifest.ron b/assets/common/abilities/weapon_ability_manifest.ron index 21a39831a8..b952cbe08a 100644 --- a/assets/common/abilities/weapon_ability_manifest.ron +++ b/assets/common/abilities/weapon_ability_manifest.ron @@ -207,6 +207,11 @@ (None, "common.abilities.unique.mindflayer.summonminions"), ], ), + Unique(BirdLargeBreathe): ( + primary: "common.abilities.unique.birdlargebreathe.flamethrower", + secondary: "common.abilities.unique.birdlargebreathe.triplestrike", + abilities: [], + ), Debug: ( primary: "common.abilities.debug.forwardboost", secondary: "common.abilities.debug.upboost", diff --git a/assets/common/items/npc_weapons/unique/birdlargebreathe.ron b/assets/common/items/npc_weapons/unique/birdlargebreathe.ron new file mode 100644 index 0000000000..15b41aa6ed --- /dev/null +++ b/assets/common/items/npc_weapons/unique/birdlargebreathe.ron @@ -0,0 +1,18 @@ +ItemDef( + name: "Bird Large Breathe", + description: "testing123", + kind: Tool(( + kind: Unique(BirdLargeBreathe), + hands: Two, + stats: Direct(( + equip_time_secs: 0.01, + power: 1.0, + poise_strength: 1.0, + speed: 1.0, + crit_chance: 0.0625, + crit_mult: 1.9142857, + )), + )), + quality: Low, + tags: [], +) \ No newline at end of file diff --git a/assets/voxygen/voxel/npc/cockatrice/male/foot_l.vox b/assets/voxygen/voxel/npc/cockatrice/male/foot_l.vox deleted file mode 100644 index bace2b8ceefd0de444a30664d4447b046bfe070d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1588 zcmeH{v5Qkd5XNU`W@q1vK}1NZLyi;DC>$KvX>kuk!XwJdCdYY=z0HAGX{;=4JggK6 z2qIi(oibo!mH#3Cf~2xrzdi7pa6*K$a^c4}^UcoezV|{79vyCfAfmy3{~?67jVShy z?zeM-6+n(#;PCNL|A#|>^%4l$!~v#Slp61;Q6QcwhB{yd-zX|n3_jE$C#VIgHBgj5 z4MTBQZ3PXqz`Oz;l#p8q4@gNhLkU<5Il#pRas@btd$?M}HL7kX>MB+j)jQnB4H@bU zHFHr6In*Jy06)NkIw(P{z#KrpO3Ve6wFQKuDn|jF)dSI>8n12x2evj(6g`b*SH)DatEz*$4t3?Eecq-UR=>bg#w^1a3Ce7;tnPN(U~NheJvleOBW_saV}A0J;h z=iA*I=}qry`gHqRik(h+esk+=@5dLEzPzRj`|{kqUN`OS-by>&kao7Z=X>8hqx8G~ l>sv~*_kW)AH_!CHXY=3aWxTW1YIXkIFJu0{o|*pvegW|fA0Ge! diff --git a/assets/voxygen/voxel/npc/cockatrice/male/leg_l.vox b/assets/voxygen/voxel/npc/cockatrice/male/leg_l.vox deleted file mode 100644 index 2e5f7d36210452a61cb8de707c5fa7bbfca00281..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2224 zcmeH}J&RvO6vof%%oP?9VHf7kMI^#D3W0!~772(fiwXupOk*!XK&)h~ENl!`N>~&m zkj{F`1skjV4*MOpvKxQ1XqqL$3bFDJhq*IzUY_%u=U)EzKDz(bDfm)yeeSNtSeYHVhe)tgB+uXU4RoRucUo2GTcS5h}g_S2s;4J;M&P27l$CgJ$S zDusHbb_bUgYYsM>v2NX?t{T2+Q7|R>ql?+*D&!$oM03$KsNrF2l+r!f*X%}oL|J3@ zWI~@#ek_Ij(bT}?%DFZTsEXJ$lm4xVc^9WvBz=@E)3&(GPtGJJy}N@Wbjz4L82o_+ z4$`J+H?vi}cD2$iWHYcT+*sAks&^}OwGv>-RqKQ8dgN?*2s zvyuyrEK84q9%{fFdRV;0f>vuD)7KbZXQy=AiDY z7_n&Fr*_etqd6na98Q^o(X+bXJKu$}Cb^OtNxq;?grha$(2-Yqh7UuN#`~hPZxisq zLsqzjGoQMrCY(BRbZ2hN$LZIEdUK{G%#(fAHHk@j%w(qU9O&HfbMiWzxT6aJug>?L z+16lw_%7V>y>QNb>e2aDNc4~2zjIsu`tBSa9**nU?>jqV{{GnCAFm$0J$4_xJ8nL` zH$Hp%^_5&FCnw|V>};H$pO4ei(<}GK$H(K-N4w+b=;+Gbb#YDqUmiYuZl2%Xd2M{V z|MK|xtyjj~-QDr!8{4Jt7a diff --git a/assets/voxygen/voxel/npc/cockatrice/male/wing_in_l.vox b/assets/voxygen/voxel/npc/cockatrice/male/wing_in_l.vox deleted file mode 100644 index 6ce0366e02218f45001bf36cc8f854f92f00589c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1412 zcmeIyziU%b6u|Lw&i(OP2qHoP@3j=uKo*5U!Oem#h=gDZ7E0~n?otsaiIani#YvGs zL4wrFKJuM zDuo2&Ovah43XsSchJYA}C1ZbMe`9}QNXe>2&Jj@pvvj91i`(Nz(^|!CY?9 zbJg>|9Uos?=ltQCf9T%w)BCr5zuELRoz;uEU*C#l^W4fq^AW z0R{U9^>T?Z!3c^qPBkUY7+~g%0A)@aAb}^G7NEi@gOYNZfheaG7$D;m12NP=4NwRk zkPtkm0Sdzd0u)%7vlQVh1Bt+cT7>fydEkWwc?5X`dEkWwc?5X`dEh}QD##+pBFF-b3GOe| zoz3+%vX5h=N~PkX@*0nOFy0CUpKY#q;(FDK?L9wgpAYrwbUNPa^}OHjd$-#iy54HF z{O~&A&1Q4x+NimL`o~7&+deO)dA}@7`@_(9-pA4ZuX_f6fOk3*|DXT> diff --git a/assets/voxygen/voxel/npc/cockatrice/male/wing_out_l.vox b/assets/voxygen/voxel/npc/cockatrice/male/wing_out_l.vox deleted file mode 100644 index 5f65901f5f47f1dc6f37cc400318e273ed497016..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1624 zcmeIyKWkHA6vy#%p8rn_B4Pqfig+8yqEIMw)1jhB2r5`8wTrt;L7XH`3N98WMFIs8 zx;bYIxH#oKyj1Gx@$5l5c+LYE$%O3b3-ks00)3<-JsA*@xXzGBU!)HfF49K^M9veL zNlj82*hCLJ0%D?v%rIGcHcRE;5i)foWY~uE96Ulps)&S)a7lU*2^ng%t5!S4l8%IR z+I3K`%+EC(Jc4IEXk~(~*}yvK$$%KBMQV{696Um#h6Iz8*{H!IAR;B+Co3~4GbuB% zj)aWRQ07qPQ0CC;$%DH)Jjc^xI2@K${b)8zef;eA%hl0#X&v1unWiwJC=?0_F3I Si^byXxnIQo|9a2*58xM1=mu8+ diff --git a/assets/voxygen/voxel/npc/phoenix/male/foot_l.vox b/assets/voxygen/voxel/npc/phoenix/male/foot_l.vox deleted file mode 100644 index cc16a4fb37ec3453fbdd1a49daa07f94facdbe8c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1156 zcmW;KL2lDP6b9huv~-1pasw1`gV`WfEea|X0whXM>4x2`VU;UHbdU6oK(3H0%oQTh z3uM*(@Ok3=vHiy$e;)hW=ew7GBI4uK%_pn)c^+|lbN}x7yyJ5w7ov{4@Ao&C*7N6u zOc9UM6c^LNA!pXI5C=a_=eFndv*^=e@}Ddg57Tt^FfGo0f4P2pH3l_KAD`Br{{6L8 zrSzxvtM_KJ`DrQoR>a|Ud+O6h>QkTk`P8bf`f8`@tKO;?ebE^%sXUdZ@>HJjICv^gg{UrA4L%-`ld1!%oyt?q&vj3(eo6Z!c%w(PvIG_0Z-v6JcXz5 zj8}uF@D!c`Qh3JeVzA=G;-I1^=4XC|iuox#g{SZop7C0ZH)eiH^UTZ|3s3L_Pw)g! z@Hk!cJi!w@!DC3#^8`=u1dq#zo+o&MCtzG!^aczkhXYT_qa0)}JK4%c*0PeN#7T~F zkiG0=D;rtMN|q8QIm$uyvXiZBWGyRM_S5OqkH=#_91eZI-}l{a*SFhk-)uI0yt1_mNE)G(zgV%g>1oFh{Os~ z(VJt_kjc%=-1$4P=i{4C4}M0(tDD=`R`LCA#QWPX&+d&KpF98M)8{X@Z)Y2E|KIw! zia5q9jw@NldWj#aSREIM)k<$$VzDjDSjXbn;@IMt^|Bn_J%4%=o;^N&ep>(Y`|msWk%S36Z-^;W&;i@xZKqUe2l=tCd+(Bo)i zEGbor=pAT&8c*YCJdJ1acp6XRX*`W*l6e|W<7qsNXZ{VI#?yEjPvaSXt?@LT#?yEj z58afc-{eW2Bi|Mb66Zq8_APA$uozFCwY=5d0sr!``gKrJjs(h$uovD zd6Fl2k|%lQFwc=0nz0no8=uNkc`8rksXTK!cq&hYs2*6|d`^*zsuQZ6%2SQcoHm}y zQ+X=Shv$#`%O{We*Y{gfs^_UZm1j=xoTV``Lo*gVPvI#%g{SZop7|Q^6rRFUcnZ&a zHFye7;VB@6XTC0OR$N$|R20Sdj8~`_pTbjk3Qysguho2G#w?AI85#>u@B~lr1W)j| zT=YD_6FkA=mZIkgp5O@{j}bjj@B~l5c(mx410GRe}=0R!n$@iq$ZJBhrrV0V zXcT#o7kPg*%dWpldEtpL{w#ZA_fUJ}GWc3-_#wYX1d@`Sp zFQ2s6Pan6hAGWqs&L{K9e8%>UUFwzTHvK2>Th1r(NqiEY#Ah4>K8a7_llUY)<7n_n zd=j5TB=H%?#bn8g$)F@i>eG)p@kx9VpK+|liRru4E7NWK`cdN(e1cE#2|iwj zoKNrxKEcP7kn;&X!6*1w268^ZC-?*zOG9qL@NyV<6<%=02?y-4!xkH?5m#Jr#t8@P zvBMS{tPxjSaK;G-?6JcZ8?4*)dTp1>rJc{`b~>Hf;c#gC{l4vXySCkK+h((A>-E~$ Fh=1v{dxQW0 diff --git a/assets/voxygen/voxel/npc/phoenix/male/wing_mid_l.vox b/assets/voxygen/voxel/npc/phoenix/male/wing_mid_l.vox deleted file mode 100644 index d21b85cdc2ddba8de1ec9eab92b0d5f2f022169d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1204 zcmW;KJ#G_07zW^PZ6`D|u4$0P4W>a<1Q~@00TRI|q@kM{Dqq3Esgb6m#aHkZ<_eZL zfvfbxGqdY=e_rp;v$Om4)7|sm5%J;b=A%{oSw!64+`oO=cT5-ZL}qgL?f&M6b)TN~ z#YODnGWL_qdScqors-iW(`@=Rllg3VSjaT@-`u_|Bo@r)$3ZIA!{+RBpJWBu7$wOK7{5q(YKaJ#MOi;bc$`l9crW_{LYJ6WIg zRxRn1KIxMp>HS*hLm&Fk<7lMMMY1H(`=eS^p2|~sD$n5YRG!LHc`DB!^HiS7Q+X=S zcnqG(Q+X;+<>`-Bc`8rksXUd3Zb*t{;VC?YXAA*P0Vr-(T)4Q%7#4pP8-=Iv6rM3u zJcXz56rOi4^d7tL6rRFUcnVJ+(!x`C3QyrFJY$&0$aGDgljx04=E*#nC-Y>U@j7@i zPlm{Dm|c9lA}3`BWILHB8=vvocrs7s$vj`KU$ytIU$*bJYp+z#lX)`Fc)jB-^^xhC zKI?fBPvS{Di6`-l&wwZKB%Z{Rc*du}lXwzO0!cjMb8)fc#N?o)NXDmsLdp0fp2U-Q z63_Uo#uwAyQXiSFG4ljZ@B~lr1dr22&l5bs6Fe>{dY<43p5SpC(engP@C1xoi{1sp z$>G4W@+3z&$WFGhk+rO3DRGvQ9OWQ8*~&)NvXZ65Sx$14gY0B08(GUrmhF5#x6|p= gj>ls=91d-_+qLa>+cuj`Td&t`wOY01a_QKJ{{R4V!~g&Q diff --git a/assets/voxygen/voxel/npc/phoenix/male/wing_out_l.vox b/assets/voxygen/voxel/npc/phoenix/male/wing_out_l.vox deleted file mode 100644 index 7ff492352a305838a4671679c919fbd5d9947160..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1272 zcmW;Kv92RE7>410lF8((M3Xx}5ha})M8%@4mLmj6y96!U(7E3#w-B9k>Xm#6q^8R) z_!c5jP^a47JoZf9nei9f^UiPo@!N;5KS#tbuOEMPi~r_`_m3aneAOF{S9rh*AO8IK z_>X%(zQw$a=M5IDV|tzu)AT%H#s&+vm{!yC3Tw=(>1BguH9c*$A9}u)ucxOCmi6?q z)qcP<8;A8gz0BBP(cbpvSQ+5l2I0a77v9_;ZyjGQt>H#3ZKH~r*GswyYMM| z3ZKHK@aZKjdC!Mwv%tx=lY5ls=91d;2-?!avr*FjnKpTIC diff --git a/common/src/comp/body.rs b/common/src/comp/body.rs index abaa7672e6..b48678bd18 100644 --- a/common/src/comp/body.rs +++ b/common/src/comp/body.rs @@ -122,7 +122,6 @@ impl<'a, BodyMeta, SpeciesMeta> core::ops::Index<&'a Body> for AllBodies &self.bird_large.body, Body::FishMedium(_) => &self.fish_medium.body, Body::Dragon(_) => &self.dragon.body, - Body::BirdLarge(_) => &self.bird_large.body, Body::FishSmall(_) => &self.fish_small.body, Body::BipedLarge(_) => &self.biped_large.body, Body::BipedSmall(_) => &self.biped_small.body, @@ -433,7 +432,7 @@ impl Body { }, Body::FishMedium(_) => 50, Body::Dragon(_) => 5000, - Body::BirdLarge(_) => 50, + Body::BirdLarge(_) => 9999999, Body::FishSmall(_) => 20, Body::BipedLarge(biped_large) => match biped_large.species { biped_large::Species::Ogre => 2500, diff --git a/common/src/comp/body/bird_large.rs b/common/src/comp/body/bird_large.rs index 3d6ac3601d..629317082a 100644 --- a/common/src/comp/body/bird_large.rs +++ b/common/src/comp/body/bird_large.rs @@ -60,10 +60,7 @@ impl<'a, SpeciesMeta> core::ops::Index<&'a Species> for AllSpecies } } -pub const ALL_SPECIES: [Species; 2] = [ - Species::Phoenix, - Species::Cockatrice, -]; +pub const ALL_SPECIES: [Species; 2] = [Species::Phoenix, Species::Cockatrice]; impl<'a, SpeciesMeta: 'a> IntoIterator for &'a AllSpecies { type IntoIter = std::iter::Copied>; diff --git a/common/src/comp/inventory/item/tool.rs b/common/src/comp/inventory/item/tool.rs index f9c0701742..13f7f2b56d 100644 --- a/common/src/comp/inventory/item/tool.rs +++ b/common/src/comp/inventory/item/tool.rs @@ -441,4 +441,5 @@ pub enum UniqueKind { ObjectTurret, WoodenSpear, MindflayerStaff, + BirdLargeBreathe, } diff --git a/common/src/comp/inventory/loadout_builder.rs b/common/src/comp/inventory/loadout_builder.rs index 6e7668d62c..3bba61a08d 100644 --- a/common/src/comp/inventory/loadout_builder.rs +++ b/common/src/comp/inventory/loadout_builder.rs @@ -1,6 +1,6 @@ use crate::{ comp::{ - biped_large, biped_small, golem, + biped_large, biped_small, bird_large, golem, inventory::{ loadout::Loadout, slot::{ArmorSlot, EquipSlot}, @@ -333,6 +333,14 @@ impl LoadoutBuilder { )); }, }, + Body::BirdLarge(bird_large) => match (bird_large.species, bird_large.body_type) { + (bird_large::Species::Cockatrice, _) => { + main_tool = Some(Item::new_from_asset_expect( + "common.items.npc_weapons.unique.birdlargebreathe", + )); + }, + _ => {}, + }, _ => {}, }; } diff --git a/voxygen/anim/src/bird_large/alpha.rs b/voxygen/anim/src/bird_large/alpha.rs new file mode 100644 index 0000000000..378851e192 --- /dev/null +++ b/voxygen/anim/src/bird_large/alpha.rs @@ -0,0 +1,87 @@ +use super::{ + super::{vek::*, Animation}, + BirdLargeSkeleton, SkeletonAttr, +}; +use common::states::utils::StageSection; + +pub struct AlphaAnimation; + +impl Animation for AlphaAnimation { + type Dependency = (Option, f32, f32); + type Skeleton = BirdLargeSkeleton; + + #[cfg(feature = "use-dyn-lib")] + const UPDATE_FN: &'static [u8] = b"bird_large_alpha\0"; + + #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_alpha")] + fn update_skeleton_inner( + skeleton: &Self::Skeleton, + (stage_section, _global_time, _timer): Self::Dependency, + anim_time: f32, + _rate: &mut f32, + s_a: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let (move1base, move2base, move3) = match stage_section { + Some(StageSection::Buildup) => (anim_time.powf(0.25), 0.0, 0.0), + Some(StageSection::Swing) => (1.0, anim_time, 0.0), + Some(StageSection::Recover) => (1.0, 1.0, anim_time.powf(4.0)), + _ => (0.0, 0.0, 0.0), + }; + + let wave_slow_cos = (anim_time * 4.5).cos(); + + let pullback = 1.0 - move3; + + let move1 = move1base * pullback; + let move2 = move2base * pullback; + + next.head.scale = Vec3::one() * 0.98; + next.neck.scale = Vec3::one() * 1.02; + next.beak.scale = Vec3::one() * 0.98; + next.leg_l.scale = Vec3::one() / 8.0 * 0.98; + next.leg_r.scale = Vec3::one() / 8.0 * 0.98; + next.foot_l.scale = Vec3::one() * 1.02; + next.foot_r.scale = Vec3::one() * 1.02; + next.chest.scale = Vec3::one() * s_a.scaler / 8.0; + + next.chest.position = + Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + wave_slow_cos * 0.06) * s_a.scaler / 8.0; + next.chest.orientation = Quaternion::rotation_x(move1 * 0.5 - move2 * 0.8); + + next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); + next.neck.orientation = Quaternion::rotation_x(move1 * 0.5 - move2 * 0.8); + + next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); + next.head.orientation = Quaternion::rotation_x(move1 * 0.5 - move2 * 0.8); + + next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); + next.beak.orientation = Quaternion::rotation_x(wave_slow_cos * -0.02 - 0.02); + + next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); + next.tail_front.orientation = Quaternion::rotation_x(-move1 * 0.2); + next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); + next.tail_rear.orientation = Quaternion::rotation_x(0.0); + + next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + + next.wing_in_l.orientation = + Quaternion::rotation_y(-1.0 + wave_slow_cos * 0.06) * Quaternion::rotation_z(0.2); + next.wing_in_r.orientation = + Quaternion::rotation_y(1.0 - wave_slow_cos * 0.06) * Quaternion::rotation_z(-0.2); + + next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_l.orientation = Quaternion::rotation_y(-0.1) * Quaternion::rotation_z(0.7); + next.wing_mid_r.orientation = Quaternion::rotation_y(0.1) * Quaternion::rotation_z(-0.7); + + next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_l.orientation = Quaternion::rotation_y(-0.2) * Quaternion::rotation_z(0.2); + next.wing_out_r.orientation = Quaternion::rotation_y(0.2) * Quaternion::rotation_z(-0.2); + + next + } +} diff --git a/voxygen/anim/src/bird_large/breathe.rs b/voxygen/anim/src/bird_large/breathe.rs new file mode 100644 index 0000000000..96cc860f9b --- /dev/null +++ b/voxygen/anim/src/bird_large/breathe.rs @@ -0,0 +1,137 @@ +use super::{ + super::{vek::*, Animation}, + BirdLargeSkeleton, SkeletonAttr, +}; +use common::{states::utils::StageSection, util::Dir}; + +pub struct BreatheAnimation; + +type BreatheAnimationDependency = ( + f32, + f32, + Vec3, + Vec3, + Option, + f32, + Dir, + bool, +); + +impl Animation for BreatheAnimation { + type Dependency = BreatheAnimationDependency; + type Skeleton = BirdLargeSkeleton; + + #[cfg(feature = "use-dyn-lib")] + const UPDATE_FN: &'static [u8] = b"bird_large_breathe\0"; + + #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_breathe")] + fn update_skeleton_inner( + skeleton: &Self::Skeleton, + (_velocity, global_time, orientation, last_ori, stage_section, timer, look_dir, on_ground): Self::Dependency, + anim_time: f32, + _rate: &mut f32, + s_a: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let (movement1base, movement2base, movement3, twitch) = match stage_section { + Some(StageSection::Buildup) => (anim_time.powf(0.25), 0.0, 0.0, 0.0), + Some(StageSection::Cast) => (1.0, anim_time.min(1.0).powf(0.1), 0.0, anim_time), + Some(StageSection::Recover) => (1.0, 1.0, anim_time, 1.0), + _ => (0.0, 0.0, 0.0, 0.0), + }; + + let pullback = 1.0 - movement3; + let subtract = global_time - timer; + let check = subtract - subtract.trunc(); + let mirror = (check - 0.5).signum(); + let twitch2 = mirror * (twitch * 20.0).sin() * pullback; + + let movement1abs = movement1base * pullback; + let movement2abs = movement2base * pullback; + + let wave_slow_cos = (anim_time * 4.5).cos(); + + next.head.scale = Vec3::one() * 0.98; + next.neck.scale = Vec3::one() * 1.02; + next.beak.scale = Vec3::one() * 0.98; + next.leg_l.scale = Vec3::one() / 8.0 * 0.98; + next.leg_r.scale = Vec3::one() / 8.0 * 0.98; + next.foot_l.scale = Vec3::one() * 1.02; + next.foot_r.scale = Vec3::one() * 1.02; + next.chest.scale = Vec3::one() * s_a.scaler / 8.0; + + next.chest.position = Vec3::new( + 0.0, + s_a.chest.0, + s_a.chest.1 + wave_slow_cos * 0.06 + twitch2 * 0.1, + ) * s_a.scaler + / 8.0; + + next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); + next.neck.orientation = Quaternion::rotation_x(movement1abs * 0.5 - movement2abs * 0.5); + + next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); + next.head.orientation = + Quaternion::rotation_x(movement1abs * 0.5 - movement2abs * 0.5 + look_dir.z * 0.4); + + next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); + next.beak.orientation = Quaternion::rotation_x(movement1abs * -0.7 + twitch2 * 0.1); + + if on_ground { + next.chest.orientation = + Quaternion::rotation_x(movement1abs * 0.1 - movement2abs * 0.1); + + next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + + next.wing_in_l.orientation = + Quaternion::rotation_y(-1.0 + movement1abs * 0.8 - movement2abs * 0.4) + * Quaternion::rotation_z(0.2 - movement1abs * 0.8 + movement2abs * 0.4); + next.wing_in_r.orientation = + Quaternion::rotation_y(1.0 - movement1abs * 0.8 + movement2abs * 0.4) + * Quaternion::rotation_z(-0.2 + movement1abs * 0.8 - movement2abs * 0.4); + + next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_l.orientation = + Quaternion::rotation_y(-0.1) * Quaternion::rotation_z(0.7); + next.wing_mid_r.orientation = + Quaternion::rotation_y(0.1) * Quaternion::rotation_z(-0.7); + + next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_l.orientation = + Quaternion::rotation_y(-0.2) * Quaternion::rotation_z(0.2); + next.wing_out_r.orientation = + Quaternion::rotation_y(0.2) * Quaternion::rotation_z(-0.2); + + next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); + next.tail_front.orientation = + Quaternion::rotation_x(-movement1abs * 0.1 + movement2abs * 0.1 + twitch2 * 0.02); + next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); + next.tail_rear.orientation = + Quaternion::rotation_x(-movement1abs * 0.1 + movement2abs * 0.1 + twitch2 * 0.02); + } else { + let ori: Vec2 = Vec2::from(orientation); + let last_ori = Vec2::from(last_ori); + let tilt = if ::vek::Vec2::new(ori, last_ori) + .map(|o| 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.2) + * last_ori.determine_side(Vec2::zero(), ori).signum() + } else { + 0.0 + } * 1.3; + + next.chest.orientation = + Quaternion::rotation_x(movement1abs * 0.1 - movement2abs * 0.1) + * Quaternion::rotation_y(tilt * 1.8); + } + + next + } +} diff --git a/voxygen/anim/src/bird_large/feed.rs b/voxygen/anim/src/bird_large/feed.rs index 3626fac9f1..8b8acf44fe 100644 --- a/voxygen/anim/src/bird_large/feed.rs +++ b/voxygen/anim/src/bird_large/feed.rs @@ -42,15 +42,15 @@ impl Animation for FeedAnimation { next.head.scale = Vec3::one() * 0.98; next.neck.scale = Vec3::one() * 1.02; next.beak.scale = Vec3::one() * 0.98; - next.leg_l.scale = Vec3::one() * 0.98; - next.leg_r.scale = Vec3::one() * 0.98; - next.foot_l.scale = Vec3::one() * 0.98; - next.foot_r.scale = Vec3::one() * 0.98; - next.chest.scale = Vec3::one() * s_a.scaler / 4.0; + next.leg_l.scale = Vec3::one() / 8.0 * 0.98; + next.leg_r.scale = Vec3::one() / 8.0 * 0.98; + next.foot_l.scale = Vec3::one() * 1.02; + next.foot_r.scale = Vec3::one() * 1.02; + next.chest.scale = Vec3::one() * s_a.scaler / 8.0; next.chest.position = Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + wave_slow_cos * 0.06 - 1.8) * s_a.scaler - / 4.0; + / 8.0; next.chest.orientation = Quaternion::rotation_x(-0.5); next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); @@ -72,9 +72,11 @@ impl Animation for FeedAnimation { next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_l.orientation = - Quaternion::rotation_y(-s_a.wings_angle + 0.3 + wave_fast * 0.08) * Quaternion::rotation_z(0.2); + Quaternion::rotation_y(-s_a.wings_angle + 0.3 + wave_fast * 0.08) + * Quaternion::rotation_z(0.2); next.wing_in_r.orientation = - Quaternion::rotation_y(s_a.wings_angle - 0.3 - wave_fast * 0.08) * Quaternion::rotation_z(-0.2); + Quaternion::rotation_y(s_a.wings_angle - 0.3 - wave_fast * 0.08) + * Quaternion::rotation_z(-0.2); next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); @@ -86,9 +88,9 @@ impl Animation for FeedAnimation { next.wing_out_l.orientation = Quaternion::rotation_y(-0.2) * Quaternion::rotation_z(0.2); next.wing_out_r.orientation = Quaternion::rotation_y(0.2) * Quaternion::rotation_z(-0.2); - next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2 - wave_slow_cos * 0.06); + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; next.leg_l.orientation = Quaternion::rotation_x(0.5); - next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2 - wave_slow_cos * 0.06); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; next.leg_r.orientation = Quaternion::rotation_x(0.5); next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); diff --git a/voxygen/anim/src/bird_large/fly.rs b/voxygen/anim/src/bird_large/fly.rs index eb30962272..5d987d74a4 100644 --- a/voxygen/anim/src/bird_large/fly.rs +++ b/voxygen/anim/src/bird_large/fly.rs @@ -15,7 +15,7 @@ impl Animation for FlyAnimation { #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_fly")] fn update_skeleton_inner( skeleton: &Self::Skeleton, - (velocity, orientation, last_ori, _global_time, avg_vel, _acc_vel): Self::Dependency, + (velocity, orientation, last_ori, _global_time, _avg_vel, _acc_vel): Self::Dependency, anim_time: f32, _rate: &mut f32, s_a: &SkeletonAttr, @@ -60,25 +60,27 @@ impl Animation for FlyAnimation { next.head.scale = Vec3::one() * 0.98; next.neck.scale = Vec3::one() * 1.02; next.beak.scale = Vec3::one() * 0.98; - next.leg_l.scale = Vec3::one() * 0.98; - next.leg_r.scale = Vec3::one() * 0.98; - next.foot_l.scale = Vec3::one() * 0.98; - next.foot_r.scale = Vec3::one() * 0.98; - next.chest.scale = Vec3::one() * s_a.scaler / 4.0; + next.leg_l.scale = Vec3::one() / 8.0 * 0.98; + next.leg_r.scale = Vec3::one() / 8.0 * 0.98; + next.foot_l.scale = Vec3::one() * 1.02; + next.foot_r.scale = Vec3::one() * 1.02; + next.chest.scale = Vec3::one() * s_a.scaler / 8.0; next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); next.neck.orientation = - Quaternion::rotation_x((-0.4 + 0.2 * velocity.xy().magnitude() / 5.0).max(-0.4)); + Quaternion::rotation_x((-0.4 + 0.2 * velocity.xy().magnitude() / 5.0).min(0.15)); next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); - next.head.orientation = - Quaternion::rotation_x((-0.6 + 0.2 * velocity.xy().magnitude() / 5.0).max(-0.6) + fast * 0.05); + + next.head.orientation = Quaternion::rotation_x( + (-0.6 + 0.2 * velocity.xy().magnitude() / 5.0).min(-0.3) + fast * 0.05, + ); next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); if velocity.z > 2.0 || velocity.xy().magnitude() < 1.8 { next.chest.position = - Vec3::new(0.0, s_a.chest.0, s_a.chest.1 - flap4 * 1.5) * s_a.scaler / 4.0; + Vec3::new(0.0, s_a.chest.0, s_a.chest.1 - flap4 * 1.5) * s_a.scaler / 8.0; next.chest.orientation = Quaternion::rotation_x( (0.8 - 0.8 * velocity.xy().magnitude() / 5.0).max(-0.2) - flap1 * 0.2, ) * Quaternion::rotation_y(tilt * 1.8 + fast * 0.01); @@ -108,14 +110,14 @@ impl Animation for FlyAnimation { next.tail_rear.orientation = Quaternion::rotation_x(-flap3 * 0.3) * Quaternion::rotation_z(-tilt * 1.0); - next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2); + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2 - flap4 * 1.5) / 8.0; next.leg_l.orientation = Quaternion::rotation_x( - (-1.0 - 0.8 * velocity.xy().magnitude() / 5.0).max(-0.8) + flap1 * -0.1, - ); - next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2); + (-1.0 * velocity.xy().magnitude() / 5.0).max(-1.0) + flap1 * -0.1, + ) * Quaternion::rotation_y(tilt * 1.6 + fast * 0.01); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2 - flap4 * 1.5) / 8.0; next.leg_r.orientation = Quaternion::rotation_x( - (-1.0 - 0.8 * velocity.xy().magnitude() / 5.0).max(-0.8) + flap1 * -0.1, - ); + (-1.0 * velocity.xy().magnitude() / 5.0).max(-1.0) + flap1 * -0.1, + ) * Quaternion::rotation_y(tilt * 1.6 + fast * 0.01); next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); next.foot_l.orientation = Quaternion::rotation_x(flap1 * -0.05); @@ -123,7 +125,7 @@ impl Animation for FlyAnimation { next.foot_r.orientation = Quaternion::rotation_x(flap1 * -0.05); } else { next.chest.position = - Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + slow * 0.05) * s_a.scaler / 4.0; + Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + slow * 0.05) * s_a.scaler / 8.0; next.chest.orientation = Quaternion::rotation_x(-0.2 + slow * 0.05 + (0.8 * velocity.z / 80.0).min(0.8)) * Quaternion::rotation_y(tilt * 1.8 + fast * 0.01); @@ -157,10 +159,12 @@ impl Animation for FlyAnimation { next.tail_rear.orientation = Quaternion::rotation_x(-0.2 + slow * 0.08) * Quaternion::rotation_z(-tilt * 1.0); - next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2); - next.leg_l.orientation = Quaternion::rotation_x(-1.0 + slow * -0.05); - next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2); - next.leg_r.orientation = Quaternion::rotation_x(-1.0 + slow * -0.05); + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2 + slow * 0.05) / 8.0; + next.leg_l.orientation = Quaternion::rotation_x(-1.0 + slow * -0.05) + * Quaternion::rotation_y(tilt * 1.6 + fast * 0.01); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2 + slow * 0.05) / 8.0; + next.leg_r.orientation = Quaternion::rotation_x(-1.0 + slow * -0.05) + * Quaternion::rotation_y(tilt * 1.6 + fast * 0.01); next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); next.foot_l.orientation = Quaternion::rotation_x(slow * -0.05); diff --git a/voxygen/anim/src/bird_large/idle.rs b/voxygen/anim/src/bird_large/idle.rs index 29947a21b2..1b4a8243df 100644 --- a/voxygen/anim/src/bird_large/idle.rs +++ b/voxygen/anim/src/bird_large/idle.rs @@ -40,14 +40,15 @@ impl Animation for IdleAnimation { next.head.scale = Vec3::one() * 0.98; next.neck.scale = Vec3::one() * 1.02; next.beak.scale = Vec3::one() * 0.98; - next.leg_l.scale = Vec3::one() * 0.98; - next.leg_r.scale = Vec3::one() * 0.98; - next.foot_l.scale = Vec3::one() * 0.98; - next.foot_r.scale = Vec3::one() * 0.98; - next.chest.scale = Vec3::one() * s_a.scaler / 4.0; + next.leg_l.scale = Vec3::one() / 8.0 * 0.98; + next.leg_r.scale = Vec3::one() / 8.0 * 0.98; + next.foot_l.scale = Vec3::one() * 1.02; + next.foot_r.scale = Vec3::one() * 1.02; + next.chest.scale = Vec3::one() * s_a.scaler / 8.0; - next.chest.position = - Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + wave_slow_cos * 0.06) * s_a.scaler / 4.0; + next.chest.position = Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + wave_slow_cos * 0.06 + 1.5) + * s_a.scaler + / 8.0; next.chest.orientation = Quaternion::rotation_x(0.0); next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); @@ -83,9 +84,9 @@ impl Animation for IdleAnimation { next.wing_out_l.orientation = Quaternion::rotation_y(-0.2) * Quaternion::rotation_z(0.2); next.wing_out_r.orientation = Quaternion::rotation_y(0.2) * Quaternion::rotation_z(-0.2); - next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2 - wave_slow_cos * 0.06); + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; next.leg_l.orientation = Quaternion::rotation_x(0.0); - next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2 - wave_slow_cos * 0.06); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; next.leg_r.orientation = Quaternion::rotation_x(0.0); next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); diff --git a/voxygen/anim/src/bird_large/mod.rs b/voxygen/anim/src/bird_large/mod.rs index 9c7e190c58..eef7b229c6 100644 --- a/voxygen/anim/src/bird_large/mod.rs +++ b/voxygen/anim/src/bird_large/mod.rs @@ -1,10 +1,16 @@ +pub mod alpha; +pub mod breathe; pub mod feed; pub mod fly; pub mod idle; pub mod run; +pub mod stunned; // Reexports -pub use self::{feed::FeedAnimation, fly::FlyAnimation, idle::IdleAnimation, run::RunAnimation}; +pub use self::{ + alpha::AlphaAnimation, breathe::BreatheAnimation, feed::FeedAnimation, fly::FlyAnimation, + idle::IdleAnimation, run::RunAnimation, stunned::StunnedAnimation, +}; use super::{make_bone, vek::*, FigureBoneData, Skeleton}; use common::comp::{self}; @@ -58,8 +64,8 @@ impl Skeleton for BirdLargeSkeleton { let wing_mid_r_mat = wing_in_r_mat * Mat4::::from(self.wing_mid_r); let wing_out_l_mat = wing_mid_l_mat * Mat4::::from(self.wing_out_l); let wing_out_r_mat = wing_mid_r_mat * Mat4::::from(self.wing_out_r); - let leg_l_mat = chest_mat * Mat4::::from(self.leg_l); - let leg_r_mat = chest_mat * Mat4::::from(self.leg_r); + let leg_l_mat = base_mat * Mat4::::from(self.leg_l); + let leg_r_mat = base_mat * Mat4::::from(self.leg_r); let foot_l_mat = leg_l_mat * Mat4::::from(self.foot_l); let foot_r_mat = leg_r_mat * Mat4::::from(self.foot_r); @@ -99,7 +105,6 @@ pub struct SkeletonAttr { foot: (f32, f32, f32), scaler: f32, wings_angle: f32, - flight_angle: f32, } impl<'a> std::convert::TryFrom<&'a comp::Body> for SkeletonAttr { @@ -129,7 +134,6 @@ impl Default for SkeletonAttr { foot: (0.0, 0.0, 0.0), scaler: 0.0, wings_angle: 0.0, - flight_angle: 0.0, } } } @@ -176,11 +180,11 @@ impl<'a> From<&'a Body> for SkeletonAttr { }, leg: match (body.species, body.body_type) { (Phoenix, _) => (2.5, -2.5, -3.5), - (Cockatrice, _) => (2.5, 2.5, -3.5), + (Cockatrice, _) => (3.5, 2.5, 13.0), }, foot: match (body.species, body.body_type) { (Phoenix, _) => (0.0, -0.5, -0.5), - (Cockatrice, _) => (1.5, -3.0, -3.0), + (Cockatrice, _) => (0.5, -3.0, -3.0), }, scaler: match (body.species, body.body_type) { (Phoenix, _) => (1.0), @@ -190,10 +194,6 @@ impl<'a> From<&'a Body> for SkeletonAttr { (Phoenix, _) => (1.3), (Cockatrice, _) => (0.9), }, - flight_angle: match (body.species, body.body_type) { - (Phoenix, _) => (-0.5), - (Cockatrice, _) => (1.0), - }, } } } diff --git a/voxygen/anim/src/bird_large/run.rs b/voxygen/anim/src/bird_large/run.rs index 0243871d25..b2088f0c4e 100644 --- a/voxygen/anim/src/bird_large/run.rs +++ b/voxygen/anim/src/bird_large/run.rs @@ -16,7 +16,7 @@ impl Animation for RunAnimation { #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_run")] fn update_skeleton_inner( skeleton: &Self::Skeleton, - (velocity, orientation, last_ori, _global_time, avg_vel, acc_vel): Self::Dependency, + (velocity, orientation, last_ori, _global_time, _avg_vel, acc_vel): Self::Dependency, _anim_time: f32, rate: &mut f32, s_a: &SkeletonAttr, @@ -64,11 +64,11 @@ impl Animation for RunAnimation { next.head.scale = Vec3::one() * 0.98; next.neck.scale = Vec3::one() * 1.02; next.beak.scale = Vec3::one() * 0.98; - next.leg_l.scale = Vec3::one() * 0.98; - next.leg_r.scale = Vec3::one() * 0.98; - next.foot_l.scale = Vec3::one() * 0.98; - next.foot_r.scale = Vec3::one() * 0.98; - next.chest.scale = Vec3::one() * s_a.scaler / 4.0; + next.leg_l.scale = Vec3::one() / 8.0 * 0.98; + next.leg_r.scale = Vec3::one() / 8.0 * 0.98; + next.foot_l.scale = Vec3::one() * 1.02; + next.foot_r.scale = Vec3::one() * 1.02; + next.chest.scale = Vec3::one() * s_a.scaler / 8.0; next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); next.head.orientation = Quaternion::rotation_x(-0.1 * speednorm + short * -0.05) @@ -88,7 +88,7 @@ impl Animation for RunAnimation { s_a.chest.0, s_a.chest.1 + short * 0.5 + 0.5 * speednorm, ) * s_a.scaler - / 4.0; + / 8.0; next.chest.orientation = Quaternion::rotation_x(short * 0.07) * Quaternion::rotation_y(tilt * 0.8) * Quaternion::rotation_z(shortalt * 0.10); @@ -102,8 +102,10 @@ impl Animation for RunAnimation { next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); - next.wing_in_l.orientation = Quaternion::rotation_y(-s_a.wings_angle) * Quaternion::rotation_z(0.2); - next.wing_in_r.orientation = Quaternion::rotation_y(s_a.wings_angle) * Quaternion::rotation_z(-0.2); + next.wing_in_l.orientation = + Quaternion::rotation_y(-s_a.wings_angle) * Quaternion::rotation_z(0.2); + next.wing_in_r.orientation = + Quaternion::rotation_y(s_a.wings_angle) * Quaternion::rotation_z(-0.2); next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); @@ -121,7 +123,7 @@ impl Animation for RunAnimation { -s_a.leg.0 + speednorm * 1.5, s_a.leg.1 + foot1b * -2.3, s_a.leg.2, - ); + ) / 8.0; next.leg_l.orientation = Quaternion::rotation_x(-0.2 * speednorm + foot1a * 0.15) * Quaternion::rotation_y(tilt * 0.5); @@ -129,7 +131,7 @@ impl Animation for RunAnimation { s_a.leg.0 + speednorm * -1.5, s_a.leg.1 + foot2b * -2.3, s_a.leg.2, - ); + ) / 8.0; next.leg_r.orientation = Quaternion::rotation_x(-0.2 * speednorm + foot2a * 0.15) * Quaternion::rotation_y(tilt * 0.5); diff --git a/voxygen/anim/src/bird_large/stunned.rs b/voxygen/anim/src/bird_large/stunned.rs new file mode 100644 index 0000000000..782af0f881 --- /dev/null +++ b/voxygen/anim/src/bird_large/stunned.rs @@ -0,0 +1,111 @@ +use super::{ + super::{vek::*, Animation}, + BirdLargeSkeleton, SkeletonAttr, +}; +use common::states::utils::StageSection; +use std::ops::Mul; + +pub struct StunnedAnimation; + +impl Animation for StunnedAnimation { + type Dependency = (f32, f32, Option, f32); + type Skeleton = BirdLargeSkeleton; + + #[cfg(feature = "use-dyn-lib")] + const UPDATE_FN: &'static [u8] = b"bird_large_stunned\0"; + + #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_stunned")] + fn update_skeleton_inner( + skeleton: &Self::Skeleton, + (_velocity, global_time, stage_section, timer): Self::Dependency, + anim_time: f32, + _rate: &mut f32, + s_a: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let duck_head_look = Vec2::new( + (global_time / 2.0 + anim_time / 8.0) + .floor() + .mul(7331.0) + .sin() + * 0.5, + (global_time / 2.0 + anim_time / 8.0) + .floor() + .mul(1337.0) + .sin() + * 0.25, + ); + let wave_slow_cos = (anim_time * 4.5).cos(); + + let (movement1base, movement2, twitch) = match stage_section { + Some(StageSection::Buildup) => (anim_time.powf(0.1), 0.0, anim_time), + Some(StageSection::Recover) => (1.0, anim_time.powf(4.0), 1.0), + _ => (0.0, 0.0, 0.0), + }; + let pullback = 1.0 - movement2; + let subtract = global_time - timer; + let check = subtract - subtract.trunc(); + let mirror = (check - 0.5).signum(); + let twitch2 = mirror * (twitch * 20.0).sin() * pullback; + let movement1abs = movement1base * pullback; + + next.head.scale = Vec3::one() * 0.98; + next.neck.scale = Vec3::one() * 1.02; + next.beak.scale = Vec3::one() * 0.98; + next.leg_l.scale = Vec3::one() / 8.0 * 0.98; + next.leg_r.scale = Vec3::one() / 8.0 * 0.98; + next.foot_l.scale = Vec3::one() * 1.02; + next.foot_r.scale = Vec3::one() * 1.02; + next.chest.scale = Vec3::one() * s_a.scaler / 8.0; + + next.chest.position = + Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + wave_slow_cos * 0.06) * s_a.scaler / 8.0; + next.chest.orientation = Quaternion::rotation_x(movement1base * 0.5); + + next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); + next.neck.orientation = Quaternion::rotation_x(0.0); + + next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); + next.head.orientation = Quaternion::rotation_z(twitch2 * 0.8) + * Quaternion::rotation_x(-duck_head_look.y.abs() + wave_slow_cos * 0.01); + + next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); + next.beak.orientation = Quaternion::rotation_x(-movement1abs * 0.8); + + next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); + next.tail_front.orientation = Quaternion::rotation_x(0.0); + next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); + next.tail_rear.orientation = Quaternion::rotation_x(0.0); + + next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + + next.wing_in_l.orientation = Quaternion::rotation_y(wave_slow_cos * 0.06 + twitch2 * 0.8) + * Quaternion::rotation_z(0.2 - movement1abs); + next.wing_in_r.orientation = Quaternion::rotation_y(wave_slow_cos * 0.06 - twitch2 * 0.8) + * Quaternion::rotation_z(-0.2 + movement1abs); + + next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_l.orientation = Quaternion::rotation_y(-0.1) * Quaternion::rotation_z(0.7); + next.wing_mid_r.orientation = Quaternion::rotation_y(0.1) * Quaternion::rotation_z(-0.7); + + next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_l.orientation = Quaternion::rotation_y(-0.2) * Quaternion::rotation_z(0.2); + next.wing_out_r.orientation = Quaternion::rotation_y(0.2) * Quaternion::rotation_z(-0.2); + + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; + next.leg_l.orientation = Quaternion::rotation_x(movement1abs * 0.8); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; + next.leg_r.orientation = Quaternion::rotation_x(0.0); + + next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_l.orientation = Quaternion::rotation_x(0.0); + next.foot_r.position = Vec3::new(s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_r.orientation = Quaternion::rotation_x(0.0); + + next + } +} diff --git a/voxygen/src/scene/figure/load.rs b/voxygen/src/scene/figure/load.rs index e2a4ddeb10..1a0dde31f7 100644 --- a/voxygen/src/scene/figure/load.rs +++ b/voxygen/src/scene/figure/load.rs @@ -3334,7 +3334,7 @@ impl BirdLargeLateralSpec { return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); }, }; - let lateral = graceful_load_segment(&spec.wing_in_l.lateral.0); + let lateral = graceful_load_segment_flipped(&spec.wing_in_l.lateral.0, true); (lateral, Vec3::from(spec.wing_in_l.offset)) } @@ -3366,7 +3366,7 @@ impl BirdLargeLateralSpec { return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); }, }; - let lateral = graceful_load_segment(&spec.wing_mid_l.lateral.0); + let lateral = graceful_load_segment_flipped(&spec.wing_mid_l.lateral.0, true); (lateral, Vec3::from(spec.wing_mid_l.offset)) } @@ -3398,7 +3398,7 @@ impl BirdLargeLateralSpec { return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); }, }; - let lateral = graceful_load_segment(&spec.wing_out_l.lateral.0); + let lateral = graceful_load_segment_flipped(&spec.wing_out_l.lateral.0, true); (lateral, Vec3::from(spec.wing_out_l.offset)) } @@ -3430,7 +3430,7 @@ impl BirdLargeLateralSpec { return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); }, }; - let lateral = graceful_load_segment(&spec.leg_l.lateral.0); + let lateral = graceful_load_segment_flipped(&spec.leg_l.lateral.0, true); (lateral, Vec3::from(spec.leg_l.offset)) } @@ -3462,7 +3462,7 @@ impl BirdLargeLateralSpec { return load_mesh("not_found", Vec3::new(-5.0, -5.0, -2.5)); }, }; - let lateral = graceful_load_segment(&spec.foot_l.lateral.0); + let lateral = graceful_load_segment_flipped(&spec.foot_l.lateral.0, true); (lateral, Vec3::from(spec.foot_l.offset)) } diff --git a/voxygen/src/scene/figure/mod.rs b/voxygen/src/scene/figure/mod.rs index b243d73c7b..ab1e900fe7 100644 --- a/voxygen/src/scene/figure/mod.rs +++ b/voxygen/src/scene/figure/mod.rs @@ -3359,7 +3359,100 @@ impl FigureMgr { skeleton_attr, ) }, + CharacterState::BasicBeam(s) => { + let stage_time = s.timer.as_secs_f32(); + let stage_progress = match s.stage_section { + StageSection::Buildup => { + stage_time / s.static_data.buildup_duration.as_secs_f32() + }, + StageSection::Cast => s.timer.as_secs_f32(), + StageSection::Recover => { + stage_time / s.static_data.recover_duration.as_secs_f32() + }, + _ => 0.0, + }; + anim::bird_large::BreatheAnimation::update_skeleton( + &target_base, + ( + rel_vel.magnitude(), + time, + ori * anim::vek::Vec3::::unit_y(), + state.last_ori * anim::vek::Vec3::::unit_y(), + Some(s.stage_section), + state.state_time, + look_dir, + physics.on_ground, + ), + stage_progress, + &mut state_animation_rate, + skeleton_attr, + ) + }, + CharacterState::ComboMelee(s) => { + let stage_index = (s.stage - 1) as usize; + let stage_time = s.timer.as_secs_f32(); + let stage_progress = match s.stage_section { + StageSection::Buildup => { + stage_time + / s.static_data.stage_data[stage_index] + .base_buildup_duration + .as_secs_f32() + }, + StageSection::Swing => { + stage_time + / s.static_data.stage_data[stage_index] + .base_swing_duration + .as_secs_f32() + }, + StageSection::Recover => { + stage_time + / s.static_data.stage_data[stage_index] + .base_recover_duration + .as_secs_f32() + }, + _ => 0.0, + }; + anim::bird_large::AlphaAnimation::update_skeleton( + &target_base, + (Some(s.stage_section), time, state.state_time), + stage_progress, + &mut state_animation_rate, + skeleton_attr, + ) + }, + CharacterState::Stunned(s) => { + let stage_time = s.timer.as_secs_f32(); + let stage_progress = match s.stage_section { + StageSection::Buildup => { + stage_time / s.static_data.buildup_duration.as_secs_f32() + }, + StageSection::Recover => { + stage_time / s.static_data.recover_duration.as_secs_f32() + }, + _ => 0.0, + }; + match s.static_data.poise_state { + PoiseState::Normal + | PoiseState::Interrupted + | PoiseState::Stunned + | PoiseState::Dazed + | PoiseState::KnockedDown => { + anim::bird_large::StunnedAnimation::update_skeleton( + &target_base, + ( + rel_vel.magnitude(), + time, + Some(s.stage_section), + state.state_time, + ), + stage_progress, + &mut state_animation_rate, + skeleton_attr, + ) + }, + } + }, // TODO! _ => target_base, }; diff --git a/world/src/layer/mod.rs b/world/src/layer/mod.rs index 854eb18072..9785b7b01a 100644 --- a/world/src/layer/mod.rs +++ b/world/src/layer/mod.rs @@ -282,7 +282,7 @@ pub fn apply_caves_supplement<'a>( }, _ => { is_hostile = true; - let species = match dynamic_rng.gen_range(0..4) { + let species = match dynamic_rng.gen_range(0..5) { 0 => comp::biped_large::Species::Ogre, 1 => comp::biped_large::Species::Cyclops, 2 => comp::biped_large::Species::Wendigo, diff --git a/world/src/layer/wildlife.rs b/world/src/layer/wildlife.rs index 0bdef79828..062b5900f4 100644 --- a/world/src/layer/wildlife.rs +++ b/world/src/layer/wildlife.rs @@ -1,8 +1,8 @@ use crate::{column::ColumnSample, sim::SimChunk, IndexRef, CONFIG}; use common::{ comp::{ - biped_large, bird_medium, bird_large, fish_medium, fish_small, quadruped_low, quadruped_medium, - quadruped_small, theropod, Alignment, + biped_large, bird_large, bird_medium, fish_medium, fish_small, quadruped_low, + quadruped_medium, quadruped_small, theropod, Alignment, }, generation::{ChunkSupplement, EntityInfo}, terrain::Block, From ff5d267535815d8d2a58ff23f664179960a58bc6 Mon Sep 17 00:00:00 2001 From: Snowram Date: Fri, 16 Apr 2021 13:17:15 +0200 Subject: [PATCH 05/10] Day period dependant wildlife spawns --- common/src/time.rs | 2 +- server/src/chunk_generator.rs | 7 ++++-- server/src/lib.rs | 2 ++ server/src/state_ext.rs | 3 ++- world/src/layer/wildlife.rs | 43 +++++++++++++++++++++++++++++++++++ world/src/lib.rs | 7 +++++- 6 files changed, 59 insertions(+), 5 deletions(-) diff --git a/common/src/time.rs b/common/src/time.rs index 023c2f5089..a13ba253f4 100644 --- a/common/src/time.rs +++ b/common/src/time.rs @@ -1,4 +1,4 @@ -#[derive(Copy, Clone, PartialEq, Eq)] +#[derive(Debug, Copy, Clone, PartialEq, Eq)] pub enum DayPeriod { Night, Morning, diff --git a/server/src/chunk_generator.rs b/server/src/chunk_generator.rs index 06f27de086..0559b4a460 100644 --- a/server/src/chunk_generator.rs +++ b/server/src/chunk_generator.rs @@ -1,7 +1,9 @@ use crate::metrics::ChunkGenMetrics; #[cfg(not(feature = "worldgen"))] use crate::test_world::{IndexOwned, World}; -use common::{generation::ChunkSupplement, slowjob::SlowJobPool, terrain::TerrainChunk}; +use common::{ + generation::ChunkSupplement, resources::TimeOfDay, slowjob::SlowJobPool, terrain::TerrainChunk, +}; use hashbrown::{hash_map::Entry, HashMap}; use specs::Entity as EcsEntity; use std::sync::{ @@ -42,6 +44,7 @@ impl ChunkGenerator { slowjob_pool: &SlowJobPool, world: Arc, index: IndexOwned, + time: TimeOfDay, ) { let v = if let Entry::Vacant(v) = self.pending_chunks.entry(key) { v @@ -55,7 +58,7 @@ impl ChunkGenerator { slowjob_pool.spawn("CHUNK_GENERATOR", move || { let index = index.as_index_ref(); let payload = world - .generate_chunk(index, key, || cancel.load(Ordering::Relaxed)) + .generate_chunk(index, key, || cancel.load(Ordering::Relaxed), Some(time)) .map_err(|_| entity); let _ = chunk_tx.send((key, payload)); }); diff --git a/server/src/lib.rs b/server/src/lib.rs index acc5d9c254..692ae02c0e 100644 --- a/server/src/lib.rs +++ b/server/src/lib.rs @@ -725,6 +725,7 @@ impl Server { &slow_jobs, Arc::clone(&world), index.clone(), + *ecs.read_resource::(), ); }); } @@ -930,6 +931,7 @@ impl Server { &slow_jobs, Arc::clone(&self.world), self.index.clone(), + *ecs.read_resource::(), ); } diff --git a/server/src/state_ext.rs b/server/src/state_ext.rs index 93dd7cdc77..89a037d95b 100644 --- a/server/src/state_ext.rs +++ b/server/src/state_ext.rs @@ -11,6 +11,7 @@ use common::{ Group, Inventory, }, effect::Effect, + resources::TimeOfDay, slowjob::SlowJobPool, uid::{Uid, UidAllocator}, }; @@ -382,7 +383,7 @@ impl StateExt for State { .for_each(|chunk_key| { #[cfg(feature = "worldgen")] { - chunk_generator.generate_chunk(None, chunk_key, &slow_jobs, Arc::clone(world), index.clone()); + chunk_generator.generate_chunk(None, chunk_key, &slow_jobs, Arc::clone(world), index.clone(), *ecs.read_resource::()); } }); } diff --git a/world/src/layer/wildlife.rs b/world/src/layer/wildlife.rs index 062b5900f4..92b4204ed2 100644 --- a/world/src/layer/wildlife.rs +++ b/world/src/layer/wildlife.rs @@ -5,7 +5,9 @@ use common::{ quadruped_medium, quadruped_small, theropod, Alignment, }, generation::{ChunkSupplement, EntityInfo}, + resources::TimeOfDay, terrain::Block, + time::DayPeriod::{self, Evening, Morning, Night, Noon}, vol::{BaseVol, ReadVol, RectSizedVol, WriteVol}, }; use rand::prelude::*; @@ -28,11 +30,13 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( _index: IndexRef, chunk: &SimChunk, supplement: &mut ChunkSupplement, + time: Option, ) { struct Entry { make_entity: fn(Vec3, &mut R) -> EntityInfo, // Entity group_size: Range, // Group size range is_underwater: bool, // Underwater? + day_period: Vec, // Period of the day get_density: fn(&SimChunk, &ColumnSample) -> f32, // Density } @@ -60,6 +64,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..4, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, col| { close(c.temp, CONFIG.snow_temp, 0.3) * BASE_DENSITY @@ -91,6 +96,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, col| { close(c.temp, CONFIG.snow_temp, 0.3) * col.tree_density * BASE_DENSITY * 1.4 }, @@ -106,6 +112,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| close(c.temp, CONFIG.snow_temp, 0.15) * BASE_DENSITY * 0.5, }, // Tundra rarer solitary ennemies @@ -119,6 +126,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| close(c.temp, CONFIG.snow_temp, 0.15) * BASE_DENSITY * 0.1, }, // Tundra rock solitary ennemies @@ -133,6 +141,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, col| { close(c.temp, CONFIG.snow_temp, 0.15) * BASE_DENSITY * col.rock * 1.0 }, @@ -154,6 +163,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, col| { close(c.temp, CONFIG.snow_temp + 0.2, 0.2) * col.tree_density * BASE_DENSITY * 0.4 }, @@ -170,6 +180,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 3..8, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, col| { close(c.temp, CONFIG.snow_temp + 0.2, 0.6) * col.tree_density * BASE_DENSITY * 0.9 }, @@ -199,6 +210,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..4, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| close(c.temp, CONFIG.snow_temp + 0.2, 0.2) * BASE_DENSITY * 1.0, }, // Taiga solitary wild @@ -237,6 +249,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| close(c.temp, CONFIG.snow_temp + 0.2, 0.6) * BASE_DENSITY * 5.0, }, // Temperate solitary ennemies @@ -271,6 +284,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, col| { close(c.temp, CONFIG.temperate_temp + 0.1, 0.5) * col.tree_density @@ -341,6 +355,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..8, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| { close(c.temp, CONFIG.temperate_temp + 0.1, 0.6) * close(c.humidity, CONFIG.forest_hum, 0.6) @@ -408,6 +423,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| { close(c.temp, CONFIG.temperate_temp + 0.1, 0.6) * BASE_DENSITY @@ -433,6 +449,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| close(c.temp, CONFIG.temperate_temp, 0.8) * BASE_DENSITY * 0.08, }, // Temperate river wildlife @@ -463,6 +480,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |_c, col| { close(col.temp, CONFIG.temperate_temp, 0.6) * if col.water_dist.map(|d| d < 10.0).unwrap_or(false) { @@ -487,6 +505,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |_c, col| { close(col.temp, CONFIG.temperate_temp, 0.6) * if col.water_dist.map(|d| d < 10.0).unwrap_or(false) { @@ -508,6 +527,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |_c, col| { close(col.temp, CONFIG.temperate_temp, 0.6) * if col.water_dist.map(|d| d < 10.0).unwrap_or(false) { @@ -532,6 +552,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, col| { close(c.temp, CONFIG.tropical_temp + 0.1, 0.5) * col.rock * BASE_DENSITY * 5.0 }, @@ -557,6 +578,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| { close(c.temp, CONFIG.tropical_temp + 0.2, 0.2) * close(c.humidity, CONFIG.jungle_hum, 0.2) @@ -585,6 +607,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| { close(c.temp, CONFIG.tropical_temp + 0.2, 0.2) * close(c.humidity, CONFIG.jungle_hum, 0.2) @@ -619,6 +642,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| { close(c.temp, CONFIG.tropical_temp + 0.2, 0.3) * close(c.humidity, CONFIG.jungle_hum, 0.2) @@ -638,6 +662,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..3, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |_c, col| { close(col.temp, CONFIG.tropical_temp + 0.2, 0.5) * if col.water_dist.map(|d| d < 10.0).unwrap_or(false) { @@ -671,6 +696,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..3, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |_c, col| { close(col.temp, CONFIG.tropical_temp, 0.5) * if col.water_dist.map(|d| d < 10.0).unwrap_or(false) { @@ -700,6 +726,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..3, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| { close(c.temp, CONFIG.tropical_temp + 0.1, 0.4) * close(c.humidity, CONFIG.desert_hum, 0.4) @@ -727,6 +754,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 3..7, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| { close(c.temp, CONFIG.tropical_temp + 0.1, 0.4) * close(c.humidity, CONFIG.desert_hum, 0.4) @@ -757,6 +785,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| { close(c.temp, CONFIG.desert_temp + 0.2, 0.3) * close(c.humidity, CONFIG.desert_hum, 0.5) @@ -781,6 +810,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| { close(c.temp, CONFIG.desert_temp + 0.2, 0.3) * close(c.humidity, CONFIG.desert_hum, 0.5) @@ -800,6 +830,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..3, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |_c, col| { close(col.temp, CONFIG.desert_temp + 0.2, 0.3) * if col.water_dist.map(|d| d < 10.0).unwrap_or(false) { @@ -824,6 +855,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..3, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| { close(c.temp, CONFIG.desert_temp + 0.2, 0.3) * close(c.humidity, CONFIG.desert_hum, 0.5) @@ -875,6 +907,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, _col| { close(c.temp, CONFIG.desert_temp + 0.2, 0.3) * BASE_DENSITY * 5.0 }, @@ -896,6 +929,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 3..5, is_underwater: true, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, col| { close(c.temp, CONFIG.temperate_temp, 1.0) * col.tree_density * BASE_DENSITY * 5.0 }, @@ -911,6 +945,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..3, is_underwater: true, + day_period: vec![Night, Morning, Noon, Evening], get_density: |c, col| { close(c.temp, CONFIG.snow_temp, 0.15) * col.tree_density * BASE_DENSITY * 5.0 }, @@ -931,6 +966,12 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }; let underwater = col_sample.water_level > col_sample.alt; + let current_day_period; + if let Some(time) = time { + current_day_period = DayPeriod::from(time.0) + } else { + current_day_period = Noon + } let entity_group = scatter.iter().enumerate().find_map( |( @@ -939,6 +980,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( make_entity, group_size, is_underwater, + day_period, get_density, }, )| { @@ -946,6 +988,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( if density > 0.0 && dynamic_rng.gen::() < density * col_sample.spawn_rate && underwater == *is_underwater + && day_period.contains(¤t_day_period) && col_sample.gradient < Some(1.3) { Some((make_entity, group_size.clone())) diff --git a/world/src/lib.rs b/world/src/lib.rs index 87eae3c622..a349e5bf7a 100644 --- a/world/src/lib.rs +++ b/world/src/lib.rs @@ -50,6 +50,7 @@ use crate::{ use common::{ assets, generation::{ChunkSupplement, EntityInfo}, + resources::TimeOfDay, terrain::{Block, BlockKind, SpriteKind, TerrainChunk, TerrainChunkMeta, TerrainChunkSize}, vol::{ReadVol, RectVolSize, WriteVol}, }; @@ -171,7 +172,9 @@ impl World { // Unwrapping because generate_chunk only returns err when should_continue evals // to true - let (tc, _cs) = self.generate_chunk(index, chunk_pos, || false).unwrap(); + let (tc, _cs) = self + .generate_chunk(index, chunk_pos, || false, None) + .unwrap(); let min_z = tc.get_min_z(); let max_z = tc.get_max_z(); @@ -209,6 +212,7 @@ impl World { chunk_pos: Vec2, // TODO: misleading name mut should_continue: impl FnMut() -> bool, + time: Option, ) -> Result<(TerrainChunk, ChunkSupplement), ()> { let mut sampler = self.sample_blocks(); @@ -390,6 +394,7 @@ impl World { index, sim_chunk, &mut supplement, + time, ); // Apply site supplementary information From 738be81943084cc6148ee66516fdfb89ea70a8cb Mon Sep 17 00:00:00 2001 From: Snowram Date: Sat, 17 Apr 2021 12:38:00 +0200 Subject: [PATCH 06/10] Improves bird_large anims --- .../voxel/bird_large_lateral_manifest.ron | 40 +++--- common/src/states/utils.rs | 2 +- voxygen/anim/src/bird_large/feed.rs | 8 +- voxygen/anim/src/bird_large/fly.rs | 30 ++--- voxygen/anim/src/bird_large/run.rs | 37 +++--- voxygen/anim/src/bird_large/stunned.rs | 27 +--- voxygen/src/scene/figure/mod.rs | 15 +-- voxygen/src/ui/widgets/item_tooltip.rs | 2 +- world/src/layer/wildlife.rs | 119 ++++++++++++++---- 9 files changed, 157 insertions(+), 123 deletions(-) diff --git a/assets/voxygen/voxel/bird_large_lateral_manifest.ron b/assets/voxygen/voxel/bird_large_lateral_manifest.ron index b470d45287..f739190d0e 100644 --- a/assets/voxygen/voxel/bird_large_lateral_manifest.ron +++ b/assets/voxygen/voxel/bird_large_lateral_manifest.ron @@ -2,7 +2,7 @@ (Phoenix, Male): ( wing_in_l: ( offset: (-6.0, -5.0, -2.0), - lateral: ("npc.phoenix.male.wing_in_l"), + lateral: ("npc.phoenix.male.wing_in_r"), ), wing_in_r: ( offset: (0.0, -5.0, -2.0), @@ -10,7 +10,7 @@ ), wing_mid_l: ( offset: (-2.5, -7.0, -2.0), - lateral: ("npc.phoenix.male.wing_mid_l"), + lateral: ("npc.phoenix.male.wing_mid_r"), ), wing_mid_r: ( offset: (-2.5, -7.0, -2.0), @@ -18,7 +18,7 @@ ), wing_out_l: ( offset: (-9.0, -8.0, -2.0), - lateral: ("npc.phoenix.male.wing_out_l"), + lateral: ("npc.phoenix.male.wing_out_r"), ), wing_out_r: ( offset: (0.0, -8.0, -2.0), @@ -26,7 +26,7 @@ ), leg_l: ( offset: (-1.5, -1.5, -1.5), - lateral: ("npc.phoenix.male.leg_l"), + lateral: ("npc.phoenix.male.leg_r"), ), leg_r: ( offset: (-1.5, -1.5, -1.5), @@ -34,7 +34,7 @@ ), foot_l: ( offset: (-1.5, -2.0, -4.0), - lateral: ("npc.phoenix.male.foot_l"), + lateral: ("npc.phoenix.male.foot_r"), ), foot_r: ( offset: (-1.5, -2.0, -4.0), @@ -44,7 +44,7 @@ (Phoenix, Female): ( wing_in_l: ( offset: (-6.0, -5.0, -2.0), - lateral: ("npc.phoenix.male.wing_in_l"), + lateral: ("npc.phoenix.male.wing_in_r"), ), wing_in_r: ( offset: (0.0, -5.0, -2.0), @@ -52,7 +52,7 @@ ), wing_mid_l: ( offset: (-2.5, -7.0, -2.0), - lateral: ("npc.phoenix.male.wing_mid_l"), + lateral: ("npc.phoenix.male.wing_mid_r"), ), wing_mid_r: ( offset: (-2.5, -7.0, -2.0), @@ -60,7 +60,7 @@ ), wing_out_l: ( offset: (-9.0, -8.0, -2.0), - lateral: ("npc.phoenix.male.wing_out_l"), + lateral: ("npc.phoenix.male.wing_out_r"), ), wing_out_r: ( offset: (0.0, -8.0, -2.0), @@ -68,7 +68,7 @@ ), leg_l: ( offset: (-1.5, -1.5, -1.5), - lateral: ("npc.phoenix.male.leg_l"), + lateral: ("npc.phoenix.male.leg_r"), ), leg_r: ( offset: (-1.5, -1.5, -1.5), @@ -76,7 +76,7 @@ ), foot_l: ( offset: (-1.5, -2.0, -4.0), - lateral: ("npc.phoenix.male.foot_l"), + lateral: ("npc.phoenix.male.foot_r"), ), foot_r: ( offset: (-1.5, -2.0, -4.0), @@ -86,7 +86,7 @@ (Cockatrice, Male): ( wing_in_l: ( offset: (-7.0, -8.0, -2.0), - lateral: ("npc.cockatrice.male.wing_in_l"), + lateral: ("npc.cockatrice.male.wing_in_r"), ), wing_in_r: ( offset: (0.0, -8.0, -2.0), @@ -94,7 +94,7 @@ ), wing_mid_l: ( offset: (-5.0, -9.0, -2.0), - lateral: ("npc.cockatrice.male.wing_mid_l"), + lateral: ("npc.cockatrice.male.wing_mid_r"), ), wing_mid_r: ( offset: (0.0, -9.0, -2.0), @@ -102,7 +102,7 @@ ), wing_out_l: ( offset: (-10.0, -11.0, -2.0), - lateral: ("npc.cockatrice.male.wing_out_l"), + lateral: ("npc.cockatrice.male.wing_out_r"), ), wing_out_r: ( offset: (0.0, -11.0, -2.0), @@ -110,7 +110,7 @@ ), leg_l: ( offset: (-3.0, -4.5, -4.0), - lateral: ("npc.cockatrice.male.leg_l"), + lateral: ("npc.cockatrice.male.leg_r"), ), leg_r: ( offset: (-3.0, -4.5, -4.0), @@ -118,7 +118,7 @@ ), foot_l: ( offset: (-3.5, -5.5, -10.0), - lateral: ("npc.cockatrice.male.foot_l"), + lateral: ("npc.cockatrice.male.foot_r"), ), foot_r: ( offset: (-3.5, -5.5, -10.0), @@ -128,7 +128,7 @@ (Cockatrice, Female): ( wing_in_l: ( offset: (-7.0, -8.0, -2.0), - lateral: ("npc.cockatrice.male.wing_in_l"), + lateral: ("npc.cockatrice.male.wing_in_r"), ), wing_in_r: ( offset: (0.0, -8.0, -2.0), @@ -136,7 +136,7 @@ ), wing_mid_l: ( offset: (-5.0, -9.0, -2.0), - lateral: ("npc.cockatrice.male.wing_mid_l"), + lateral: ("npc.cockatrice.male.wing_mid_r"), ), wing_mid_r: ( offset: (0.0, -9.0, -2.0), @@ -144,7 +144,7 @@ ), wing_out_l: ( offset: (-10.0, -11.0, -2.0), - lateral: ("npc.cockatrice.male.wing_out_l"), + lateral: ("npc.cockatrice.male.wing_out_r"), ), wing_out_r: ( offset: (0.0, -11.0, -2.0), @@ -152,7 +152,7 @@ ), leg_l: ( offset: (-3.0, -4.5, -4.0), - lateral: ("npc.cockatrice.male.leg_l"), + lateral: ("npc.cockatrice.male.leg_r"), ), leg_r: ( offset: (-3.0, -4.5, -4.0), @@ -160,7 +160,7 @@ ), foot_l: ( offset: (-3.5, -5.5, -10.0), - lateral: ("npc.cockatrice.male.foot_l"), + lateral: ("npc.cockatrice.male.foot_r"), ), foot_r: ( offset: (-3.5, -5.5, -10.0), diff --git a/common/src/states/utils.rs b/common/src/states/utils.rs index d5a5acdf23..9b8e4f75de 100644 --- a/common/src/states/utils.rs +++ b/common/src/states/utils.rs @@ -77,7 +77,7 @@ impl Body { Body::BirdMedium(_) => 80.0, Body::FishMedium(_) => 80.0, Body::Dragon(_) => 250.0, - Body::BirdLarge(_) => 75.0, + Body::BirdLarge(_) => 110.0, Body::FishSmall(_) => 60.0, Body::BipedSmall(biped_small) => match biped_small.species { biped_small::Species::Haniwa => 65.0, diff --git a/voxygen/anim/src/bird_large/feed.rs b/voxygen/anim/src/bird_large/feed.rs index 8b8acf44fe..9e61647de6 100644 --- a/voxygen/anim/src/bird_large/feed.rs +++ b/voxygen/anim/src/bird_large/feed.rs @@ -54,11 +54,11 @@ impl Animation for FeedAnimation { next.chest.orientation = Quaternion::rotation_x(-0.5); next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); - next.neck.orientation = Quaternion::rotation_x(-0.8); + next.neck.orientation = Quaternion::rotation_x(-0.2); next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); next.head.orientation = Quaternion::rotation_z(duck_head_look.x) - * Quaternion::rotation_x(0.2 - duck_head_look.y.abs() + wave_slow_cos * 0.01); + * Quaternion::rotation_x(-0.2 - duck_head_look.y.abs() + wave_slow_cos * 0.01); next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); next.beak.orientation = Quaternion::rotation_x(beak * -0.1 - 0.1); @@ -89,9 +89,9 @@ impl Animation for FeedAnimation { next.wing_out_r.orientation = Quaternion::rotation_y(0.2) * Quaternion::rotation_z(-0.2); next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; - next.leg_l.orientation = Quaternion::rotation_x(0.5); + next.leg_l.orientation = Quaternion::rotation_x(0.0); next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; - next.leg_r.orientation = Quaternion::rotation_x(0.5); + next.leg_r.orientation = Quaternion::rotation_x(0.0); next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); next.foot_l.orientation = Quaternion::rotation_x(0.0); diff --git a/voxygen/anim/src/bird_large/fly.rs b/voxygen/anim/src/bird_large/fly.rs index 5d987d74a4..227be64ae0 100644 --- a/voxygen/anim/src/bird_large/fly.rs +++ b/voxygen/anim/src/bird_large/fly.rs @@ -6,7 +6,7 @@ use super::{ pub struct FlyAnimation; impl Animation for FlyAnimation { - type Dependency = (Vec3, Vec3, Vec3, f32, Vec3, f32); + type Dependency = (Vec3, Vec3, Vec3); type Skeleton = BirdLargeSkeleton; #[cfg(feature = "use-dyn-lib")] @@ -15,7 +15,7 @@ impl Animation for FlyAnimation { #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_fly")] fn update_skeleton_inner( skeleton: &Self::Skeleton, - (velocity, orientation, last_ori, _global_time, _avg_vel, _acc_vel): Self::Dependency, + (velocity, orientation, last_ori): Self::Dependency, anim_time: f32, _rate: &mut f32, s_a: &SkeletonAttr, @@ -25,6 +25,7 @@ impl Animation for FlyAnimation { let slow = (anim_time * 2.0).sin(); let fast = (anim_time * 4.0).sin(); + // Harmonic series hack to get a sine/saw mix let freq = 8.0; let off1 = 0.0; let off2 = -1.7; @@ -59,7 +60,6 @@ impl Animation for FlyAnimation { next.head.scale = Vec3::one() * 0.98; next.neck.scale = Vec3::one() * 1.02; - next.beak.scale = Vec3::one() * 0.98; next.leg_l.scale = Vec3::one() / 8.0 * 0.98; next.leg_r.scale = Vec3::one() / 8.0 * 0.98; next.foot_l.scale = Vec3::one() * 1.02; @@ -73,12 +73,12 @@ impl Animation for FlyAnimation { next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); next.head.orientation = Quaternion::rotation_x( - (-0.6 + 0.2 * velocity.xy().magnitude() / 5.0).min(-0.3) + fast * 0.05, + (-0.5 + 0.2 * velocity.xy().magnitude() / 5.0).min(-0.3) + fast * 0.05, ); next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); - if velocity.z > 2.0 || velocity.xy().magnitude() < 1.8 { + if velocity.z > 2.0 || velocity.xy().magnitude() < 12.0 { next.chest.position = Vec3::new(0.0, s_a.chest.0, s_a.chest.1 - flap4 * 1.5) * s_a.scaler / 8.0; next.chest.orientation = Quaternion::rotation_x( @@ -105,24 +105,24 @@ impl Animation for FlyAnimation { next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); next.tail_front.orientation = - Quaternion::rotation_x(-flap2 * 0.2) * Quaternion::rotation_z(-tilt * 1.0); + Quaternion::rotation_x(-flap2 * 0.2 + 0.1) * Quaternion::rotation_z(-tilt * 1.0); next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); next.tail_rear.orientation = - Quaternion::rotation_x(-flap3 * 0.3) * Quaternion::rotation_z(-tilt * 1.0); + Quaternion::rotation_x(-flap3 * 0.3 + 0.15) * Quaternion::rotation_z(-tilt * 0.8); next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2 - flap4 * 1.5) / 8.0; next.leg_l.orientation = Quaternion::rotation_x( - (-1.0 * velocity.xy().magnitude() / 5.0).max(-1.0) + flap1 * -0.1, + (-1.0 * velocity.xy().magnitude() / 5.0).max(-1.2) + flap1 * -0.1, ) * Quaternion::rotation_y(tilt * 1.6 + fast * 0.01); next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2 - flap4 * 1.5) / 8.0; next.leg_r.orientation = Quaternion::rotation_x( - (-1.0 * velocity.xy().magnitude() / 5.0).max(-1.0) + flap1 * -0.1, + (-1.0 * velocity.xy().magnitude() / 5.0).max(-1.2) + flap1 * -0.1, ) * Quaternion::rotation_y(tilt * 1.6 + fast * 0.01); next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); - next.foot_l.orientation = Quaternion::rotation_x(flap1 * -0.05); + next.foot_l.orientation = Quaternion::rotation_x(flap1 * -0.1); next.foot_r.position = Vec3::new(s_a.foot.0, s_a.foot.1, s_a.foot.2); - next.foot_r.orientation = Quaternion::rotation_x(flap1 * -0.05); + next.foot_r.orientation = Quaternion::rotation_x(flap1 * -0.1); } else { next.chest.position = Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + slow * 0.05) * s_a.scaler / 8.0; @@ -154,16 +154,16 @@ impl Animation for FlyAnimation { next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); next.tail_front.orientation = - Quaternion::rotation_x(slow * 0.04) * Quaternion::rotation_z(-tilt * 1.0); + Quaternion::rotation_x(0.04 - slow * 0.04) * Quaternion::rotation_z(-tilt * 1.0); next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); next.tail_rear.orientation = - Quaternion::rotation_x(-0.2 + slow * 0.08) * Quaternion::rotation_z(-tilt * 1.0); + Quaternion::rotation_x(slow * 0.08) * Quaternion::rotation_z(-tilt * 0.8); next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2 + slow * 0.05) / 8.0; - next.leg_l.orientation = Quaternion::rotation_x(-1.0 + slow * -0.05) + next.leg_l.orientation = Quaternion::rotation_x(-1.2 + slow * -0.05) * Quaternion::rotation_y(tilt * 1.6 + fast * 0.01); next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2 + slow * 0.05) / 8.0; - next.leg_r.orientation = Quaternion::rotation_x(-1.0 + slow * -0.05) + next.leg_r.orientation = Quaternion::rotation_x(-1.2 + slow * -0.05) * Quaternion::rotation_y(tilt * 1.6 + fast * 0.01); next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); diff --git a/voxygen/anim/src/bird_large/run.rs b/voxygen/anim/src/bird_large/run.rs index b2088f0c4e..cde5f7aba5 100644 --- a/voxygen/anim/src/bird_large/run.rs +++ b/voxygen/anim/src/bird_large/run.rs @@ -7,7 +7,7 @@ use std::f32::consts::PI; pub struct RunAnimation; impl Animation for RunAnimation { - type Dependency = (Vec3, Vec3, Vec3, f32, Vec3, f32); + type Dependency = (Vec3, Vec3, Vec3, f32); type Skeleton = BirdLargeSkeleton; #[cfg(feature = "use-dyn-lib")] @@ -16,8 +16,8 @@ impl Animation for RunAnimation { #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_run")] fn update_skeleton_inner( skeleton: &Self::Skeleton, - (velocity, orientation, last_ori, _global_time, _avg_vel, acc_vel): Self::Dependency, - _anim_time: f32, + (velocity, orientation, last_ori, acc_vel): Self::Dependency, + anim_time: f32, rate: &mut f32, s_a: &SkeletonAttr, ) -> Self::Skeleton { @@ -28,25 +28,29 @@ impl Animation for RunAnimation { //let speednorm = speed / 13.0; let speednorm = (speed / 13.0).powf(0.25); - let speedmult = 2.0; + let speedmult = 0.8; let lab: f32 = 0.6; //6 + // acc_vel and anim_time mix to make sure phase lenght isn't starting at + // +infinite + let mixed_vel = acc_vel + anim_time * 5.0; //sets run frequency using speed, with anim_time setting a floor + let short = ((1.0 / (0.72 - + 0.28 * ((acc_vel * 1.0 * lab * speedmult + PI * -0.15 - 0.5).sin()).powi(2))) + + 0.28 * ((mixed_vel * 1.0 * lab * speedmult + PI * -0.15 - 0.5).sin()).powi(2))) .sqrt()) - * ((acc_vel * 1.0 * lab * speedmult + PI * -0.15 - 0.5).sin()) + * ((mixed_vel * 1.0 * lab * speedmult + PI * -0.15 - 0.5).sin()) * speednorm; // - let shortalt = (acc_vel * 1.0 * lab * speedmult + PI * 3.0 / 8.0 - 0.5).sin() * speednorm; + let shortalt = (mixed_vel * 1.0 * lab * speedmult + PI * 3.0 / 8.0 - 0.5).sin() * speednorm; //FL - let foot1a = (acc_vel * 1.0 * lab * speedmult + 0.0 + PI).sin() * speednorm; //1.5 - let foot1b = (acc_vel * 1.0 * lab * speedmult + 1.57 + PI).sin() * speednorm; //1.9 + let foot1a = (mixed_vel * 1.0 * lab * speedmult + 0.0 + PI).sin() * speednorm; //1.5 + let foot1b = (mixed_vel * 1.0 * lab * speedmult + 1.57 + PI).sin() * speednorm; //1.9 //FR - let foot2a = (acc_vel * 1.0 * lab * speedmult).sin() * speednorm; //1.2 - let foot2b = (acc_vel * 1.0 * lab * speedmult + 1.57).sin() * speednorm; //1.6 + let foot2a = (mixed_vel * 1.0 * lab * speedmult).sin() * speednorm; //1.2 + let foot2b = (mixed_vel * 1.0 * lab * speedmult + 1.57).sin() * speednorm; //1.6 let ori: Vec2 = Vec2::from(orientation); let last_ori = Vec2::from(last_ori); let tilt = if ::vek::Vec2::new(ori, last_ori) @@ -63,7 +67,6 @@ impl Animation for RunAnimation { next.head.scale = Vec3::one() * 0.98; next.neck.scale = Vec3::one() * 1.02; - next.beak.scale = Vec3::one() * 0.98; next.leg_l.scale = Vec3::one() / 8.0 * 0.98; next.leg_r.scale = Vec3::one() / 8.0 * 0.98; next.foot_l.scale = Vec3::one() * 1.02; @@ -94,18 +97,16 @@ impl Animation for RunAnimation { * Quaternion::rotation_z(shortalt * 0.10); next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); - next.tail_front.orientation = Quaternion::rotation_x(short * -0.02); + next.tail_front.orientation = Quaternion::rotation_x(0.6 + short * -0.02); next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); - next.tail_rear.orientation = Quaternion::rotation_x(short * -0.1); + next.tail_rear.orientation = Quaternion::rotation_x(-0.2 + short * -0.1); next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); - next.wing_in_l.orientation = - Quaternion::rotation_y(-s_a.wings_angle) * Quaternion::rotation_z(0.2); - next.wing_in_r.orientation = - Quaternion::rotation_y(s_a.wings_angle) * Quaternion::rotation_z(-0.2); + next.wing_in_l.orientation = Quaternion::rotation_y(-0.8) * Quaternion::rotation_z(0.2); + next.wing_in_r.orientation = Quaternion::rotation_y(0.8) * Quaternion::rotation_z(-0.2); next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); diff --git a/voxygen/anim/src/bird_large/stunned.rs b/voxygen/anim/src/bird_large/stunned.rs index 782af0f881..62a1905a7e 100644 --- a/voxygen/anim/src/bird_large/stunned.rs +++ b/voxygen/anim/src/bird_large/stunned.rs @@ -3,12 +3,11 @@ use super::{ BirdLargeSkeleton, SkeletonAttr, }; use common::states::utils::StageSection; -use std::ops::Mul; pub struct StunnedAnimation; impl Animation for StunnedAnimation { - type Dependency = (f32, f32, Option, f32); + type Dependency = (f32, Option, f32); type Skeleton = BirdLargeSkeleton; #[cfg(feature = "use-dyn-lib")] @@ -17,25 +16,13 @@ impl Animation for StunnedAnimation { #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_stunned")] fn update_skeleton_inner( skeleton: &Self::Skeleton, - (_velocity, global_time, stage_section, timer): Self::Dependency, + (global_time, stage_section, timer): Self::Dependency, anim_time: f32, _rate: &mut f32, s_a: &SkeletonAttr, ) -> Self::Skeleton { let mut next = (*skeleton).clone(); - let duck_head_look = Vec2::new( - (global_time / 2.0 + anim_time / 8.0) - .floor() - .mul(7331.0) - .sin() - * 0.5, - (global_time / 2.0 + anim_time / 8.0) - .floor() - .mul(1337.0) - .sin() - * 0.25, - ); let wave_slow_cos = (anim_time * 4.5).cos(); let (movement1base, movement2, twitch) = match stage_section { @@ -64,19 +51,16 @@ impl Animation for StunnedAnimation { next.chest.orientation = Quaternion::rotation_x(movement1base * 0.5); next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); - next.neck.orientation = Quaternion::rotation_x(0.0); next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); - next.head.orientation = Quaternion::rotation_z(twitch2 * 0.8) - * Quaternion::rotation_x(-duck_head_look.y.abs() + wave_slow_cos * 0.01); + next.head.orientation = + Quaternion::rotation_z(twitch2 * 0.8) * Quaternion::rotation_x(wave_slow_cos * 0.01); next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); next.beak.orientation = Quaternion::rotation_x(-movement1abs * 0.8); next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); - next.tail_front.orientation = Quaternion::rotation_x(0.0); next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); - next.tail_rear.orientation = Quaternion::rotation_x(0.0); next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); @@ -99,12 +83,9 @@ impl Animation for StunnedAnimation { next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; next.leg_l.orientation = Quaternion::rotation_x(movement1abs * 0.8); next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; - next.leg_r.orientation = Quaternion::rotation_x(0.0); next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); - next.foot_l.orientation = Quaternion::rotation_x(0.0); next.foot_r.position = Vec3::new(s_a.foot.0, s_a.foot.1, s_a.foot.2); - next.foot_r.orientation = Quaternion::rotation_x(0.0); next } diff --git a/voxygen/src/scene/figure/mod.rs b/voxygen/src/scene/figure/mod.rs index ab1e900fe7..baaacc7f60 100644 --- a/voxygen/src/scene/figure/mod.rs +++ b/voxygen/src/scene/figure/mod.rs @@ -3283,9 +3283,6 @@ impl FigureMgr { FigureState::new(renderer, BirdLargeSkeleton::default()) }); - // Average velocity relative to the current ground - let rel_avg_vel = state.avg_vel - physics.ground_vel; - let (character, last_character) = match (character, last_character) { (Some(c), Some(l)) => (c, l), _ => continue, @@ -3316,8 +3313,6 @@ impl FigureMgr { // TODO: Update to use the quaternion. ori * anim::vek::Vec3::::unit_y(), state.last_ori * anim::vek::Vec3::::unit_y(), - time, - rel_avg_vel, state.acc_vel, ), state.state_time, @@ -3332,9 +3327,6 @@ impl FigureMgr { // TODO: Update to use the quaternion. ori * anim::vek::Vec3::::unit_y(), state.last_ori * anim::vek::Vec3::::unit_y(), - time, - rel_avg_vel, - state.acc_vel, ), state.state_time, &mut state_animation_rate, @@ -3440,12 +3432,7 @@ impl FigureMgr { | PoiseState::KnockedDown => { anim::bird_large::StunnedAnimation::update_skeleton( &target_base, - ( - rel_vel.magnitude(), - time, - Some(s.stage_section), - state.state_time, - ), + (time, Some(s.stage_section), state.state_time), stage_progress, &mut state_animation_rate, skeleton_attr, diff --git a/voxygen/src/ui/widgets/item_tooltip.rs b/voxygen/src/ui/widgets/item_tooltip.rs index 62b106643a..7d66c154d9 100644 --- a/voxygen/src/ui/widgets/item_tooltip.rs +++ b/voxygen/src/ui/widgets/item_tooltip.rs @@ -1058,7 +1058,7 @@ impl<'a> Widget for ItemTooltip<'a> { .w(text_w) .get_h(ui) .unwrap_or(0.0) - + V_PAD + + V_PAD * 2.0 } else { 0.0 }; diff --git a/world/src/layer/wildlife.rs b/world/src/layer/wildlife.rs index 92b4204ed2..8e2c007608 100644 --- a/world/src/layer/wildlife.rs +++ b/world/src/layer/wildlife.rs @@ -367,7 +367,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( Entry { make_entity: |pos, rng| { EntityInfo::at(pos) - .with_body(match rng.gen_range(0..11) { + .with_body(match rng.gen_range(0..10) { 0 => quadruped_small::Body { species: quadruped_small::Species::Fox, body_type: quadruped_small::BodyType::Male, @@ -408,14 +408,9 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( &quadruped_medium::Species::Hirdrasil, ) .into(), - 9 => quadruped_small::Body::random_with( - rng, - &quadruped_small::Species::Truffler, - ) - .into(), _ => quadruped_small::Body::random_with( rng, - &quadruped_small::Species::Batfox, + &quadruped_small::Species::Truffler, ) .into(), }) @@ -431,6 +426,26 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( * 8.0 }, }, + // Temperate solitary wild night + Entry { + make_entity: |pos, rng| { + EntityInfo::at(pos) + .with_body( + quadruped_small::Body::random_with(rng, &quadruped_small::Species::Batfox) + .into(), + ) + .with_alignment(Alignment::Enemy) + }, + group_size: 1..2, + is_underwater: false, + day_period: vec![Night], + get_density: |c, _col| { + close(c.temp, CONFIG.temperate_temp + 0.1, 0.6) + * BASE_DENSITY + * close(c.humidity, CONFIG.forest_hum, 0.6) + * 0.8 + }, + }, // Rare temperate solitary enemies Entry { make_entity: |pos, rng| { @@ -583,10 +598,29 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( close(c.temp, CONFIG.tropical_temp + 0.2, 0.2) * close(c.humidity, CONFIG.jungle_hum, 0.2) * BASE_DENSITY - * 3.0 + * 2.8 }, }, - // Jungle rare solitary wild + // Jungle solitary ennemies day + Entry { + make_entity: |pos, rng| { + EntityInfo::at(pos) + .with_body( + theropod::Body::random_with(rng, &theropod::Species::Sunlizard).into(), + ) + .with_alignment(Alignment::Enemy) + }, + group_size: 1..2, + is_underwater: false, + day_period: vec![Morning, Noon, Evening], + get_density: |c, _col| { + close(c.temp, CONFIG.tropical_temp + 0.2, 0.2) + * close(c.humidity, CONFIG.jungle_hum, 0.2) + * BASE_DENSITY + * 0.5 + }, + }, + // Jungle rare solitary wild day Entry { make_entity: |pos, rng| { EntityInfo::at(pos) @@ -607,7 +641,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, - day_period: vec![Night, Morning, Noon, Evening], + day_period: vec![Morning, Noon, Evening], get_density: |c, _col| { close(c.temp, CONFIG.tropical_temp + 0.2, 0.2) * close(c.humidity, CONFIG.jungle_hum, 0.2) @@ -619,16 +653,12 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( Entry { make_entity: |pos, rng| { EntityInfo::at(pos) - .with_body(match rng.gen_range(0..5) { + .with_body(match rng.gen_range(0..4) { 0 => bird_medium::Body::random_with(rng, &bird_medium::Species::Parrot) .into(), - 1 => { - quadruped_low::Body::random_with(rng, &quadruped_low::Species::Monitor) - .into() - }, - 2 => bird_large::Body::random_with(rng, &bird_large::Species::Cockatrice) + 1 => bird_large::Body::random_with(rng, &bird_large::Species::Cockatrice) .into(), - 3 => quadruped_small::Body::random_with( + 2 => quadruped_small::Body::random_with( rng, &quadruped_small::Species::Quokka, ) @@ -650,6 +680,26 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( * 8.0 }, }, + // Jungle solitary wild day + Entry { + make_entity: |pos, rng| { + EntityInfo::at(pos) + .with_body( + quadruped_low::Body::random_with(rng, &quadruped_low::Species::Monitor) + .into(), + ) + .with_alignment(Alignment::Enemy) + }, + group_size: 1..2, + is_underwater: false, + day_period: vec![Morning, Noon, Evening], + get_density: |c, _col| { + close(c.temp, CONFIG.tropical_temp + 0.2, 0.3) + * close(c.humidity, CONFIG.jungle_hum, 0.2) + * BASE_DENSITY + * 2.0 + }, + }, // Tropical rare river enemy Entry { make_entity: |pos, rng| { @@ -867,7 +917,7 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( Entry { make_entity: |pos, rng| { EntityInfo::at(pos) - .with_body(match rng.gen_range(0..7) { + .with_body(match rng.gen_range(0..4) { 0 => quadruped_small::Body::random_with( rng, &quadruped_small::Species::Holladon, @@ -882,21 +932,36 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( &quadruped_medium::Species::Camel, ) .into(), - 3 => quadruped_low::Body { - species: quadruped_low::Species::Salamander, - body_type: quadruped_low::BodyType::Male, - } - .into(), - 4 => quadruped_small::Body::random_with( + 3 => quadruped_small::Body::random_with( rng, &quadruped_small::Species::Porcupine, ) .into(), - 5 => quadruped_small::Body { + _ => quadruped_small::Body { species: quadruped_small::Species::Hare, body_type: quadruped_small::BodyType::Male, } .into(), + }) + .with_alignment(Alignment::Wild) + }, + group_size: 1..2, + is_underwater: false, + day_period: vec![Night, Morning, Noon, Evening], + get_density: |c, _col| { + close(c.temp, CONFIG.desert_temp + 0.2, 0.3) * BASE_DENSITY * 3.8 + }, + }, + // Desert solitary wild day + Entry { + make_entity: |pos, rng| { + EntityInfo::at(pos) + .with_body(match rng.gen_range(0..3) { + 1 => quadruped_low::Body { + species: quadruped_low::Species::Salamander, + body_type: quadruped_low::BodyType::Male, + } + .into(), _ => quadruped_small::Body::random_with( rng, &quadruped_small::Species::Gecko, @@ -907,9 +972,9 @@ pub fn apply_wildlife_supplement<'a, R: Rng>( }, group_size: 1..2, is_underwater: false, - day_period: vec![Night, Morning, Noon, Evening], + day_period: vec![Morning, Noon, Evening], get_density: |c, _col| { - close(c.temp, CONFIG.desert_temp + 0.2, 0.3) * BASE_DENSITY * 5.0 + close(c.temp, CONFIG.desert_temp + 0.2, 0.3) * BASE_DENSITY * 1.0 }, }, // Underwater temperate From 4249d0ddccc30169aef7e082b21c73ff1dcb7849 Mon Sep 17 00:00:00 2001 From: Snowram Date: Wed, 21 Apr 2021 00:36:40 +0200 Subject: [PATCH 07/10] Phoenix npc + AI (WIP) --- .../unique/birdlargefire/firebomb.ron | 17 ++ .../unique/birdlargefire/fireshockwave.ron | 15 ++ .../unique/birdlargefire/flamethrower.ron | 14 ++ .../unique/birdlargefire/triplestrike.ron | 53 +++++ .../abilities/weapon_ability_manifest.ron | 8 + .../npc_weapons/unique/birdlargefire.ron | 18 ++ .../voxel/bird_large_central_manifest.ron | 28 +-- .../voxel/bird_large_lateral_manifest.ron | 40 ++-- .../voxygen/voxel/npc/phoenix/male/beak.vox | Bin 26739 -> 1248 bytes .../voxygen/voxel/npc/phoenix/male/chest.vox | Bin 2120 -> 8448 bytes .../voxygen/voxel/npc/phoenix/male/foot_r.vox | Bin 1156 -> 1628 bytes .../voxygen/voxel/npc/phoenix/male/head.vox | Bin 1408 -> 3392 bytes .../voxygen/voxel/npc/phoenix/male/leg_r.vox | Bin 1172 -> 2644 bytes .../voxygen/voxel/npc/phoenix/male/neck.vox | Bin 1240 -> 4536 bytes .../voxel/npc/phoenix/male/tail_front.vox | Bin 1136 -> 2768 bytes .../voxel/npc/phoenix/male/tail_rear.vox | Bin 1120 -> 3032 bytes .../voxel/npc/phoenix/male/wing_in_r.vox | Bin 1248 -> 1840 bytes .../voxel/npc/phoenix/male/wing_mid_r.vox | Bin 1204 -> 1748 bytes .../voxel/npc/phoenix/male/wing_out_r.vox | Bin 1272 -> 2648 bytes common/src/comp/agent.rs | 4 +- common/src/comp/body.rs | 6 +- common/src/comp/inventory/item/tool.rs | 1 + common/src/comp/inventory/loadout_builder.rs | 6 +- common/src/states/utils.rs | 2 +- server/src/sys/agent.rs | 220 ++++++++++++++++++ voxygen/anim/src/bird_large/alpha.rs | 22 +- voxygen/anim/src/bird_large/breathe.rs | 1 - voxygen/anim/src/bird_large/feed.rs | 9 +- voxygen/anim/src/bird_large/idle.rs | 13 +- voxygen/anim/src/bird_large/mod.rs | 38 +-- voxygen/anim/src/bird_large/shockwave.rs | 99 ++++++++ voxygen/anim/src/bird_large/shoot.rs | 136 +++++++++++ voxygen/anim/src/bird_large/stunned.rs | 1 - voxygen/anim/src/bird_large/swim.rs | 102 ++++++++ voxygen/src/scene/figure/mod.rs | 74 +++++- voxygen/src/session/mod.rs | 1 + 36 files changed, 853 insertions(+), 75 deletions(-) create mode 100644 assets/common/abilities/unique/birdlargefire/firebomb.ron create mode 100644 assets/common/abilities/unique/birdlargefire/fireshockwave.ron create mode 100644 assets/common/abilities/unique/birdlargefire/flamethrower.ron create mode 100644 assets/common/abilities/unique/birdlargefire/triplestrike.ron create mode 100644 assets/common/items/npc_weapons/unique/birdlargefire.ron create mode 100644 voxygen/anim/src/bird_large/shockwave.rs create mode 100644 voxygen/anim/src/bird_large/shoot.rs create mode 100644 voxygen/anim/src/bird_large/swim.rs diff --git a/assets/common/abilities/unique/birdlargefire/firebomb.ron b/assets/common/abilities/unique/birdlargefire/firebomb.ron new file mode 100644 index 0000000000..def79061ea --- /dev/null +++ b/assets/common/abilities/unique/birdlargefire/firebomb.ron @@ -0,0 +1,17 @@ +BasicRanged( + energy_cost: 0, + buildup_duration: 0.5, + recover_duration: 0.35, + projectile: Fireball( + damage: 100.0, + radius: 5.0, + energy_regen: 50, + ), + projectile_body: Object(BoltFire), + /*projectile_light: Some(LightEmitter { + col: (1.0, 0.75, 0.11).into(), + ..Default::default() + }),*/ + projectile_gravity: Some(Gravity(0.15)), + projectile_speed: 60.0, +) diff --git a/assets/common/abilities/unique/birdlargefire/fireshockwave.ron b/assets/common/abilities/unique/birdlargefire/fireshockwave.ron new file mode 100644 index 0000000000..e7e1f67a0f --- /dev/null +++ b/assets/common/abilities/unique/birdlargefire/fireshockwave.ron @@ -0,0 +1,15 @@ +Shockwave( + energy_cost: 600, + buildup_duration: 0.7, + swing_duration: 0.1, + recover_duration: 0.2, + damage: 200, + poise_damage: 0, + knockback: ( strength: 25.0, direction: Away), + shockwave_angle: 360.0, + shockwave_vertical_angle: 90.0, + shockwave_speed: 20.0, + shockwave_duration: 0.5, + requires_ground: false, + move_efficiency: 0.1, +) diff --git a/assets/common/abilities/unique/birdlargefire/flamethrower.ron b/assets/common/abilities/unique/birdlargefire/flamethrower.ron new file mode 100644 index 0000000000..4bf507fa86 --- /dev/null +++ b/assets/common/abilities/unique/birdlargefire/flamethrower.ron @@ -0,0 +1,14 @@ +BasicBeam( + buildup_duration: 0.4, + recover_duration: 0.25, + beam_duration: 0.5, + damage: 50, + tick_rate: 3.0, + range: 15.0, + max_angle: 22.5, + damage_effect: None, + energy_regen: 0, + energy_drain: 0, + orientation_behavior: Normal, + specifier: Flamethrower, +) \ No newline at end of file diff --git a/assets/common/abilities/unique/birdlargefire/triplestrike.ron b/assets/common/abilities/unique/birdlargefire/triplestrike.ron new file mode 100644 index 0000000000..2eaa09c043 --- /dev/null +++ b/assets/common/abilities/unique/birdlargefire/triplestrike.ron @@ -0,0 +1,53 @@ +ComboMelee( + stage_data: [ + ( + stage: 1, + base_damage: 100, + damage_increase: 0, + base_poise_damage: 0, + poise_damage_increase: 0, + knockback: 5.0, + range: 4.5, + angle: 30.0, + base_buildup_duration: 0.4, + base_swing_duration: 0.1, + base_recover_duration: 0.3, + forward_movement: 2.0, + ), + ( + stage: 2, + base_damage: 80, + damage_increase: 0, + base_poise_damage: 0, + poise_damage_increase: 0, + knockback: 5.0, + range: 3.5, + angle: 30.0, + base_buildup_duration: 0.4, + base_swing_duration: 0.1, + base_recover_duration: 0.3, + forward_movement: 1.5, + ), + ( + stage: 3, + base_damage: 130, + damage_increase: 0, + base_poise_damage: 0, + poise_damage_increase: 0, + knockback: 10.0, + range: 3.5, + angle: 30.0, + base_buildup_duration: 0.65, + base_swing_duration: 0.1, + base_recover_duration: 0.3, + forward_movement: 1.5, + ), + ], + initial_energy_gain: 0, + max_energy_gain: 0, + energy_increase: 0, + speed_increase: 0.0, + max_speed_increase: 0.0, + scales_from_combo: 0, + is_interruptible: false, +) diff --git a/assets/common/abilities/weapon_ability_manifest.ron b/assets/common/abilities/weapon_ability_manifest.ron index b952cbe08a..d990e18d61 100644 --- a/assets/common/abilities/weapon_ability_manifest.ron +++ b/assets/common/abilities/weapon_ability_manifest.ron @@ -212,6 +212,14 @@ secondary: "common.abilities.unique.birdlargebreathe.triplestrike", abilities: [], ), + Unique(BirdLargeFire): ( + primary: "common.abilities.unique.birdlargefire.flamethrower", + secondary: "common.abilities.unique.birdlargefire.triplestrike", + abilities: [ + (None, "common.abilities.staff.fireshockwave"), + (None, "common.abilities.staff.firebomb"), + ], + ), Debug: ( primary: "common.abilities.debug.forwardboost", secondary: "common.abilities.debug.upboost", diff --git a/assets/common/items/npc_weapons/unique/birdlargefire.ron b/assets/common/items/npc_weapons/unique/birdlargefire.ron new file mode 100644 index 0000000000..b092a15f0f --- /dev/null +++ b/assets/common/items/npc_weapons/unique/birdlargefire.ron @@ -0,0 +1,18 @@ +ItemDef( + name: "Bird Large Fire", + description: "testing123", + kind: Tool(( + kind: Unique(BirdLargeFire), + hands: Two, + stats: Direct(( + equip_time_secs: 0.01, + power: 1.0, + poise_strength: 1.0, + speed: 1.0, + crit_chance: 0.0625, + crit_mult: 1.9142857, + )), + )), + quality: Low, + tags: [], +) \ No newline at end of file diff --git a/assets/voxygen/voxel/bird_large_central_manifest.ron b/assets/voxygen/voxel/bird_large_central_manifest.ron index c09bbae0b3..f8a0999986 100644 --- a/assets/voxygen/voxel/bird_large_central_manifest.ron +++ b/assets/voxygen/voxel/bird_large_central_manifest.ron @@ -1,53 +1,53 @@ ({ (Phoenix, Male): ( head: ( - offset: (-2.0, -1.0, -0.0), + offset: (-4.0, -4.0, 0.0), central: ("npc.phoenix.male.head"), ), beak: ( - offset: (-2.0, 0.0, -1.0), + offset: (-2.0, 0.0, -3.0), central: ("npc.phoenix.male.beak"), ), neck: ( - offset: (-2.0, -0.0, -0.0), + offset: (-4.0, 0.0, 0.0), central: ("npc.phoenix.male.neck"), ), chest: ( - offset: (-3.0, -5.5, -4.0), + offset: (-6.0, -9.5, -7.5), central: ("npc.phoenix.male.chest"), ), tail_front: ( - offset: (-2.0, -3.0, -3.0), + offset: (-10.0, -14.0, -3.0), central: ("npc.phoenix.male.tail_front"), ), tail_rear: ( - offset: (-1.0, -3.0, -3.0), + offset: (-9.0, -30.0, -2.0), central: ("npc.phoenix.male.tail_rear"), ) ), (Phoenix, Female): ( head: ( - offset: (-2.0, -1.0, -0.0), + offset: (-4.0, -4.0, 0.0), central: ("npc.phoenix.male.head"), ), beak: ( - offset: (-2.0, 0.0, -1.0), + offset: (-2.0, 0.0, -3.0), central: ("npc.phoenix.male.beak"), ), neck: ( - offset: (-2.0, -0.0, -0.0), + offset: (-4.0, 0.0, 0.0), central: ("npc.phoenix.male.neck"), ), chest: ( - offset: (-3.0, -5.5, -4.0), + offset: (-6.0, -9.5, -7.5), central: ("npc.phoenix.male.chest"), ), tail_front: ( - offset: (-2.0, -3.0, -3.0), + offset: (-10.0, -14.0, -3.0), central: ("npc.phoenix.male.tail_front"), ), tail_rear: ( - offset: (-1.0, -3.0, -3.0), + offset: (-9.0, -30.0, -2.0), central: ("npc.phoenix.male.tail_rear"), ) ), @@ -61,7 +61,7 @@ central: ("npc.cockatrice.male.beak"), ), neck: ( - offset: (-3.5, -0.0, -6.0), + offset: (-3.5, 0.0, -6.0), central: ("npc.cockatrice.male.neck"), ), chest: ( @@ -87,7 +87,7 @@ central: ("npc.cockatrice.male.beak"), ), neck: ( - offset: (-3.5, -0.0, -6.0), + offset: (-3.5, 0.0, -6.0), central: ("npc.cockatrice.male.neck"), ), chest: ( diff --git a/assets/voxygen/voxel/bird_large_lateral_manifest.ron b/assets/voxygen/voxel/bird_large_lateral_manifest.ron index f739190d0e..3eca741462 100644 --- a/assets/voxygen/voxel/bird_large_lateral_manifest.ron +++ b/assets/voxygen/voxel/bird_large_lateral_manifest.ron @@ -1,85 +1,85 @@ ({ (Phoenix, Male): ( wing_in_l: ( - offset: (-6.0, -5.0, -2.0), + offset: (-10.0, -12.0, -1.5), lateral: ("npc.phoenix.male.wing_in_r"), ), wing_in_r: ( - offset: (0.0, -5.0, -2.0), + offset: (0.0, -12.0, -1.5), lateral: ("npc.phoenix.male.wing_in_r"), ), wing_mid_l: ( - offset: (-2.5, -7.0, -2.0), + offset: (-7.0, -15.0, -0.5), lateral: ("npc.phoenix.male.wing_mid_r"), ), wing_mid_r: ( - offset: (-2.5, -7.0, -2.0), + offset: (0.0, -15.0, -0.5), lateral: ("npc.phoenix.male.wing_mid_r"), ), wing_out_l: ( - offset: (-9.0, -8.0, -2.0), + offset: (-18.0, -18.0, -2.0), lateral: ("npc.phoenix.male.wing_out_r"), ), wing_out_r: ( - offset: (0.0, -8.0, -2.0), + offset: (0.0, -18.0, -2.0), lateral: ("npc.phoenix.male.wing_out_r"), ), leg_l: ( - offset: (-1.5, -1.5, -1.5), + offset: (-4.0, -5.0, -4.5), lateral: ("npc.phoenix.male.leg_r"), ), leg_r: ( - offset: (-1.5, -1.5, -1.5), + offset: (-4.0, -5.0, -4.5), lateral: ("npc.phoenix.male.leg_r"), ), foot_l: ( - offset: (-1.5, -2.0, -4.0), + offset: (-3.5, -4.5, -9.0), lateral: ("npc.phoenix.male.foot_r"), ), foot_r: ( - offset: (-1.5, -2.0, -4.0), + offset: (-3.5, -4.5, -9.0), lateral: ("npc.phoenix.male.foot_r"), ) ), (Phoenix, Female): ( wing_in_l: ( - offset: (-6.0, -5.0, -2.0), + offset: (-10.0, -12.0, -1.5), lateral: ("npc.phoenix.male.wing_in_r"), ), wing_in_r: ( - offset: (0.0, -5.0, -2.0), + offset: (0.0, -12.0, -1.5), lateral: ("npc.phoenix.male.wing_in_r"), ), wing_mid_l: ( - offset: (-2.5, -7.0, -2.0), + offset: (-7.0, -15.0, -0.5), lateral: ("npc.phoenix.male.wing_mid_r"), ), wing_mid_r: ( - offset: (-2.5, -7.0, -2.0), + offset: (0.0, -15.0, -0.5), lateral: ("npc.phoenix.male.wing_mid_r"), ), wing_out_l: ( - offset: (-9.0, -8.0, -2.0), + offset: (-18.0, -18.0, -2.0), lateral: ("npc.phoenix.male.wing_out_r"), ), wing_out_r: ( - offset: (0.0, -8.0, -2.0), + offset: (0.0, -18.0, -2.0), lateral: ("npc.phoenix.male.wing_out_r"), ), leg_l: ( - offset: (-1.5, -1.5, -1.5), + offset: (-4.0, -5.0, -4.5), lateral: ("npc.phoenix.male.leg_r"), ), leg_r: ( - offset: (-1.5, -1.5, -1.5), + offset: (-4.0, -5.0, -4.5), lateral: ("npc.phoenix.male.leg_r"), ), foot_l: ( - offset: (-1.5, -2.0, -4.0), + offset: (-3.5, -4.5, -9.0), lateral: ("npc.phoenix.male.foot_r"), ), foot_r: ( - offset: (-1.5, -2.0, -4.0), + offset: (-3.5, -4.5, -9.0), lateral: ("npc.phoenix.male.foot_r"), ) ), diff --git a/assets/voxygen/voxel/npc/phoenix/male/beak.vox b/assets/voxygen/voxel/npc/phoenix/male/beak.vox index d3fc53871879bb0d9bbc1ac27f062039eb6aca5d..bd6b55aab0c8375174672330d87f5e50b6db3dd8 100644 GIT binary patch literal 1248 zcmeIyJxat-6vgp-&yAuYMk~Qe5UoVlFoI|ZMsPsJS&CZ_Yy@S{+SRxQD=P~jf)>Ke z^&fYT+6nx4=Rn>EFObXg(e}NWoeipn!CnfpVKqKp?FtXo77-X-jjJcd?%9LOTEW9U z1uy7u^HwnT)*%E;ba^QR*@7N)1lbCD;cG&4d4ahw6Kkpzg3QLOt92pBCc5Yd^2N!~ z;P2p^v99avYj(St)7Qhy?@8~uefOHb-~ao}@9clv*|n5gX1R+4oXYZHSvf2$A6Jw& eOP&q+c_Z?Dr^uhC%-Li=r*BPuecWaDpZW#5IRhmC literal 26739 zcmc(ncYGUX8OJYi>MoXP*-N;EGR;XR)ge%*Q-vgLRvo(}q2)MBck(roPCZF>!lo=k z2eiD& z_hqg%3}ffm_Fece>}kW;yZykHTS)=i2E=iQb%=fY4{R^uQ?}csAA`u>vK}#D7=x%Z zxNaTFhym;y>rqCmM-0gR0QM2~|S^x9vGq64>OpY|F=Ff5+JV@w>`SXOm;nZ@cc)Tq&3$wX&8ZkExYi zT}zyfsg+w@ON@@GWv#9yb8$?q{OVfEv(;l?SY2yyE!LT5pzIF4ME)LX? za+1W!k(+PAr-!hTBk?7YLf|cq5Xi^qQjF>-CK2)xb|T~>(PE8`bNAQW@6TwG^ z06q}}A~Y5O908)kf`buuBKSn`iQuC{1)m5$5qu)}Y&#u!bUT7i1fK{#5qu)}$dE?x ziQp5#CxTA|A01{L8L^RXq9}7Mfd@T4_ z@X=+ZD@JBXMn-J#f{y_o13m_P4EPxE!RZL(;bXwZfR6zmAQ8yJ$AFIk9|JyU27x?$ z4EPxEF(8Ac5y%5*;A8_8myldUEFk6)bBI~QBm(#1WEe4oFqRODhy}zvVh%Bjm_!s2 z!-ye-v4mJeEFk6)bBI~QB%+8IMhqoOOH0Y(;$pI}u#n8p&nI(pbII)NY%)1HnG}n~ zWO#Tu85$ZwUHREDvEw}1!6x^o<)0aL>u|$80SUY8dtSTg%1y@(gLY+FZg}QMR%+UQ z5T->-ZkBdglIDn6Sc#P*1gx}6zzULpl{-2CE3-nt%Fhl0R@M^%dm;yW5(j%S2P}@$%`2tHATip8@*xPfkr*N?H#hx&>ci>>}$id!;gS|5cdlwE?euX2f zgu8LD^2;y5_8uJUJvrEWaj+Q<_TC&UzQ!|F!f71r=^X5RIN1AguxD_v_v2vi&%r)` zgMA!B#lfX%5!sU@zui4|1?o4z|X@);ZXK zgKcoImvFEl2ixReTO4eggPq}E@xK-s-^LGdu!lL=$8)ex;9#G~!9IzDeKH686b|;O z9PHCL*h@Lsr*p8+;9#H0!9I(FeKrR>$H6{_gMBUsdl?6NIR|?M2YV$4`#cWz`5f$3 z9PHH`>`OS}xpK8#vh4aHwXJ34)(nq?E5&_1rGN89P9@;*bj2BAL3v?%)x$ygZ(H6`!Npo z;~eZKIM`2eu%F^!Kh42@hJ*bq2m3h=Hs)YI&%u6ygZ&~0`y~$c%N*=iIM}apuwUa~ zzs|vagM7;q?{ly};9zg&V1LNL{)mJ9F$en-4)&)U z?9VvZpL4Lk;9!5r!TySa{WS;s8xHoj9PAPY`#TQy_Z;jWIM_dOuz%uUk8rSm=3xKA z!QR5b{*{CM8wdM$4)z}$>_0i!e{rz?=3o;JR-RP50qElp|Ko#d!`++Dy$~4L1lL6# zkAgiFwCkmw$Kax6BX^=4Yc||cyXu6JDV(60lGAFrX+lqT>uy*%EEOzi=QQldwjR-K zNBUvA^%+gnf^z!6a;ff1U2KDh-s*nc6Nc>%;N2V<)xyD(QOFevnNG-Qwo|WET{*d6 zW(uZKoX~ZoQ9P!1R1;;tc~BfpX>=4jvI)7j(RSVc?~Mv#mjYJJkap9bsw%VLw5Cr* zt!>3|+stR}Jt*5->N6wLk8Z$(M0IfQbck`CC$g0>o1c3GX4OL|_`Oou+3)qyN_T;gKTTG;eP44mD7fm@o4V5ogNwDk4z?G}2qIt5)#^0du+pBtH zqu6xj&74-jHQ%T@dTUbMOFIq8!EQq-%x$#*TQO`prCkYKJX(^D9W6~&>{3t3(USV; zOv*Xg3|@M2$*oqULN+{SY-hV;BOh{t){uLn^Z^UGsT40LwRPd^plhmoyVI09hfdjV z>oPR8KVH~ww%Rq>H%Ic>d@h^U8Sh>$D+R=;uL5!f_f)ikV diff --git a/assets/voxygen/voxel/npc/phoenix/male/chest.vox b/assets/voxygen/voxel/npc/phoenix/male/chest.vox index 86e1d808ba6ae9883cc5eb1954cd2ca999d6f156..017abd28ebdae0c766471fdc8c7926474cb23a92 100644 GIT binary patch literal 8448 zcmeI%%c^ZjRtDfX<2FV_tlLf-g4ie!ds03_356zwDOez7J&La&*b$V%zE9z^*s>)e zHqO4}fvWAd$|uOSm4}ZQYmEQQ9dquvZYTfduYU2Pe>#riFMsmwcb4P7esml^|Mr(Z zy*wZI@45aScYg7=zx?*!{!rn+AIEX@e!LIbIKK9wCe$rF4vz_S%j3Q0@j3`~oqAJm z>JjSHt4n_!uYDaKgHWel-TM3Z=nR6?s~7!?<6QI!bv}Dn$6ye&oBiBm2z9&rI}GAk zKaRISujj|0wexFl=j$NUwVz)bz3UO`qz--hIGy|H-1RA8MvL>kGYC>|Qqnxn_m<~t zXAq>`G%M45p0915ADuyvdeiJ&GSnrrWEcdgH_aC3K0nGn->bFJ83f5RTh{a9cvRZO zo)`qFH(8ulx$F5_ua2kg41&~~W=nnE{9G#SCdVL1y~(27{Y={3b+QgZoz_WxIKPK| zKg0ZlR=Prlf{-jX8Aa$^R?D#57YTD2vTpFEtzdSR&g&m{&N1ywX=2xLF!HGlG(Dy zwx8|z+up5hyVeYX)SG6@x@C`x{ajp=i{H(~xvD#ZAoZr%Qg7MgDu0!~%3tNJok5U# z)4F8#GkcT0IoF%L&=~~DG+Q#~UG6S-m!mTXl4-V{nTO27b)!FAH$t7v=j_Go#Wj6# z&V)Ljy>H4}|E7Eygt{H|Yd`AyAk^ioul1~7o3%3tQg2$%B2#2aeQ%|@e`N@D>P_{M zIjdZiv$iT%{R~2#dedxK&sN_{lcTA7tM2LQp04kyH(Bb<(d)#}ayW7*G=r>fg>z6^rYn`VoxKDPaA*L^$RUGI&~AV{X! zlG(Dy#eOdKPrb;~83f5RTWs0mDu0!?)~h_7L6A(dC9|!^&F|}0U(3z;-sI^Ff@GR4 znJs(V?VoaY{JT7zL6A(dC9|K|hjXRU9Ue}Y^`>>nY>w7EXPWl``6h4eO`cGv-n1?@N4MYZ_*5Fb%Mf6aC{1N z$dBeZBTuMPZ<;MON7#SJNAsL%-Xq$*GYC>|nl0;=J<@ea*CpkxE#(;msW;7*b;};p zey06f>umQ%XAmUQY{_ivvE-NIFXy}D=?sEonk|_vdu;pJ^3*L)XAmUQY_VmJi|a|b z$Y12?41#2uEt$=6b$uvT`?<=~83f5RTQZyDCVz81t@S2PXAmUQY{_ivakn4J-SzQ| zdH(4Pf@GR4nf=T@?B!uE5BK&%mO+qu(`-F6FYQZx@%wp^VGyL=e9pdUU)ncj41&}v zM}MufqkG-tc>*PH9N1>kGdyLE}L#R`4nk~-K-%IJ9JKb|fuE-JU)SG6@ zdRE7yR>!OTwO2VpoqE%3v1|9->7G0CRCR0eO`cGv-ZWcmwJ!}mPxaT*?W6aPX6;>` zP^aFsE}1QR40+0M4QTWsPpFg0lGz;LJj3y$zgKIE?ztmRkb2W>S-0$w_Mi4oRkxI9 z5TxETTWs|)9iKv-?g~1CAem;1&9Ur%*$;Ke(-{QGG+S)hV>|wKzSg?s=?sEonk|`a zJuZ%a={~CE z=D6GcU7qJV&GVV_{4)rWX|~wU>_hh9clB^DJR}(e`RH%cY&|nC{jJhoWEcdgHz~(> zugCD&%g&DBIWjy)hUdt5n`Y-ZUduVY)^mKcQ|6Q*)TuYkmdsh?ik!6-xgtlXQ*W9r z>sg1-V#8-K$E)MLS2;qRdedyloK3FDHMu6&< z-ek#akB{N_L!LV1hy0Ky)TuYk7F+fR*CCu=I6rj@zZC{S>P@pHv(-n+Q>f|qDbFBC zy=k`C98;b`on9^J41#2uEjGup|7Abcy5#8$f@GR4nQc9`O`S$kQ1F$uwIso8#*ESNZGss8@M9gCLn^OJ>U+H~E|2ueILf=?sEonk||A z%-&s_yS@DOokZ69{1YV8Y{{ID@ufV*TcthZ7zC*|Su*`apL&G*-m~#=Bto5f<&3v_ zhW7-s(~MB3-ZVQ)_}&p8)!K?oks;KnH_evyta4OZ4fnbnp-#QY;%stMT9a#XggW&m zi?fG&Jlx~q^b5$e>NW=nmH@Y)|A#ad}2y!Oiy>eQPonQc8Hy!Jb`wbH^X4uc@| zCQD{pkF=L`{B(R;$}@n?UI!9}rJ_*nn1j#g8GTVAA`&shTB~ND%B-3oM zWsfbt!{;W)-|}<@K{Cyj%$7Ya@)zezy~xuU1j#g8>}U3BFIU(7DobY&B-3okoHu*8 z+4D_~&LBvp*^)W$em2i<63d5TxE@$@IgsfO_Uz zJH4}<9uw-+D<$7*$&XgdiW#9!y=ivV^qD{1r|CXTpZR47buw8pXOnAkGuGG+Q!fq|fyETCFY8_h30foqE%3 zSeQQNi!NX3Kic`9Eb&_lvzv=glBU zy=k^&&Nbi4a=azCP@pHb6(^w)BPfMkz){~-ZWb> z=XHK5SGlY6y~;5NQg50qne!%hle;+=;U_4?`E zlU95RWf0Wr8(`IA5Tssd>-*Zw=*^5!C)2ES^S-{c?(Z@PQg51-vA$H=FlP{?-lW8O ztFgXsVa^~(y=hj`->0QHgCO-LW%~KFY0e->y-8W-YrVB)&LBv=X;!x5Z-0L~zim#a zQ*WA;i~U{9FXk6>LY;cktX$2n_IGuB20`l0=j`V1-pm*Tsn^SX3|jR~fI(23Z{bak zL6Ca2>(jeFp-$@9-?yPZ^a*w9)zA<9&?nTXS98C$)K7gvoqBcZPyMM+s8g>l{iVP3 z33ck#t-tlRKA}#%da+OXMW0Y7^{Rh$UJQcN>-F*&w03dM7w3F&&X;@Z9y18)xZE0j z=n?9qhCV&?33XCapPu@JI;m5iKJ^K8QkVYnGZ_S_SGWGwXAtDGchO@Iw5!Jq;#s%r zW6-;mHm>gftG!?C{d#ZVF@vC`#|(ltJ!TNJ!|NT!tKi_`!ga7Ui{aL@u|M1_>@2=VZk8AdK zrF{OuaeV$UKjtsW@%hiQeEz&0pZ~JT=ile~`JMV-U+U+7{!#n<`rZBWH^2Im&tLz? Pcb~8S{g0o|SAhQoBu^Sf literal 2120 zcmXZcL2l$k7(n6QRe!lVf=zCKB5q(7h!ul^h7kfJAP8o|&U;q5LX@60QXPR@Ay=p? zMB)Tlu{WO$-SfI!Z>sIrWh=k@@ardkM#Oh--hFR~&(0A)y?g)l$NR)<#z&yz*Wcd1 z%XF~6Ce(Orh-rFkh%!A+NHj#9E>oQz3n~(gi{wzz5N*1YW{&1~igGDY9xKvuOyzMx zK}Dh=3Xg({L_<^_1r>>g$a0yIM@6C`n)N8CNVI)Eo1>hJh^byCl&R{dmozqG6uwj) zJQ9tI%2ys0iN;0N#|Z@$iH2y_Hy#yrqgVXlHA$ns&~Z#!_az^QBZHBIgfNnx{F^apR)R?)R*VN1|~t^D~c%L_?fr`^lps z(GX|r&pav;4bv5OHQQ6pSn_nKd9oKL*F3rA>5|5nITTbR8m80fQcfJw@pQHZ6^S3; ze)T5)|9h|rULA*;#Bo@1Mjo!M*l_X2+3 zdOfS2RnMwt)pOhqJ*%Eo#Oi{plOMOpmsSf_lU2`Ze~#OxXVtUnS@rz<_KWS?FF)UY z|LN8(<@Kz3Rz1hkaHqG!>w=vnj}kA|K_&!T4$ zvgkP;7bjbMaj|G27W=b5LW})b^elQ7J&T^>u{xfZ{Vwf0v)6dTp74Ns+~F2CxZYl0U$>W+m+krad3$<#+8!Prw)^}0?e6YwyS=^L OZf+5T4BmM;T5)Cra2S=iTLre-sqV@KgHnP_{P~|*2 znNwYtpjznSh)~W&Mx=5cGP02e#t&qwj$|ZsMMn0zhTCjmd{$d5jkS!ms0{;s;kqM1sMU;HapbXY;XkB8*>_S+V7xh@E(rn zIms{^m@_bbm<^7A#=t(b5oAr?!x234x2`VU;UHbdU6oK(3H0%oQTh z3uM*(@Ok3=vHiy$e;)hW=ew7GBI4uK%_pn)c^+|lbN}x7yyJ5w7ov{4@Ao&C*7N6u zOc9UM6c^LNA!pXI5C=a_=eFndv*^=e@}Ddg57Tt^FfGo0f4P2pH3l_KAD`Br{{6L8 zrSzxvtM_KJ`DrQoR>a|Ud+O6h>QkTk`P8bf`f8`@tKO;?ebE^%sXUdZ@>HJjICv^gg{UrA4L%-`ld1!%oyt?q&vj3(eo6Z!c%w(PvIG_0Z-v6JcXz5 zj8}uF@D!c`Qh3JeVzA=G;-I1^=4XC|iuox#g{SZop7C0ZH)eiH^UTZ|3s3L_Pw)g! z@Hk!cJi!w@!DC3#^8`=u1dq#zo+o&MCtzG!^aczkhXYT_qa0)}JK4%c*0PeN#7T~F zkiG0=D;rtMN|q8QIm$uyvXiZBWGyRM_S5OqkH=#_91eZI-}l{a*SFhk-)uI0yh{$izJ7Nb3Vy%T)~xE)dp$>IL=G^X?Lkfsl6h&B&suqw4pTnMoUiI}v8YQ{J*)ds z+|OR^65vNY)sz>1(36^vu}^!A-p<+E#i~EHIiCH&PRSlNwW~u+ZOf@0t@ld= zYeXJxZch#|w`m#QNdar9jxW%&^q>t7tJ_J##JYO@$nw` z)L#u`@~g71&T(y%KW56+k9N}TFVg+m7`2egcb)UOmERP<=#;yrUuA%=TF=*(zG?W2 zamLEH<=8XvVX#j4F(YmfvDYM3XP7VpVq~w&4H%k5pd1Ay< zh^wZMo5FX@d(&emRU3P~+PAF`RILaiuf=zOE2;_uD3}wbP z%hkXkAkj7AD2tqQk$*wTt9I;+4uPDEg{@6`lS#P) z*e4~|bPwc2SJ>LD)C}g&dNxLfKu*TO))qaLe4IB)`j^C>q8(Xx*E2XTfN%ZR_+%Ba-x$19yl-u90GFaHp4fp1A*-LC1&m} zehK`PqsA}!TGX+mz6E$E-v;O{sT~m19_Km6yYvaG(Fdq4)X1bp7WFZyhXGz=zH@J! zdxM>NT8xHXSJqrdAE0hQKvIi~8tmwUx@j|eK@hiu8nIg(zfg-8>K>y*ASYuD+e#N& z*<-~9ft={%K)pI}KMs5^1ac1D=J>@g4l(3~zBoww5XAV-49*^$J zqfS6LH8X^I4P(s4J!#yT>CM4xdGJBZZ#Jk2KTXt~`Q~7;+#D>I&B1cTr|Z#beQ+?Z ztAknX4rcw$!L;8TOsCDkY`Qs^&PN9e_JLga?*iQB6WxvfGW=Jde+d0d=pn!R&96Rd z{=ff0@87?_e=Ohq`A7F}|M`pi{qoI6&;8iFet7@m#{`{N+9>z<&WD3**cH literal 1408 zcmXZaPtF@P7{~Ev?4%TlO>Tf9Zr}~7SWpyHULioDq$=I8^NJ0tTp=ew@)5`tas^)@ z5+}&2;?2%{#%d-{rNEH6C~OR`R~PI_d|Nb00U){bn2tVY(3O&ZHajjSEp z^t^17b&_?ot+HN`GT;12FpO`0EIFGenV-$Ij7kLpBx$g*h$U`1- zIufi`Lq&ny58bWv>3lk$&S&cJ>3lk$&ZqO4%6vMX&ZqO~eCFTa)A@8holobp{XQPvg`0%q`&42sDc|6E~C0ZSiB%XnY!<#%FF7pT?*0X?%Y8O73qr zK8;V~)A%$#>y|b?jZfp#_%uFqo9E7~HtSU&w?CCnC!=4-&G@F{!>pTcLp8hi?$!lw`^eCF$7vSMJ-sR)YwSzn=Ie+r+%r|>C!=4&^g#RX@aaKr(7?8g0mAGh0WT(8%0xm?Ei Zd>*IMX&jHoaX1{te!m~P-A><#{{bIzp^E?j diff --git a/assets/voxygen/voxel/npc/phoenix/male/leg_r.vox b/assets/voxygen/voxel/npc/phoenix/male/leg_r.vox index 2390b7c75752cd1c539c2b0407e09b95bc739cf7..7e732b8c74879b6437d0acd7d2897d3a1b0b7832 100644 GIT binary patch literal 2644 zcmeH|&59mX5Qa}xy;XHO1k{z_N(k=6+(Hr%9fpixKw>T>TbF_x!4R^~Wo6~cMRuKT z8dvenXdhX5sYw7`mbF;6j1F%Meg-gZQ@UK@L`wTNTdD#*zk z3w`ECm~S#)(|kAg)gjz>xo_!y6pV28>$?XYh&ZVgI^?T?L&y(7%J-6Q4I}Z>4^LyYLzhqS+6c&6>nMowG20QiHnW3;TjxSZBm8FpJeK_CN(` zu?>&aJh+eGoQ(6}K7w<`&Si5>yoe(}K@VDJQy-#N_z=Q}Brdekk@P2iGB*g4TA_`O zqQCGVDmFnvkvCZ5K{Os@vo?7X`((~Zea4=|HQD3B8j#>aoLh&6wnWXu{Za!NHPc$e z!^J)0L4X|3#rd^1Izp=jJ_IQ6sTFafL)uLE5Yj3%(SjEJNq^!)fKo@Mcm_HetneU; zUdY0OF1*HrXuM`!w9vlPr#L@#$Vtx>=hvo%e4%F~cb=R*x$|7Z$c^0cC}Q*E%*mOl zdvfQwl_zJ;!vYTi6!btrlbi4$K*}|Rr{unqyT*e6C3B($op{AQ3J*k_Xccp!-s}(5 z$-9}Hk$n)yOzxMlld+(MctJ<)31P*Vh^@HGI^3y&AfddrMlEXdzym>&7ur$$ev5Y% z4+N)cBKlGS-^!O0(DIo6@{%%|6MY z$UT~GA2moQa4}BQAfceud^?&wQHO+r&fdJ3qnh7Q^L(4{2K~)ANC=;-IoCuzChJYU zU6b`D&tUQlCf{>>w8%Xt?=3xOqIJoOo}vf+`G-djhyMHDaD9D!-j)x4e0jcqbN@Ww zy}WhrZP)rV{`U9t>6!gMp4oHv^Q$q=Z{R!l(Z~5>@$-qs`Rb0JzfU`Fh+lsZ=Rfy! Y9ygzzA6|ZYzW@8?T;KkAUcLYy0VgkT>Hq)$ literal 1172 zcmYk)J+2cm7zNNT10_ww22jKf+#o6*3W6uZkANVcp>t1_mNE)G(zgV%g>1oFh{Os~ z(VJt_kjc%=-1$4P=i{4C4}M0(tDD=`R`LCA#QWPX&+d&KpF98M)8{X@Z)Y2E|KIw! zia5q9jw@NldWj#aSREIM)k<$$VzDjDSjXbn;@IMt^|Bn_J%4%=o;^N&ep>(Y`|msWk%S36Z-^;W&;i@xZKqUe2l=tCd+(Bo)i zEGbor=pAT&8c*YCJdJ1acp6XRX*`W*l6e|W<7qsNXZ{VI#?yEjPvaSXt?@LT#?yEj z58afc-{eW2Bi|Mb66Zq8_APA$uozFCwY=5d0sr!``gKrJjs(h$uovD zd6Fl2k|%lQFwc=0nz0no8=uNkc`8rksXTK!cq&hYs2*6|d`^*zsuQZ6%2SQcoHm}y zQ+X=Shv$#`%O{We*Y{gfs^_UZm1j=xoTV``Lo*gVPvI#%g{SZop7|Q^6rRFUcnZ&a zHFye7;VB@6XTC0OR$N$|R20Sdj8~`_pTbjk3Qysguho2G#w?AI85#>u@B~lr1W)j| zT=YD_6FkA=mZIkgp5O@{j}bjj@B~l5c(mxhx4-%FM}HNO&pv(k0VDtTMC6NiU;XkpANU(E zeZunP@4kBXhZ7Eezju-7)3ZbL5qo@&JU=@av7B0+p98Ws?8fu>$_I2dH`3=Rfa z^a&0InLLAoff7?}VQW6~?gE3HjKi}*^>WW%_Vm-;o4);c);QfQPG^UJgd!GbX~(Vl zuxB60=|Df^mFl$cChKwSJBqdtJhoh%KWTn2g6rlHpabcy;rT*td)Mz!SI!s zpmnZW<;r@l%nv#kC^13nKo8mnIye|8F+poz&*kXY00#plCfKr-`9T8*1BFujh4lnq z7&|DjT*k9OuDu&LYiZO<%vzxq!cLom#jKm-&6p!)t^_)c)C_e5f?v#A@tfqg0X{9i zcd(QfPk9YUd`*3^)ZJ70jh+*~h%^MH~`e++WrGX6}sFnLDR$ow@W|yT3>{??VrC2uLVmjV6hn=!p&i2}P`@R{m;* z1_230%$sKri&wSS&4P6a)+bn}V7+2*3&s~O>cRNXAt0fMHJU^)zPMVTLqI|iYcxsP zC;cZn1SAx(o>~Rti<=i3B!10U^Lq%$=?{Dm%~#cAFXh=GAfbr0=8G7*p&L2`BowhW zbk8UC+>OJ~At0fMHJSu^B=e090SQH{(Ik_6XrJg1kWj=LO$xn|y_HwT1|)uA8_k+_ z5RlU!xdn(8c@=pTc@=pTc@=pTc@=q~Td_{)5Rg#B+R(kotFFeu?huer#2QVaUNmUW zq3;loP{bNd58tkERm}fc!wcxWJ*1_}GZEz=mSQfqoyc6r#7+4Tpm_yD3 z2;~fi>Wy#zwl|vC!gpo48rC5op@=nB9Nic08|V;_P{bN5fgb2shk%44)>z49(_Gjm zj3?0{@rziaNud|-FZ4o(#4ln!HSRsey~iP@ggIiHPu z!JK;*IDUZR$9xX%ci=*Ma770<^ogT^xjrQrV@+^p%%J35AncWf&wurcPbI$l|Lgtx z_m7X|+du!{@y)+}{&>86{n2ed_Fdoq{^Qr@`#Jmnn6p2cJ$@+i_zCyzIR@$SX#dqezv6!Gcy`>V%e;$=FMj9c^@JR~MPf;>&8M z@>HJ6Gna~|@>HJ6^X5kHZ&#kmQ+X;+EC}(MV?P`YeZSxL-EP;n+il-$Hja(>4_^U^D*ylh diff --git a/assets/voxygen/voxel/npc/phoenix/male/tail_front.vox b/assets/voxygen/voxel/npc/phoenix/male/tail_front.vox index 7247b3cdf671ee6b7b56fdf093b48d503df63372..3f9568990e3b5a5b17fc50684646e81f10d27789 100644 GIT binary patch literal 2768 zcmeH|!HOhB5QeiNvoa$xBCEP*1;LXb-b9~a1<_$~1Q%r8NAVRDFM_go^WanXES@}h z5D)gw&^^25SNH^Tb_YJ<&o3goYci*MzWU@L!s1rD67gY{qyzGs=14w|QX%Vcn}7S@F|IJwQ!MQ7ZM11ATr)#y{l0tYyP2p&}AY$BYkEKa6z zIPSeHYk`*KtW?W#>fq$StyErMEk)M3$kGaPRHJrqa!?oQ%FHzLo=|34=2^HW;hvbE znFl8aZlSl5kIY^{s&bfMu5w?iaF24*%06lbCkJ)VTbQ|G_7SAUK9I)qHM!I#+_l_l zV=uLXlY=_wEzBI5y<*R!eXq(Ykd^0E_AC2U?v%CRJ2*LTgWkdnhq)Z_S0Kh-lckv4 z&&Ga}+Xjj=dL5h`xB&|@E3*gOkzVQS>+I{iFKB09XCF8@a09rw%ev1x=i|cP6z}28 zI4|oE*3TeA!&8=Tq~{*)rqIfRh6^_FnKT zK`-cmlLOZVe&7cloE*Dba0?E6&np>B3TIjPyivgl6>O%0*1}nl!O5)^&Is3V;N-wH z#VcO%;N-wHD}Sru4G&HZT;F_0z)bWl7CkL)Yd8vU9JUBUUZQuuf;K9j( zvxPNSoq6_owMGR~Dwt8hIu*>RU_k{NRQU4Kj~}P<-~Xj2Po8WK<@NV(Z?As;Xxm=C zc<|nbzSo=ihu_beGy8v>**7BFJ1K4N!-w#hr0pk@?P*Ed&t0}}b=q#ozaI1U=X+&4 X->lo~7awe|{Vt<6b9gDhYHd72FRkMn+8!4WE3I$`W)#)*&9xcHyj?|0WPN9pra z77_Q$h=*7#9u~4(EbcGtW4U}-ESL97`wROoS8p%pKQg^%t@^63PO84@?RwD{ebE<1(fjM64}IuEkE4;XrBo@R_mAeQ z@id;s(|9J2r|~qN#?yEvnWynIp2pL7=Fi}1JdLOEG@kL(8c*YCJdLOE&`nADPM+jR zo;d|P36LI^22O*_Y4Oi=kUYtgJaejek|%kR=lu)4pPf9(lRU|jJY!0eCwY=5d6H*N z^PHKX8Cwy(`KdgWr}9*u$}^XPr}9*Y>VehZbBWwkT~M7=o@##Pvhh@&%2Rp1UcK%g z-n{BRZ#I@x&r^9S&s^TQN@He*W^8(%!c%w(PvI#%^EKcpJcXz56rTBN@D!fHQ$PyO zd|eDy+*n*x6vh0ESE!hu!c%w(PvM!b)qG>dDvg;L8XHgW1W)h;Pw=>1^gO{6Ji%i~ z(engP@C1*?h@K~Sf+t`+TJ#1CH-`()%99-BAbZ)#RyMMhmBd+2a+HJYWhYzN$XZqs zXF17H4zib>Y-J;BS@rYz+)t-dKOT?$a5(h+e&2VyUEglEeY4s0^?Kb`tCedb{sCA& BYGwcc diff --git a/assets/voxygen/voxel/npc/phoenix/male/tail_rear.vox b/assets/voxygen/voxel/npc/phoenix/male/tail_rear.vox index 580e90a07136b50c82c064056835a3eb7f36665b..306cba6437dc9f7f838e1a6fd42e52a7e96f58e7 100644 GIT binary patch literal 3032 zcmeH|OOD+{6o$)H<$9cY*uL*4LPD7#SVNEyv|2P0kx#80uA0>e=0kFY|KXYXu)c3(tRe zardhy@b$V;Dh#Sd2rDMowHm{k425hrYHT<35Ftas*)4vns@tk5Fe4;#W)?cTqu;5? z?Ns$UbqmZ0LQc#~SJ<(8_V(=U*@FNP1QKMrvg{nx=niV~2X-KWK*D9Z_DFxEKXR`l zJqS7BGF^L8Z{3OhM1P_OAtzj3y0aSnneI#nLQXh$!C&}{7d!|#;oKE}#b5Ct-_hV-R*B zlMAlWbqu^yG0NaSaGL;|g76r-L5XWUkA)Gnwm{ZRR>$$Xo|P&bUHX z+3A=)t|@$u!sjTuv(R$(oI~L{5OT&9x(XfVez~UdJgTm?a!u7Y4}o(8A!jIbm2YLl*Awq(TYj^k^?|~V1x{Dy>gv(2} zr`yx*=|IQ{=MFrF1Fv_$gOC%>9q~u}5f4I6i#y>?I1sWwb3SthA^QvGi{|H7^YhF3 z%K6F}WXX-0B6w}4pv^6Uki9XtU`!W5$j;)d>4G(V1R*D!Yt7BK_!bXBPB`cA4)5?F z1Sj0T+N^*p_&2O%e1rR$&xX0+ggsUh$_K*$BxgNkNs!AIU_qz55q+yF}G!3(_@ zey>awa>fmy67K-nVA`~*IOueF+gk9vM) o)%=I(`KQp#59B`wGynaGoyUi6e*f~*`Q7V>`R$uu=H)NIzXy~^L;wH) literal 1120 zcmXZaJ#G_07zW_a22s(tr$81rm$LpME@uV5uT(su-W1z%yVV2KmB zNZ?B$>()Q%v z*WDuGJ{F7n#bR+6%jNxIxxD*w`Q{>wGqygr)*t@-c2uSGhxW7g=CJu_Df(8#*M>)t|cCwX?tYsx}mXjRiAbZ)# sRyMMhRX?B4{d79@$d`5$T0d zrjwcHH%G|YY53$-RjNDiOY-{V*}ab<^5S^4=8(HZ>(i1`+m06OG7{nM#e_QMyAy8WXq6^44I*1 zBSRxY=EcTXjNEvy7`fTD!l|v=7^*EJQZ@22^0K+Ysl{!h&YtreMC#;5UPfN*!6BW^ z+S%4RBQM)U>clK}MqWlDsUY5K9S^aiLxT)+!r0#pQ_5Fi0Xz{2L8saL4@n2{`T0eXdA!B?on z2|BTxpPlrH?SB*J$!mXl_xZuEhx=t+;JCp(%=rA}>+NrCKOSMd zio0cryES4RcN1o;u=M_oekZ1%(=`2@u|mw#-Gq5I#)4SMFt22*>BpkK*2W@VrXRC* zrCqSrkF^{#R#ld|o}2`|QG}@F{!>pTeh?wD2i>3ZKHK@EK(umFYJ9 zOpvQj=9BqkKABJEGnRu-=95WeFU%$%OXQ*KhHQ}eWc3-##wYX1d@`SpFQ2s6Pan5$ zA9j{h&L{K9e8%#QRqB=LHvN?INqiEY#3%7de8x85llUY)iBIA)wg#WXC-F%{5}&bM zOqM*D+>``Kefkzk>XY~+K8a7_Gq%;(F@2SKWx9=%Pw)vo!6*0xACE)MC-?-P;A2Y2 z`2?Tf6MVb|az4Q)_yifRhTMeV;c(+sc)=Mb9C5%NJ8ZE*Tyen}CmeCW9y@HYL0oac x87CZZz#cnnv1!-qwOuZkc0Qlm>2zwx7{ diff --git a/assets/voxygen/voxel/npc/phoenix/male/wing_mid_r.vox b/assets/voxygen/voxel/npc/phoenix/male/wing_mid_r.vox index 94585e73ca9a30533c6940811aa19f5ba6b67ed8..6917189474f4845cb0c1b3b3b32e68bbb3ddd236 100644 GIT binary patch literal 1748 zcmeIy&x#X45XbRfRd-kBUnbd;;7Jf~qR+5`Xb48w1zGn|d<8)SL0LQsK8H`@$&&}k zMPVoIw%_;$Iqwd9sP6hrGeb?ttCwf@K1zCVyxLIoEu_=c`O{&);hNzHjyPDOzd`=o_a3j)eoSmqvN6a-Z0!wY>B)_(CqqI;=`{v4XtF^&M?xOR5YSv( zn`?3;BC$4$Sr*Y)Y;BDGNg_S1<{f9Hd${w@5zvRGHOsYkiiFIK4d5e zXl~4hY)*!Zf&gE5%bPPFn-Ve#0=}?qLPkO0(!4<{r&4h-Twjn1A|0JMF0Q* literal 1204 zcmW;KJ&w~r7zW_aju#r5)HEpK2Gbxa76mOU1V}7F%QkdVLzOE;o2`+iqa|0!73K<& zI6HZAFsKIxMp>3v%0Lm&Fk<7i}Fi)2Zn_e1qnc`8rksXU#>Q+X;+<*7WK%u{(P zPvxmR{crG8p2|~sD$o4aDo^F9Je8;N&~-`iU3dyl;ps!bQviyG6*n$!(uc*5#ZKWV zJcXwZ6;I(QJcZ}|OTE8ccnVM9DLjQ|9@4^7cnVM9DLj3c`^e0ic}=1>KA9);WS-2E zdHQtlWS$I>Jutg@pCT7!Cu9eiCmWwWZ9JJL^JJc{SFhWLH?P`{o3$y`^JJdP)2Fx3 z(mXP=W?uC?i6`+Sp2U-Q`fI?GcoI+INj&}4;7L4*CxIlM{<^qXa$#~(QY7Ouze36Q zB%Z{RcoI*4t@;}?&(b_Hv&PC3Ji!w@!4o_#7d=n#1W)j|rRaHrCwPL#V?@srJi!w% z9xZw|3>SwJ&&rb=|`q&S<6b65@$KdQ4X?`t!!j1D_OSl m`P@#YQ#&4y?Ql4>-EP;m+ilxyHf_CLx7BLZmdmAcBmM&_P;|}! diff --git a/assets/voxygen/voxel/npc/phoenix/male/wing_out_r.vox b/assets/voxygen/voxel/npc/phoenix/male/wing_out_r.vox index e45c808905cbf23f58bb7022711bf9b31ba8d31d..62b8ab25cf84c5ca312511085dc982566ec0cf9e 100644 GIT binary patch literal 2648 zcmeIz&59&N6o%n*;{QLQL=apF;!gA$MjW&mY{3B;=Tcm|6Wj>O;NHt}<;sQRMxiom zCVdCJfnGZe58sLGiYPkk%j&PceD>Z?BJ#zfyL&=DEh10voHmDjFu1k!h_ghk$5hCbFV2cXDDeU#tx4WID%d5t$PT zb0r%(2J1;KC*$JeW<2lj*+&L4A|a!oqM>^kgPi2@a=UmrOki(dFFXPw5;6)ZT9DB( zFoWDykmC|$*C@+G{zf=>1jHyKkr@S*b&GQAS5s@yCbJnKM4BxDrSDo3lb_bS_{a-Aea zyGoOxjn5ah$>B(kfQZDkOct^>*;|vXH@OaSlG5d-FkKD{2hTbnGACq|E_>~=wJz6= z0U2^A3@jWx0^*Q8A)^f0>X2(ghfF#F0}BU_fH-AK$S6~;6%DdvQy5q{cm%{H*My7$ zxyV&vVBz2q(5&*z=xXSgu%2s3UU?4q9oaFMCuxH&7nIH`omV=qtC9^J12gC_k+Y)G zaa{qCIU%F4u4IckIyuN0^`=ReNxTOfJnKM4q@;A3y{ODBDV^3a$eHwztaMrFGW+4+ z5fHPEgv?yXik8`rk@c8acPZ?Lfh{^5=@FPCQc*grbeR3j4ZY|w3VSPiVX6*WmG0{B ztOFTYCuHVAR{?hU=`#uGwMm7Xd+ReGxQROzYGQ>CX$PnDkPUWaZCJehP<>8R3CrK3tmm5wSM zRXVD4ROu+MT}n5VZYteWx~X(iw}7uc`{a?hfB!!`dGh3Zs8>I{bH4oTExM$xQbG|Ebet?hgxsmhdY|f`n&R>c-zfXGJGyi(A=O6F8bHAVG Us}~=hFMq#3|N8UC^Yu5tKYTCJ4gdfE literal 1272 zcmW;Kv98-T7>40rvTUb77T-Z(bnM2Z$kYT5(u1Nckkmzzr5pE7z6Il)GY8-$)N?oA z0&l@Uht8g|vkxixwZu;(_39`8`0eAjUn1g{*AKtC#D7b~hlkH^U-pLM8@#}bkDoq2 z{Nvh>?=f%VX@#W|)6*K$G(F8&VZjzJ5VL2P=jm}ZPtOb1+8b;UE6*{nrsvgadR(yX z*)l!Pi}_p^tg+GFVp>m+GgfQwTI;uQzj2Q>Hkh}@V!``&Z(fJ*>qi?$ZD0TW+g(O# zU+Z73Rkzh&Es)m)-8Ea07dJ&-C!MhW;70>#UUiHk`_Sv)Lm3ZKHK@EN7zQ}`4yR}v zpR7J(+4y8WnNQ~P`@2`|=O2G)e||VvQaPW@C-WK0J65SzrrY#W&L{Cnd=j6;C-E8E zfKTF+_#{4w&)6D#5}(8;5lMW;b}?BpFzJ*8NqzbjO6rsNBtD5x;xo3@*fD*TdS$we zlTYvoKEWsW1Ruj8=M#K_Pw+7%XFgSF)3vamMf-_Dy z;($GNh&yh$;({|yIO2dkc8EJ}xZ;8{PB`L#J$CJWzqi}%)~?rUyId~qd_K3+>C}$L RV>=uUZNJ~Q-EOCE#Q%5wj diff --git a/common/src/comp/agent.rs b/common/src/comp/agent.rs index 68b30b314f..237a480119 100644 --- a/common/src/comp/agent.rs +++ b/common/src/comp/agent.rs @@ -38,6 +38,8 @@ pub enum Tactic { FixedTurret, RotatingTurret, Mindflayer, + BirdLargeBreathe, + BirdLargeFire, } #[derive(Copy, Clone, Debug, PartialEq)] @@ -247,7 +249,7 @@ impl<'a> From<&'a Body> for Psyche { }, Body::BipedSmall(_) => 0.5, Body::BirdMedium(_) => 0.5, - Body::BirdLarge(_) => 0.4, + Body::BirdLarge(_) => 0.9, Body::FishMedium(_) => 0.15, Body::FishSmall(_) => 0.0, Body::BipedLarge(_) => 1.0, diff --git a/common/src/comp/body.rs b/common/src/comp/body.rs index b48678bd18..2c00e94280 100644 --- a/common/src/comp/body.rs +++ b/common/src/comp/body.rs @@ -372,6 +372,10 @@ impl Body { biped_large::Species::Dullahan => 4000, _ => 3000, }, + Body::BirdLarge(body) => match body.species { + bird_large::Species::Cockatrice => 4000, + bird_large::Species::Phoenix => 6000, + }, Body::Humanoid(_) => 750, _ => 1000, } @@ -583,7 +587,7 @@ impl Body { pub fn flying_height(&self) -> f32 { match self { - Body::BirdLarge(_) => 30.0, + Body::BirdLarge(_) => 50.0, Body::BirdMedium(_) => 40.0, Body::Dragon(_) => 60.0, Body::Ship(ship::Body::DefaultAirship) => 60.0, diff --git a/common/src/comp/inventory/item/tool.rs b/common/src/comp/inventory/item/tool.rs index 13f7f2b56d..9bf489e6e8 100644 --- a/common/src/comp/inventory/item/tool.rs +++ b/common/src/comp/inventory/item/tool.rs @@ -442,4 +442,5 @@ pub enum UniqueKind { WoodenSpear, MindflayerStaff, BirdLargeBreathe, + BirdLargeFire, } diff --git a/common/src/comp/inventory/loadout_builder.rs b/common/src/comp/inventory/loadout_builder.rs index 3bba61a08d..e1fb785727 100644 --- a/common/src/comp/inventory/loadout_builder.rs +++ b/common/src/comp/inventory/loadout_builder.rs @@ -339,7 +339,11 @@ impl LoadoutBuilder { "common.items.npc_weapons.unique.birdlargebreathe", )); }, - _ => {}, + (bird_large::Species::Phoenix, _) => { + main_tool = Some(Item::new_from_asset_expect( + "common.items.npc_weapons.unique.birdlargefire", + )); + }, }, _ => {}, }; diff --git a/common/src/states/utils.rs b/common/src/states/utils.rs index 9b8e4f75de..8a30596df9 100644 --- a/common/src/states/utils.rs +++ b/common/src/states/utils.rs @@ -188,7 +188,7 @@ impl Body { pub fn fly_thrust(&self) -> Option { match self { Body::BirdMedium(_) => Some(GRAVITY * self.mass().0 * 2.0), - Body::BirdLarge(_) => Some(GRAVITY * self.mass().0 * 2.0), + Body::BirdLarge(_) => Some(GRAVITY * self.mass().0 * 3.0), Body::Dragon(_) => Some(200_000.0), Body::Ship(ship::Body::DefaultAirship) => Some(300_000.0), _ => None, diff --git a/server/src/sys/agent.rs b/server/src/sys/agent.rs index 68ba3e23a1..cc46acc65b 100644 --- a/server/src/sys/agent.rs +++ b/server/src/sys/agent.rs @@ -1538,6 +1538,8 @@ impl<'a> AgentData<'a> { Some(ToolKind::Unique(UniqueKind::TheropodBird)) => Tactic::Theropod, Some(ToolKind::Unique(UniqueKind::ObjectTurret)) => Tactic::Turret, Some(ToolKind::Unique(UniqueKind::MindflayerStaff)) => Tactic::Mindflayer, + Some(ToolKind::Unique(UniqueKind::BirdLargeBreathe)) => Tactic::BirdLargeBreathe, + Some(ToolKind::Unique(UniqueKind::BirdLargeFire)) => Tactic::BirdLargeFire, _ => Tactic::Melee, }; @@ -2587,6 +2589,224 @@ impl<'a> AgentData<'a> { controller.inputs.move_z = bearing.z; } }, + Tactic::BirdLargeFire => { + if self.physics_state.on_ground { + controller + .actions + .push(ControlAction::CancelInput(InputKind::Fly)); + if self.energy.current() > 600 && thread_rng().gen_bool(0.3) { + if thread_rng().gen_bool(0.99) { + if dist_sqrd < (2.5 * min_attack_dist).powi(2) + && self.energy.current() > 600 + { + dbg!("shockwave"); + controller + .actions + .push(ControlAction::basic_input(InputKind::Ability(0))); + } else { + if let Some((bearing, speed)) = agent.chaser.chase( + &*terrain, + self.pos.0, + self.vel.0, + tgt_pos.0, + TraversalConfig { + min_tgt_dist: 1.25, + ..self.traversal_config + }, + ) { + if can_see_tgt(&*terrain, self.pos, tgt_pos, dist_sqrd) + && agent.action_timer > 1.0 + { + dbg!("fireball1"); + controller.inputs.move_dir = bearing + .xy() + .rotated_z(thread_rng().gen_range(-1.57..-0.5)) + .try_normalized() + .unwrap_or_else(Vec2::zero) + * speed; + controller.actions.push(ControlAction::basic_input( + InputKind::Ability(1), + )); + agent.action_timer = 0.0; + } else { + dbg!("walk to player1"); + controller.inputs.move_dir = bearing + .xy() + .try_normalized() + .unwrap_or_else(Vec2::zero) + * speed; + self.jump_if(controller, bearing.z > 1.5); + controller.inputs.move_z = bearing.z; + agent.action_timer += dt.0; + } + } + } + } else { + dbg!("fly"); + controller + .actions + .push(ControlAction::basic_input(InputKind::Fly)); + controller + .actions + .push(ControlAction::basic_input(InputKind::Jump)); + controller.inputs.move_z = 6.0; + } + } else { + if dist_sqrd < (2.0 * min_attack_dist).powi(2) { + dbg!("triple strike"); + controller + .actions + .push(ControlAction::basic_input(InputKind::Secondary)); + } else if let Some((bearing, speed)) = agent.chaser.chase( + &*terrain, + self.pos.0, + self.vel.0, + tgt_pos.0, + TraversalConfig { + min_tgt_dist: 1.25, + ..self.traversal_config + }, + ) { + dbg!("walk to player2"); + controller.inputs.move_dir = + bearing.xy().try_normalized().unwrap_or_else(Vec2::zero) * speed; + self.jump_if(controller, bearing.z > 1.5); + controller.inputs.move_z = bearing.z; + } + } + } else { + if read_data + .terrain + .ray(self.pos.0, self.pos.0 - (Vec3::unit_z() * 5.0)) + .until(Block::is_solid) + .cast() + .1 + .map_or(true, |b| b.is_some()) + { + controller.inputs.move_z = 1.0; + } + if self.energy.current() > 50 { + if thread_rng().gen_bool(0.99) { + if dist_sqrd < (3.5 * min_attack_dist).powi(2) { + dbg!("flamethrower1"); + controller + .actions + .push(ControlAction::basic_input(InputKind::Primary)); + } else { + if let Some((bearing, speed)) = agent.chaser.chase( + &*terrain, + self.pos.0, + self.vel.0, + tgt_pos.0, + TraversalConfig { + min_tgt_dist: 1.25, + ..self.traversal_config + }, + ) { + if can_see_tgt(&*terrain, self.pos, tgt_pos, dist_sqrd) { + if agent.action_timer > 3.0 { + dbg!("fireball2"); + controller.actions.push(ControlAction::basic_input( + InputKind::Ability(1), + )); + agent.action_timer = 0.0; + } else { + dbg!("fly to player1"); + controller.inputs.move_dir = bearing + .xy() + .rotated_z(thread_rng().gen_range(-1.57..-0.5)) + .try_normalized() + .unwrap_or_else(Vec2::zero) + * speed; + agent.action_timer += dt.0; + } + } else { + dbg!("fly to player2"); + controller.inputs.move_dir = bearing + .xy() + .try_normalized() + .unwrap_or_else(Vec2::zero) + * speed; + self.jump_if(controller, bearing.z > 1.5); + controller.inputs.move_z = bearing.z; + } + } + } + } else { + dbg!("land1"); + controller.inputs.move_z = -1.0; + controller + .actions + .push(ControlAction::CancelInput(InputKind::Fly)); + } + } else { + if dist_sqrd < (2.5 * min_attack_dist).powi(2) { + dbg!("land2"); + controller.inputs.move_z = -1.0; + controller + .actions + .push(ControlAction::CancelInput(InputKind::Fly)); + } else { + dbg!("fly to player"); + } + } + } + }, + Tactic::BirdLargeBreathe => { + if dist_sqrd < (2.5 * min_attack_dist).powi(2) { + controller.inputs.move_dir = Vec2::zero(); + controller + .actions + .push(ControlAction::basic_input(InputKind::Secondary)); + } else if dist_sqrd < (7.0 * min_attack_dist).powi(2) { + if agent.action_timer < 2.0 { + controller.inputs.move_dir = (tgt_pos.0 - self.pos.0) + .xy() + .rotated_z(0.47 * PI) + .try_normalized() + .unwrap_or_else(Vec2::unit_y); + controller + .actions + .push(ControlAction::basic_input(InputKind::Primary)); + agent.action_timer += dt.0; + } else if agent.action_timer < 4.0 { + controller.inputs.move_dir = (tgt_pos.0 - self.pos.0) + .xy() + .rotated_z(-0.47 * PI) + .try_normalized() + .unwrap_or_else(Vec2::unit_y); + controller + .actions + .push(ControlAction::basic_input(InputKind::Primary)); + agent.action_timer += dt.0; + } else if agent.action_timer < 6.0 { + controller + .actions + .push(ControlAction::basic_input(InputKind::Ability(0))); + agent.action_timer += dt.0; + } else { + agent.action_timer = 0.0; + } + } else if dist_sqrd < MAX_CHASE_DIST.powi(2) { + if let Some((bearing, speed)) = agent.chaser.chase( + &*terrain, + self.pos.0, + self.vel.0, + tgt_pos.0, + TraversalConfig { + min_tgt_dist: 1.25, + ..self.traversal_config + }, + ) { + controller.inputs.move_dir = + bearing.xy().try_normalized().unwrap_or_else(Vec2::zero) * speed; + self.jump_if(controller, bearing.z > 1.5); + controller.inputs.move_z = bearing.z; + } + } else { + agent.target = None; + } + }, } } diff --git a/voxygen/anim/src/bird_large/alpha.rs b/voxygen/anim/src/bird_large/alpha.rs index 378851e192..013d8416c1 100644 --- a/voxygen/anim/src/bird_large/alpha.rs +++ b/voxygen/anim/src/bird_large/alpha.rs @@ -7,7 +7,7 @@ use common::states::utils::StageSection; pub struct AlphaAnimation; impl Animation for AlphaAnimation { - type Dependency = (Option, f32, f32); + type Dependency = (Option, Vec3, Vec3, f32, f32); type Skeleton = BirdLargeSkeleton; #[cfg(feature = "use-dyn-lib")] @@ -16,7 +16,7 @@ impl Animation for AlphaAnimation { #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_alpha")] fn update_skeleton_inner( skeleton: &Self::Skeleton, - (stage_section, _global_time, _timer): Self::Dependency, + (stage_section, orientation, last_ori, _global_time, _timer): Self::Dependency, anim_time: f32, _rate: &mut f32, s_a: &SkeletonAttr, @@ -37,9 +37,22 @@ impl Animation for AlphaAnimation { let move1 = move1base * pullback; let move2 = move2base * pullback; + let ori: Vec2 = Vec2::from(orientation); + let last_ori = Vec2::from(last_ori); + let tilt = if ::vek::Vec2::new(ori, last_ori) + .map(|o| 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.2) + * last_ori.determine_side(Vec2::zero(), ori).signum() + } else { + 0.0 + } * 1.3; + next.head.scale = Vec3::one() * 0.98; next.neck.scale = Vec3::one() * 1.02; - next.beak.scale = Vec3::one() * 0.98; next.leg_l.scale = Vec3::one() / 8.0 * 0.98; next.leg_r.scale = Vec3::one() / 8.0 * 0.98; next.foot_l.scale = Vec3::one() * 1.02; @@ -51,7 +64,8 @@ impl Animation for AlphaAnimation { next.chest.orientation = Quaternion::rotation_x(move1 * 0.5 - move2 * 0.8); next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); - next.neck.orientation = Quaternion::rotation_x(move1 * 0.5 - move2 * 0.8); + next.neck.orientation = Quaternion::rotation_x(move1 * 0.5 - move2 * 0.8) + * Quaternion::rotation_z(move1 * tilt * 1.5); next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); next.head.orientation = Quaternion::rotation_x(move1 * 0.5 - move2 * 0.8); diff --git a/voxygen/anim/src/bird_large/breathe.rs b/voxygen/anim/src/bird_large/breathe.rs index 96cc860f9b..904fc13d29 100644 --- a/voxygen/anim/src/bird_large/breathe.rs +++ b/voxygen/anim/src/bird_large/breathe.rs @@ -54,7 +54,6 @@ impl Animation for BreatheAnimation { next.head.scale = Vec3::one() * 0.98; next.neck.scale = Vec3::one() * 1.02; - next.beak.scale = Vec3::one() * 0.98; next.leg_l.scale = Vec3::one() / 8.0 * 0.98; next.leg_r.scale = Vec3::one() / 8.0 * 0.98; next.foot_l.scale = Vec3::one() * 1.02; diff --git a/voxygen/anim/src/bird_large/feed.rs b/voxygen/anim/src/bird_large/feed.rs index 9e61647de6..59daa60311 100644 --- a/voxygen/anim/src/bird_large/feed.rs +++ b/voxygen/anim/src/bird_large/feed.rs @@ -41,7 +41,6 @@ impl Animation for FeedAnimation { next.head.scale = Vec3::one() * 0.98; next.neck.scale = Vec3::one() * 1.02; - next.beak.scale = Vec3::one() * 0.98; next.leg_l.scale = Vec3::one() / 8.0 * 0.98; next.leg_r.scale = Vec3::one() / 8.0 * 0.98; next.foot_l.scale = Vec3::one() * 1.02; @@ -51,7 +50,7 @@ impl Animation for FeedAnimation { next.chest.position = Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + wave_slow_cos * 0.06 - 1.8) * s_a.scaler / 8.0; - next.chest.orientation = Quaternion::rotation_x(-0.5); + next.chest.orientation = Quaternion::rotation_x(s_a.feed); next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); next.neck.orientation = Quaternion::rotation_x(-0.2); @@ -72,11 +71,9 @@ impl Animation for FeedAnimation { next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_l.orientation = - Quaternion::rotation_y(-s_a.wings_angle + 0.3 + wave_fast * 0.08) - * Quaternion::rotation_z(0.2); + Quaternion::rotation_y(-0.7 + wave_fast * 0.08) * Quaternion::rotation_z(0.2); next.wing_in_r.orientation = - Quaternion::rotation_y(s_a.wings_angle - 0.3 - wave_fast * 0.08) - * Quaternion::rotation_z(-0.2); + Quaternion::rotation_y(0.7 - wave_fast * 0.08) * Quaternion::rotation_z(-0.2); next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); diff --git a/voxygen/anim/src/bird_large/idle.rs b/voxygen/anim/src/bird_large/idle.rs index 1b4a8243df..ced671cb26 100644 --- a/voxygen/anim/src/bird_large/idle.rs +++ b/voxygen/anim/src/bird_large/idle.rs @@ -39,7 +39,6 @@ impl Animation for IdleAnimation { next.head.scale = Vec3::one() * 0.98; next.neck.scale = Vec3::one() * 1.02; - next.beak.scale = Vec3::one() * 0.98; next.leg_l.scale = Vec3::one() / 8.0 * 0.98; next.leg_r.scale = Vec3::one() / 8.0 * 0.98; next.foot_l.scale = Vec3::one() * 1.02; @@ -62,17 +61,17 @@ impl Animation for IdleAnimation { next.beak.orientation = Quaternion::rotation_x(wave_slow_cos * -0.02 - 0.02); next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); - next.tail_front.orientation = Quaternion::rotation_x(0.0); + next.tail_front.orientation = Quaternion::rotation_x(0.6); next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); - next.tail_rear.orientation = Quaternion::rotation_x(0.0); + next.tail_rear.orientation = Quaternion::rotation_x(-0.2); next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); next.wing_in_l.orientation = - Quaternion::rotation_y(-1.0 + wave_slow_cos * 0.06) * Quaternion::rotation_z(0.2); + Quaternion::rotation_y(-0.8 + wave_slow_cos * 0.06) * Quaternion::rotation_z(0.2); next.wing_in_r.orientation = - Quaternion::rotation_y(1.0 - wave_slow_cos * 0.06) * Quaternion::rotation_z(-0.2); + Quaternion::rotation_y(0.8 - wave_slow_cos * 0.06) * Quaternion::rotation_z(-0.2); next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); @@ -81,8 +80,8 @@ impl Animation for IdleAnimation { next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); - next.wing_out_l.orientation = Quaternion::rotation_y(-0.2) * Quaternion::rotation_z(0.2); - next.wing_out_r.orientation = Quaternion::rotation_y(0.2) * Quaternion::rotation_z(-0.2); + next.wing_out_l.orientation = Quaternion::rotation_y(-0.4) * Quaternion::rotation_z(0.2); + next.wing_out_r.orientation = Quaternion::rotation_y(0.4) * Quaternion::rotation_z(-0.2); next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; next.leg_l.orientation = Quaternion::rotation_x(0.0); diff --git a/voxygen/anim/src/bird_large/mod.rs b/voxygen/anim/src/bird_large/mod.rs index eef7b229c6..55ac715d38 100644 --- a/voxygen/anim/src/bird_large/mod.rs +++ b/voxygen/anim/src/bird_large/mod.rs @@ -4,12 +4,16 @@ pub mod feed; pub mod fly; pub mod idle; pub mod run; +pub mod shockwave; +pub mod shoot; pub mod stunned; +pub mod swim; // Reexports pub use self::{ alpha::AlphaAnimation, breathe::BreatheAnimation, feed::FeedAnimation, fly::FlyAnimation, - idle::IdleAnimation, run::RunAnimation, stunned::StunnedAnimation, + idle::IdleAnimation, run::RunAnimation, shockwave::ShockwaveAnimation, shoot::ShootAnimation, + stunned::StunnedAnimation, swim::SwimAnimation, }; use super::{make_bone, vek::*, FigureBoneData, Skeleton}; @@ -104,7 +108,7 @@ pub struct SkeletonAttr { leg: (f32, f32, f32), foot: (f32, f32, f32), scaler: f32, - wings_angle: f32, + feed: f32, } impl<'a> std::convert::TryFrom<&'a comp::Body> for SkeletonAttr { @@ -133,7 +137,7 @@ impl Default for SkeletonAttr { leg: (0.0, 0.0, 0.0), foot: (0.0, 0.0, 0.0), scaler: 0.0, - wings_angle: 0.0, + feed: 0.0, } } } @@ -143,56 +147,56 @@ impl<'a> From<&'a Body> for SkeletonAttr { use comp::bird_large::Species::*; Self { chest: match (body.species, body.body_type) { - (Phoenix, _) => (2.5, 8.0), + (Phoenix, _) => (2.5, 16.0), (Cockatrice, _) => (2.5, 16.0), }, neck: match (body.species, body.body_type) { - (Phoenix, _) => (0.5, 3.0), + (Phoenix, _) => (2.5, -5.5), (Cockatrice, _) => (5.0, -1.5), }, head: match (body.species, body.body_type) { - (Phoenix, _) => (2.0, 2.0), + (Phoenix, _) => (6.0, 12.0), (Cockatrice, _) => (8.0, 4.5), }, beak: match (body.species, body.body_type) { - (Phoenix, _) => (2.0, 1.0), + (Phoenix, _) => (5.0, 3.0), (Cockatrice, _) => (2.0, -3.0), }, tail_front: match (body.species, body.body_type) { - (Phoenix, _) => (-5.5, -2.0), + (Phoenix, _) => (-9.5, -1.0), (Cockatrice, _) => (-5.0, -2.5), }, tail_rear: match (body.species, body.body_type) { - (Phoenix, _) => (-3.0, -3.0), + (Phoenix, _) => (-11.0, 0.0), (Cockatrice, _) => (-8.0, -3.0), }, wing_in: match (body.species, body.body_type) { - (Phoenix, _) => (3.0, 2.5, 3.0), + (Phoenix, _) => (3.0, 2.5, 2.0), (Cockatrice, _) => (3.5, 7.0, 3.5), }, wing_mid: match (body.species, body.body_type) { - (Phoenix, _) => (6.5, -1.0, 0.0), + (Phoenix, _) => (10.0, 1.0, 0.0), (Cockatrice, _) => (6.0, 0.0, 0.0), }, wing_out: match (body.species, body.body_type) { - (Phoenix, _) => (0.5, -1.0, 0.0), + (Phoenix, _) => (7.0, 2.0, 1.5), (Cockatrice, _) => (4.0, -1.0, 1.0), }, leg: match (body.species, body.body_type) { - (Phoenix, _) => (2.5, -2.5, -3.5), + (Phoenix, _) => (4.0, 1.5, 12.0), (Cockatrice, _) => (3.5, 2.5, 13.0), }, foot: match (body.species, body.body_type) { - (Phoenix, _) => (0.0, -0.5, -0.5), + (Phoenix, _) => (0.5, -0.5, -2.5), (Cockatrice, _) => (0.5, -3.0, -3.0), }, scaler: match (body.species, body.body_type) { (Phoenix, _) => (1.0), (Cockatrice, _) => (1.0), }, - wings_angle: match (body.species, body.body_type) { - (Phoenix, _) => (1.3), - (Cockatrice, _) => (0.9), + feed: match (body.species, body.body_type) { + (Phoenix, _) => (-0.65), + (Cockatrice, _) => (-0.5), }, } } diff --git a/voxygen/anim/src/bird_large/shockwave.rs b/voxygen/anim/src/bird_large/shockwave.rs new file mode 100644 index 0000000000..dfe31308ba --- /dev/null +++ b/voxygen/anim/src/bird_large/shockwave.rs @@ -0,0 +1,99 @@ +use super::{ + super::{vek::*, Animation}, + BirdLargeSkeleton, SkeletonAttr, +}; +use common::states::utils::StageSection; + +pub struct ShockwaveAnimation; + +impl Animation for ShockwaveAnimation { + #[allow(clippy::type_complexity)] + type Dependency = (f32, f32, Option); + type Skeleton = BirdLargeSkeleton; + + #[cfg(feature = "use-dyn-lib")] + const UPDATE_FN: &'static [u8] = b"bird_large_shockwave\0"; + + #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_shockwave")] + fn update_skeleton_inner( + skeleton: &Self::Skeleton, + (_global_time, _velocity, stage_section): Self::Dependency, + anim_time: f32, + _rate: &mut f32, + s_a: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let (movement1base, movement2base, movement3, _twitch) = match stage_section { + Some(StageSection::Buildup) => (anim_time.powf(0.5), 0.0, 0.0, 0.0), + Some(StageSection::Cast) => (1.0, anim_time.min(1.0).powf(0.1), 0.0, anim_time), + Some(StageSection::Recover) => (1.0, 1.0, anim_time, 1.0), + _ => (0.0, 0.0, 0.0, 0.0), + }; + + let pullback = 1.0 - movement3; + + let movement1abs = movement1base * pullback; + let movement2abs = movement2base * pullback; + + let wave_slow_cos = (anim_time * 4.5).cos(); + + next.head.scale = Vec3::one() * 0.98; + next.neck.scale = Vec3::one() * 1.02; + next.leg_l.scale = Vec3::one() / 8.0 * 0.98; + next.leg_r.scale = Vec3::one() / 8.0 * 0.98; + next.foot_l.scale = Vec3::one() * 1.02; + next.foot_r.scale = Vec3::one() * 1.02; + next.chest.scale = Vec3::one() * s_a.scaler / 8.0; + + next.chest.position = + Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + movement1abs * 1.5) * s_a.scaler / 8.0; + next.chest.orientation = Quaternion::rotation_x(movement1abs * 1.0 + movement2abs * -1.0); + + next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); + next.neck.orientation = Quaternion::rotation_x(-0.2); + + next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); + next.head.orientation = Quaternion::rotation_x(wave_slow_cos * 0.01); + + next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); + next.beak.orientation = Quaternion::rotation_x(wave_slow_cos * -0.02 - 0.02); + + next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); + next.tail_front.orientation = Quaternion::rotation_x(0.6); + next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); + next.tail_rear.orientation = Quaternion::rotation_x(-0.2); + + next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + + next.wing_in_l.orientation = + Quaternion::rotation_y(-0.8 + movement1abs * 1.6 + movement2abs * -1.6) + * Quaternion::rotation_z(0.2 + movement1abs * -0.8); + next.wing_in_r.orientation = + Quaternion::rotation_y(0.8 + movement1abs * -1.6 + movement2abs * 1.6) + * Quaternion::rotation_z(-0.2 + movement1abs * 0.8); + + next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_l.orientation = Quaternion::rotation_y(-0.1) * Quaternion::rotation_z(0.7); + next.wing_mid_r.orientation = Quaternion::rotation_y(0.1) * Quaternion::rotation_z(-0.7); + + next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_l.orientation = Quaternion::rotation_y(-0.4) * Quaternion::rotation_z(0.2); + next.wing_out_r.orientation = Quaternion::rotation_y(0.4) * Quaternion::rotation_z(-0.2); + + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; + next.leg_l.orientation = Quaternion::rotation_x(0.0); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; + next.leg_r.orientation = Quaternion::rotation_x(0.0); + + next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_l.orientation = Quaternion::rotation_x(0.0); + next.foot_r.position = Vec3::new(s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_r.orientation = Quaternion::rotation_x(0.0); + + next + } +} diff --git a/voxygen/anim/src/bird_large/shoot.rs b/voxygen/anim/src/bird_large/shoot.rs new file mode 100644 index 0000000000..16a67df318 --- /dev/null +++ b/voxygen/anim/src/bird_large/shoot.rs @@ -0,0 +1,136 @@ +use super::{ + super::{vek::*, Animation}, + BirdLargeSkeleton, SkeletonAttr, +}; +use common::{states::utils::StageSection, util::Dir}; + +pub struct ShootAnimation; + +type ShootAnimationDependency = ( + f32, + f32, + Vec3, + Vec3, + Option, + f32, + Dir, + bool, +); + +impl Animation for ShootAnimation { + type Dependency = ShootAnimationDependency; + type Skeleton = BirdLargeSkeleton; + + #[cfg(feature = "use-dyn-lib")] + const UPDATE_FN: &'static [u8] = b"bird_large_shoot\0"; + + #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_shoot")] + fn update_skeleton_inner( + skeleton: &Self::Skeleton, + (_velocity, global_time, orientation, last_ori, stage_section, timer, look_dir, on_ground): Self::Dependency, + anim_time: f32, + _rate: &mut f32, + s_a: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let (movement1base, movement2base, movement3, twitch) = match stage_section { + Some(StageSection::Buildup) => (anim_time.powf(0.25), 0.0, 0.0, 0.0), + Some(StageSection::Cast) => (1.0, anim_time.min(1.0).powf(0.1), 0.0, anim_time), + Some(StageSection::Recover) => (1.0, 1.0, anim_time, 1.0), + _ => (0.0, 0.0, 0.0, 0.0), + }; + + let pullback = 1.0 - movement3; + let subtract = global_time - timer; + let check = subtract - subtract.trunc(); + let mirror = (check - 0.5).signum(); + let twitch2 = mirror * (twitch * 20.0).sin() * pullback; + + let movement1abs = movement1base * pullback; + let movement2abs = movement2base * pullback; + + let wave_slow_cos = (anim_time * 4.5).cos(); + + next.head.scale = Vec3::one() * 0.98; + next.neck.scale = Vec3::one() * 1.02; + next.leg_l.scale = Vec3::one() / 8.0 * 0.98; + next.leg_r.scale = Vec3::one() / 8.0 * 0.98; + next.foot_l.scale = Vec3::one() * 1.02; + next.foot_r.scale = Vec3::one() * 1.02; + next.chest.scale = Vec3::one() * s_a.scaler / 8.0; + + next.chest.position = Vec3::new( + 0.0, + s_a.chest.0, + s_a.chest.1 + wave_slow_cos * 0.06 + twitch2 * 0.1, + ) * s_a.scaler + / 8.0; + + next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); + next.neck.orientation = Quaternion::rotation_x(movement1abs * 0.5 - movement2abs * 0.5); + + next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); + next.head.orientation = + Quaternion::rotation_x(movement1abs * 0.5 - movement2abs * 0.5 + look_dir.z * 0.4); + + next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); + next.beak.orientation = Quaternion::rotation_x(movement1abs * -0.7 + twitch2 * 0.1); + + if on_ground { + next.chest.orientation = + Quaternion::rotation_x(movement1abs * 0.1 - movement2abs * 0.1); + + next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + + next.wing_in_l.orientation = + Quaternion::rotation_y(-1.0 + movement1abs * 0.8 - movement2abs * 0.4) + * Quaternion::rotation_z(0.2 - movement1abs * 0.8 + movement2abs * 0.4); + next.wing_in_r.orientation = + Quaternion::rotation_y(1.0 - movement1abs * 0.8 + movement2abs * 0.4) + * Quaternion::rotation_z(-0.2 + movement1abs * 0.8 - movement2abs * 0.4); + + next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_l.orientation = + Quaternion::rotation_y(-0.1) * Quaternion::rotation_z(0.7); + next.wing_mid_r.orientation = + Quaternion::rotation_y(0.1) * Quaternion::rotation_z(-0.7); + + next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_l.orientation = + Quaternion::rotation_y(-0.2) * Quaternion::rotation_z(0.2); + next.wing_out_r.orientation = + Quaternion::rotation_y(0.2) * Quaternion::rotation_z(-0.2); + + next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); + next.tail_front.orientation = + Quaternion::rotation_x(-movement1abs * 0.1 + movement2abs * 0.1 + twitch2 * 0.02); + next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); + next.tail_rear.orientation = + Quaternion::rotation_x(-movement1abs * 0.1 + movement2abs * 0.1 + twitch2 * 0.02); + } else { + let ori: Vec2 = Vec2::from(orientation); + let last_ori = Vec2::from(last_ori); + let tilt = if ::vek::Vec2::new(ori, last_ori) + .map(|o| 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.2) + * last_ori.determine_side(Vec2::zero(), ori).signum() + } else { + 0.0 + } * 1.3; + + next.chest.orientation = + Quaternion::rotation_x(movement1abs * 0.1 - movement2abs * 0.1) + * Quaternion::rotation_y(tilt * 1.8); + } + + next + } +} diff --git a/voxygen/anim/src/bird_large/stunned.rs b/voxygen/anim/src/bird_large/stunned.rs index 62a1905a7e..5377b3cdbb 100644 --- a/voxygen/anim/src/bird_large/stunned.rs +++ b/voxygen/anim/src/bird_large/stunned.rs @@ -39,7 +39,6 @@ impl Animation for StunnedAnimation { next.head.scale = Vec3::one() * 0.98; next.neck.scale = Vec3::one() * 1.02; - next.beak.scale = Vec3::one() * 0.98; next.leg_l.scale = Vec3::one() / 8.0 * 0.98; next.leg_r.scale = Vec3::one() / 8.0 * 0.98; next.foot_l.scale = Vec3::one() * 1.02; diff --git a/voxygen/anim/src/bird_large/swim.rs b/voxygen/anim/src/bird_large/swim.rs new file mode 100644 index 0000000000..d81cc092d1 --- /dev/null +++ b/voxygen/anim/src/bird_large/swim.rs @@ -0,0 +1,102 @@ +use super::{ + super::{vek::*, Animation}, + BirdLargeSkeleton, SkeletonAttr, +}; +use std::ops::Mul; + +pub struct SwimAnimation; + +impl Animation for SwimAnimation { + #[allow(clippy::type_complexity)] + type Dependency = (Vec3, Vec3, Vec3, f32); + type Skeleton = BirdLargeSkeleton; + + #[cfg(feature = "use-dyn-lib")] + const UPDATE_FN: &'static [u8] = b"bird_large_swim\0"; + + #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_swim")] + fn update_skeleton_inner( + skeleton: &Self::Skeleton, + (_velocity, _orientation, _last_ori, global_time): Self::Dependency, + anim_time: f32, + _rate: &mut f32, + s_a: &SkeletonAttr, + ) -> Self::Skeleton { + let mut next = (*skeleton).clone(); + + let duck_head_look = Vec2::new( + (global_time / 2.0 + anim_time / 8.0) + .floor() + .mul(7331.0) + .sin() + * 0.5, + (global_time / 2.0 + anim_time / 8.0) + .floor() + .mul(1337.0) + .sin() + * 0.25, + ); + let wave_slow_cos = (anim_time * 4.5).cos(); + + let wave_fast = (anim_time * 6.0).sin(); + let wave_fast_cos = (anim_time * 6.0).cos(); + + next.head.scale = Vec3::one() * 0.98; + next.neck.scale = Vec3::one() * 1.02; + next.leg_l.scale = Vec3::one() / 8.0 * 0.98; + next.leg_r.scale = Vec3::one() / 8.0 * 0.98; + next.foot_l.scale = Vec3::one() * 1.02; + next.foot_r.scale = Vec3::one() * 1.02; + next.chest.scale = Vec3::one() * s_a.scaler / 8.0; + + next.chest.position = Vec3::new(0.0, s_a.chest.0, s_a.chest.1 + wave_slow_cos * 0.06 + 1.5) + * s_a.scaler + / 8.0; + next.chest.orientation = Quaternion::rotation_x(0.0); + + next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); + next.neck.orientation = Quaternion::rotation_x(0.0); + + next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); + next.head.orientation = Quaternion::rotation_z(duck_head_look.x) + * Quaternion::rotation_x(-duck_head_look.y.abs() + wave_slow_cos * 0.01); + + next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); + next.beak.orientation = Quaternion::rotation_x(wave_slow_cos * -0.02 - 0.02); + + next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); + next.tail_front.orientation = Quaternion::rotation_x(0.6); + next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); + next.tail_rear.orientation = Quaternion::rotation_x(-0.2); + + next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + + next.wing_in_l.orientation = + Quaternion::rotation_y(-0.8 + wave_slow_cos * 0.06) * Quaternion::rotation_z(0.2); + next.wing_in_r.orientation = + Quaternion::rotation_y(0.8 - wave_slow_cos * 0.06) * Quaternion::rotation_z(-0.2); + + next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_l.orientation = Quaternion::rotation_y(-0.1) * Quaternion::rotation_z(0.7); + next.wing_mid_r.orientation = Quaternion::rotation_y(0.1) * Quaternion::rotation_z(-0.7); + + next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_l.orientation = Quaternion::rotation_y(-0.4) * Quaternion::rotation_z(0.2); + next.wing_out_r.orientation = Quaternion::rotation_y(0.4) * Quaternion::rotation_z(-0.2); + + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; + next.leg_l.orientation = Quaternion::rotation_x(-0.8 + wave_fast * 0.5); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; + next.leg_r.orientation = Quaternion::rotation_x(-0.8 + wave_fast_cos * 0.5); + + next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_l.orientation = Quaternion::rotation_x(0.0); + next.foot_r.position = Vec3::new(s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_r.orientation = Quaternion::rotation_x(0.0); + + next + } +} diff --git a/voxygen/src/scene/figure/mod.rs b/voxygen/src/scene/figure/mod.rs index baaacc7f60..6e63d4547b 100644 --- a/voxygen/src/scene/figure/mod.rs +++ b/voxygen/src/scene/figure/mod.rs @@ -3332,6 +3332,20 @@ impl FigureMgr { &mut state_animation_rate, skeleton_attr, ), + // Swim + (_, true, _) => anim::bird_large::SwimAnimation::update_skeleton( + &BirdLargeSkeleton::default(), + ( + rel_vel, + // TODO: Update to use the quaternion. + ori * anim::vek::Vec3::::unit_y(), + state.last_ori * anim::vek::Vec3::::unit_y(), + time, + ), + state.state_time, + &mut state_animation_rate, + skeleton_attr, + ), // TODO! _ => anim::bird_large::IdleAnimation::update_skeleton( &BirdLargeSkeleton::default(), @@ -3407,7 +3421,65 @@ impl FigureMgr { anim::bird_large::AlphaAnimation::update_skeleton( &target_base, - (Some(s.stage_section), time, state.state_time), + ( + Some(s.stage_section), + ori * anim::vek::Vec3::::unit_y(), + state.last_ori * anim::vek::Vec3::::unit_y(), + time, + state.state_time, + ), + stage_progress, + &mut state_animation_rate, + skeleton_attr, + ) + }, + CharacterState::BasicRanged(s) => { + let stage_time = s.timer.as_secs_f32(); + + let stage_progress = match s.stage_section { + StageSection::Buildup => { + stage_time / s.static_data.buildup_duration.as_secs_f32() + }, + StageSection::Recover => { + stage_time / s.static_data.recover_duration.as_secs_f32() + }, + + _ => 0.0, + }; + anim::bird_large::ShootAnimation::update_skeleton( + &target_base, + ( + rel_vel.magnitude(), + time, + ori * anim::vek::Vec3::::unit_y(), + state.last_ori * anim::vek::Vec3::::unit_y(), + Some(s.stage_section), + state.state_time, + look_dir, + physics.on_ground, + ), + stage_progress, + &mut state_animation_rate, + skeleton_attr, + ) + }, + CharacterState::Shockwave(s) => { + let stage_time = s.timer.as_secs_f32(); + let stage_progress = match s.stage_section { + StageSection::Buildup => { + stage_time / s.static_data.buildup_duration.as_secs_f32() + }, + StageSection::Swing => { + stage_time / s.static_data.swing_duration.as_secs_f32() + }, + StageSection::Recover => { + stage_time / s.static_data.recover_duration.as_secs_f32() + }, + _ => 0.0, + }; + anim::bird_large::ShockwaveAnimation::update_skeleton( + &target_base, + (time, rel_vel.magnitude(), Some(s.stage_section)), stage_progress, &mut state_animation_rate, skeleton_attr, diff --git a/voxygen/src/session/mod.rs b/voxygen/src/session/mod.rs index fa1b98732c..a0e9895f88 100644 --- a/voxygen/src/session/mod.rs +++ b/voxygen/src/session/mod.rs @@ -522,6 +522,7 @@ impl PlayState for SessionState { select_pos, target_entity.map(|t| t.0), ); + dbg!(self.key_state.fly); }, GameInput::Climb => { self.key_state.climb_up = state; From 7cfd8debede2ce4910675c6d1830575c706b51e5 Mon Sep 17 00:00:00 2001 From: James Melkonian Date: Wed, 21 Apr 2021 20:55:02 -0700 Subject: [PATCH 08/10] Phoenix AI pass --- assets/common/abilities/bow/basic.ron | 2 +- server/src/sys/agent.rs | 147 ++++++++++++++++---------- 2 files changed, 90 insertions(+), 59 deletions(-) diff --git a/assets/common/abilities/bow/basic.ron b/assets/common/abilities/bow/basic.ron index 6fdc1a9087..56d9bb99e7 100644 --- a/assets/common/abilities/bow/basic.ron +++ b/assets/common/abilities/bow/basic.ron @@ -4,7 +4,7 @@ BasicRanged( recover_duration: 0.3, projectile: Arrow( damage: 70.0, - knockback: 5.0, + knockback: 1.0, energy_regen: 40, ), projectile_body: Object(Arrow), diff --git a/server/src/sys/agent.rs b/server/src/sys/agent.rs index cc46acc65b..703f730b4b 100644 --- a/server/src/sys/agent.rs +++ b/server/src/sys/agent.rs @@ -2590,74 +2590,89 @@ impl<'a> AgentData<'a> { } }, Tactic::BirdLargeFire => { - if self.physics_state.on_ground { - controller - .actions - .push(ControlAction::CancelInput(InputKind::Fly)); - if self.energy.current() > 600 && thread_rng().gen_bool(0.3) { - if thread_rng().gen_bool(0.99) { - if dist_sqrd < (2.5 * min_attack_dist).powi(2) - && self.energy.current() > 600 - { - dbg!("shockwave"); - controller - .actions - .push(ControlAction::basic_input(InputKind::Ability(0))); - } else { - if let Some((bearing, speed)) = agent.chaser.chase( - &*terrain, - self.pos.0, - self.vel.0, - tgt_pos.0, - TraversalConfig { - min_tgt_dist: 1.25, - ..self.traversal_config - }, - ) { - if can_see_tgt(&*terrain, self.pos, tgt_pos, dist_sqrd) - && agent.action_timer > 1.0 - { - dbg!("fireball1"); - controller.inputs.move_dir = bearing - .xy() - .rotated_z(thread_rng().gen_range(-1.57..-0.5)) - .try_normalized() - .unwrap_or_else(Vec2::zero) - * speed; - controller.actions.push(ControlAction::basic_input( - InputKind::Ability(1), - )); - agent.action_timer = 0.0; - } else { - dbg!("walk to player1"); - controller.inputs.move_dir = bearing - .xy() - .try_normalized() - .unwrap_or_else(Vec2::zero) - * speed; - self.jump_if(controller, bearing.z > 1.5); - controller.inputs.move_z = bearing.z; - agent.action_timer += dt.0; - } - } - } - } else { - dbg!("fly"); + // Set fly to false + controller + .actions + .push(ControlAction::CancelInput(InputKind::Fly)); + if dist_sqrd > 30.0_f32.powi(2) { + if thread_rng().gen_bool(0.05) + && can_see_tgt(&*terrain, self.pos, tgt_pos, dist_sqrd) + { + controller + .actions + .push(ControlAction::basic_input(InputKind::Ability(1))); + } + if let Some((bearing, speed)) = agent.chaser.chase( + &*terrain, + self.pos.0, + self.vel.0, + tgt_pos.0, + TraversalConfig { + min_tgt_dist: 1.25, + ..self.traversal_config + }, + ) { + dbg!("fly towards target"); + controller.inputs.move_dir = + bearing.xy().try_normalized().unwrap_or_else(Vec2::zero) * speed; + if (self.pos.0.z - tgt_pos.0.z) < 20.0 { controller .actions .push(ControlAction::basic_input(InputKind::Fly)); controller .actions .push(ControlAction::basic_input(InputKind::Jump)); - controller.inputs.move_z = 6.0; + controller.inputs.move_z = 1.0; + } else { + self.jump_if(controller, bearing.z > 1.5); + controller.inputs.move_z = bearing.z; } - } else { - if dist_sqrd < (2.0 * min_attack_dist).powi(2) { + } + } else if agent.action_timer < 3.0 { + if thread_rng().gen_bool(0.1) { + if thread_rng().gen_bool(0.5) { + controller + .actions + .push(ControlAction::basic_input(InputKind::Ability(1))); + } + } else if thread_rng().gen_bool(0.5) && (self.pos.0.z - tgt_pos.0.z) < 15.0 { + dbg!("fly"); + controller + .actions + .push(ControlAction::basic_input(InputKind::Fly)); + controller + .actions + .push(ControlAction::basic_input(InputKind::Jump)); + controller.inputs.move_z = 1.0; + } + agent.action_timer += dt.0; + } else if !self.physics_state.on_ground { + // Do not increment the timer during this movement + // The next stage shouldn't trigger until the entity + // is on the ground + controller + .actions + .push(ControlAction::basic_input(InputKind::Fly)); + dbg!("fly to player2"); + let move_dir = tgt_pos.0 - self.pos.0; + controller.inputs.move_dir = + move_dir.xy().try_normalized().unwrap_or_else(Vec2::zero) * 2.0; + controller.inputs.move_z = move_dir.z - 0.5; + } else if agent.action_timer < 7.0 { + if dist_sqrd < (2.5 * min_attack_dist).powi(2) { + if self.energy.current() > 600 && thread_rng().gen_bool(0.3) { + dbg!("shockwave"); + controller + .actions + .push(ControlAction::basic_input(InputKind::Ability(0))); + } else { dbg!("triple strike"); controller .actions .push(ControlAction::basic_input(InputKind::Secondary)); - } else if let Some((bearing, speed)) = agent.chaser.chase( + } + } else { + if let Some((bearing, speed)) = agent.chaser.chase( &*terrain, self.pos.0, self.vel.0, @@ -2667,13 +2682,28 @@ impl<'a> AgentData<'a> { ..self.traversal_config }, ) { - dbg!("walk to player2"); + if thread_rng().gen_bool(0.2) + && can_see_tgt(&*terrain, self.pos, tgt_pos, dist_sqrd) + { + dbg!("fireball1"); + controller + .actions + .push(ControlAction::basic_input(InputKind::Ability(1))); + agent.action_timer = 0.0; + } + dbg!("walk to player1"); controller.inputs.move_dir = bearing.xy().try_normalized().unwrap_or_else(Vec2::zero) * speed; self.jump_if(controller, bearing.z > 1.5); controller.inputs.move_z = bearing.z; } } + agent.action_timer += dt.0; + } + if agent.action_timer > 7.0 { + agent.action_timer = 0.0; + } + /* } else { if read_data .terrain @@ -2751,6 +2781,7 @@ impl<'a> AgentData<'a> { } } } + */ }, Tactic::BirdLargeBreathe => { if dist_sqrd < (2.5 * min_attack_dist).powi(2) { From 58300592ca8261ea448f713e44a3dcd2259fcf16 Mon Sep 17 00:00:00 2001 From: Snowram Date: Fri, 23 Apr 2021 01:34:37 +0200 Subject: [PATCH 09/10] More phoenix AI work More phoenix AI work --- assets/common/abilities/bow/basic.ron | 2 +- .../unique/birdlargebreathe/firebomb.ron | 17 + .../abilities/weapon_ability_manifest.ron | 11 +- .../voxygen/voxel/npc/phoenix/male/chest.vox | Bin 8448 -> 8448 bytes common/src/comp/body.rs | 11 +- common/src/states/utils.rs | 2 +- server/src/rtsim/entity.rs | 10 +- server/src/sys/agent.rs | 351 +++++++++--------- voxygen/anim/src/bird_large/alpha.rs | 47 ++- voxygen/anim/src/bird_large/breathe.rs | 3 +- voxygen/anim/src/bird_large/shockwave.rs | 59 +-- voxygen/anim/src/bird_large/shoot.rs | 34 +- voxygen/anim/src/bird_large/swim.rs | 4 +- voxygen/src/scene/figure/mod.rs | 14 +- voxygen/src/session/mod.rs | 1 - 15 files changed, 287 insertions(+), 279 deletions(-) create mode 100644 assets/common/abilities/unique/birdlargebreathe/firebomb.ron diff --git a/assets/common/abilities/bow/basic.ron b/assets/common/abilities/bow/basic.ron index 56d9bb99e7..6fdc1a9087 100644 --- a/assets/common/abilities/bow/basic.ron +++ b/assets/common/abilities/bow/basic.ron @@ -4,7 +4,7 @@ BasicRanged( recover_duration: 0.3, projectile: Arrow( damage: 70.0, - knockback: 1.0, + knockback: 5.0, energy_regen: 40, ), projectile_body: Object(Arrow), diff --git a/assets/common/abilities/unique/birdlargebreathe/firebomb.ron b/assets/common/abilities/unique/birdlargebreathe/firebomb.ron new file mode 100644 index 0000000000..def79061ea --- /dev/null +++ b/assets/common/abilities/unique/birdlargebreathe/firebomb.ron @@ -0,0 +1,17 @@ +BasicRanged( + energy_cost: 0, + buildup_duration: 0.5, + recover_duration: 0.35, + projectile: Fireball( + damage: 100.0, + radius: 5.0, + energy_regen: 50, + ), + projectile_body: Object(BoltFire), + /*projectile_light: Some(LightEmitter { + col: (1.0, 0.75, 0.11).into(), + ..Default::default() + }),*/ + projectile_gravity: Some(Gravity(0.15)), + projectile_speed: 60.0, +) diff --git a/assets/common/abilities/weapon_ability_manifest.ron b/assets/common/abilities/weapon_ability_manifest.ron index d990e18d61..e36d47685c 100644 --- a/assets/common/abilities/weapon_ability_manifest.ron +++ b/assets/common/abilities/weapon_ability_manifest.ron @@ -208,16 +208,17 @@ ], ), Unique(BirdLargeBreathe): ( - primary: "common.abilities.unique.birdlargebreathe.flamethrower", + primary: "common.abilities.unique.birdlargebreathe.firebomb", secondary: "common.abilities.unique.birdlargebreathe.triplestrike", - abilities: [], + abilities: [ + (None, "common.abilities.unique.birdlargebreathe.flamethrower"), + ], ), Unique(BirdLargeFire): ( - primary: "common.abilities.unique.birdlargefire.flamethrower", + primary: "common.abilities.unique.birdlargefire.firebomb", secondary: "common.abilities.unique.birdlargefire.triplestrike", abilities: [ - (None, "common.abilities.staff.fireshockwave"), - (None, "common.abilities.staff.firebomb"), + (None, "common.abilities.unique.birdlargefire.fireshockwave"), ], ), Debug: ( diff --git a/assets/voxygen/voxel/npc/phoenix/male/chest.vox b/assets/voxygen/voxel/npc/phoenix/male/chest.vox index 017abd28ebdae0c766471fdc8c7926474cb23a92..0dc30cb4b5896407eca56551849c5095f7272cbd 100644 GIT binary patch delta 84 zcmZp0YH->xnVa_;s}RcxnVXlNRfvV34T#x+c=G~oVP>F^AS*u`5VHgE<^z0>nH2e11=#r6fS4VK aIRx0gaRM=y0NWQ 700.0, - Body::BirdLarge(_) => 700.0, + Body::BirdLarge(_) => 2_200.0, // based on its mass divided by the volume of a bird scaled up to the size of the dragon Body::Dragon(_) => 3_700.0, @@ -184,8 +184,7 @@ impl Body { // ravens are 0.69-2 kg, crows are 0.51 kg on average Body::BirdMedium(_) => 1.0, - // australian magpies are around 0.22-0.35 kg - Body::BirdLarge(_) => 0.3, + Body::BirdLarge(_) => 200.0, Body::Dragon(_) => 20_000.0, Body::FishMedium(_) => 2.5, @@ -283,7 +282,7 @@ impl Body { }, Body::BipedSmall(_) => Vec3::new(1.0, 0.75, 1.4), Body::BirdMedium(_) => Vec3::new(2.0, 1.0, 1.1), - Body::BirdLarge(_) => Vec3::new(2.0, 1.0, 1.8), + Body::BirdLarge(_) => Vec3::new(2.0, 5.5, 3.8), Body::Dragon(_) => Vec3::new(16.0, 10.0, 16.0), Body::FishMedium(_) => Vec3::new(0.5, 2.0, 0.8), Body::FishSmall(_) => Vec3::new(0.3, 1.2, 0.6), @@ -436,7 +435,7 @@ impl Body { }, Body::FishMedium(_) => 50, Body::Dragon(_) => 5000, - Body::BirdLarge(_) => 9999999, + Body::BirdLarge(_) => 3000, Body::FishSmall(_) => 20, Body::BipedLarge(biped_large) => match biped_large.species { biped_large::Species::Ogre => 2500, @@ -548,7 +547,7 @@ impl Body { }, Body::FishMedium(_) => 10, Body::Dragon(_) => 500, - Body::BirdLarge(_) => 10, + Body::BirdLarge(_) => 120, Body::FishSmall(_) => 10, Body::BipedLarge(biped_large) => match biped_large.species { biped_large::Species::Ogre => 70, diff --git a/common/src/states/utils.rs b/common/src/states/utils.rs index 8a30596df9..245169fc3b 100644 --- a/common/src/states/utils.rs +++ b/common/src/states/utils.rs @@ -188,7 +188,7 @@ impl Body { pub fn fly_thrust(&self) -> Option { match self { Body::BirdMedium(_) => Some(GRAVITY * self.mass().0 * 2.0), - Body::BirdLarge(_) => Some(GRAVITY * self.mass().0 * 3.0), + Body::BirdLarge(_) => Some(GRAVITY * self.mass().0 * 0.5), Body::Dragon(_) => Some(200_000.0), Body::Ship(ship::Body::DefaultAirship) => Some(300_000.0), _ => None, diff --git a/server/src/rtsim/entity.rs b/server/src/rtsim/entity.rs index c79a4e3b43..04e59de0cf 100644 --- a/server/src/rtsim/entity.rs +++ b/server/src/rtsim/entity.rs @@ -38,12 +38,18 @@ impl Entity { match self.rng(PERM_GENUS).gen::() { // we want 5% airships, 45% birds, 50% humans x if x < 0.05 => comp::Body::Ship(comp::ship::Body::DefaultAirship), - x if x < 0.50 => { + x if x < 0.45 => { let species = *(&comp::bird_medium::ALL_SPECIES) .choose(&mut self.rng(PERM_SPECIES)) .unwrap(); comp::bird_medium::Body::random_with(&mut self.rng(PERM_BODY), &species).into() }, + x if x < 0.50 => { + let species = *(&comp::bird_large::ALL_SPECIES) + .choose(&mut self.rng(PERM_SPECIES)) + .unwrap(); + comp::bird_large::Body::random_with(&mut self.rng(PERM_BODY), &species).into() + }, _ => { let species = *(&comp::humanoid::ALL_SPECIES) .choose(&mut self.rng(PERM_SPECIES)) @@ -60,7 +66,7 @@ impl Entity { comp::Body::BirdMedium(b) => { get_npc_name(&npc_names.bird_medium, b.species).to_string() }, - comp::Body::BirdLarge(_) => "Warbler".to_string(), + comp::Body::BirdLarge(b) => get_npc_name(&npc_names.bird_large, b.species).to_string(), comp::Body::Dragon(b) => get_npc_name(&npc_names.dragon, b.species).to_string(), comp::Body::Humanoid(b) => get_npc_name(&npc_names.humanoid, b.species).to_string(), comp::Body::Ship(_) => "Veloren Air".to_string(), diff --git a/server/src/sys/agent.rs b/server/src/sys/agent.rs index 703f730b4b..2afa0d5ebd 100644 --- a/server/src/sys/agent.rs +++ b/server/src/sys/agent.rs @@ -2590,6 +2590,137 @@ impl<'a> AgentData<'a> { } }, Tactic::BirdLargeFire => { + // Set fly to false + controller + .actions + .push(ControlAction::CancelInput(InputKind::Fly)); + // If further than 30 blocks + if dist_sqrd > 30.0_f32.powi(2) { + // If random chance and can see target + if thread_rng().gen_bool(0.05) + && can_see_tgt(&*terrain, self.pos, tgt_pos, dist_sqrd) + { + // Fireball + controller + .actions + .push(ControlAction::basic_input(InputKind::Primary)); + } + // If some target + if let Some((bearing, speed)) = agent.chaser.chase( + &*terrain, + self.pos.0, + self.vel.0, + tgt_pos.0, + TraversalConfig { + min_tgt_dist: 1.25, + ..self.traversal_config + }, + ) { + // Walk to target + controller.inputs.move_dir = + bearing.xy().try_normalized().unwrap_or_else(Vec2::zero) * speed; + // If less than 20 blocks higher than target + if (self.pos.0.z - tgt_pos.0.z) < 20.0 { + // Fly upward + controller + .actions + .push(ControlAction::basic_input(InputKind::Fly)); + controller + .actions + .push(ControlAction::basic_input(InputKind::Jump)); + controller.inputs.move_z = 1.0; + } else { + // Jump + self.jump_if(controller, bearing.z > 1.5); + controller.inputs.move_z = bearing.z; + } + } + } + // If higher than 2 blocks + else if !read_data + .terrain + .ray(self.pos.0, self.pos.0 - (Vec3::unit_z() * 2.0)) + .until(Block::is_solid) + .cast() + .1 + .map_or(true, |b| b.is_some()) + { + // Do not increment the timer during this movement + // The next stage shouldn't trigger until the entity + // is on the ground + // Fly to target + controller + .actions + .push(ControlAction::basic_input(InputKind::Fly)); + let move_dir = tgt_pos.0 - self.pos.0; + controller.inputs.move_dir = + move_dir.xy().try_normalized().unwrap_or_else(Vec2::zero) * 2.0; + controller.inputs.move_z = move_dir.z - 0.5; + // If further than 4 blocks and random chance + if thread_rng().gen_bool(0.05) && dist_sqrd > (4.0 * min_attack_dist).powi(2) { + // Fireball + controller + .actions + .push(ControlAction::basic_input(InputKind::Primary)); + } + } + // If further than 4 blocks and random chance + else if thread_rng().gen_bool(0.05) && dist_sqrd > (4.0 * min_attack_dist).powi(2) + { + // Fireball + controller + .actions + .push(ControlAction::basic_input(InputKind::Primary)); + } + // If random chance and less than 20 blocks higher than target and further than 4 + // blocks + else if thread_rng().gen_bool(0.5) + && (self.pos.0.z - tgt_pos.0.z) < 15.0 + && dist_sqrd > (4.0 * min_attack_dist).powi(2) + { + controller + .actions + .push(ControlAction::basic_input(InputKind::Fly)); + controller + .actions + .push(ControlAction::basic_input(InputKind::Jump)); + controller.inputs.move_z = 1.0; + } + // If further than 2.5 blocks and random chance + else if dist_sqrd > (2.5 * min_attack_dist).powi(2) { + // If some target + if let Some((bearing, speed)) = agent.chaser.chase( + &*terrain, + self.pos.0, + self.vel.0, + tgt_pos.0, + TraversalConfig { + min_tgt_dist: 1.25, + ..self.traversal_config + }, + ) { + // Walk to target + controller.inputs.move_dir = + bearing.xy().try_normalized().unwrap_or_else(Vec2::zero) * speed; + self.jump_if(controller, bearing.z > 1.5); + controller.inputs.move_z = bearing.z; + } + } + // If energy higher than 600 and random chance + else if self.energy.current() > 600 && thread_rng().gen_bool(0.4) { + // Shockwave + controller + .actions + .push(ControlAction::basic_input(InputKind::Ability(0))); + } else { + // Triple strike + controller + .actions + .push(ControlAction::basic_input(InputKind::Secondary)); + } + }, + // Mostly identical to BirdLargeFire but tweaked for flamethrower instead of shockwave + Tactic::BirdLargeBreathe => { // Set fly to false controller .actions @@ -2600,7 +2731,7 @@ impl<'a> AgentData<'a> { { controller .actions - .push(ControlAction::basic_input(InputKind::Ability(1))); + .push(ControlAction::basic_input(InputKind::Primary)); } if let Some((bearing, speed)) = agent.chaser.chase( &*terrain, @@ -2612,7 +2743,6 @@ impl<'a> AgentData<'a> { ..self.traversal_config }, ) { - dbg!("fly towards target"); controller.inputs.move_dir = bearing.xy().try_normalized().unwrap_or_else(Vec2::zero) * speed; if (self.pos.0.z - tgt_pos.0.z) < 20.0 { @@ -2628,197 +2758,46 @@ impl<'a> AgentData<'a> { controller.inputs.move_z = bearing.z; } } - } else if agent.action_timer < 3.0 { - if thread_rng().gen_bool(0.1) { - if thread_rng().gen_bool(0.5) { - controller - .actions - .push(ControlAction::basic_input(InputKind::Ability(1))); - } - } else if thread_rng().gen_bool(0.5) && (self.pos.0.z - tgt_pos.0.z) < 15.0 { - dbg!("fly"); - controller - .actions - .push(ControlAction::basic_input(InputKind::Fly)); - controller - .actions - .push(ControlAction::basic_input(InputKind::Jump)); - controller.inputs.move_z = 1.0; - } - agent.action_timer += dt.0; - } else if !self.physics_state.on_ground { + } else if !read_data + .terrain + .ray(self.pos.0, self.pos.0 - (Vec3::unit_z() * 2.0)) + .until(Block::is_solid) + .cast() + .1 + .map_or(true, |b| b.is_some()) + { // Do not increment the timer during this movement // The next stage shouldn't trigger until the entity // is on the ground controller .actions .push(ControlAction::basic_input(InputKind::Fly)); - dbg!("fly to player2"); let move_dir = tgt_pos.0 - self.pos.0; controller.inputs.move_dir = move_dir.xy().try_normalized().unwrap_or_else(Vec2::zero) * 2.0; controller.inputs.move_z = move_dir.z - 0.5; - } else if agent.action_timer < 7.0 { - if dist_sqrd < (2.5 * min_attack_dist).powi(2) { - if self.energy.current() > 600 && thread_rng().gen_bool(0.3) { - dbg!("shockwave"); - controller - .actions - .push(ControlAction::basic_input(InputKind::Ability(0))); - } else { - dbg!("triple strike"); - controller - .actions - .push(ControlAction::basic_input(InputKind::Secondary)); - } - } else { - if let Some((bearing, speed)) = agent.chaser.chase( - &*terrain, - self.pos.0, - self.vel.0, - tgt_pos.0, - TraversalConfig { - min_tgt_dist: 1.25, - ..self.traversal_config - }, - ) { - if thread_rng().gen_bool(0.2) - && can_see_tgt(&*terrain, self.pos, tgt_pos, dist_sqrd) - { - dbg!("fireball1"); - controller - .actions - .push(ControlAction::basic_input(InputKind::Ability(1))); - agent.action_timer = 0.0; - } - dbg!("walk to player1"); - controller.inputs.move_dir = - bearing.xy().try_normalized().unwrap_or_else(Vec2::zero) * speed; - self.jump_if(controller, bearing.z > 1.5); - controller.inputs.move_z = bearing.z; - } + if thread_rng().gen_bool(0.05) && dist_sqrd > (4.0 * min_attack_dist).powi(2) { + controller + .actions + .push(ControlAction::basic_input(InputKind::Primary)); } - agent.action_timer += dt.0; - } - if agent.action_timer > 7.0 { - agent.action_timer = 0.0; - } - /* - } else { - if read_data - .terrain - .ray(self.pos.0, self.pos.0 - (Vec3::unit_z() * 5.0)) - .until(Block::is_solid) - .cast() - .1 - .map_or(true, |b| b.is_some()) - { - controller.inputs.move_z = 1.0; - } - if self.energy.current() > 50 { - if thread_rng().gen_bool(0.99) { - if dist_sqrd < (3.5 * min_attack_dist).powi(2) { - dbg!("flamethrower1"); - controller - .actions - .push(ControlAction::basic_input(InputKind::Primary)); - } else { - if let Some((bearing, speed)) = agent.chaser.chase( - &*terrain, - self.pos.0, - self.vel.0, - tgt_pos.0, - TraversalConfig { - min_tgt_dist: 1.25, - ..self.traversal_config - }, - ) { - if can_see_tgt(&*terrain, self.pos, tgt_pos, dist_sqrd) { - if agent.action_timer > 3.0 { - dbg!("fireball2"); - controller.actions.push(ControlAction::basic_input( - InputKind::Ability(1), - )); - agent.action_timer = 0.0; - } else { - dbg!("fly to player1"); - controller.inputs.move_dir = bearing - .xy() - .rotated_z(thread_rng().gen_range(-1.57..-0.5)) - .try_normalized() - .unwrap_or_else(Vec2::zero) - * speed; - agent.action_timer += dt.0; - } - } else { - dbg!("fly to player2"); - controller.inputs.move_dir = bearing - .xy() - .try_normalized() - .unwrap_or_else(Vec2::zero) - * speed; - self.jump_if(controller, bearing.z > 1.5); - controller.inputs.move_z = bearing.z; - } - } - } - } else { - dbg!("land1"); - controller.inputs.move_z = -1.0; - controller - .actions - .push(ControlAction::CancelInput(InputKind::Fly)); - } - } else { - if dist_sqrd < (2.5 * min_attack_dist).powi(2) { - dbg!("land2"); - controller.inputs.move_z = -1.0; - controller - .actions - .push(ControlAction::CancelInput(InputKind::Fly)); - } else { - dbg!("fly to player"); - } - } - } - */ - }, - Tactic::BirdLargeBreathe => { - if dist_sqrd < (2.5 * min_attack_dist).powi(2) { - controller.inputs.move_dir = Vec2::zero(); + } else if thread_rng().gen_bool(0.05) && dist_sqrd > (4.0 * min_attack_dist).powi(2) + { controller .actions - .push(ControlAction::basic_input(InputKind::Secondary)); - } else if dist_sqrd < (7.0 * min_attack_dist).powi(2) { - if agent.action_timer < 2.0 { - controller.inputs.move_dir = (tgt_pos.0 - self.pos.0) - .xy() - .rotated_z(0.47 * PI) - .try_normalized() - .unwrap_or_else(Vec2::unit_y); - controller - .actions - .push(ControlAction::basic_input(InputKind::Primary)); - agent.action_timer += dt.0; - } else if agent.action_timer < 4.0 { - controller.inputs.move_dir = (tgt_pos.0 - self.pos.0) - .xy() - .rotated_z(-0.47 * PI) - .try_normalized() - .unwrap_or_else(Vec2::unit_y); - controller - .actions - .push(ControlAction::basic_input(InputKind::Primary)); - agent.action_timer += dt.0; - } else if agent.action_timer < 6.0 { - controller - .actions - .push(ControlAction::basic_input(InputKind::Ability(0))); - agent.action_timer += dt.0; - } else { - agent.action_timer = 0.0; - } - } else if dist_sqrd < MAX_CHASE_DIST.powi(2) { + .push(ControlAction::basic_input(InputKind::Primary)); + } else if thread_rng().gen_bool(0.5) + && (self.pos.0.z - tgt_pos.0.z) < 15.0 + && dist_sqrd > (4.0 * min_attack_dist).powi(2) + { + controller + .actions + .push(ControlAction::basic_input(InputKind::Fly)); + controller + .actions + .push(ControlAction::basic_input(InputKind::Jump)); + controller.inputs.move_z = 1.0; + } else if dist_sqrd > (3.0 * min_attack_dist).powi(2) { if let Some((bearing, speed)) = agent.chaser.chase( &*terrain, self.pos.0, @@ -2834,8 +2813,18 @@ impl<'a> AgentData<'a> { self.jump_if(controller, bearing.z > 1.5); controller.inputs.move_z = bearing.z; } + } else if self.energy.current() > 600 && agent.action_timer < 3.0 { + controller + .actions + .push(ControlAction::basic_input(InputKind::Ability(0))); + agent.action_timer += dt.0; + } else if agent.action_timer < 6.0 { + controller + .actions + .push(ControlAction::basic_input(InputKind::Secondary)); + agent.action_timer += dt.0; } else { - agent.target = None; + agent.action_timer = 0.0; } }, } diff --git a/voxygen/anim/src/bird_large/alpha.rs b/voxygen/anim/src/bird_large/alpha.rs index 013d8416c1..99f5c56566 100644 --- a/voxygen/anim/src/bird_large/alpha.rs +++ b/voxygen/anim/src/bird_large/alpha.rs @@ -7,7 +7,7 @@ use common::states::utils::StageSection; pub struct AlphaAnimation; impl Animation for AlphaAnimation { - type Dependency = (Option, Vec3, Vec3, f32, f32); + type Dependency = (Option, Vec3, Vec3, bool); type Skeleton = BirdLargeSkeleton; #[cfg(feature = "use-dyn-lib")] @@ -16,7 +16,7 @@ impl Animation for AlphaAnimation { #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_alpha")] fn update_skeleton_inner( skeleton: &Self::Skeleton, - (stage_section, orientation, last_ori, _global_time, _timer): Self::Dependency, + (stage_section, orientation, last_ori, on_ground): Self::Dependency, anim_time: f32, _rate: &mut f32, s_a: &SkeletonAttr, @@ -73,28 +73,35 @@ impl Animation for AlphaAnimation { next.beak.position = Vec3::new(0.0, s_a.beak.0, s_a.beak.1); next.beak.orientation = Quaternion::rotation_x(wave_slow_cos * -0.02 - 0.02); - next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); - next.tail_front.orientation = Quaternion::rotation_x(-move1 * 0.2); - next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); - next.tail_rear.orientation = Quaternion::rotation_x(0.0); + if on_ground { + next.tail_front.position = Vec3::new(0.0, s_a.tail_front.0, s_a.tail_front.1); + next.tail_front.orientation = Quaternion::rotation_x(-move1 * 0.2); + next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); + next.tail_rear.orientation = Quaternion::rotation_x(0.0); - next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); - next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); - next.wing_in_l.orientation = - Quaternion::rotation_y(-1.0 + wave_slow_cos * 0.06) * Quaternion::rotation_z(0.2); - next.wing_in_r.orientation = - Quaternion::rotation_y(1.0 - wave_slow_cos * 0.06) * Quaternion::rotation_z(-0.2); + next.wing_in_l.orientation = + Quaternion::rotation_y(-1.0 + wave_slow_cos * 0.06) * Quaternion::rotation_z(0.2); + next.wing_in_r.orientation = + Quaternion::rotation_y(1.0 - wave_slow_cos * 0.06) * Quaternion::rotation_z(-0.2); - next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); - next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); - next.wing_mid_l.orientation = Quaternion::rotation_y(-0.1) * Quaternion::rotation_z(0.7); - next.wing_mid_r.orientation = Quaternion::rotation_y(0.1) * Quaternion::rotation_z(-0.7); + next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_l.orientation = + Quaternion::rotation_y(-0.1) * Quaternion::rotation_z(0.7); + next.wing_mid_r.orientation = + Quaternion::rotation_y(0.1) * Quaternion::rotation_z(-0.7); - next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); - next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); - next.wing_out_l.orientation = Quaternion::rotation_y(-0.2) * Quaternion::rotation_z(0.2); - next.wing_out_r.orientation = Quaternion::rotation_y(0.2) * Quaternion::rotation_z(-0.2); + next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_l.orientation = + Quaternion::rotation_y(-0.2) * Quaternion::rotation_z(0.2); + next.wing_out_r.orientation = + Quaternion::rotation_y(0.2) * Quaternion::rotation_z(-0.2); + } else { + } next } diff --git a/voxygen/anim/src/bird_large/breathe.rs b/voxygen/anim/src/bird_large/breathe.rs index 904fc13d29..9d9c632619 100644 --- a/voxygen/anim/src/bird_large/breathe.rs +++ b/voxygen/anim/src/bird_large/breathe.rs @@ -7,7 +7,6 @@ use common::{states::utils::StageSection, util::Dir}; pub struct BreatheAnimation; type BreatheAnimationDependency = ( - f32, f32, Vec3, Vec3, @@ -27,7 +26,7 @@ impl Animation for BreatheAnimation { #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_breathe")] fn update_skeleton_inner( skeleton: &Self::Skeleton, - (_velocity, global_time, orientation, last_ori, stage_section, timer, look_dir, on_ground): Self::Dependency, + (global_time, orientation, last_ori, stage_section, timer, look_dir, on_ground): Self::Dependency, anim_time: f32, _rate: &mut f32, s_a: &SkeletonAttr, diff --git a/voxygen/anim/src/bird_large/shockwave.rs b/voxygen/anim/src/bird_large/shockwave.rs index dfe31308ba..c6dd88dd57 100644 --- a/voxygen/anim/src/bird_large/shockwave.rs +++ b/voxygen/anim/src/bird_large/shockwave.rs @@ -8,7 +8,7 @@ pub struct ShockwaveAnimation; impl Animation for ShockwaveAnimation { #[allow(clippy::type_complexity)] - type Dependency = (f32, f32, Option); + type Dependency = (Option, bool); type Skeleton = BirdLargeSkeleton; #[cfg(feature = "use-dyn-lib")] @@ -17,7 +17,7 @@ impl Animation for ShockwaveAnimation { #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_shockwave")] fn update_skeleton_inner( skeleton: &Self::Skeleton, - (_global_time, _velocity, stage_section): Self::Dependency, + (stage_section, on_ground): Self::Dependency, anim_time: f32, _rate: &mut f32, s_a: &SkeletonAttr, @@ -64,35 +64,42 @@ impl Animation for ShockwaveAnimation { next.tail_rear.position = Vec3::new(0.0, s_a.tail_rear.0, s_a.tail_rear.1); next.tail_rear.orientation = Quaternion::rotation_x(-0.2); - next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); - next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + if on_ground { + next.wing_in_l.position = Vec3::new(-s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); + next.wing_in_r.position = Vec3::new(s_a.wing_in.0, s_a.wing_in.1, s_a.wing_in.2); - next.wing_in_l.orientation = - Quaternion::rotation_y(-0.8 + movement1abs * 1.6 + movement2abs * -1.6) - * Quaternion::rotation_z(0.2 + movement1abs * -0.8); - next.wing_in_r.orientation = - Quaternion::rotation_y(0.8 + movement1abs * -1.6 + movement2abs * 1.6) - * Quaternion::rotation_z(-0.2 + movement1abs * 0.8); + next.wing_in_l.orientation = + Quaternion::rotation_y(-0.8 + movement1abs * 1.6 + movement2abs * -1.6) + * Quaternion::rotation_z(0.2 + movement1abs * -0.8); + next.wing_in_r.orientation = + Quaternion::rotation_y(0.8 + movement1abs * -1.6 + movement2abs * 1.6) + * Quaternion::rotation_z(-0.2 + movement1abs * 0.8); - next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); - next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); - next.wing_mid_l.orientation = Quaternion::rotation_y(-0.1) * Quaternion::rotation_z(0.7); - next.wing_mid_r.orientation = Quaternion::rotation_y(0.1) * Quaternion::rotation_z(-0.7); + next.wing_mid_l.position = Vec3::new(-s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_r.position = Vec3::new(s_a.wing_mid.0, s_a.wing_mid.1, s_a.wing_mid.2); + next.wing_mid_l.orientation = + Quaternion::rotation_y(-0.1) * Quaternion::rotation_z(0.7); + next.wing_mid_r.orientation = + Quaternion::rotation_y(0.1) * Quaternion::rotation_z(-0.7); - next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); - next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); - next.wing_out_l.orientation = Quaternion::rotation_y(-0.4) * Quaternion::rotation_z(0.2); - next.wing_out_r.orientation = Quaternion::rotation_y(0.4) * Quaternion::rotation_z(-0.2); + next.wing_out_l.position = Vec3::new(-s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_r.position = Vec3::new(s_a.wing_out.0, s_a.wing_out.1, s_a.wing_out.2); + next.wing_out_l.orientation = + Quaternion::rotation_y(-0.4) * Quaternion::rotation_z(0.2); + next.wing_out_r.orientation = + Quaternion::rotation_y(0.4) * Quaternion::rotation_z(-0.2); - next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; - next.leg_l.orientation = Quaternion::rotation_x(0.0); - next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; - next.leg_r.orientation = Quaternion::rotation_x(0.0); + next.leg_l.position = Vec3::new(-s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; + next.leg_l.orientation = Quaternion::rotation_x(0.0); + next.leg_r.position = Vec3::new(s_a.leg.0, s_a.leg.1, s_a.leg.2) / 8.0; + next.leg_r.orientation = Quaternion::rotation_x(0.0); - next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); - next.foot_l.orientation = Quaternion::rotation_x(0.0); - next.foot_r.position = Vec3::new(s_a.foot.0, s_a.foot.1, s_a.foot.2); - next.foot_r.orientation = Quaternion::rotation_x(0.0); + next.foot_l.position = Vec3::new(-s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_l.orientation = Quaternion::rotation_x(0.0); + next.foot_r.position = Vec3::new(s_a.foot.0, s_a.foot.1, s_a.foot.2); + next.foot_r.orientation = Quaternion::rotation_x(0.0); + } else { + } next } diff --git a/voxygen/anim/src/bird_large/shoot.rs b/voxygen/anim/src/bird_large/shoot.rs index 16a67df318..cd56d3020b 100644 --- a/voxygen/anim/src/bird_large/shoot.rs +++ b/voxygen/anim/src/bird_large/shoot.rs @@ -27,7 +27,16 @@ impl Animation for ShootAnimation { #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_shoot")] fn update_skeleton_inner( skeleton: &Self::Skeleton, - (_velocity, global_time, orientation, last_ori, stage_section, timer, look_dir, on_ground): Self::Dependency, + ( + _velocity, + global_time, + _orientation, + _last_ori, + stage_section, + timer, + look_dir, + on_ground, + ): Self::Dependency, anim_time: f32, _rate: &mut f32, s_a: &SkeletonAttr, @@ -67,9 +76,6 @@ impl Animation for ShootAnimation { ) * s_a.scaler / 8.0; - next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); - next.neck.orientation = Quaternion::rotation_x(movement1abs * 0.5 - movement2abs * 0.5); - next.head.position = Vec3::new(0.0, s_a.head.0, s_a.head.1); next.head.orientation = Quaternion::rotation_x(movement1abs * 0.5 - movement2abs * 0.5 + look_dir.z * 0.4); @@ -78,6 +84,9 @@ impl Animation for ShootAnimation { next.beak.orientation = Quaternion::rotation_x(movement1abs * -0.7 + twitch2 * 0.1); if on_ground { + next.neck.position = Vec3::new(0.0, s_a.neck.0, s_a.neck.1); + next.neck.orientation = Quaternion::rotation_x(movement1abs * 0.5 - movement2abs * 0.5); + next.chest.orientation = Quaternion::rotation_x(movement1abs * 0.1 - movement2abs * 0.1); @@ -112,23 +121,6 @@ impl Animation for ShootAnimation { next.tail_rear.orientation = Quaternion::rotation_x(-movement1abs * 0.1 + movement2abs * 0.1 + twitch2 * 0.02); } else { - let ori: Vec2 = Vec2::from(orientation); - let last_ori = Vec2::from(last_ori); - let tilt = if ::vek::Vec2::new(ori, last_ori) - .map(|o| 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.2) - * last_ori.determine_side(Vec2::zero(), ori).signum() - } else { - 0.0 - } * 1.3; - - next.chest.orientation = - Quaternion::rotation_x(movement1abs * 0.1 - movement2abs * 0.1) - * Quaternion::rotation_y(tilt * 1.8); } next diff --git a/voxygen/anim/src/bird_large/swim.rs b/voxygen/anim/src/bird_large/swim.rs index d81cc092d1..f914cc20e6 100644 --- a/voxygen/anim/src/bird_large/swim.rs +++ b/voxygen/anim/src/bird_large/swim.rs @@ -8,7 +8,7 @@ pub struct SwimAnimation; impl Animation for SwimAnimation { #[allow(clippy::type_complexity)] - type Dependency = (Vec3, Vec3, Vec3, f32); + type Dependency = f32; type Skeleton = BirdLargeSkeleton; #[cfg(feature = "use-dyn-lib")] @@ -17,7 +17,7 @@ impl Animation for SwimAnimation { #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_swim")] fn update_skeleton_inner( skeleton: &Self::Skeleton, - (_velocity, _orientation, _last_ori, global_time): Self::Dependency, + global_time: Self::Dependency, anim_time: f32, _rate: &mut f32, s_a: &SkeletonAttr, diff --git a/voxygen/src/scene/figure/mod.rs b/voxygen/src/scene/figure/mod.rs index 6e63d4547b..9eb7f1b45f 100644 --- a/voxygen/src/scene/figure/mod.rs +++ b/voxygen/src/scene/figure/mod.rs @@ -3335,13 +3335,7 @@ impl FigureMgr { // Swim (_, true, _) => anim::bird_large::SwimAnimation::update_skeleton( &BirdLargeSkeleton::default(), - ( - rel_vel, - // TODO: Update to use the quaternion. - ori * anim::vek::Vec3::::unit_y(), - state.last_ori * anim::vek::Vec3::::unit_y(), - time, - ), + time, state.state_time, &mut state_animation_rate, skeleton_attr, @@ -3380,7 +3374,6 @@ impl FigureMgr { anim::bird_large::BreatheAnimation::update_skeleton( &target_base, ( - rel_vel.magnitude(), time, ori * anim::vek::Vec3::::unit_y(), state.last_ori * anim::vek::Vec3::::unit_y(), @@ -3425,8 +3418,7 @@ impl FigureMgr { Some(s.stage_section), ori * anim::vek::Vec3::::unit_y(), state.last_ori * anim::vek::Vec3::::unit_y(), - time, - state.state_time, + physics.on_ground, ), stage_progress, &mut state_animation_rate, @@ -3479,7 +3471,7 @@ impl FigureMgr { }; anim::bird_large::ShockwaveAnimation::update_skeleton( &target_base, - (time, rel_vel.magnitude(), Some(s.stage_section)), + (Some(s.stage_section), physics.on_ground), stage_progress, &mut state_animation_rate, skeleton_attr, diff --git a/voxygen/src/session/mod.rs b/voxygen/src/session/mod.rs index a0e9895f88..fa1b98732c 100644 --- a/voxygen/src/session/mod.rs +++ b/voxygen/src/session/mod.rs @@ -522,7 +522,6 @@ impl PlayState for SessionState { select_pos, target_entity.map(|t| t.0), ); - dbg!(self.key_state.fly); }, GameInput::Climb => { self.key_state.climb_up = state; From abfb14626a28753a86c661beea0cc64031fe1211 Mon Sep 17 00:00:00 2001 From: Snowram Date: Sat, 24 Apr 2021 02:03:07 +0200 Subject: [PATCH 10/10] Body specific beam offsets --- CHANGELOG.md | 2 ++ common/src/comp/body.rs | 2 +- common/src/states/basic_beam.rs | 12 ++++++++---- voxygen/anim/src/bird_large/shoot.rs | 22 ++-------------------- voxygen/benches/meshing_benchmark.rs | 7 ++++++- voxygen/src/scene/figure/mod.rs | 3 --- 6 files changed, 19 insertions(+), 29 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 6cb1304efb..b0aa14db29 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -45,6 +45,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Drag is now calculated based on physical properties - Terrain chunks are now deflate-compressed when sent over the network. - Missing translations can be displayed in English. +- New large birds npcs +- Day period dependant wildlife spawns ### Changed diff --git a/common/src/comp/body.rs b/common/src/comp/body.rs index d11aca639a..1fe48767b6 100644 --- a/common/src/comp/body.rs +++ b/common/src/comp/body.rs @@ -282,7 +282,7 @@ impl Body { }, Body::BipedSmall(_) => Vec3::new(1.0, 0.75, 1.4), Body::BirdMedium(_) => Vec3::new(2.0, 1.0, 1.1), - Body::BirdLarge(_) => Vec3::new(2.0, 5.5, 3.8), + Body::BirdLarge(_) => Vec3::new(2.0, 5.0, 2.4), Body::Dragon(_) => Vec3::new(16.0, 10.0, 16.0), Body::FishMedium(_) => Vec3::new(0.5, 2.0, 0.8), Body::FishSmall(_) => Vec3::new(0.3, 1.2, 0.6), diff --git a/common/src/states/basic_beam.rs b/common/src/states/basic_beam.rs index 766959470f..ca700bf91b 100644 --- a/common/src/states/basic_beam.rs +++ b/common/src/states/basic_beam.rs @@ -3,7 +3,7 @@ use crate::{ Attack, AttackDamage, AttackEffect, CombatEffect, CombatRequirement, Damage, DamageSource, GroupTarget, }, - comp::{beam, CharacterState, EnergyChange, EnergySource, Ori, Pos, StateUpdate}, + comp::{beam, Body, CharacterState, EnergyChange, EnergySource, Ori, Pos, StateUpdate}, event::ServerEvent, states::{ behavior::{CharacterBehavior, JoinData}, @@ -136,11 +136,15 @@ impl CharacterBehavior for Data { owner: Some(*data.uid), specifier: self.static_data.specifier, }; + let body_offsets_z = match data.body { + Body::BirdLarge(_) => data.body.height() * 0.9, + _ => data.body.height() * 0.5, + }; // Gets offsets let body_offsets = Vec3::new( - (data.body.radius() + 0.2) * data.inputs.look_dir.x, - (data.body.radius() + 0.2) * data.inputs.look_dir.y, - data.body.eye_height() * 0.6, + (data.body.radius() + 1.0) * data.inputs.look_dir.x, + (data.body.radius() + 1.0) * data.inputs.look_dir.y, + body_offsets_z, ); let pos = Pos(data.pos.0 + body_offsets); // Create beam segment diff --git a/voxygen/anim/src/bird_large/shoot.rs b/voxygen/anim/src/bird_large/shoot.rs index cd56d3020b..23cd8c3f08 100644 --- a/voxygen/anim/src/bird_large/shoot.rs +++ b/voxygen/anim/src/bird_large/shoot.rs @@ -6,16 +6,7 @@ use common::{states::utils::StageSection, util::Dir}; pub struct ShootAnimation; -type ShootAnimationDependency = ( - f32, - f32, - Vec3, - Vec3, - Option, - f32, - Dir, - bool, -); +type ShootAnimationDependency = (f32, Option, f32, Dir, bool); impl Animation for ShootAnimation { type Dependency = ShootAnimationDependency; @@ -27,16 +18,7 @@ impl Animation for ShootAnimation { #[cfg_attr(feature = "be-dyn-lib", export_name = "bird_large_shoot")] fn update_skeleton_inner( skeleton: &Self::Skeleton, - ( - _velocity, - global_time, - _orientation, - _last_ori, - stage_section, - timer, - look_dir, - on_ground, - ): Self::Dependency, + (global_time, stage_section, timer, look_dir, on_ground): Self::Dependency, anim_time: f32, _rate: &mut f32, s_a: &SkeletonAttr, diff --git a/voxygen/benches/meshing_benchmark.rs b/voxygen/benches/meshing_benchmark.rs index 31f12a9215..a203913312 100644 --- a/voxygen/benches/meshing_benchmark.rs +++ b/voxygen/benches/meshing_benchmark.rs @@ -27,7 +27,12 @@ pub fn criterion_benchmark(c: &mut Criterion) { (0..GEN_SIZE) .flat_map(|x| (0..GEN_SIZE).map(move |y| Vec2::new(x, y))) .map(|offset| offset + CENTER) - .map(|pos| (pos, world.generate_chunk(index, pos, || false).unwrap())) + .map(|pos| { + ( + pos, + world.generate_chunk(index, pos, || false, None).unwrap(), + ) + }) .for_each(|(key, chunk)| { terrain.insert(key, Arc::new(chunk.0)); }); diff --git a/voxygen/src/scene/figure/mod.rs b/voxygen/src/scene/figure/mod.rs index 9eb7f1b45f..ff3845c199 100644 --- a/voxygen/src/scene/figure/mod.rs +++ b/voxygen/src/scene/figure/mod.rs @@ -3441,10 +3441,7 @@ impl FigureMgr { anim::bird_large::ShootAnimation::update_skeleton( &target_base, ( - rel_vel.magnitude(), time, - ori * anim::vek::Vec3::::unit_y(), - state.last_ori * anim::vek::Vec3::::unit_y(), Some(s.stage_section), state.state_time, look_dir,