implement settings

This commit is contained in:
Quaternions 2023-10-11 23:34:32 -07:00
parent 3b7a1d5dff
commit 7f7b0d92e6
2 changed files with 119 additions and 33 deletions

View File

@ -57,7 +57,17 @@ pub struct Ratio64{
den:std::num::NonZeroU64, den:std::num::NonZeroU64,
} }
impl Ratio64{ impl Ratio64{
pub const ZERO:Self=Ratio64{num:0,den:unsafe{std::num::NonZeroU64::new_unchecked(1)}};
pub const ONE:Self=Ratio64{num:1,den:unsafe{std::num::NonZeroU64::new_unchecked(1)}}; pub const ONE:Self=Ratio64{num:1,den:unsafe{std::num::NonZeroU64::new_unchecked(1)}};
pub fn new(num:i64,den:u64)->Option<Ratio64>{
match std::num::NonZeroU64::new(den){
Some(den)=>{
//TODO: gcd
Some(Self{num,den})
},
None=>None,
}
}
pub fn mul_int(self,rhs:i64)->i64{ pub fn mul_int(self,rhs:i64)->i64{
rhs*self.num/self.den.get() as i64 rhs*self.num/self.den.get() as i64
} }
@ -65,6 +75,68 @@ impl Ratio64{
rhs*self.den.get() as i64/self.num rhs*self.den.get() as i64/self.num
} }
} }
//from num_traits crate
#[inline]
fn integer_decode_f64(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)
}
#[derive(Debug)]
enum Ratio64TryFromFloat64Error{
Nan,
Infinite,
Subnormal,
HighlyNegativeExponent(i16),
HighlyPositiveExponent(i16),
}
impl TryFrom<f64> for Ratio64{
type Error=Ratio64TryFromFloat64Error;
fn try_from(value:f64)->Result<Self,Self::Error>{
match value.classify(){
std::num::FpCategory::Nan=>Err(Self::Error::Nan),
std::num::FpCategory::Infinite=>Err(Self::Error::Infinite),
std::num::FpCategory::Zero=>Ok(Self::ZERO),
std::num::FpCategory::Subnormal=>Err(Self::Error::Subnormal),
std::num::FpCategory::Normal=>{
let (m,e,s)=integer_decode_f64(value);
if e< -127{
//bye bye
Err(Self::Error::HighlyNegativeExponent(e))
}else if e< -63{
//TODO
Err(Self::Error::HighlyNegativeExponent(e))
}else if e<0{
Ok(Ratio64::new((m as i64)*(s as i64),1<<-e).unwrap())
}else if e<62-52{
Ok(Ratio64::new((m as i64)*(s as i64)*(1<<e),1).unwrap())
}else{
Err(Self::Error::HighlyPositiveExponent(e))
}
},
}
}
}
impl std::ops::Mul<Ratio64> for Ratio64{
type Output=Ratio64;
#[inline]
fn mul(self,rhs:Ratio64)->Self::Output{
let (num,den)=(self.num*rhs.num,self.den.get()*rhs.den.get());
//TODO: gcd
Self{
num,
den:unsafe{std::num::NonZeroU64::new_unchecked(den)},
}
}
}
impl std::ops::Mul<i64> for Ratio64{ impl std::ops::Mul<i64> for Ratio64{
type Output=Ratio64; type Output=Ratio64;
#[inline] #[inline]
@ -92,6 +164,9 @@ pub struct Ratio64Vec2{
} }
impl Ratio64Vec2{ impl Ratio64Vec2{
pub const ONE:Self=Self{x:Ratio64::ONE,y:Ratio64::ONE}; pub const ONE:Self=Self{x:Ratio64::ONE,y:Ratio64::ONE};
pub fn new(x:Ratio64,y:Ratio64)->Self{
Self{x,y}
}
pub fn mul_int(self,rhs:glam::I64Vec2)->glam::I64Vec2{ pub fn mul_int(self,rhs:glam::I64Vec2)->glam::I64Vec2{
glam::i64vec2( glam::i64vec2(
self.x.mul_int(rhs.x), self.x.mul_int(rhs.x),

View File

@ -1,3 +1,4 @@
use crate::integer::{Ratio64,Ratio64Vec2};
struct Ratio{ struct Ratio{
ratio:f64, ratio:f64,
} }
@ -7,23 +8,25 @@ enum DerivedFov{
} }
enum Fov{ enum Fov{
Exactly{x:f64,y:f64}, Exactly{x:f64,y:f64},
DeriveX{x:DerivedFov,y:f64}, SpecifyXDeriveY{x:f64,y:DerivedFov},
DeriveY{x:f64,y:DerivedFov}, SpecifyYDeriveX{x:DerivedFov,y:f64},
} }
impl Default for Fov{ impl Default for Fov{
fn default()->Self{ fn default()->Self{
Fov::DeriveX{x:DerivedFov::FromScreenAspect,y:1.0} Fov::SpecifyYDeriveX{x:DerivedFov::FromScreenAspect,y:1.0}
} }
} }
enum DerivedSensitivity{
FromRatio(Ratio64),
}
enum Sensitivity{ enum Sensitivity{
Exactly{x:f64,y:f64}, Exactly{x:Ratio64,y:Ratio64},
DeriveX{x:Ratio,y:f64}, SpecifyXDeriveY{x:Ratio64,y:DerivedSensitivity},
DeriveY{x:f64,y:Ratio}, SpecifyYDeriveX{x:DerivedSensitivity,y:Ratio64},
} }
impl Default for Sensitivity{ impl Default for Sensitivity{
fn default()->Self{ fn default()->Self{
Sensitivity::DeriveY{x:0.001,y:Ratio{ratio:1.0}} Sensitivity::SpecifyXDeriveY{x:Ratio64::ONE/1000,y:DerivedSensitivity::FromRatio(Ratio64::ONE)}
} }
} }
@ -36,21 +39,25 @@ impl UserSettings{
pub fn calculate_fov(&self,zoom:f64,screen_size:&glam::UVec2)->glam::DVec2{ pub fn calculate_fov(&self,zoom:f64,screen_size:&glam::UVec2)->glam::DVec2{
zoom*match &self.fov{ zoom*match &self.fov{
&Fov::Exactly{x,y}=>glam::dvec2(x,y), &Fov::Exactly{x,y}=>glam::dvec2(x,y),
Fov::DeriveX{x,y}=>match x{ Fov::SpecifyXDeriveY{x,y}=>match y{
DerivedFov::FromScreenAspect=>glam::dvec2(y*(screen_size.x as f64/screen_size.y as f64),*y),
DerivedFov::FromAspect(ratio)=>glam::dvec2(y*ratio.ratio,*y),
},
Fov::DeriveY{x,y}=>match y{
DerivedFov::FromScreenAspect=>glam::dvec2(*x,x*(screen_size.y as f64/screen_size.x as f64)), DerivedFov::FromScreenAspect=>glam::dvec2(*x,x*(screen_size.y as f64/screen_size.x as f64)),
DerivedFov::FromAspect(ratio)=>glam::dvec2(*x,x*ratio.ratio), DerivedFov::FromAspect(ratio)=>glam::dvec2(*x,x*ratio.ratio),
}, },
Fov::SpecifyYDeriveX{x,y}=>match x{
DerivedFov::FromScreenAspect=>glam::dvec2(y*(screen_size.x as f64/screen_size.y as f64),*y),
DerivedFov::FromAspect(ratio)=>glam::dvec2(y*ratio.ratio,*y),
},
} }
} }
pub fn calculate_sensitivity(&self)->glam::DVec2{ pub fn calculate_sensitivity(&self)->Ratio64Vec2{
match &self.sensitivity{ match &self.sensitivity{
&Sensitivity::Exactly{x,y}=>glam::dvec2(x,y), &Sensitivity::Exactly{x,y}=>Ratio64Vec2::new(x,y),
Sensitivity::DeriveX{x,y}=>glam::dvec2(y*x.ratio,*y), Sensitivity::SpecifyXDeriveY{x,y}=>match y{
Sensitivity::DeriveY{x,y}=>glam::dvec2(*x,x*y.ratio), &DerivedSensitivity::FromRatio(ratio)=>Ratio64Vec2::new(*x,*x*ratio),
}
Sensitivity::SpecifyYDeriveX{x,y}=>match x{
&DerivedSensitivity::FromRatio(ratio)=>Ratio64Vec2::new(*y*ratio,*y),
}
} }
} }
} }
@ -71,7 +78,7 @@ pub fn read_user_settings()->UserSettings{
x:fov_x, x:fov_x,
y:fov_y y:fov_y
}, },
(Ok(Some(fov_x)),Ok(None))=>Fov::DeriveY{ (Ok(Some(fov_x)),Ok(None))=>Fov::SpecifyXDeriveY{
x:fov_x, x:fov_x,
y:if let Ok(Some(fov_y_from_x_ratio))=cfg.getfloat("camera","fov_y_from_x_ratio"){ y:if let Ok(Some(fov_y_from_x_ratio))=cfg.getfloat("camera","fov_y_from_x_ratio"){
DerivedFov::FromAspect(Ratio{ratio:fov_y_from_x_ratio}) DerivedFov::FromAspect(Ratio{ratio:fov_y_from_x_ratio})
@ -79,7 +86,7 @@ pub fn read_user_settings()->UserSettings{
DerivedFov::FromScreenAspect DerivedFov::FromScreenAspect
} }
}, },
(Ok(None),Ok(Some(fov_y)))=>Fov::DeriveX{ (Ok(None),Ok(Some(fov_y)))=>Fov::SpecifyYDeriveX{
x:if let Ok(Some(fov_x_from_y_ratio))=cfg.getfloat("camera","fov_x_from_y_ratio"){ x:if let Ok(Some(fov_x_from_y_ratio))=cfg.getfloat("camera","fov_x_from_y_ratio"){
DerivedFov::FromAspect(Ratio{ratio:fov_x_from_y_ratio}) DerivedFov::FromAspect(Ratio{ratio:fov_x_from_y_ratio})
}else{ }else{
@ -94,20 +101,24 @@ pub fn read_user_settings()->UserSettings{
let (cfg_sensitivity_x,cfg_sensitivity_y)=(cfg.getfloat("camera","sensitivity_x"),cfg.getfloat("camera","sensitivity_y")); let (cfg_sensitivity_x,cfg_sensitivity_y)=(cfg.getfloat("camera","sensitivity_x"),cfg.getfloat("camera","sensitivity_y"));
let sensitivity=match(cfg_sensitivity_x,cfg_sensitivity_y){ let sensitivity=match(cfg_sensitivity_x,cfg_sensitivity_y){
(Ok(Some(sensitivity_x)),Ok(Some(sensitivity_y)))=>Sensitivity::Exactly { (Ok(Some(sensitivity_x)),Ok(Some(sensitivity_y)))=>Sensitivity::Exactly {
x:sensitivity_x, x:Ratio64::try_from(sensitivity_x).unwrap(),
y:sensitivity_y y:Ratio64::try_from(sensitivity_y).unwrap(),
}, },
(Ok(Some(sensitivity_x)),Ok(None))=>Sensitivity::DeriveY{ (Ok(Some(sensitivity_x)),Ok(None))=>Sensitivity::SpecifyXDeriveY{
x:sensitivity_x, x:Ratio64::try_from(sensitivity_x).unwrap(),
y:Ratio{ y:if let Ok(Some(sensitivity_y_from_x_ratio))=cfg.getfloat("camera","sensitivity_y_from_x_ratio"){
ratio:if let Ok(Some(sensitivity_y_from_x_ratio))=cfg.getfloat("camera","sensitivity_y_from_x_ratio"){sensitivity_y_from_x_ratio}else{1.0} DerivedSensitivity::FromRatio(Ratio64::try_from(sensitivity_y_from_x_ratio).unwrap())
} }else{
DerivedSensitivity::FromRatio(Ratio64::ONE)
}, },
(Ok(None),Ok(Some(sensitivity_y)))=>Sensitivity::DeriveX{
x:Ratio{
ratio:if let Ok(Some(sensitivity_x_from_y_ratio))=cfg.getfloat("camera","sensitivity_x_from_y_ratio"){sensitivity_x_from_y_ratio}else{1.0}
}, },
y:sensitivity_y, (Ok(None),Ok(Some(sensitivity_y)))=>Sensitivity::SpecifyYDeriveX{
x:if let Ok(Some(sensitivity_x_from_y_ratio))=cfg.getfloat("camera","sensitivity_x_from_y_ratio"){
DerivedSensitivity::FromRatio(Ratio64::try_from(sensitivity_x_from_y_ratio).unwrap())
}else{
DerivedSensitivity::FromRatio(Ratio64::ONE)
},
y:Ratio64::try_from(sensitivity_y).unwrap(),
}, },
_=>{ _=>{
Sensitivity::default() Sensitivity::default()