Skip to content

Commit

Permalink
Smooth flights using requestAnimationFrame
Browse files Browse the repository at this point in the history
  • Loading branch information
jaredkhan committed Nov 30, 2024
1 parent 3a2deb0 commit 989d5c9
Show file tree
Hide file tree
Showing 2 changed files with 49 additions and 47 deletions.
90 changes: 44 additions & 46 deletions OZprivate/rawJS/OZTreeModule/src/position_helper.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,16 +4,14 @@ import {max, min} from './util/index';
let x_add = null;
let y_add = null;
let r_mult = null;
let intro_step_num = null;
let intro_sec_step_num = null;
let fly_start_time = null;
let global_anim_speed = 10; // set this to 15 for recoding smoother flight animations larger numbers mean slower
let length_intro = null;
let num_intro_steps = null;
let duration = null;
let more_flying_needed = null;
let flight_fps = 1000/60; // set this to 1000/500 for recording much slower flight animations that can then be sped up after the screen casting process is complete.
let original_flight_fps = 1000/60; // the frame rate used when global_anim_speed was calibrated
let into_node;
let pre_xp, pre_yp, pre_ws;
let fly_timer = null
let fly_frame_request_id = null

function drawreg_target(node,x,y,r) {
// we assume that only those for whom graphref is true will call this routine
Expand Down Expand Up @@ -182,7 +180,7 @@ function get_v_horizon_by_arc(node, factor, x, y, r) {
let vxmax = (x+r*(node.arcx+node.arcr*factor));
let vxmin = (x+r*(node.arcx-node.arcr*factor));
let vymax = (y+r*(node.arcy+node.arcr*factor));
let vymin = (y+r*(node.arcy-node.arcr*factor)); // add for leaf points *1.305 doesn'fly_timer care about the leaf direction - could change this later perhaps
let vymin = (y+r*(node.arcy-node.arcr*factor)); // add for leaf points *1.305 doesn't care about the leaf direction - could change this later perhaps
return [vxmax, vxmin, vymax, vymin]
}

Expand Down Expand Up @@ -220,8 +218,7 @@ function perform_actual_fly(controller, into_node, speed=1, accel_type="linear",
pre_yp = tree_state.yp;
pre_ws = tree_state.ws;
get_xyr_target(controller.root, tree_state.xp, tree_state.yp, 220*tree_state.ws, into_node);
intro_step_num = 0;
intro_sec_step_num = 0;
fly_start_time = performance.now();
if(((r_mult>0.9999)&&(r_mult<1.00001))&&(x_add*x_add<1)&&(y_add*y_add<1)) {
// nothing to zoom to so better to do nothing and return false or it feels like a bug
if (typeof finalize_func === "function") {
Expand All @@ -244,9 +241,10 @@ function perform_actual_fly(controller, into_node, speed=1, accel_type="linear",
}
return true;
} else {
length_intro = Math.abs(Math.log(r_mult))*global_anim_speed;
num_intro_steps = Math.max(Math.floor(length_intro),12)/speed;
perform_fly_b2(controller, into_node, speed, accel_type, finalize_func, abrupt_func);
duration = Math.max(Math.floor(Math.abs(Math.log(r_mult))*global_anim_speed),12) / speed / original_flight_fps;
fly_frame_request_id = requestAnimationFrame(function () {
perform_fly_b2(controller, into_node, speed, accel_type, finalize_func, abrupt_func);
});
return true;
}
}
Expand All @@ -267,17 +265,19 @@ function perform_actual_fly(controller, into_node, speed=1, accel_type="linear",
* @param {func} abrupt_func is optional, and gives a function to call when fly is abrupted
*/
function perform_fly_b2(controller, into_node, speed, accel_type, finalize_func, abrupt_func) {
function pan_proportion(step, total) {
var x = step / total,
out = (Math.sin((x+0.25) * Math.PI*2) + 1) / 2;

const t = (performance.now() - fly_start_time) / 1000 / duration;
const clamped_t = Math.min(1, Math.max(0, t));

function pan_proportion(x) {
var out = (Math.sin((x+0.25) * Math.PI*2) + 1) / 2;

return accel_type === 'accel' ? (x > 0.5 ? out : 0)
: accel_type === 'decel' ? 1 - (x < 0.5 ? out : 0)
: x;
}
function zoom_proportion(step, total) {
var x = step / total,
out = (Math.sin((x+1.5) * Math.PI) + 1) / 2;
function zoom_proportion(x) {
var out = (Math.sin((x+1.5) * Math.PI) + 1) / 2;

return accel_type === 'accel' ? out
: accel_type === 'decel' ? out
Expand All @@ -288,46 +288,44 @@ function perform_fly_b2(controller, into_node, speed, accel_type, finalize_func,
//need to reanchor, this sometimes causes jerkiness
//also we may not know how many steps we will need to take
// NB: Approximate accel/decel by not panning at all, let the final non-reanchor step handle it
pan_zoom(accel_type === 'accel' || accel_type === 'decel'? 0 : 1/num_intro_steps, 1/num_intro_steps);
pan_zoom(accel_type === 'accel' || accel_type === 'decel'? 0 : clamped_t, clamped_t);
tree_state.set_action(r_mult > 1 ? "fly-in" : "fly-out");
controller.re_calc();
controller.reanchor();
controller.trigger_refresh_loop();

clearTimeout(fly_timer);
fly_timer = setTimeout(function () {
if (tree_state.flying) {
perform_actual_fly(controller, into_node, speed, 'linear', finalize_func, abrupt_func);
} else if (typeof abrupt_func === 'function') {
abrupt_func()
}
},1000.0/flight_fps);
} else if (!more_flying_needed && intro_step_num <num_intro_steps) {

if (tree_state.flying) {
perform_actual_fly(controller, into_node, speed, 'linear', finalize_func, abrupt_func);
} else if (typeof abrupt_func === 'function') {
abrupt_func()
}

} else {
//don't need to reanchor - this is more normal, and is smoother
intro_step_num++;
intro_sec_step_num++;
pan_zoom(
pan_proportion(intro_sec_step_num, num_intro_steps),
zoom_proportion(intro_sec_step_num, num_intro_steps)
pan_proportion(clamped_t),
zoom_proportion(clamped_t)
);
tree_state.set_action(r_mult > 1 ? "fly-in" : "fly-out");
controller.re_calc();
controller.trigger_refresh_loop();

clearTimeout(fly_timer);
fly_timer = setTimeout(function () {
if (tree_state.flying) {
perform_fly_b2(controller, into_node, speed, accel_type, finalize_func, abrupt_func);
} else if (typeof abrupt_func === 'function') {
abrupt_func()
cancelAnimationFrame(fly_frame_request_id);

if (t < 1) {
fly_frame_request_id = requestAnimationFrame(function () {
if (tree_state.flying) {
perform_fly_b2(controller, into_node, speed, accel_type, finalize_func, abrupt_func);
} else if (typeof abrupt_func === 'function') {
abrupt_func()
}
});
} else {
tree_state.flying = false;
tree_state.set_action(null);
if (typeof finalize_func === "function") {
finalize_func()
}
},1000.0/flight_fps);
} else {
clearTimeout(fly_timer);
tree_state.flying = false;
tree_state.set_action(null);
if (typeof finalize_func === "function") {
finalize_func()
}
}
}
Expand Down
6 changes: 5 additions & 1 deletion OZprivate/rawJS/OZTreeModule/tests/test_position_helper.js
Original file line number Diff line number Diff line change
Expand Up @@ -98,8 +98,9 @@ function test_cur_location(test, controller, node_latin_name, exp_xp, exp_yp, ex
exp_xp = round(exp_xp);
}


test('perform_actual_fly', function (test) {
global.requestAnimationFrame = (callback) => setTimeout(callback, 16);
global.cancelAnimationFrame = clearTimeout;
var nodes = {};

return populate_factory().then((factory) => {
Expand Down Expand Up @@ -151,6 +152,9 @@ test('perform_actual_fly', function (test) {


test.onFinish(function() {

global.requestAnimationFrame = undefined;
global.cancelAnimationFrame = undefined;
// NB: Something data_repo includes in is holding node open.
// Can't find it so force our tests to end.
process.exit(0)
Expand Down

0 comments on commit 989d5c9

Please sign in to comment.