2020-02-08 18:43:18 -05:00
|
|
|
use std::convert::From;
|
|
|
|
use std::fmt::{self, Display, Write};
|
2020-02-15 08:51:00 -05:00
|
|
|
use std::ops::{Add, AddAssign, Div, DivAssign, Mul, MulAssign, Rem, RemAssign, Sub, SubAssign, Neg};
|
2020-02-08 18:43:18 -05:00
|
|
|
|
|
|
|
use num_bigint::BigInt;
|
|
|
|
use num_rational::BigRational;
|
2020-02-14 12:30:48 -05:00
|
|
|
use num_traits::sign::Signed;
|
2020-02-08 18:43:18 -05:00
|
|
|
|
|
|
|
const PRECISION: usize = 10;
|
|
|
|
|
|
|
|
#[derive(Clone, Debug, Eq, PartialEq, Ord, PartialOrd)]
|
|
|
|
pub(crate) struct Number {
|
|
|
|
val: BigRational,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Number {
|
|
|
|
pub const fn new(val: BigRational) -> Number {
|
|
|
|
Number { val }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn to_integer(&self) -> BigInt {
|
|
|
|
self.val.to_integer()
|
|
|
|
}
|
2020-02-09 03:13:31 -05:00
|
|
|
|
|
|
|
pub fn ratio<A: Into<BigInt>, B: Into<BigInt>>(a: A, b: B) -> Self {
|
|
|
|
Number::new(BigRational::new(a.into(), b.into()))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn round(self) -> Self {
|
|
|
|
Number {
|
|
|
|
val: self.val.round(),
|
|
|
|
}
|
|
|
|
}
|
2020-02-09 16:14:24 -05:00
|
|
|
|
|
|
|
pub fn ceil(self) -> Self {
|
|
|
|
Number {
|
|
|
|
val: self.val.ceil(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn floor(self) -> Self {
|
|
|
|
Number {
|
|
|
|
val: self.val.floor(),
|
|
|
|
}
|
|
|
|
}
|
2020-02-14 12:30:48 -05:00
|
|
|
|
|
|
|
pub fn abs(self) -> Self {
|
|
|
|
Number {
|
|
|
|
val: self.val.abs(),
|
|
|
|
}
|
|
|
|
}
|
2020-02-09 03:13:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::LowerHex for Number {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
write!(f, "{:0>2x}", self.val.to_integer())
|
|
|
|
}
|
2020-02-08 18:43:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl From<BigInt> for Number {
|
|
|
|
fn from(b: BigInt) -> Self {
|
|
|
|
Number {
|
|
|
|
val: BigRational::from_integer(b),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-14 08:14:30 -05:00
|
|
|
impl From<Number> for BigInt {
|
2020-02-14 18:28:09 -05:00
|
|
|
#[inline]
|
2020-02-14 08:14:30 -05:00
|
|
|
fn from(b: Number) -> Self {
|
|
|
|
b.to_integer()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-08 20:38:37 -05:00
|
|
|
macro_rules! from_integer {
|
|
|
|
($ty:ty) => {
|
|
|
|
impl From<$ty> for Number {
|
|
|
|
fn from(b: $ty) -> Self {
|
|
|
|
Number {
|
|
|
|
val: BigRational::from_integer(BigInt::from(b)),
|
|
|
|
}
|
|
|
|
}
|
2020-02-08 20:07:20 -05:00
|
|
|
}
|
2020-02-08 20:38:37 -05:00
|
|
|
};
|
2020-02-08 20:07:20 -05:00
|
|
|
}
|
|
|
|
|
2020-02-08 20:38:37 -05:00
|
|
|
from_integer!(u16);
|
|
|
|
from_integer!(usize);
|
|
|
|
from_integer!(i32);
|
2020-02-09 13:31:58 -05:00
|
|
|
from_integer!(u8);
|
2020-02-08 20:38:37 -05:00
|
|
|
|
2020-02-08 18:43:18 -05:00
|
|
|
impl Display for Number {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
write!(f, "{}", self.val.to_integer())?;
|
|
|
|
let mut frac = self.val.fract();
|
|
|
|
if frac != BigRational::from_integer(BigInt::from(0)) {
|
|
|
|
f.write_char('.')?;
|
2020-02-09 20:26:14 -05:00
|
|
|
for _ in 0..(PRECISION - 1) {
|
2020-02-08 18:43:18 -05:00
|
|
|
frac *= BigRational::from_integer(BigInt::from(10));
|
|
|
|
write!(f, "{}", frac.to_integer())?;
|
|
|
|
frac = frac.fract();
|
|
|
|
if frac == BigRational::from_integer(BigInt::from(0)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2020-02-08 18:51:29 -05:00
|
|
|
if frac != BigRational::from_integer(BigInt::from(0)) {
|
2020-02-08 20:20:03 -05:00
|
|
|
write!(
|
|
|
|
f,
|
|
|
|
"{}",
|
|
|
|
(frac * BigRational::from_integer(BigInt::from(10)))
|
|
|
|
.round()
|
|
|
|
.to_integer()
|
|
|
|
)?;
|
2020-02-08 18:51:29 -05:00
|
|
|
}
|
2020-02-08 18:43:18 -05:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Add for Number {
|
|
|
|
type Output = Self;
|
|
|
|
|
|
|
|
fn add(self, other: Self) -> Self {
|
|
|
|
Number {
|
|
|
|
val: self.val + other.val,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-09 03:13:31 -05:00
|
|
|
impl AddAssign for Number {
|
|
|
|
fn add_assign(&mut self, other: Self) {
|
|
|
|
self.val += other.val
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-08 18:43:18 -05:00
|
|
|
impl Sub for Number {
|
|
|
|
type Output = Self;
|
|
|
|
|
|
|
|
fn sub(self, other: Self) -> Self {
|
|
|
|
Number {
|
|
|
|
val: self.val - other.val,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-02-08 20:07:20 -05:00
|
|
|
|
2020-02-09 03:13:31 -05:00
|
|
|
impl SubAssign for Number {
|
|
|
|
fn sub_assign(&mut self, other: Self) {
|
|
|
|
self.val -= other.val
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-08 20:07:20 -05:00
|
|
|
impl Mul for Number {
|
|
|
|
type Output = Self;
|
|
|
|
|
|
|
|
fn mul(self, other: Self) -> Self {
|
|
|
|
Number {
|
|
|
|
val: self.val * other.val,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-09 12:18:41 -05:00
|
|
|
impl MulAssign for Number {
|
|
|
|
fn mul_assign(&mut self, other: Self) {
|
|
|
|
self.val *= other.val
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-08 20:07:20 -05:00
|
|
|
impl Div for Number {
|
|
|
|
type Output = Self;
|
|
|
|
|
|
|
|
fn div(self, other: Self) -> Self {
|
|
|
|
Number {
|
|
|
|
val: self.val / other.val,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-02-09 12:18:41 -05:00
|
|
|
|
|
|
|
impl DivAssign for Number {
|
|
|
|
fn div_assign(&mut self, other: Self) {
|
|
|
|
self.val /= other.val
|
|
|
|
}
|
|
|
|
}
|
2020-02-14 09:44:46 -05:00
|
|
|
|
|
|
|
impl Rem for Number {
|
|
|
|
type Output = Self;
|
|
|
|
|
|
|
|
fn rem(self, other: Self) -> Self {
|
|
|
|
Number {
|
|
|
|
val: self.val % other.val,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl RemAssign for Number {
|
|
|
|
fn rem_assign(&mut self, other: Self) {
|
|
|
|
self.val %= other.val
|
|
|
|
}
|
|
|
|
}
|
2020-02-15 08:51:00 -05:00
|
|
|
|
|
|
|
impl Neg for Number {
|
|
|
|
type Output = Self;
|
|
|
|
|
|
|
|
fn neg(self) -> Self {
|
|
|
|
Number {
|
|
|
|
val: -self.val,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|