use malachite_base::num::arithmetic::traits::Abs;
use malachite_base::num::basic::floats::PrimitiveFloat;
use malachite_base::num::basic::traits::{Infinity, NaN, NegativeInfinity};
use malachite_base::num::comparison::traits::PartialOrdAbs;
use malachite_base::num::conversion::traits::ExactFrom;
use malachite_base::test_util::generators::{primitive_float_gen, primitive_float_gen_var_11};
use malachite_float::Float;
use malachite_float::test_util::common::{
ORDERED_F32S, ORDERED_F64S, ORDERED_FLOAT_HEX_STRINGS, parse_hex_string,
};
use malachite_float::test_util::generators::{
float_float_primitive_float_triple_gen, float_gen, float_gen_var_2,
float_primitive_float_pair_gen, float_primitive_float_pair_gen_var_1,
float_primitive_float_primitive_float_triple_gen,
};
use std::cmp::Ordering::{self, *};
const fn encode(oo: Option<Ordering>) -> u8 {
match oo {
None => 9,
Some(Less) => 0,
Some(Equal) => 1,
Some(Greater) => 2,
}
}
#[rustfmt::skip]
const MATRIX_F32: [[u8; 17]; 21] = [
[1, 2, 2, 2, 2, 2, 2, 2, 9, 2, 2, 2, 2, 2, 2, 2, 1],
[0, 0, 2, 2, 2, 2, 2, 2, 9, 2, 2, 2, 2, 2, 2, 0, 0],
[0, 0, 1, 2, 2, 2, 2, 2, 9, 2, 2, 2, 2, 2, 1, 0, 0],
[0, 0, 0, 2, 2, 2, 2, 2, 9, 2, 2, 2, 2, 2, 0, 0, 0],
[0, 0, 0, 0, 1, 2, 2, 2, 9, 2, 2, 2, 1, 0, 0, 0, 0], [0, 0, 0, 0, 1, 2, 2, 2, 9, 2, 2, 2, 1, 0, 0, 0, 0], [0, 0, 0, 0, 1, 2, 2, 2, 9, 2, 2, 2, 1, 0, 0, 0, 0], [0, 0, 0, 0, 0, 1, 2, 2, 9, 2, 2, 1, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 0, 0, 2, 9, 2, 0, 0, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 0, 0, 1, 9, 1, 0, 0, 0, 0, 0, 0, 0], [9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9], [0, 0, 0, 0, 0, 0, 0, 1, 9, 1, 0, 0, 0, 0, 0, 0, 0], [0, 0, 0, 0, 0, 0, 0, 2, 9, 2, 0, 0, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 1, 2, 2, 9, 2, 2, 1, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 1, 2, 2, 2, 9, 2, 2, 2, 1, 0, 0, 0, 0], [0, 0, 0, 0, 1, 2, 2, 2, 9, 2, 2, 2, 1, 0, 0, 0, 0], [0, 0, 0, 0, 1, 2, 2, 2, 9, 2, 2, 2, 1, 0, 0, 0, 0], [0, 0, 0, 2, 2, 2, 2, 2, 9, 2, 2, 2, 2, 2, 0, 0, 0],
[0, 0, 1, 2, 2, 2, 2, 2, 9, 2, 2, 2, 2, 2, 1, 0, 0],
[0, 0, 2, 2, 2, 2, 2, 2, 9, 2, 2, 2, 2, 2, 2, 0, 0],
[1, 2, 2, 2, 2, 2, 2, 2, 9, 2, 2, 2, 2, 2, 2, 2, 1],
];
#[rustfmt::skip]
const MATRIX_F64: [[u8; 17]; 21] = [
[1, 2, 2, 2, 2, 2, 2, 2, 9, 2, 2, 2, 2, 2, 2, 2, 1],
[0, 1, 2, 2, 2, 2, 2, 2, 9, 2, 2, 2, 2, 2, 2, 1, 0],
[0, 0, 1, 2, 2, 2, 2, 2, 9, 2, 2, 2, 2, 2, 1, 0, 0],
[0, 0, 0, 1, 2, 2, 2, 2, 9, 2, 2, 2, 2, 1, 0, 0, 0],
[0, 0, 0, 0, 1, 2, 2, 2, 9, 2, 2, 2, 1, 0, 0, 0, 0], [0, 0, 0, 0, 1, 2, 2, 2, 9, 2, 2, 2, 1, 0, 0, 0, 0], [0, 0, 0, 0, 1, 2, 2, 2, 9, 2, 2, 2, 1, 0, 0, 0, 0], [0, 0, 0, 0, 0, 1, 2, 2, 9, 2, 2, 1, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 0, 1, 2, 9, 2, 1, 0, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 0, 0, 1, 9, 1, 0, 0, 0, 0, 0, 0, 0], [9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9], [0, 0, 0, 0, 0, 0, 0, 1, 9, 1, 0, 0, 0, 0, 0, 0, 0], [0, 0, 0, 0, 0, 0, 1, 2, 9, 2, 1, 0, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 1, 2, 2, 9, 2, 2, 1, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 1, 2, 2, 2, 9, 2, 2, 2, 1, 0, 0, 0, 0], [0, 0, 0, 0, 1, 2, 2, 2, 9, 2, 2, 2, 1, 0, 0, 0, 0], [0, 0, 0, 0, 1, 2, 2, 2, 9, 2, 2, 2, 1, 0, 0, 0, 0], [0, 0, 0, 1, 2, 2, 2, 2, 9, 2, 2, 2, 2, 1, 0, 0, 0],
[0, 0, 1, 2, 2, 2, 2, 2, 9, 2, 2, 2, 2, 2, 1, 0, 0],
[0, 1, 2, 2, 2, 2, 2, 2, 9, 2, 2, 2, 2, 2, 2, 1, 0],
[1, 2, 2, 2, 2, 2, 2, 2, 9, 2, 2, 2, 2, 2, 2, 2, 1],
];
#[test]
fn test_partial_cmp_abs_primitive_float() {
for (sx, row) in ORDERED_FLOAT_HEX_STRINGS.iter().zip(MATRIX_F64.iter()) {
let x = parse_hex_string(sx);
for (&y, &e) in ORDERED_F64S.iter().zip(row.iter()) {
assert_eq!(encode(x.partial_cmp_abs(&y)), e);
assert_eq!(encode(y.partial_cmp_abs(&x).map(Ordering::reverse)), e);
}
}
for (sx, row) in ORDERED_FLOAT_HEX_STRINGS.iter().zip(MATRIX_F32.iter()) {
let x = parse_hex_string(sx);
for (&y, &e) in ORDERED_F32S.iter().zip(row.iter()) {
assert_eq!(encode(x.partial_cmp_abs(&y)), e);
assert_eq!(encode(y.partial_cmp_abs(&x).map(Ordering::reverse)), e);
}
}
}
#[allow(clippy::needless_pass_by_value)]
fn partial_cmp_abs_primitive_float_properties_helper_helper<
T: PartialOrd<T> + PartialOrdAbs<Float> + PrimitiveFloat,
>(
n: Float,
u: T,
) where
Float: TryFrom<T> + PartialOrd<T> + PartialOrdAbs<T>,
{
let cmp = n.partial_cmp_abs(&u);
assert_eq!((&n).abs().partial_cmp(&u.abs()), cmp);
let cmp_rev = cmp.map(Ordering::reverse);
assert_eq!(u.partial_cmp_abs(&n), cmp_rev);
assert_eq!(
PartialOrdAbs::<Float>::partial_cmp_abs(&n, &Float::exact_from(u)),
cmp
);
}
fn partial_cmp_abs_primitive_float_properties_helper<
T: PartialOrd<T> + PartialOrdAbs<Float> + PrimitiveFloat,
>()
where
Float: TryFrom<T> + PartialOrd<T> + PartialOrdAbs<T>,
{
float_primitive_float_pair_gen::<T>().test_properties(|(n, u)| {
partial_cmp_abs_primitive_float_properties_helper_helper(n, u);
});
float_primitive_float_pair_gen_var_1::<T>().test_properties(|(n, u)| {
partial_cmp_abs_primitive_float_properties_helper_helper(n, u);
});
float_float_primitive_float_triple_gen::<T>().test_properties(|(n, m, u)| {
if n.lt_abs(&u) && u.lt_abs(&m) {
assert_eq!(PartialOrdAbs::<Float>::partial_cmp_abs(&n, &m), Some(Less));
} else if n.gt_abs(&u) && u.gt_abs(&m) {
assert_eq!(
PartialOrdAbs::<Float>::partial_cmp_abs(&n, &m),
Some(Greater)
);
}
});
float_primitive_float_primitive_float_triple_gen::<T>().test_properties(|(n, u, v)| {
if u.lt_abs(&n) && n.lt_abs(&v) {
assert!(PartialOrdAbs::<T>::lt_abs(&u, &v));
} else if u.gt_abs(&n) && n.gt_abs(&v) {
assert!(PartialOrdAbs::<T>::gt_abs(&u, &v));
}
});
float_gen().test_properties(|x| {
assert!(!(x.lt_abs(&T::NAN)));
assert!(!(x.gt_abs(&T::NAN)));
});
float_gen_var_2().test_properties(|x| {
assert!(x.le_abs(&T::NEGATIVE_INFINITY));
assert!(x.le_abs(&T::INFINITY));
});
primitive_float_gen::<T>().test_properties(|x| {
assert!(!(x.lt_abs(&Float::NAN)));
assert!(!(x.gt_abs(&Float::NAN)));
});
primitive_float_gen_var_11::<T>().test_properties(|x| {
assert!(x.le_abs(&Float::NEGATIVE_INFINITY));
assert!(x.le_abs(&Float::INFINITY));
});
}
#[test]
fn partial_cmp_abs_primitive_float_properties() {
apply_fn_to_primitive_floats!(partial_cmp_abs_primitive_float_properties_helper);
}