8 Commits
video ... ratio

Author SHA1 Message Date
51bbc93116 more test 2026-02-25 08:38:28 -08:00
8c5e80d860 kill tabs 2026-02-25 08:38:28 -08:00
413e88d1ec no heap 2026-02-25 08:15:27 -08:00
8c5ff71af2 build entire continued fraction every loop 2026-02-25 08:02:38 -08:00
4528e6b0c2 work 2026-02-24 09:58:53 -08:00
834aa4263f js work 2026-02-24 09:12:20 -08:00
98e2b72418 implement it - easier than expected 2026-02-24 09:06:31 -08:00
385fa57f0a ratio 2026-02-24 08:50:49 -08:00
3 changed files with 145 additions and 22 deletions

View File

@@ -5,6 +5,8 @@ use strafesnet_roblox_bot_player::{bot,head,time,graphics};
use strafesnet_graphics::setup;
use strafesnet_common::physics::Time as PhysicsTime;
mod ratio;
// Hack to keep the code compiling,
// SurfaceTarget::Canvas is not available in IDE for whatever reason.
struct ToSurfaceTarget(web_sys::HtmlCanvasElement);
@@ -141,9 +143,10 @@ impl PlaybackHead{
self.head.set_paused(time,paused);
}
#[wasm_bindgen]
pub fn set_scale(&mut self,time:f64,scale_num:i64,scale_den:u64){
pub fn set_scale(&mut self,time:f64,scale:f64){
let time=time::from_float(time).unwrap();
self.head.set_scale(time,strafesnet_common::integer::Ratio64::new(scale_num,scale_den).unwrap());
let scale=crate::ratio::ratio_from_float(scale).unwrap();
self.head.set_scale(time,scale);
}
#[wasm_bindgen]
pub fn seek_to(&mut self,time:f64,new_time:f64){

130
wasm-module/src/ratio.rs Normal file
View File

@@ -0,0 +1,130 @@
use strafesnet_common::integer::Ratio64;
#[derive(Debug)]
pub enum RatioFromFloatError{
Nan,
Overflow,
Underflow,
}
fn f64_into_parts(f: f64) -> (u64, i16, i8) {
let bits: u64 = f.to_bits();
let sign: i8 = if bits >> 63 == 0 { 1 } else { -1 };
let mut exponent: i16 = ((bits >> 52) & 0x7ff) as i16;
let mantissa = if exponent == 0 {
(bits & 0xfffffffffffff) << 1
} else {
(bits & 0xfffffffffffff) | 0x10000000000000
};
// Exponent bias + mantissa shift
exponent -= 1023 + 52;
(mantissa, exponent, sign)
}
pub fn ratio_from_float(value:f64)->Result<Ratio64,RatioFromFloatError>{
// Handle special values first
match value.classify(){
core::num::FpCategory::Nan=>return Err(RatioFromFloatError::Nan),
core::num::FpCategory::Zero=>return Ok(Ratio64::ZERO),
core::num::FpCategory::Subnormal
|core::num::FpCategory::Normal
|core::num::FpCategory::Infinite=>{
if value<i64::MIN as f64{
return Err(RatioFromFloatError::Underflow);
}
if (i64::MAX as f64)<value{
return Err(RatioFromFloatError::Overflow);
}
}
}
// value = sign * mantissa * 2 ^ exponent
let (mantissa,exponent,sign)=f64_into_parts(value);
if 0<=exponent{
// we know value < i64::MIN, so it must just be an integer.
return Ok(Ratio64::new((sign as i64*mantissa as i64)<<exponent,1).unwrap());
}
if exponent< -62-52{
// very small number is not representable
return Ok(Ratio64::ZERO);
}
if -63<exponent{
// exactly representable
return Ok(Ratio64::new(sign as i64*mantissa as i64,1<<-exponent).unwrap());
}
// the denominator is necessarily bigger than the numerator at this point.
// create exact float num/den ratio
let mut in_num=mantissa as u128;
let mut in_den=1u128<<-exponent;
// a+b*c
fn ma(a:i64,b:i64,c:i64)->Option<i64>{
a.checked_add(b.checked_mul(c)?)
}
let mut p_prev=1i64;
let mut q_prev=0i64;
let mut p_cur=0i64;
let mut q_cur=1i64;
// compute continued fraction
loop{
let whole=in_den/in_num;
if whole==0{
// we have depleted the input fraction and created an exact representation
break;
}
if (i64::MAX as u128)<whole{
// cannot continue fraction
break;
}
let Some(p_next)=ma(p_prev,p_cur,whole as i64)else{
// overflow
break;
};
let Some(q_next)=ma(q_prev,q_cur,whole as i64)else{
// overflow
break;
};
p_prev=p_cur;
q_prev=q_cur;
p_cur=p_next;
q_cur=q_next;
(in_num,in_den)=(in_den-in_num*whole,in_num);
}
Ok(Ratio64::new(p_cur,q_cur as u64).unwrap())
}
#[cfg(test)]
fn req(r0:Ratio64,r1:Ratio64){
println!("r0={r0:?} r1={r1:?}");
assert_eq!(r0.num(),r1.num(),"Nums not eq");
assert_eq!(r0.den(),r1.den(),"Dens not eq");
}
#[test]
fn test(){
req(ratio_from_float(2.0).unwrap(),Ratio64::new(2,1).unwrap());
req(ratio_from_float(1.0).unwrap(),Ratio64::new(1,1).unwrap());
req(ratio_from_float(0.5).unwrap(),Ratio64::new(1,2).unwrap());
req(ratio_from_float(1.1).unwrap(),Ratio64::new(2476979795053773,2251799813685248).unwrap());
req(ratio_from_float(0.8).unwrap(),Ratio64::new(3602879701896397,4503599627370496).unwrap());
req(ratio_from_float(0.61).unwrap(),Ratio64::new(5494391545392005,9007199254740992).unwrap());
req(ratio_from_float(0.01).unwrap(),Ratio64::new(5764607523034235,576460752303423488).unwrap());
req(ratio_from_float(0.001).unwrap(),Ratio64::new(1152921504606847,1152921504606846976).unwrap());
req(ratio_from_float(0.00001).unwrap(),Ratio64::new(89605456633725,8960545663372499267).unwrap());
req(ratio_from_float(0.00000000001).unwrap(),Ratio64::new(35204848,3520484800000000213).unwrap());
req(ratio_from_float(0.000000000000000001).unwrap(),Ratio64::new(2,1999999999999999857).unwrap());
req(ratio_from_float(2222222222222.0).unwrap(),Ratio64::new(2222222222222,1).unwrap());
req(ratio_from_float(core::f64::consts::PI).unwrap(),Ratio64::new(884279719003555,281474976710656).unwrap());
}

View File

@@ -44,20 +44,11 @@ function elapsed() {
const control_speed = document.getElementById("control_speed");
var paused = false;
var speed = 0;
function set_speed(new_speed) {
speed = new_speed;
var speed_num = null;
var speed_den = null;
if (new_speed < 0) {
speed_num = BigInt(4) ** BigInt(-new_speed);
speed_den = BigInt(5) ** BigInt(-new_speed);
} else {
speed_num = BigInt(5) ** BigInt(new_speed);
speed_den = BigInt(4) ** BigInt(new_speed);
}
playback.set_scale(elapsed(), speed_num, speed_den);
control_speed.value = `${((5 / 4) ** new_speed).toPrecision(3)}x`;
var scale = 1;
function set_scale(new_scale) {
scale = new_scale;
playback.set_scale(elapsed(), scale);
control_speed.value = `${scale.toPrecision(3)}x`;
}
// Controls
@@ -75,25 +66,24 @@ document.getElementById("control_backward").addEventListener("click", (e) => {
playback.seek_backward(2.0);
});
document.getElementById("control_slower").addEventListener("click", (e) => {
set_speed(Math.max(speed - 1, -27));
set_scale((scale * 4) / 5);
});
const regex = new RegExp("^([^x]*)x?$");
control_speed.addEventListener("change", (e) => {
const parsed = regex.exec(e.target.value);
if (!parsed) {
set_speed(0);
set_scale(1);
return;
}
const input = Number(parsed.at(1));
if (Number.isNaN(input)) {
set_speed(0);
set_scale(1);
return;
}
const rounded = Math.round(Math.log(input) / Math.log(5 / 4));
set_speed(Math.max(-27, Math.min(27, rounded)));
set_scale(input);
});
document.getElementById("control_faster").addEventListener("click", (e) => {
set_speed(Math.min(speed + 1, 27));
set_scale((scale * 5) / 4);
});
// Rendering