use crate::errors::NonEmptyError;
use crate::errors::NonEmptyError::Empty;
use std::collections::btree_set::{IntoIter, Iter};
use std::collections::BTreeSet;
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord)]
pub struct NEOrderedSet<T: Ord>(BTreeSet<T>);
impl<T: Ord> NEOrderedSet<T> {
pub fn new(head: T, tail: Vec<T>) -> Self {
let mut set = BTreeSet::new();
set.insert(head);
set.extend(tail);
Self(set)
}
pub fn singleton(value: T) -> Self {
let mut set = BTreeSet::new();
set.insert(value);
Self(set)
}
pub fn from(set: BTreeSet<T>) -> Result<Self, NonEmptyError> {
if set.is_empty() {
Err(Empty)
} else {
Ok(Self(set))
}
}
#[doc(hidden)]
pub fn __from_set_unsafe(set: BTreeSet<T>) -> Self {
debug_assert!(!set.is_empty());
Self(set)
}
pub fn into_set(self) -> BTreeSet<T> {
self.0
}
pub fn is_empty(&self) -> bool {
false
}
pub fn insert(&mut self, value: T) -> bool {
self.0.insert(value)
}
pub fn remove(&mut self, value: &T) -> bool {
if self.0.len() == 1 && self.0.contains(&value) {
false } else {
self.0.remove(&value)
}
}
pub fn contains(&self, value: &T) -> bool {
self.0.contains(value)
}
pub fn len(&self) -> usize {
self.0.len()
}
}
impl<T: Ord> From<NEOrderedSet<T>> for BTreeSet<T> {
fn from(set: NEOrderedSet<T>) -> Self {
set.into_set()
}
}
impl<T: Ord> TryFrom<BTreeSet<T>> for NEOrderedSet<T> {
type Error = NonEmptyError;
fn try_from(set: BTreeSet<T>) -> Result<Self, Self::Error> {
Self::from(set)
}
}
impl<T: Ord> IntoIterator for NEOrderedSet<T> {
type Item = T;
type IntoIter = IntoIter<T>;
fn into_iter(self) -> Self::IntoIter {
self.0.into_iter()
}
}
impl<'a, T: Ord> IntoIterator for &'a NEOrderedSet<T> {
type Item = &'a T;
type IntoIter = Iter<'a, T>;
fn into_iter(self) -> Self::IntoIter {
self.0.iter()
}
}