+ /* camera */
+ v3_copy( localplayer.cam.pos, frame->cam_pos );
+ if( localplayer.gate_waiting ){
+ m4x3_mulv( localplayer.gate_waiting->transport,
+ frame->cam_pos, frame->cam_pos );
+ }
+ v3_copy( localplayer.cam.angles, frame->cam_angles );
+ frame->cam_fov = localplayer.cam.fov;
+
+ /* animator */
+ void *dst = replay_frame_data( frame, k_replay_framedata_animator );
+
+ if( localplayer.subsystem == k_player_subsystem_walk )
+ memcpy( dst, &player_walk.animator, animator_size );
+ else if( localplayer.subsystem == k_player_subsystem_skate )
+ memcpy( dst, &player_skate.animator, animator_size );
+ else if( localplayer.subsystem == k_player_subsystem_dead ){
+ memcpy( dst, &player_dead.animator, animator_size );
+ }
+}
+
+static
+void skaterift_restore_frame( replay_frame *frame ){
+ replay_gamestate *gs =
+ replay_frame_data( frame, k_replay_framedata_internal_gamestate );
+ void *src = replay_frame_data( frame, k_replay_framedata_gamestate );
+ u16 src_size = frame->data_table[ k_replay_framedata_gamestate ][1];
+ assert( src_size );
+
+ if(frame->system == k_player_subsystem_walk ){
+ memcpy( &player_walk.state, src, src_size );
+ }
+ else if( frame->system == k_player_subsystem_skate ){
+ memcpy( &player_skate.state, src, src_size );
+ }
+ else if( frame->system == k_player_subsystem_dead ){
+ player__dead_transition();
+ struct replay_rb *arr = src;
+
+ for( u32 i=0; i<localplayer.ragdoll.part_count; i ++ ){
+ struct ragdoll_part *part = &localplayer.ragdoll.parts[i];
+ rigidbody *rb = &part->obj.rb;
+
+ v3_copy( arr[i].co, rb->co );
+ v3_copy( arr[i].w, rb->w );
+ v3_copy( arr[i].v, rb->v );
+ v4_copy( arr[i].q, rb->q );
+
+ v3_copy( arr[i].co, part->prev_co );
+ v4_copy( arr[i].q, part->prev_q );
+ }
+ }
+
+ localplayer.subsystem = frame->system;
+
+ memcpy( &localplayer.rb, &gs->rb, sizeof(rigidbody) );
+ v3_copy( gs->angles, localplayer.angles );
+
+ v3_copy( frame->cam_pos, localplayer.cam.pos );
+ v3_copy( frame->cam_angles, localplayer.cam.angles );
+ localplayer.cam.fov = frame->cam_fov;
+
+ memcpy( &localplayer.cam_control, &gs->cam_control,
+ sizeof(struct player_cam_controller) );
+
+ /* chop end off replay */
+ frame->r = NULL;
+ skaterift.replay.statehead = frame;
+ skaterift.replay.head = frame;
+ skaterift.replay.cursor_frame = frame;
+ skaterift.replay.cursor = frame->time;
+ skaterift.replay_control = k_replay_control_scrub;
+ skaterift.activity = k_skaterift_default;
+ vg.time = frame->time;
+}
+
+static void skaterift_replay_resume(void){
+ replay_frame *prev = replay_find_recent_stateframe(&skaterift.replay);
+
+ if( prev ){
+ skaterift.replay_control = k_replay_control_resume;
+ skaterift.resume_target = prev;
+ skaterift.resume_begin = skaterift.replay.cursor;
+ skaterift.resume_transition = 0.0f;
+ }
+}
+
+static void skaterift_replay_pre_update(void){
+ if( skaterift.activity != k_skaterift_replay ) return;
+
+ if( skaterift.replay_control == k_replay_control_resume ){
+ if( skaterift.replay.cursor_frame == skaterift.resume_target ||
+ skaterift.replay.cursor_frame == NULL ){
+ skaterift_restore_frame( skaterift.resume_target );
+ }
+ else {
+ vg_slewf( &skaterift.resume_transition, 1.0f,
+ vg.time_frame_delta * (1.0f/1.0f) );
+
+ if( skaterift.resume_transition >= 1.0f )
+ skaterift_restore_frame( skaterift.resume_target );
+ else {
+ f64 target = vg_lerp( skaterift.resume_begin,
+ skaterift.resume_target->time,
+ vg_smoothstepf( skaterift.resume_transition ) );
+ replay_seek( &skaterift.replay, target );