#![allow(clippy::manual_clamp)]
use crate::geometry::Size;
use crate::style::AvailableSpace;
pub(crate) trait MaybeMath<In, Out> {
fn maybe_min(self, rhs: In) -> Out;
fn maybe_max(self, rhs: In) -> Out;
fn maybe_clamp(self, min: In, max: In) -> Out;
fn maybe_add(self, rhs: In) -> Out;
fn maybe_sub(self, rhs: In) -> Out;
}
impl MaybeMath<Option<f32>, Option<f32>> for Option<f32> {
fn maybe_min(self, rhs: Option<f32>) -> Option<f32> {
match (self, rhs) {
(Some(l), Some(r)) => Some(l.min(r)),
(Some(_l), None) => self,
(None, Some(_r)) => None,
(None, None) => None,
}
}
fn maybe_max(self, rhs: Option<f32>) -> Option<f32> {
match (self, rhs) {
(Some(l), Some(r)) => Some(l.max(r)),
(Some(_l), None) => self,
(None, Some(_r)) => None,
(None, None) => None,
}
}
fn maybe_clamp(self, min: Option<f32>, max: Option<f32>) -> Option<f32> {
match (self, min, max) {
(Some(base), Some(min), Some(max)) => Some(base.min(max).max(min)),
(Some(base), None, Some(max)) => Some(base.min(max)),
(Some(base), Some(min), None) => Some(base.max(min)),
(Some(_), None, None) => self,
(None, _, _) => None,
}
}
fn maybe_add(self, rhs: Option<f32>) -> Option<f32> {
match (self, rhs) {
(Some(l), Some(r)) => Some(l + r),
(Some(_l), None) => self,
(None, Some(_r)) => None,
(None, None) => None,
}
}
fn maybe_sub(self, rhs: Option<f32>) -> Option<f32> {
match (self, rhs) {
(Some(l), Some(r)) => Some(l - r),
(Some(_l), None) => self,
(None, Some(_r)) => None,
(None, None) => None,
}
}
}
impl MaybeMath<f32, Option<f32>> for Option<f32> {
fn maybe_min(self, rhs: f32) -> Option<f32> {
self.map(|val| val.min(rhs))
}
fn maybe_max(self, rhs: f32) -> Option<f32> {
self.map(|val| val.max(rhs))
}
fn maybe_clamp(self, min: f32, max: f32) -> Option<f32> {
self.map(|val| val.min(max).max(min))
}
fn maybe_add(self, rhs: f32) -> Option<f32> {
self.map(|val| val + rhs)
}
fn maybe_sub(self, rhs: f32) -> Option<f32> {
self.map(|val| val - rhs)
}
}
impl MaybeMath<Option<f32>, f32> for f32 {
fn maybe_min(self, rhs: Option<f32>) -> f32 {
match rhs {
Some(val) => self.min(val),
None => self,
}
}
fn maybe_max(self, rhs: Option<f32>) -> f32 {
match rhs {
Some(val) => self.max(val),
None => self,
}
}
fn maybe_clamp(self, min: Option<f32>, max: Option<f32>) -> f32 {
match (min, max) {
(Some(min), Some(max)) => self.min(max).max(min),
(None, Some(max)) => self.min(max),
(Some(min), None) => self.max(min),
(None, None) => self,
}
}
fn maybe_add(self, rhs: Option<f32>) -> f32 {
match rhs {
Some(val) => self + val,
None => self,
}
}
fn maybe_sub(self, rhs: Option<f32>) -> f32 {
match rhs {
Some(val) => self - val,
None => self,
}
}
}
impl MaybeMath<f32, AvailableSpace> for AvailableSpace {
fn maybe_min(self, rhs: f32) -> AvailableSpace {
match self {
AvailableSpace::Definite(val) => AvailableSpace::Definite(val.min(rhs)),
AvailableSpace::MinContent => AvailableSpace::Definite(rhs),
AvailableSpace::MaxContent => AvailableSpace::Definite(rhs),
}
}
fn maybe_max(self, rhs: f32) -> AvailableSpace {
match self {
AvailableSpace::Definite(val) => AvailableSpace::Definite(val.max(rhs)),
AvailableSpace::MinContent => AvailableSpace::MinContent,
AvailableSpace::MaxContent => AvailableSpace::MaxContent,
}
}
fn maybe_clamp(self, min: f32, max: f32) -> AvailableSpace {
match self {
AvailableSpace::Definite(val) => AvailableSpace::Definite(val.min(max).max(min)),
AvailableSpace::MinContent => AvailableSpace::MinContent,
AvailableSpace::MaxContent => AvailableSpace::MaxContent,
}
}
fn maybe_add(self, rhs: f32) -> AvailableSpace {
match self {
AvailableSpace::Definite(val) => AvailableSpace::Definite(val + rhs),
AvailableSpace::MinContent => AvailableSpace::MinContent,
AvailableSpace::MaxContent => AvailableSpace::MaxContent,
}
}
fn maybe_sub(self, rhs: f32) -> AvailableSpace {
match self {
AvailableSpace::Definite(val) => AvailableSpace::Definite(val - rhs),
AvailableSpace::MinContent => AvailableSpace::MinContent,
AvailableSpace::MaxContent => AvailableSpace::MaxContent,
}
}
}
impl MaybeMath<Option<f32>, AvailableSpace> for AvailableSpace {
fn maybe_min(self, rhs: Option<f32>) -> AvailableSpace {
match (self, rhs) {
(AvailableSpace::Definite(val), Some(rhs)) => AvailableSpace::Definite(val.min(rhs)),
(AvailableSpace::Definite(val), None) => AvailableSpace::Definite(val),
(AvailableSpace::MinContent, Some(rhs)) => AvailableSpace::Definite(rhs),
(AvailableSpace::MinContent, None) => AvailableSpace::MinContent,
(AvailableSpace::MaxContent, Some(rhs)) => AvailableSpace::Definite(rhs),
(AvailableSpace::MaxContent, None) => AvailableSpace::MaxContent,
}
}
fn maybe_max(self, rhs: Option<f32>) -> AvailableSpace {
match (self, rhs) {
(AvailableSpace::Definite(val), Some(rhs)) => AvailableSpace::Definite(val.max(rhs)),
(AvailableSpace::Definite(val), None) => AvailableSpace::Definite(val),
(AvailableSpace::MinContent, _) => AvailableSpace::MinContent,
(AvailableSpace::MaxContent, _) => AvailableSpace::MaxContent,
}
}
fn maybe_clamp(self, min: Option<f32>, max: Option<f32>) -> AvailableSpace {
match (self, min, max) {
(AvailableSpace::Definite(val), Some(min), Some(max)) => AvailableSpace::Definite(val.min(max).max(min)),
(AvailableSpace::Definite(val), None, Some(max)) => AvailableSpace::Definite(val.min(max)),
(AvailableSpace::Definite(val), Some(min), None) => AvailableSpace::Definite(val.max(min)),
(AvailableSpace::Definite(val), None, None) => AvailableSpace::Definite(val),
(AvailableSpace::MinContent, _, _) => AvailableSpace::MinContent,
(AvailableSpace::MaxContent, _, _) => AvailableSpace::MaxContent,
}
}
fn maybe_add(self, rhs: Option<f32>) -> AvailableSpace {
match (self, rhs) {
(AvailableSpace::Definite(val), Some(rhs)) => AvailableSpace::Definite(val + rhs),
(AvailableSpace::Definite(val), None) => AvailableSpace::Definite(val),
(AvailableSpace::MinContent, _) => AvailableSpace::MinContent,
(AvailableSpace::MaxContent, _) => AvailableSpace::MaxContent,
}
}
fn maybe_sub(self, rhs: Option<f32>) -> AvailableSpace {
match (self, rhs) {
(AvailableSpace::Definite(val), Some(rhs)) => AvailableSpace::Definite(val - rhs),
(AvailableSpace::Definite(val), None) => AvailableSpace::Definite(val),
(AvailableSpace::MinContent, _) => AvailableSpace::MinContent,
(AvailableSpace::MaxContent, _) => AvailableSpace::MaxContent,
}
}
}
impl<In, Out, T: MaybeMath<In, Out>> MaybeMath<Size<In>, Size<Out>> for Size<T> {
fn maybe_min(self, rhs: Size<In>) -> Size<Out> {
Size { width: self.width.maybe_min(rhs.width), height: self.height.maybe_min(rhs.height) }
}
fn maybe_max(self, rhs: Size<In>) -> Size<Out> {
Size { width: self.width.maybe_max(rhs.width), height: self.height.maybe_max(rhs.height) }
}
fn maybe_clamp(self, min: Size<In>, max: Size<In>) -> Size<Out> {
Size {
width: self.width.maybe_clamp(min.width, max.width),
height: self.height.maybe_clamp(min.height, max.height),
}
}
fn maybe_add(self, rhs: Size<In>) -> Size<Out> {
Size { width: self.width.maybe_add(rhs.width), height: self.height.maybe_add(rhs.height) }
}
fn maybe_sub(self, rhs: Size<In>) -> Size<Out> {
Size { width: self.width.maybe_sub(rhs.width), height: self.height.maybe_sub(rhs.height) }
}
}
#[cfg(test)]
mod tests {
mod lhs_option_f32_rhs_option_f32 {
use crate::math::MaybeMath;
use rstest::rstest;
#[rstest]
#[case(Some(3.0), Some(5.0), Some(3.0))]
#[case(Some(5.0), Some(3.0), Some(3.0))]
#[case(Some(3.0), None, Some(3.0))]
#[case(None, Some(3.0), None)]
#[case(None, None, None)]
fn test_maybe_min(#[case] lhs: Option<f32>, #[case] rhs: Option<f32>, #[case] expected: Option<f32>) {
assert_eq!(lhs.maybe_min(rhs), expected);
}
#[rstest]
#[case(Some(3.0), Some(5.0), Some(5.0))]
#[case(Some(5.0), Some(3.0), Some(5.0))]
#[case(Some(3.0), None, Some(3.0))]
#[case(None, Some(3.0), None)]
#[case(None, None, None)]
fn test_maybe_max(#[case] lhs: Option<f32>, #[case] rhs: Option<f32>, #[case] expected: Option<f32>) {
assert_eq!(lhs.maybe_max(rhs), expected);
}
#[rstest]
#[case(Some(3.0), Some(5.0), Some(8.0))]
#[case(Some(5.0), Some(3.0), Some(8.0))]
#[case(Some(3.0), None, Some(3.0))]
#[case(None, Some(3.0), None)]
#[case(None, None, None)]
fn test_maybe_add(#[case] lhs: Option<f32>, #[case] rhs: Option<f32>, #[case] expected: Option<f32>) {
assert_eq!(lhs.maybe_add(rhs), expected);
}
#[rstest]
#[case(Some(3.0), Some(5.0), Some(-2.0))]
#[case(Some(5.0), Some(3.0), Some(2.0))]
#[case(Some(3.0), None, Some(3.0))]
#[case(None, Some(3.0), None)]
#[case(None, None, None)]
fn test_maybe_sub(#[case] lhs: Option<f32>, #[case] rhs: Option<f32>, #[case] expected: Option<f32>) {
assert_eq!(lhs.maybe_sub(rhs), expected);
}
}
mod lhs_option_f32_rhs_f32 {
use crate::math::MaybeMath;
use rstest::rstest;
#[rstest]
#[case(Some(3.0), 5.0, Some(3.0))]
#[case(Some(5.0), 3.0, Some(3.0))]
#[case(None, 3.0, None)]
fn test_maybe_min(#[case] lhs: Option<f32>, #[case] rhs: f32, #[case] expected: Option<f32>) {
assert_eq!(lhs.maybe_min(rhs), expected);
}
#[rstest]
#[case(Some(3.0), 5.0, Some(5.0))]
#[case(Some(5.0), 3.0, Some(5.0))]
#[case(None, 3.0, None)]
fn test_maybe_max(#[case] lhs: Option<f32>, #[case] rhs: f32, #[case] expected: Option<f32>) {
assert_eq!(lhs.maybe_max(rhs), expected);
}
#[rstest]
#[case(Some(3.0), 5.0, Some(8.0))]
#[case(Some(5.0), 3.0, Some(8.0))]
#[case(None, 3.0, None)]
fn test_maybe_add(#[case] lhs: Option<f32>, #[case] rhs: f32, #[case] expected: Option<f32>) {
assert_eq!(lhs.maybe_add(rhs), expected);
}
#[rstest]
#[case(Some(3.0), 5.0, Some(-2.0))]
#[case(Some(5.0), 3.0, Some(2.0))]
#[case(None, 3.0, None)]
fn test_maybe_sub(#[case] lhs: Option<f32>, #[case] rhs: f32, #[case] expected: Option<f32>) {
assert_eq!(lhs.maybe_sub(rhs), expected);
}
}
mod lhs_f32_rhs_option_f32 {
use crate::math::MaybeMath;
use rstest::rstest;
#[rstest]
#[case(3.0, Some(5.0), 3.0)]
#[case(5.0, Some(3.0), 3.0)]
#[case(3.0, None, 3.0)]
fn test_maybe_min(#[case] lhs: f32, #[case] rhs: Option<f32>, #[case] expected: f32) {
assert_eq!(lhs.maybe_min(rhs), expected);
}
#[rstest]
#[case(3.0, Some(5.0), 5.0)]
#[case(5.0, Some(3.0), 5.0)]
#[case(3.0, None, 3.0)]
fn test_maybe_max(#[case] lhs: f32, #[case] rhs: Option<f32>, #[case] expected: f32) {
assert_eq!(lhs.maybe_max(rhs), expected);
}
#[rstest]
#[case(3.0, Some(5.0), 8.0)]
#[case(5.0, Some(3.0), 8.0)]
#[case(3.0, None, 3.0)]
fn test_maybe_add(#[case] lhs: f32, #[case] rhs: Option<f32>, #[case] expected: f32) {
assert_eq!(lhs.maybe_add(rhs), expected);
}
#[rstest]
#[case(3.0, Some(5.0), -2.0)]
#[case(5.0, Some(3.0), 2.0)]
#[case(3.0, None, 3.0)]
fn test_maybe_sub(#[case] lhs: f32, #[case] rhs: Option<f32>, #[case] expected: f32) {
assert_eq!(lhs.maybe_sub(rhs), expected);
}
}
}