pub mod anim;
pub mod prelude;
use int_math::UVec2;
use std::fmt;
pub enum VirtualScale {
IntScale(u16),
FloatScale(f32),
}
fn gcd(a: u16, b: u16) -> u16 {
if b == 0 {
a
} else {
gcd(b, a % b)
}
}
fn aspect_ratio(size: UVec2) -> (u16, u16) {
let divisor = gcd(size.x, size.y);
(size.x / divisor, size.y / divisor)
}
#[derive(Debug)]
pub enum AspectRatio {
Ratio16By9,
Ratio21By9,
Ratio16By10,
Ratio4By3,
Other(f32),
}
impl AspectRatio {
fn convert(value: UVec2) -> Self {
let aspect = aspect_ratio(value);
match aspect {
(16, 9) => Self::Ratio16By9,
(21, 9) => Self::Ratio21By9,
(16, 10) => Self::Ratio16By10,
(4, 3) => Self::Ratio4By3,
_ => Self::Other(value.x as f32 / value.y as f32),
}
}
}
impl From<(u16, u16)> for AspectRatio {
fn from(value: (u16, u16)) -> Self {
Self::convert(value.into())
}
}
impl From<UVec2> for AspectRatio {
fn from(value: UVec2) -> Self {
Self::convert(value)
}
}
impl fmt::Display for AspectRatio {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Self::Ratio16By9 => write!(f, "16:9"),
Self::Ratio16By10 => write!(f, "16:10"),
Self::Ratio21By9 => write!(f, "21:9"),
Self::Ratio4By3 => write!(f, "4:3"),
Self::Other(vec) => write!(f, "aspect ratio: {:?}", vec),
}
}
}
pub struct Color {
r: u8,
g: u8,
b: u8,
a: u8,
}
impl Color {
pub fn from_f32(r: f32, g: f32, b: f32, a: f32) -> Self {
Self::from_octet(
(r * 255.0) as u8,
(g * 255.0) as u8,
(b * 255.0) as u8,
(a * 255.0) as u8,
)
}
pub const fn from_octet(r: u8, g: u8, b: u8, a: u8) -> Self {
Self { r, g, b, a }
}
pub fn to_f64(&self) -> (f64, f64, f64, f64) {
(
self.r as f64 / 255.0,
self.g as f64 / 255.0,
self.b as f64 / 255.0,
self.a as f64 / 255.0,
)
}
}
#[derive(Debug, Eq, PartialEq)]
pub enum ViewportStrategy {
FitIntegerScaling(UVec2),
FitFloatScaling(UVec2),
MatchPhysicalSize,
}