diff --git a/crates/fj-kernel/src/algorithms/approx/curve.rs b/crates/fj-kernel/src/algorithms/approx/curve.rs index 34c0a9cde..25b6470a9 100644 --- a/crates/fj-kernel/src/algorithms/approx/curve.rs +++ b/crates/fj-kernel/src/algorithms/approx/curve.rs @@ -61,7 +61,7 @@ pub fn approx_circle( let n = number_of_vertices_for_circle(tolerance, radius); for i in 0..n { - let angle = Scalar::PI * 2. / n as f64 * i as f64; + let angle = Scalar::TAU / n as f64 * i as f64; let point = circle.point_from_circle_coords([angle]); out.push(Local::new([angle], point)); } diff --git a/crates/fj-math/src/circle.rs b/crates/fj-math/src/circle.rs index 31ccba817..3e12173cd 100644 --- a/crates/fj-math/src/circle.rs +++ b/crates/fj-math/src/circle.rs @@ -107,7 +107,7 @@ impl Circle { let coord = if atan >= Scalar::ZERO { atan } else { - atan + Scalar::PI * 2. + atan + Scalar::TAU }; Point::from([coord]) } diff --git a/crates/fj-math/src/line.rs b/crates/fj-math/src/line.rs index 4bfed5413..9d247ee7e 100644 --- a/crates/fj-math/src/line.rs +++ b/crates/fj-math/src/line.rs @@ -74,7 +74,7 @@ impl Line { // The triangle is valid only, if the three points are not on the // same line. - Triangle::from_points([a, b, c]).is_some() + Triangle::from_points([a, b, c]).is_ok() }; if other_origin_is_not_on_self { diff --git a/crates/fj-math/src/scalar.rs b/crates/fj-math/src/scalar.rs index 96a258638..74407ac9b 100644 --- a/crates/fj-math/src/scalar.rs +++ b/crates/fj-math/src/scalar.rs @@ -1,4 +1,10 @@ -use std::{cmp, f64::consts::PI, fmt, hash::Hash, ops}; +use std::{ + cmp, + f64::consts::{PI, TAU}, + fmt, + hash::Hash, + ops, +}; use decorum::R64; @@ -28,6 +34,9 @@ impl Scalar { /// The `Scalar` instance that represents pi pub const PI: Self = Self(PI); + /// The `Scalar` instance that represents tau + pub const TAU: Self = Self(TAU); + /// Construct a `Scalar` from an `f64` /// /// # Panics @@ -61,6 +70,18 @@ impl Scalar { self.0 as u64 } + /// The sign of the scalar + /// + /// Return `Scalar::ZERO`, if the scalar is zero, `Scalar::ONE`, if it is + /// positive, `-Scalar::ONE`, if it is negative. + pub fn sign(self) -> Scalar { + if self == Self::ZERO { + Self::ZERO + } else { + Self(self.0.signum()) + } + } + /// Compute the absolute value of the scalar pub fn abs(self) -> Self { self.0.abs().into() diff --git a/crates/fj-math/src/triangle.rs b/crates/fj-math/src/triangle.rs index f4cd21888..02e87b7a4 100644 --- a/crates/fj-math/src/triangle.rs +++ b/crates/fj-math/src/triangle.rs @@ -18,10 +18,10 @@ pub struct Triangle { impl Triangle { /// Construct a triangle from three points /// - /// # Panics - /// - /// Panics, if the points don't form a triangle. - pub fn from_points(points: [impl Into>; 3]) -> Option { + /// Returns an error, if the points don't form a triangle. + pub fn from_points( + points: [impl Into>; 3], + ) -> Result> { let points = points.map(Into::into); let area = { @@ -31,9 +31,9 @@ impl Triangle { // A triangle is not valid if it doesn't span any area if area != Scalar::from(0.0) { - Some(Self { points }) + Ok(Self { points }) } else { - None + Err(NotATriangle { points }) } } @@ -107,6 +107,12 @@ where } } +/// Returned by [`Triangle::from_points`], if the points don't form a triangle +#[derive(Clone, Copy, Debug, Eq, PartialEq, Hash, Ord, PartialOrd)] +pub struct NotATriangle { + pub points: [Point; 3], +} + /// Winding direction of a triangle. pub enum Winding { /// Counter-clockwise