use super::{Keyed, KeyedVecSet};
use core::mem;
use core::ops::Deref;
pub struct Mut<'a, V>(&'a mut V);
impl<V> Mut<'_, V> {
pub unsafe fn as_mut(&mut self) -> &mut V {
self.0
}
}
impl<V> Deref for Mut<'_, V> {
type Target = V;
fn deref(&self) -> &V {
self.0
}
}
#[derive(Debug)]
pub enum Entry<'a, K, V> {
Occupied(OccupiedEntry<'a, K, V>),
Vacant(VacantEntry<'a, K, V>),
}
impl<'a, K, V> Entry<'a, K, V>
where
K: Ord,
V: Keyed<K>,
{
pub fn or_insert(self, default: V) -> Mut<'a, V> {
assert!(self.key() == default.key());
match self {
Entry::Occupied(entry) => Mut(unsafe { entry.into_mut() }),
Entry::Vacant(entry) => entry.insert(default),
}
}
pub fn or_insert_with<F>(self, call: F) -> Mut<'a, V>
where
F: FnOnce() -> V,
{
match self {
Entry::Occupied(entry) => Mut(unsafe { entry.into_mut() }),
Entry::Vacant(entry) => {
let value = call();
assert!(&entry.key == value.key());
entry.insert(value)
}
}
}
pub fn or_insert_with_key<F>(self, default: F) -> Mut<'a, V>
where
F: FnOnce(&K) -> V,
{
match self {
Entry::Occupied(entry) => Mut(unsafe { entry.into_mut() }),
Entry::Vacant(entry) => {
let value = default(&entry.key);
assert!(&entry.key == value.key());
entry.insert(value)
}
}
}
pub fn or_insert_full(self, value: V) -> (usize, bool) {
assert!(self.key() == value.key());
match self {
Entry::Occupied(e) => (e.index, false),
Entry::Vacant(e) => {
let index = e.index;
e.map.base.insert(index, value);
(index, true)
}
}
}
pub unsafe fn or_insert_full_unchecked(self, value: V) -> (usize, bool) {
match self {
Entry::Occupied(e) => (e.index, false),
Entry::Vacant(e) => {
let index = e.index;
e.map.base.insert(index, value);
(index, true)
}
}
}
pub fn or_insert_with_full<F>(self, call: F) -> (usize, bool)
where
F: FnOnce() -> V,
{
match self {
Entry::Occupied(e) => (e.index, false),
Entry::Vacant(e) => {
let index = e.index;
let value = call();
assert!(&e.key == value.key());
e.map.base.insert(index, value);
(index, true)
}
}
}
pub unsafe fn or_insert_with_full_unchecked<F>(self, call: F) -> (usize, bool)
where
F: FnOnce() -> V,
{
match self {
Entry::Occupied(e) => (e.index, false),
Entry::Vacant(e) => {
let index = e.index;
let value = call();
e.map.base.insert(index, value);
(index, true)
}
}
}
pub fn key(&self) -> &K {
match self {
Entry::Occupied(entry) => entry.key(),
Entry::Vacant(entry) => entry.key(),
}
}
pub fn index(&self) -> usize {
match self {
Entry::Occupied(entry) => entry.index(),
Entry::Vacant(entry) => entry.index(),
}
}
pub unsafe fn and_modify<F>(self, f: F) -> Self
where
F: FnOnce(&mut V),
{
match self {
Entry::Occupied(mut o) => {
unsafe { f(o.get_mut()) };
Entry::Occupied(o)
}
x => x,
}
}
}
#[derive(Debug)]
pub struct OccupiedEntry<'a, K, V> {
map: &'a mut KeyedVecSet<K, V>,
key: K,
index: usize,
}
impl<'a, K, V> OccupiedEntry<'a, K, V> {
pub(super) fn new(
map: &'a mut KeyedVecSet<K, V>,
key: K,
index: usize,
) -> OccupiedEntry<'a, K, V> {
OccupiedEntry { map, key, index }
}
pub fn key(&self) -> &K {
&self.key
}
pub fn index(&self) -> usize {
self.index
}
pub fn into_key(self) -> K {
self.key
}
pub fn get(&self) -> &V {
&self.map[self.index]
}
pub unsafe fn get_mut(&mut self) -> &mut V {
unsafe { self.map.get_index_mut(self.index).expect("unexisting key") }
}
pub unsafe fn into_mut(self) -> &'a mut V {
unsafe { self.map.get_index_mut(self.index).expect("unexisting key") }
}
pub fn insert(&mut self, value: V) -> V {
mem::replace(unsafe { self.get_mut() }, value)
}
pub fn remove_entry(self) -> V {
self.map.remove_index(self.index)
}
pub fn remove(self) -> V {
self.remove_entry()
}
}
#[derive(Debug)]
pub struct VacantEntry<'a, K, V> {
map: &'a mut KeyedVecSet<K, V>,
key: K,
index: usize,
}
impl<'a, K, V> VacantEntry<'a, K, V> {
pub(super) fn new(
map: &'a mut KeyedVecSet<K, V>,
key: K,
index: usize,
) -> VacantEntry<'a, K, V> {
VacantEntry { map, key, index }
}
pub fn key(&self) -> &K {
&self.key
}
pub fn index(&self) -> usize {
self.index
}
pub fn into_key(self) -> K {
self.key
}
pub fn insert(self, value: V) -> Mut<'a, V>
where
K: Ord,
V: Keyed<K>,
{
self.map.base.insert(self.index, value);
Mut(unsafe { self.map.base.get_unchecked_mut(self.index) })
}
}