+ f32 head_yaw = localplayer.angles[0] + VG_PIf,
+ y = vg_angle_diff( head_yaw, -walk_yaw ),
+ p = vg_clampf( localplayer.angles[1],
+ -k_sit_pitch_limit, k_sit_pitch_limit );
+
+ if( fabsf(y) > k_sit_yaw_limit ){
+ y = 0.0f;
+ p = 0.0f;
+ }
+ animator->yaw = vg_lerpf( animator->yaw, y, vg.time_delta*2.0f );
+ animator->pitch = vg_lerpf( animator->pitch, p, vg.time_delta*2.8f );
+ }
+
+ if( w->state.outro_type ){
+ struct player_avatar *av = localplayer.playeravatar;
+ struct skeleton_anim *anim =
+ player_walk_outro_anim( w->state.outro_type );
+ f32 outro_length = (f32)anim->length / anim->rate,
+ outro_time = vg.time - w->state.outro_start_time;
+ animator->outro_t = outro_time / outro_length;
+
+ if( w->state.outro_type == k_walk_outro_drop_in ){
+ float inv_rate = 1.0f / anim->rate,
+ anim_frames = anim->length * inv_rate,
+ step_frames = 12.0f * inv_rate,
+ commit_frames = 6.0f * inv_rate,
+ drop_frames = anim_frames - step_frames,
+ step_t = vg_minf( 1.0f, outro_time / step_frames ),
+ remaind_time = vg_maxf( 0.0f, outro_time - step_frames ),
+ dop_t = vg_minf( 1.0f, remaind_time / drop_frames );
+ animator->commit_t = vg_minf( 1.0f, remaind_time / commit_frames );
+
+ walk_yaw = vg_alerpf( w->state.drop_in_start_angle,
+ w->state.drop_in_angle, step_t );
+
+ v3_lerp( w->state.drop_in_start, w->state.drop_in_target,
+ step_t, localplayer.rb.co );
+ q_axis_angle( animator->root_q, (v3f){0.0f,1.0f,0.0f},
+ walk_yaw + VG_PIf );
+
+ v3_copy( w->state.drop_in_foot_anchor, animator->foot_anchor );
+
+ /* the drop in bit */
+ if( step_t >= 1.0f ){
+ v4f final_q;
+ player_walk_drop_in_overhang_transform( dop_t, localplayer.rb.co,
+ final_q );
+ q_mul( final_q, animator->root_q, animator->root_q );
+
+ v4_copy( animator->root_q, localplayer.rb.q );
+ v3_muladds( animator->root_co, localplayer.rb.to_world[1],
+ -0.1f*dop_t, animator->root_co );
+ }
+ return;
+ }
+ else{
+ v3_muladds( animator->root_co, localplayer.rb.to_world[1],
+ -0.1f*animator->outro_t, animator->root_co );
+ }