use std::{
fmt::Display,
num::{NonZeroU128, NonZeroU16, NonZeroU32, NonZeroU64, NonZeroU8, NonZeroUsize},
ops::Shr,
};
use num::{Integer, PrimInt};
pub trait NotZero: PrimInt + Integer {
fn not_zero(self) -> bool {
!self.is_zero()
}
fn to_nonzero(self) -> Option<NonZero<Self>> {
NonZero::new(self)
}
}
impl<T> NotZero for T where T: PrimInt + Integer {}
#[derive(Debug, Copy, Clone, PartialOrd, Ord, PartialEq, Eq)]
pub struct NonZero<T: PrimInt + Integer> {
value: T,
}
impl<T> Display for NonZero<T>
where
T: PrimInt + Integer + Display,
{
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "{}", self.value)
}
}
impl<T> NonZero<T>
where
T: PrimInt + Integer,
{
pub fn new(value: T) -> Option<Self> {
value.not_zero().then_some(Self { value })
}
pub const unsafe fn new_unchecked(value: T) -> Self {
Self { value }
}
pub const fn get(self) -> T {
self.value
}
pub fn is_even(self) -> bool {
self.value.is_even()
}
pub fn is_odd(self) -> bool {
self.value.is_odd()
}
pub fn set(&mut self, value: T) -> bool {
if value.not_zero() {
unsafe { self.set_unchecked(value) }
}
value.not_zero()
}
pub unsafe fn set_unchecked(&mut self, value: T) {
self.value = value;
}
pub fn map(self, f: impl Fn(T) -> T) -> Option<Self> {
Self::new(f(self.value))
}
#[must_use]
pub unsafe fn map_unchecked(self, f: fn(T) -> T) -> Self {
Self::new_unchecked(f(self.value))
}
pub fn trailing_zeros(self) -> u32 {
self.value.trailing_zeros()
}
}
impl<T> NonZero<T>
where
T: PrimInt + Integer + Shr<u32, Output = T>,
{
#[must_use]
pub fn without_trailing_zeros(self) -> Self {
unsafe { Self::new_unchecked(self.value >> self.trailing_zeros()) }
}
}
macro_rules! impl_from_primitive {
($new_name: ty, $primitive: ty) => {
impl From<$primitive> for $new_name {
fn from(value: $primitive) -> Self {
Self { value: value.get() }
}
}
};
}
impl_from_primitive!(NonZero<u8>, NonZeroU8);
impl_from_primitive!(NonZero<u16>, NonZeroU16);
impl_from_primitive!(NonZero<u32>, NonZeroU32);
impl_from_primitive!(NonZero<u64>, NonZeroU64);
impl_from_primitive!(NonZero<u128>, NonZeroU128);
impl_from_primitive!(NonZero<usize>, NonZeroUsize);
#[allow(unused_imports, clippy::unwrap_used)]
mod tests {
use std::ops::Range;
use num::{BigInt, FromPrimitive, One};
use crate::NonZero;
#[test]
fn new_works() {
assert!(NonZero::new(1u8).is_some());
assert!(NonZero::new(1u16).is_some());
assert!(NonZero::new(1u32).is_some());
assert!(NonZero::new(1u64).is_some());
assert!(NonZero::new(1u128).is_some());
assert!(NonZero::new(1usize).is_some());
}
}