Add winding order for Triangle2d (#10620)
# Objective
This PR adds some helpers for `Triangle2d` to work with its winding
order. This could also be extended to polygons (and `Triangle3d` once
it's added).
## Solution
- Add `WindingOrder` enum with `Clockwise`, `Counterclockwise` and
`Invalid` variants
- `Invalid` is for cases where the winding order can not be reliably
computed, i.e. the points lie on a single line and the area is zero
- Add `Triangle2d::winding_order` method that uses a signed surface area
to determine the winding order
- Add `Triangle2d::reverse` method that reverses the winding order by
swapping the second and third vertices
The API looks like this:
```rust
let mut triangle = Triangle2d::new(
Vec2::new(0.0, 2.0),
Vec2::new(-0.5, -1.2),
Vec2::new(-1.0, -1.0),
);
assert_eq!(triangle.winding_order(), WindingOrder::Clockwise);
// Reverse winding order
triangle.reverse();
assert_eq!(triangle.winding_order(), WindingOrder::Counterclockwise);
```
I also added tests to make sure the methods work correctly. For now,
they live in the same file as the primitives.
## Open questions
- Should it be `Counterclockwise` or `CounterClockwise`? The first one
is more correct but perhaps a bit less readable. Counter-clockwise is
also a valid spelling, but it seems to be a lot less common than
counterclockwise.
- Is `WindingOrder::Invalid` a good name? Parry uses
`TriangleOrientation::Degenerate`, but I'm not a huge fan, at least as a
non-native English speaker. Any better suggestions?
- Is `WindingOrder` fine in `bevy_math::primitives`? It's not specific
to a dimension, so I put it there for now.
This commit is contained in:
parent
96444b24fd
commit
e1c8d60f91
@ -1,4 +1,4 @@
|
||||
use super::Primitive2d;
|
||||
use super::{Primitive2d, WindingOrder};
|
||||
use crate::Vec2;
|
||||
|
||||
/// A normalized vector pointing in a direction in 2D space
|
||||
@ -174,7 +174,7 @@ impl BoxedPolyline2d {
|
||||
}
|
||||
|
||||
/// A triangle in 2D space
|
||||
#[derive(Clone, Debug)]
|
||||
#[derive(Clone, Debug, PartialEq)]
|
||||
pub struct Triangle2d {
|
||||
/// The vertices of the triangle
|
||||
pub vertices: [Vec2; 3],
|
||||
@ -182,12 +182,32 @@ pub struct Triangle2d {
|
||||
impl Primitive2d for Triangle2d {}
|
||||
|
||||
impl Triangle2d {
|
||||
/// Create a new `Triangle2d` from `a`, `b`, and `c`,
|
||||
/// Create a new `Triangle2d` from points `a`, `b`, and `c`
|
||||
pub fn new(a: Vec2, b: Vec2, c: Vec2) -> Self {
|
||||
Self {
|
||||
vertices: [a, b, c],
|
||||
}
|
||||
}
|
||||
|
||||
/// Get the [`WindingOrder`] of the triangle
|
||||
#[doc(alias = "orientation")]
|
||||
pub fn winding_order(&self) -> WindingOrder {
|
||||
let [a, b, c] = self.vertices;
|
||||
let area = (b - a).perp_dot(c - a);
|
||||
if area > f32::EPSILON {
|
||||
WindingOrder::CounterClockwise
|
||||
} else if area < -f32::EPSILON {
|
||||
WindingOrder::Clockwise
|
||||
} else {
|
||||
WindingOrder::Invalid
|
||||
}
|
||||
}
|
||||
|
||||
/// Reverse the [`WindingOrder`] of the triangle
|
||||
/// by swapping the second and third vertices
|
||||
pub fn reverse(&mut self) {
|
||||
self.vertices.swap(1, 2);
|
||||
}
|
||||
}
|
||||
|
||||
/// A rectangle primitive
|
||||
@ -298,3 +318,37 @@ impl RegularPolygon {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use super::*;
|
||||
|
||||
#[test]
|
||||
fn triangle_winding_order() {
|
||||
let mut cw_triangle = Triangle2d::new(
|
||||
Vec2::new(0.0, 2.0),
|
||||
Vec2::new(-0.5, -1.2),
|
||||
Vec2::new(-1.0, -1.0),
|
||||
);
|
||||
assert_eq!(cw_triangle.winding_order(), WindingOrder::Clockwise);
|
||||
|
||||
let ccw_triangle = Triangle2d::new(
|
||||
Vec2::new(0.0, 2.0),
|
||||
Vec2::new(-1.0, -1.0),
|
||||
Vec2::new(-0.5, -1.2),
|
||||
);
|
||||
assert_eq!(ccw_triangle.winding_order(), WindingOrder::CounterClockwise);
|
||||
|
||||
// The clockwise triangle should be the same as the counterclockwise
|
||||
// triangle when reversed
|
||||
cw_triangle.reverse();
|
||||
assert_eq!(cw_triangle, ccw_triangle);
|
||||
|
||||
let invalid_triangle = Triangle2d::new(
|
||||
Vec2::new(0.0, 2.0),
|
||||
Vec2::new(0.0, -1.0),
|
||||
Vec2::new(0.0, -1.2),
|
||||
);
|
||||
assert_eq!(invalid_triangle.winding_order(), WindingOrder::Invalid);
|
||||
}
|
||||
}
|
||||
|
||||
@ -12,3 +12,16 @@ pub trait Primitive2d {}
|
||||
|
||||
/// A marker trait for 3D primitives
|
||||
pub trait Primitive3d {}
|
||||
|
||||
/// The winding order for a set of points
|
||||
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
|
||||
pub enum WindingOrder {
|
||||
/// A clockwise winding order
|
||||
Clockwise,
|
||||
/// A counterclockwise winding order
|
||||
CounterClockwise,
|
||||
/// An invalid winding order indicating that it could not be computed reliably.
|
||||
/// This often happens in *degenerate cases* where the points lie on the same line
|
||||
#[doc(alias = "Degenerate")]
|
||||
Invalid,
|
||||
}
|
||||
|
||||
Loading…
Reference in New Issue
Block a user