mirror of
https://github.com/AppFlowy-IO/AppFlowy.git
synced 2025-07-21 07:58:13 +00:00
71 lines
1.5 KiB
Rust
71 lines
1.5 KiB
Rust
![]() |
use std::collections::HashMap;
|
||
|
use std::sync::Arc;
|
||
|
|
||
|
pub trait RefCountValue {
|
||
|
fn did_remove(&self);
|
||
|
}
|
||
|
|
||
|
struct RefCountHandler<T> {
|
||
|
ref_count: usize,
|
||
|
inner: T,
|
||
|
}
|
||
|
|
||
|
impl<T> RefCountHandler<T> {
|
||
|
pub fn new(inner: T) -> Self {
|
||
|
Self { ref_count: 1, inner }
|
||
|
}
|
||
|
|
||
|
pub fn increase_ref_count(&mut self) {
|
||
|
self.ref_count += 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
pub struct RefCountHashMap<T>(HashMap<String, RefCountHandler<T>>);
|
||
|
|
||
|
impl<T> RefCountHashMap<T>
|
||
|
where
|
||
|
T: Clone + Send + Sync + RefCountValue,
|
||
|
{
|
||
|
pub fn new() -> Self {
|
||
|
Self(Default::default())
|
||
|
}
|
||
|
|
||
|
pub fn get(&self, key: &str) -> Option<T> {
|
||
|
self.0.get(key).and_then(|handler| Some(handler.inner.clone()))
|
||
|
}
|
||
|
|
||
|
pub fn insert(&mut self, key: String, value: T) {
|
||
|
if let Some(handler) = self.0.get_mut(&key) {
|
||
|
handler.increase_ref_count();
|
||
|
} else {
|
||
|
let handler = RefCountHandler::new(value);
|
||
|
self.0.insert(key, handler);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
pub fn remove(&mut self, key: &str) {
|
||
|
let mut should_remove = false;
|
||
|
if let Some(value) = self.0.get_mut(key) {
|
||
|
if value.ref_count > 0 {
|
||
|
value.ref_count -= 1;
|
||
|
}
|
||
|
should_remove = value.ref_count == 0;
|
||
|
}
|
||
|
|
||
|
if should_remove {
|
||
|
if let Some(handler) = self.0.remove(key) {
|
||
|
handler.inner.did_remove();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl<T> RefCountValue for Arc<T>
|
||
|
where
|
||
|
T: RefCountValue,
|
||
|
{
|
||
|
fn did_remove(&self) {
|
||
|
(**self).did_remove()
|
||
|
}
|
||
|
}
|