skia_safe/core/
scalar_.rsuse std::ops::Mul;
#[allow(non_camel_case_types)]
pub type scalar = skia_bindings::SkScalar;
pub trait Scalar: Copy {
const ZERO: Self;
const NEARLY_ZERO: Self;
const ONE: Self;
const HALF: Self;
fn nearly_equal(x: scalar, y: scalar, tolerance: impl Into<Option<scalar>>) -> bool;
fn nearly_zero(&self, tolerance: impl Into<Option<scalar>>) -> bool;
}
pub trait Scalars {
fn are_finite(&self) -> bool;
}
impl Scalar for scalar {
const ZERO: Self = 0.0;
const NEARLY_ZERO: Self = 1.0 / ((1 << 12) as Self);
const ONE: Self = 1.0;
const HALF: Self = 0.5;
fn nearly_equal(x: scalar, y: scalar, tolerance: impl Into<Option<scalar>>) -> bool {
let tolerance = tolerance.into().unwrap_or(Self::NEARLY_ZERO);
debug_assert!(tolerance >= 0.0);
(x - y).abs() <= tolerance
}
fn nearly_zero(&self, tolerance: impl Into<Option<scalar>>) -> bool {
let tolerance = tolerance.into().unwrap_or(Self::NEARLY_ZERO);
debug_assert!(tolerance >= 0.0);
self.abs() <= tolerance
}
}
impl<T> Scalars for [T]
where
T: Scalar,
T: Mul<Output = T>,
T: PartialEq,
{
fn are_finite(&self) -> bool {
self.iter().fold(T::ZERO, |prod, value| prod * *value) == T::ZERO
}
}
pub fn radians_to_degrees(radians: scalar) -> scalar {
radians * (180.0 / std::f32::consts::PI)
}