// Copyright © 2024 Mikhail Hogrefe
//
// This file is part of Malachite.
//
// Malachite is free software: you can redistribute it and/or modify it under the terms of the GNU
// Lesser General Public License (LGPL) as published by the Free Software Foundation; either version
// 3 of the License, or (at your option) any later version. See .
use malachite_base::num::basic::floats::PrimitiveFloat;
use malachite_base::num::basic::traits::NegativeInfinity;
use malachite_base::num::conversion::traits::ExactFrom;
use malachite_q::test_util::generators::{
rational_gen, rational_primitive_float_pair_gen,
rational_primitive_float_primitive_float_triple_gen,
rational_rational_primitive_float_triple_gen,
};
use malachite_q::Rational;
use rug;
use std::cmp::Ordering::{self, *};
use std::str::FromStr;
#[test]
fn test_partial_cmp_primitive_float() {
let test = |u, v: f32, out: Option| {
let out_rev = out.map(Ordering::reverse);
assert_eq!(Rational::from_str(u).unwrap().partial_cmp(&v), out);
assert_eq!(rug::Rational::from_str(u).unwrap().partial_cmp(&v), out);
assert_eq!(v.partial_cmp(&Rational::from_str(u).unwrap()), out_rev);
assert_eq!(v.partial_cmp(&rug::Rational::from_str(u).unwrap()), out_rev);
let v = f64::from(v);
assert_eq!(Rational::from_str(u).unwrap().partial_cmp(&v), out);
assert_eq!(rug::Rational::from_str(u).unwrap().partial_cmp(&v), out);
assert_eq!(v.partial_cmp(&Rational::from_str(u).unwrap()), out_rev);
assert_eq!(v.partial_cmp(&rug::Rational::from_str(u).unwrap()), out_rev);
};
test("2/3", f32::NAN, None);
test("2/3", f32::INFINITY, Some(Less));
test("2/3", f32::NEGATIVE_INFINITY, Some(Greater));
test("-2/3", f32::NAN, None);
test("-2/3", f32::INFINITY, Some(Less));
test("-2/3", f32::NEGATIVE_INFINITY, Some(Greater));
test("0", 0.0, Some(Equal));
test("0", -0.0, Some(Equal));
test("0", 5.0, Some(Less));
test("0", -5.0, Some(Greater));
test("3/2", 1.5, Some(Equal));
test("3/2", 5.0, Some(Less));
test("3/2", -5.0, Some(Greater));
test("-3/2", 5.0, Some(Less));
test("-3/2", -5.0, Some(Greater));
test("-3/2", -1.5, Some(Equal));
test("1/3", 0.333, Some(Greater));
test("1/3", 0.334, Some(Less));
test("1/3", -0.333, Some(Greater));
test("1/3", -0.334, Some(Greater));
test("-1/3", -0.334, Some(Greater));
test("-1/3", -0.333, Some(Less));
test("-1/3", 0.334, Some(Less));
test("-1/3", 0.333, Some(Less));
}
#[allow(clippy::trait_duplication_in_bounds)]
fn partial_cmp_primitive_float_properties_helper<
T: PartialOrd + PartialOrd + PrimitiveFloat,
>()
where
Rational: TryFrom + PartialOrd,
rug::Rational: PartialOrd,
{
rational_primitive_float_pair_gen::().test_properties(|(n, u)| {
let cmp = n.partial_cmp(&u);
assert_eq!(rug::Rational::from(&n).partial_cmp(&u), cmp);
let cmp_rev = cmp.map(Ordering::reverse);
assert_eq!(u.partial_cmp(&n), cmp_rev);
assert_eq!(u.partial_cmp(&rug::Rational::from(&n)), cmp_rev);
if u.is_finite() {
assert_eq!(n.cmp(&Rational::exact_from(u)), cmp.unwrap());
}
});
rational_rational_primitive_float_triple_gen::().test_properties(|(n, m, u)| {
if n < u && u < m {
assert_eq!(n.cmp(&m), Less);
} else if n > u && u > m {
assert_eq!(n.cmp(&m), Greater);
}
});
rational_primitive_float_primitive_float_triple_gen::().test_properties(|(n, u, v)| {
if u < n && n < v {
assert!(u < v);
} else if u > n && n > v {
assert!(u > v);
}
});
rational_gen().test_properties(|x| {
assert!(x > T::NEGATIVE_INFINITY);
assert!(x < T::INFINITY);
});
}
#[test]
fn partial_cmp_primitive_float_properties() {
apply_fn_to_primitive_floats!(partial_cmp_primitive_float_properties_helper);
}