use crate::compat::*;
pub trait WeakElement {
type Strong;
fn new(view: &Self::Strong) -> Self;
fn view(&self) -> Option<Self::Strong>;
fn is_expired(&self) -> bool {
self.view().is_none()
}
fn clone(view: &Self::Strong) -> Self::Strong
where Self: Sized
{
Self::new(view).view().expect("WeakElement::clone")
}
}
pub trait WeakKey : WeakElement {
type Key: ?Sized + Eq + Hash;
fn with_key<F, R>(view: &Self::Strong, f: F) -> R
where F: FnOnce(&Self::Key) -> R;
fn hash<H: Hasher>(view: &Self::Strong, h: &mut H) {
Self::with_key(view, |k| k.hash(h));
}
fn equals<Q>(view: &Self::Strong, key: &Q) -> bool
where Q: ?Sized + Eq,
Self::Key: Borrow<Q>
{
Self::with_key(view, |k| k.borrow() == key)
}
}
impl<T: ?Sized> WeakElement for rc::Weak<T> {
type Strong = rc::Rc<T>;
fn new(view: &Self::Strong) -> Self {
rc::Rc::<T>::downgrade(view)
}
fn view(&self) -> Option<Self::Strong> {
self.upgrade()
}
fn clone(view: &Self::Strong) -> Self::Strong {
view.clone()
}
}
impl<T: ?Sized + Eq + Hash> WeakKey for rc::Weak<T> {
type Key = T;
fn with_key<F, R>(view: &Self::Strong, f: F) -> R
where F: FnOnce(&Self::Key) -> R
{
f(view)
}
}
impl<T: ?Sized> WeakElement for sync::Weak<T> {
type Strong = sync::Arc<T>;
fn new(view: &Self::Strong) -> Self {
sync::Arc::<T>::downgrade(view)
}
fn view(&self) -> Option<Self::Strong> {
self.upgrade()
}
fn clone(view: &Self::Strong) -> Self::Strong {
view.clone()
}
}
impl<T: ?Sized + Eq + Hash> WeakKey for sync::Weak<T>
{
type Key = T;
fn with_key<F, R>(view: &Self::Strong, f: F) -> R
where F: FnOnce(&Self::Key) -> R
{
f(view)
}
}