add Outlet component

This commit is contained in:
Adrian Wannenmacher 2022-12-06 12:02:51 +01:00
parent b67b9feef0
commit 26543a1876
No known key found for this signature in database
GPG key ID: 19D986ECB1E492D5
5 changed files with 145 additions and 48 deletions

View file

@ -20,18 +20,18 @@ impl OutletData {
/// # use dioxus_router_core::{Name, OutletData};
/// let mut d = OutletData::default();
/// let (m, a, n);
/// (m, d) = d.next(None);
/// (a, d) = d.next(None);
/// (n, d) = d.next(Some(Name::of::<bool>()));
/// (m, d) = d.next(&None);
/// (a, d) = d.next(&None);
/// (n, d) = d.next(&Some(Name::of::<bool>()));
///
/// assert_eq!(m, 0);
/// assert_eq!(a, 1);
/// assert_eq!(n, 0);
/// ```
pub fn next(&self, name: Option<Name>) -> (usize, Self) {
pub fn next(&self, name: &Option<Name>) -> (usize, Self) {
let mut next = self.clone();
let depth = next.depth(name.clone()).map(|d| d + 1).unwrap_or(0);
let depth = next.depth(name).map(|d| d + 1).unwrap_or(0);
next.set_depth(name, depth);
@ -43,17 +43,17 @@ impl OutletData {
/// ```rust
/// # use dioxus_router_core::OutletData;
/// let mut d = OutletData::default();
/// let b = d.depth(None);
/// d.set_depth(None, 18);
/// let a = d.depth(None);
/// let b = d.depth(&None);
/// d.set_depth(&None, 18);
/// let a = d.depth(&None);
///
/// assert_eq!(b, None);
/// assert_eq!(a, Some(18));
/// ```
pub fn depth(&self, name: Option<Name>) -> Option<usize> {
pub fn depth(&self, name: &Option<Name>) -> Option<usize> {
match name {
None => self.main,
Some(n) => self.named.get(&n).copied(),
Some(n) => self.named.get(n).copied(),
}
}
@ -62,17 +62,17 @@ impl OutletData {
/// ```rust
/// # use dioxus_router_core::OutletData;
/// let mut d = OutletData::default();
/// let b = d.depth(None);
/// d.set_depth(None, 18);
/// let a = d.depth(None);
/// let b = d.depth(&None);
/// d.set_depth(&None, 18);
/// let a = d.depth(&None);
///
/// assert_eq!(b, None);
/// assert_eq!(a, Some(18));
/// ```
pub fn set_depth(&mut self, name: Option<Name>, depth: usize) {
pub fn set_depth(&mut self, name: &Option<Name>, depth: usize) {
match name {
None => self.main = Some(depth),
Some(n) => _ = self.named.insert(n, depth),
Some(n) => _ = self.named.insert(n.clone(), depth),
}
}
}
@ -107,18 +107,18 @@ mod tests {
fn depth() {
let td = test_data();
assert_eq!(td.depth(None), Some(18));
assert_eq!(td.depth(Some(Name::of::<bool>())), Some(0));
assert_eq!(td.depth(&None), Some(18));
assert_eq!(td.depth(&Some(Name::of::<bool>())), Some(0));
assert_eq!(td.depth(Some(Name::of::<u8>())), Some(8));
assert_eq!(td.depth(Some(Name::of::<u16>())), Some(16));
assert_eq!(td.depth(Some(Name::of::<u32>())), Some(32));
assert_eq!(td.depth(Some(Name::of::<u64>())), Some(64));
assert_eq!(td.depth(&Some(Name::of::<u8>())), Some(8));
assert_eq!(td.depth(&Some(Name::of::<u16>())), Some(16));
assert_eq!(td.depth(&Some(Name::of::<u32>())), Some(32));
assert_eq!(td.depth(&Some(Name::of::<u64>())), Some(64));
assert_eq!(td.depth(Some(Name::of::<i8>())), None);
assert_eq!(td.depth(Some(Name::of::<i16>())), None);
assert_eq!(td.depth(Some(Name::of::<i32>())), None);
assert_eq!(td.depth(Some(Name::of::<i64>())), None);
assert_eq!(td.depth(&Some(Name::of::<i8>())), None);
assert_eq!(td.depth(&Some(Name::of::<i16>())), None);
assert_eq!(td.depth(&Some(Name::of::<i32>())), None);
assert_eq!(td.depth(&Some(Name::of::<i64>())), None);
}
#[test]
@ -126,21 +126,21 @@ mod tests {
let mut td = test_data();
// set
td.set_depth(None, 0);
td.set_depth(Some(Name::of::<bool>()), 1);
td.set_depth(&None, 0);
td.set_depth(&Some(Name::of::<bool>()), 1);
td.set_depth(Some(Name::of::<u8>()), 2);
td.set_depth(Some(Name::of::<u16>()), 4);
td.set_depth(Some(Name::of::<u32>()), 8);
td.set_depth(Some(Name::of::<u64>()), 16);
td.set_depth(&Some(Name::of::<u8>()), 2);
td.set_depth(&Some(Name::of::<u16>()), 4);
td.set_depth(&Some(Name::of::<u32>()), 8);
td.set_depth(&Some(Name::of::<u64>()), 16);
td.set_depth(Some(Name::of::<i8>()), 32);
td.set_depth(Some(Name::of::<i16>()), 64);
td.set_depth(Some(Name::of::<i32>()), 128);
td.set_depth(Some(Name::of::<i64>()), 256);
td.set_depth(&Some(Name::of::<i8>()), 32);
td.set_depth(&Some(Name::of::<i16>()), 64);
td.set_depth(&Some(Name::of::<i32>()), 128);
td.set_depth(&Some(Name::of::<i64>()), 256);
// check
assert_eq!(td.depth(None), Some(0));
assert_eq!(td.depth(&None), Some(0));
assert_eq!(*td.named.get(&Name::of::<bool>()).unwrap(), 1);
assert_eq!(*td.named.get(&Name::of::<u8>()).unwrap(), 2);
@ -158,43 +158,43 @@ mod tests {
fn next() {
let td = test_data();
let (current, next) = td.next(None);
let (current, next) = td.next(&None);
assert_eq!(current, 19);
assert_eq!(next.depth(None), Some(19));
assert_eq!(next.depth(&None), Some(19));
let (current, next) = td.next(Some(Name::of::<bool>()));
let (current, next) = td.next(&Some(Name::of::<bool>()));
assert_eq!(current, 1);
assert_eq!(*next.named.get(&Name::of::<bool>()).unwrap(), 1);
let (current, next) = td.next(Some(Name::of::<u8>()));
let (current, next) = td.next(&Some(Name::of::<u8>()));
assert_eq!(current, 9);
assert_eq!(*next.named.get(&Name::of::<u8>()).unwrap(), 9);
let (current, next) = td.next(Some(Name::of::<u16>()));
let (current, next) = td.next(&Some(Name::of::<u16>()));
assert_eq!(current, 17);
assert_eq!(*next.named.get(&Name::of::<u16>()).unwrap(), 17);
let (current, next) = td.next(Some(Name::of::<u32>()));
let (current, next) = td.next(&Some(Name::of::<u32>()));
assert_eq!(current, 33);
assert_eq!(*next.named.get(&Name::of::<u32>()).unwrap(), 33);
let (current, next) = td.next(Some(Name::of::<u64>()));
let (current, next) = td.next(&Some(Name::of::<u64>()));
assert_eq!(current, 65);
assert_eq!(*next.named.get(&Name::of::<u64>()).unwrap(), 65);
let (current, next) = td.next(Some(Name::of::<i8>()));
let (current, next) = td.next(&Some(Name::of::<i8>()));
assert_eq!(current, 0);
assert_eq!(*next.named.get(&Name::of::<i8>()).unwrap(), 0);
let (current, next) = td.next(Some(Name::of::<i16>()));
let (current, next) = td.next(&Some(Name::of::<i16>()));
assert_eq!(current, 0);
assert_eq!(*next.named.get(&Name::of::<i16>()).unwrap(), 0);
let (current, next) = td.next(Some(Name::of::<i32>()));
let (current, next) = td.next(&Some(Name::of::<i32>()));
assert_eq!(current, 0);
assert_eq!(*next.named.get(&Name::of::<i32>()).unwrap(), 0);
let (current, next) = td.next(Some(Name::of::<i64>()));
let (current, next) = td.next(&Some(Name::of::<i64>()));
assert_eq!(current, 0);
assert_eq!(*next.named.get(&Name::of::<i64>()).unwrap(), 0);
}

View file

@ -16,6 +16,7 @@ dioxus = { path="../dioxus" }
dioxus-router-core = { path = "../router-core"}
futures-channel = "0.3.25"
futures-util = "0.3.25"
log = "0.4.17"
# for wasm

View file

@ -0,0 +1,57 @@
use dioxus::prelude::*;
use dioxus_router_core::{Name, OutletData};
use log::error;
use crate::utils::use_router_internal::use_router_internal;
#[derive(Debug, Eq, PartialEq, Props)]
pub struct OutletProps {
pub depth: Option<usize>,
pub name: Option<Name>,
}
#[allow(non_snake_case)]
pub fn Outlet(cx: Scope<OutletProps>) -> Element {
let OutletProps { depth, name } = cx.props;
// hook up to router
let router = match use_router_internal(&cx) {
Some(r) => r,
None => {
error!("`Outlet` must have access to a parent router, will be inactive");
#[cfg(debug_assertions)]
panic!("`Outlet` must have access to a parent router");
#[cfg(not(debug_assertions))]
return None;
}
};
let state = loop {
if let Some(state) = router.state.try_read() {
break state;
}
};
// do depth calculation and propagation
let depth = cx.use_hook(|| {
let mut context = cx.consume_context::<OutletData>().unwrap_or_default();
let depth = depth
.or_else(|| context.depth(name))
.map(|d| d + 1)
.unwrap_or_default();
context.set_depth(name, depth);
cx.provide_context(context);
depth
});
// get content
let X: Component = match name {
None => state.content.get(*depth),
Some(n) => state.named_content.get(n).and_then(|n| n.get(*depth)),
}?
.0
.clone();
render! {
X { }
}
}

View file

@ -1,4 +1,9 @@
pub(crate) mod contexts {
pub mod components {
mod outlet;
pub use outlet::*;
}
mod contexts {
pub(crate) mod router;
}
@ -10,3 +15,7 @@ pub mod hooks {
pub mod prelude {
pub use dioxus_router_core::prelude::*;
}
mod utils {
pub(crate) mod use_router_internal;
}

View file

@ -0,0 +1,30 @@
use std::sync::Arc;
use dioxus::prelude::ScopeState;
use dioxus_router_core::RouterMessage;
use crate::contexts::router::RouterContext;
/// A private hook to subscribe to the router.
///
/// Used to reduce redundancy within other components/hooks. Safe to call multiple times for a
/// single component, but not recommended. Multiple subscriptions will be discarded.
///
/// # Return values
/// - [`None`], when the current component isn't a descendant of a [`use_router`] component.
/// - Otherwise [`Some`].
///
/// [`use_router`]: crate::hooks::use_router
pub(crate) fn use_router_internal<'a>(cx: &'a ScopeState) -> &'a mut Option<RouterContext> {
let id = cx.use_hook(|| Arc::new(cx.scope_id()));
cx.use_hook(|| {
let router = cx.consume_context::<RouterContext>()?;
let _ = router
.sender
.unbounded_send(RouterMessage::Subscribe(id.clone()));
Some(router)
})
}