use std::ops::{Add, Bound, Range, RangeBounds, Sub};
mod private {
pub trait ZeroOne {
fn zero() -> Self;
fn one() -> Self;
}
macro_rules! impl_one {
($($t:ty),*) => {
$(
impl ZeroOne for $t {
fn zero() -> Self {
0 as $t
}
fn one() -> Self {
1 as $t
}
}
)*
};
}
macro_rules! for_all_num_type {
($macro:ident) => {
$macro! { u8, u16, u32, u64, usize, i8, i16, i32, i64, isize, f32, f64 }
};
}
for_all_num_type! { impl_one }
}
use private::ZeroOne;
pub trait RangeBoundsExt<
T: PartialOrd<T> + Add<Output = T> + Sub<Output = T> + Clone + Copy + Send + Sync + 'static + ZeroOne,
>: RangeBounds<T>
{
fn start(&self) -> Option<T> {
match self.start_bound() {
Bound::Included(v) => Some(*v),
Bound::Excluded(v) => Some(*v + ZeroOne::one()),
Bound::Unbounded => None,
}
}
fn end(&self) -> Option<T> {
match self.end_bound() {
Bound::Included(v) => Some(*v + ZeroOne::one()),
Bound::Excluded(v) => Some(*v),
Bound::Unbounded => None,
}
}
fn start_with_bound(&self, bound: T) -> T {
self.start().unwrap_or(bound)
}
fn end_with_bound(&self, bound: T) -> T {
self.end().unwrap_or(bound)
}
fn bounds(&self, range: Range<T>) -> Range<T> {
let start = self.start_with_bound(range.start);
let end = self.end_with_bound(range.end);
start..end
}
fn size(&self) -> Option<T> {
let start = self.start()?;
let end = self.end()?;
Some(end - start)
}
fn is_empty(&self) -> bool {
match self.size() {
Some(len) => len == ZeroOne::zero(),
None => false,
}
}
fn is_full(&self) -> bool {
self.start_bound() == Bound::Unbounded && self.end_bound() == Bound::Unbounded
}
fn map<F, R>(&self, f: F) -> (Bound<R>, Bound<R>)
where
F: Fn(&T) -> R,
{
(self.start_bound().map(&f), self.end_bound().map(&f))
}
}
impl<
T: PartialOrd<T> + Add<Output = T> + Sub<Output = T> + Clone + Copy + Send + Sync + 'static + ZeroOne,
RB: RangeBounds<T>,
> RangeBoundsExt<T> for RB
{
}