typedmap 0.1.0

A hash map (dash map). A type of value is defined by trait implemented on key type
Documentation
use std::marker::PhantomData;

use crate::typedkey::TypedKey;
use crate::typedvalue::TypedMapValue;
use crate::TypedMapKey;

pub struct OccupiedEntry<'a, K, Marker>
where
    K: 'static + TypedMapKey<Marker>,
{
    base: std::collections::hash_map::OccupiedEntry<'a, TypedKey, TypedMapValue>,
    _phantom: std::marker::PhantomData<Marker>,
    _phantom_key: std::marker::PhantomData<K>,
}

impl<'a, K, Marker> OccupiedEntry<'a, K, Marker>
where
    K: 'static + TypedMapKey<Marker>,
{
    #[inline]
    pub fn into_mut(self) -> &'a mut K::Value {
        self.base
            .into_mut()
            .downcast_mut::<K::Value>()
            .expect("Broken TypedMap entry")
    }

    #[inline]
    pub fn key(&self) -> &K {
        self.base
            .key()
            .downcast_ref::<K>()
            .expect("Broken TypedMap entry")
    }

    #[inline]
    pub fn remove_entry(self) -> (K, K::Value) {
        let (k, v) = self.base.remove_entry();
        let v = v
            .downcast::<K::Value>()
            .ok()
            .expect("Broken TypedMap entry");
        let k = k.downcast::<K>().ok().expect("Broken TypedMap entry");
        (k, v)
    }

    #[inline]
    pub fn get(&self) -> &K::Value {
        self.base
            .get()
            .downcast_ref::<K::Value>()
            .expect("Broken TypedMap entry")
    }

    #[inline]
    pub fn get_mut(&mut self) -> &mut K::Value {
        self.base
            .get_mut()
            .downcast_mut::<K::Value>()
            .expect("Broken TypedMap entry")
    }

    #[inline]
    pub fn insert(&mut self, value: K::Value) -> K::Value {
        let value = TypedMapValue::from_value(value);
        self.base
            .insert(value)
            .downcast::<K::Value>()
            .ok()
            .expect("Broken TypedMap entry")
    }

    #[inline]
    pub fn remove(self) -> K::Value {
        self.base
            .remove()
            .downcast::<K::Value>()
            .ok()
            .expect("Broken TypedMap entry")
    }
}

pub struct VacantEntry<'a, K, Marker>
where
    K: 'static + TypedMapKey<Marker>,
{
    base: std::collections::hash_map::VacantEntry<'a, TypedKey, TypedMapValue>,
    _phantom: std::marker::PhantomData<Marker>,
    _phantom_key: std::marker::PhantomData<K>,
}

impl<'a, K, Marker> VacantEntry<'a, K, Marker>
where
    K: 'static + TypedMapKey<Marker>,
{
    #[inline]
    pub fn key(&self) -> &K {
        self.base
            .key()
            .downcast_ref::<K>()
            .expect("Broken TypedMap entry")
    }

    #[inline]
    pub fn into_key(self) -> K {
        self.base
            .into_key()
            .downcast::<K>()
            .ok()
            .expect("Broken TypedMap entry")
    }

    #[inline]
    pub fn insert(self, value: K::Value) -> &'a mut K::Value {
        let value = TypedMapValue::from_value(value);
        self.base
            .insert(value)
            .downcast_mut::<K::Value>()
            .expect("Broken TypedMap entry")
    }
}

pub enum Entry<'a, K, Marker>
where
    K: 'static + TypedMapKey<Marker>,
{
    Occupied(OccupiedEntry<'a, K, Marker>),
    Vacant(VacantEntry<'a, K, Marker>),
}

pub(crate) fn map_entry<Marker, K: 'static + TypedMapKey<Marker>>(
    entry: std::collections::hash_map::Entry<'_, TypedKey, TypedMapValue>,
) -> Entry<'_, K, Marker> {
    match entry {
        std::collections::hash_map::Entry::Occupied(base) => Entry::Occupied(OccupiedEntry {
            base,
            _phantom: PhantomData,
            _phantom_key: PhantomData,
        }),
        std::collections::hash_map::Entry::Vacant(base) => Entry::Vacant(VacantEntry {
            base,
            _phantom: PhantomData,
            _phantom_key: PhantomData,
        }),
    }
}

impl<'a, Marker, K: 'static + TypedMapKey<Marker>> Entry<'a, K, Marker> {
    #[inline]
    pub fn or_insert(self, default: K::Value) -> &'a mut K::Value {
        match self {
            Self::Occupied(entry) => entry.into_mut(),
            Self::Vacant(entry) => entry.insert(default),
        }
    }

    #[inline]
    pub fn or_insert_with<F: FnOnce() -> K::Value>(self, default: F) -> &'a mut K::Value {
        match self {
            Self::Occupied(entry) => entry.into_mut(),
            Self::Vacant(entry) => entry.insert(default()),
        }
    }

    pub fn key(&self) -> &K {
        match *self {
            Self::Occupied(ref entry) => entry.key(),
            Self::Vacant(ref entry) => entry.key(),
        }
    }

    pub fn and_modify<F>(self, f: F) -> Self
    where
        F: FnOnce(&mut K::Value),
    {
        match self {
            Self::Occupied(mut entry) => {
                f(entry.get_mut());
                Self::Occupied(entry)
            }
            Self::Vacant(entry) => Self::Vacant(entry),
        }
    }
}

impl<'a, Marker, K: 'static + TypedMapKey<Marker>> Entry<'a, K, Marker>
where
    K::Value: Default,
{
    pub fn or_default(self) -> &'a mut K::Value {
        match self {
            Self::Occupied(entry) => entry.into_mut(),
            Self::Vacant(entry) => entry.insert(Default::default()),
        }
    }
}