use std::{
ops::{Index, IndexMut},
slice::GetDisjointMutError,
};
use index_vec::Idx;
use crate::{
IndexedDomain, IndexedValue, ToIndex,
pointer::{ArcFamily, PointerFamily, RcFamily, RefFamily},
};
pub struct IndexVec<'a, K: IndexedValue + 'a, V, P: PointerFamily<'a>> {
vec: Vec<V>,
pub(crate) domain: P::Pointer<IndexedDomain<K>>,
}
impl<'a, K, V, P> IndexVec<'a, K, V, P>
where
K: IndexedValue + 'a,
P: PointerFamily<'a>,
V: Clone,
{
pub fn from_elem(elem: V, domain: &P::Pointer<IndexedDomain<K>>) -> Self {
let vec = vec![elem; domain.len()];
IndexVec {
vec,
domain: domain.clone(),
}
}
}
impl<'a, K, V, P> IndexVec<'a, K, V, P>
where
K: IndexedValue + 'a,
P: PointerFamily<'a>,
{
pub fn from_fn(f: impl FnMut(K::Index) -> V, domain: &P::Pointer<IndexedDomain<K>>) -> Self {
let vec = domain.indices().map(f).collect();
IndexVec {
vec,
domain: domain.clone(),
}
}
pub fn get<M>(&self, idx: impl ToIndex<K, M>) -> &V {
let idx = idx.to_index(&self.domain);
debug_assert!(self.domain.contains_index(idx));
unsafe { self.vec.get_unchecked(idx.index()) }
}
pub fn get_mut<M>(&mut self, idx: impl ToIndex<K, M>) -> &mut V {
let idx = idx.to_index(&self.domain);
debug_assert!(self.domain.contains_index(idx));
unsafe { self.vec.get_unchecked_mut(idx.index()) }
}
pub fn iter(&self) -> impl DoubleEndedIterator<Item = &V> + ExactSizeIterator<Item = &V> {
self.vec.iter()
}
pub fn iter_mut(
&mut self,
) -> impl DoubleEndedIterator<Item = &mut V> + ExactSizeIterator<Item = &mut V> {
self.vec.iter_mut()
}
pub fn as_slice(&self) -> &[V] {
&self.vec
}
pub fn as_slice_mut(&mut self) -> &mut [V] {
&mut self.vec
}
pub fn get_disjoint_mut<const N: usize>(
&mut self,
indices: [K::Index; N],
) -> Result<[&mut V; N], GetDisjointMutError> {
self.vec.get_disjoint_mut(indices.map(Idx::index))
}
}
impl<'a, K, V, P> Clone for IndexVec<'a, K, V, P>
where
K: IndexedValue + 'a,
P: PointerFamily<'a>,
V: Clone,
{
fn clone(&self) -> Self {
IndexVec {
vec: self.vec.clone(),
domain: self.domain.clone(),
}
}
fn clone_from(&mut self, source: &Self) {
self.vec.clone_from(&source.vec);
}
}
impl<'a, K, V, P> PartialEq for IndexVec<'a, K, V, P>
where
K: IndexedValue + 'a,
P: PointerFamily<'a>,
V: PartialEq,
{
fn eq(&self, other: &Self) -> bool {
self.vec == other.vec
}
}
impl<'a, K, V, P> Eq for IndexVec<'a, K, V, P>
where
K: IndexedValue + 'a,
P: PointerFamily<'a>,
V: Eq,
{
}
impl<'a, K, V, P> Index<K::Index> for IndexVec<'a, K, V, P>
where
K: IndexedValue + 'a,
P: PointerFamily<'a>,
{
type Output = V;
fn index(&self, index: K::Index) -> &Self::Output {
self.get(index)
}
}
impl<'a, K, V, P> IndexMut<K::Index> for IndexVec<'a, K, V, P>
where
K: IndexedValue + 'a,
P: PointerFamily<'a>,
{
fn index_mut(&mut self, index: K::Index) -> &mut Self::Output {
self.get_mut(index)
}
}
pub type RcIndexVec<K, V> = IndexVec<'static, K, V, RcFamily>;
pub type ArcIndexVec<K, V> = IndexVec<'static, K, V, ArcFamily>;
pub type RefIndexVec<'a, K, V> = IndexVec<'a, K, V, RefFamily<'a>>;