mirror of
https://github.com/DioxusLabs/dioxus
synced 2024-11-10 14:44:12 +00:00
Merge pull request #1441 from ealmloff/fix-uplink
Expose public methods for manipulating the global runtime
This commit is contained in:
commit
13f10fea1b
3 changed files with 65 additions and 8 deletions
|
@ -32,6 +32,7 @@ pub(crate) mod innerlude {
|
||||||
pub use crate::nodes::RenderReturn;
|
pub use crate::nodes::RenderReturn;
|
||||||
pub use crate::nodes::*;
|
pub use crate::nodes::*;
|
||||||
pub use crate::properties::*;
|
pub use crate::properties::*;
|
||||||
|
pub use crate::runtime::{in_runtime, override_runtime, Runtime};
|
||||||
pub use crate::scheduler::*;
|
pub use crate::scheduler::*;
|
||||||
pub use crate::scope_context::*;
|
pub use crate::scope_context::*;
|
||||||
pub use crate::scopes::*;
|
pub use crate::scopes::*;
|
||||||
|
@ -86,11 +87,11 @@ pub use crate::innerlude::{
|
||||||
pub mod prelude {
|
pub mod prelude {
|
||||||
pub use crate::innerlude::{
|
pub use crate::innerlude::{
|
||||||
consume_context, consume_context_from_scope, current_scope_id, fc_to_builder, has_context,
|
consume_context, consume_context_from_scope, current_scope_id, fc_to_builder, has_context,
|
||||||
provide_context, provide_context_to_scope, provide_root_context, push_future,
|
in_runtime, override_runtime, provide_context, provide_context_to_scope,
|
||||||
remove_future, schedule_update_any, spawn, spawn_forever, suspend, throw, AnyValue,
|
provide_root_context, push_future, remove_future, schedule_update_any, spawn,
|
||||||
Component, Element, Event, EventHandler, Fragment, IntoAttributeValue, LazyNodes,
|
spawn_forever, suspend, throw, AnyValue, Component, Element, Event, EventHandler, Fragment,
|
||||||
Properties, Scope, ScopeId, ScopeState, Scoped, TaskId, Template, TemplateAttribute,
|
IntoAttributeValue, LazyNodes, Properties, Runtime, Scope, ScopeId, ScopeState, Scoped,
|
||||||
TemplateNode, Throw, VNode, VirtualDom,
|
TaskId, Template, TemplateAttribute, TemplateNode, Throw, VNode, VirtualDom,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -7,6 +7,51 @@ thread_local! {
|
||||||
static RUNTIMES: RefCell<Vec<Rc<Runtime>>> = RefCell::new(vec![]);
|
static RUNTIMES: RefCell<Vec<Rc<Runtime>>> = RefCell::new(vec![]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Run some code within a runtime
|
||||||
|
pub fn in_runtime<R>(runtime: Rc<Runtime>, f: impl FnOnce() -> R) -> R {
|
||||||
|
let _guard = RuntimeGuard::new(runtime);
|
||||||
|
f()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Override the current runtime. This must be used to override the current runtime when importing components from a dynamic library that has it's own runtime.
|
||||||
|
///
|
||||||
|
/// ```rust
|
||||||
|
/// use dioxus::prelude::*;
|
||||||
|
///
|
||||||
|
/// fn main() {
|
||||||
|
/// let virtual_dom = VirtualDom::new(app);
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// fn app(cx: Scope) -> Element {
|
||||||
|
/// render!{ Component { runtime: Runtime::current().unwrap() } }
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// // In a dynamic library
|
||||||
|
/// #[derive(Props)]
|
||||||
|
/// struct ComponentProps {
|
||||||
|
/// runtime: std::rc::Rc<Runtime>,
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// impl PartialEq for ComponentProps {
|
||||||
|
/// fn eq(&self, _other: &Self) -> bool {
|
||||||
|
/// true
|
||||||
|
/// }
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// fn Component(cx: Scope<ComponentProps>) -> Element {
|
||||||
|
/// cx.use_hook(|| override_runtime(cx.props.runtime.clone()));
|
||||||
|
///
|
||||||
|
/// render! { div {} }
|
||||||
|
/// }
|
||||||
|
/// ```
|
||||||
|
pub fn override_runtime(runtime: Rc<Runtime>) {
|
||||||
|
RUNTIMES.with(|stack| {
|
||||||
|
let mut stack = stack.borrow_mut();
|
||||||
|
stack.pop();
|
||||||
|
stack.push(runtime);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/// Pushes a new scope onto the stack
|
/// Pushes a new scope onto the stack
|
||||||
pub(crate) fn push_runtime(runtime: Rc<Runtime>) {
|
pub(crate) fn push_runtime(runtime: Rc<Runtime>) {
|
||||||
RUNTIMES.with(|stack| stack.borrow_mut().push(runtime));
|
RUNTIMES.with(|stack| stack.borrow_mut().push(runtime));
|
||||||
|
@ -41,7 +86,8 @@ where
|
||||||
.flatten()
|
.flatten()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) struct Runtime {
|
/// A global runtime that is shared across all scopes that provides the async runtime and context API
|
||||||
|
pub struct Runtime {
|
||||||
pub(crate) scope_contexts: RefCell<Vec<Option<ScopeContext>>>,
|
pub(crate) scope_contexts: RefCell<Vec<Option<ScopeContext>>>,
|
||||||
pub(crate) scheduler: Rc<Scheduler>,
|
pub(crate) scheduler: Rc<Scheduler>,
|
||||||
|
|
||||||
|
@ -63,6 +109,11 @@ impl Runtime {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Get the current runtime
|
||||||
|
pub fn current() -> Option<Rc<Self>> {
|
||||||
|
RUNTIMES.with(|stack| stack.borrow().last().cloned())
|
||||||
|
}
|
||||||
|
|
||||||
/// Create a scope context. This slab is synchronized with the scope slab.
|
/// Create a scope context. This slab is synchronized with the scope slab.
|
||||||
pub(crate) fn create_context_at(&self, id: ScopeId, context: ScopeContext) {
|
pub(crate) fn create_context_at(&self, id: ScopeId, context: ScopeContext) {
|
||||||
let mut contexts = self.scope_contexts.borrow_mut();
|
let mut contexts = self.scope_contexts.borrow_mut();
|
||||||
|
@ -77,14 +128,14 @@ impl Runtime {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get the current scope id
|
/// Get the current scope id
|
||||||
pub fn current_scope_id(&self) -> Option<ScopeId> {
|
pub(crate) fn current_scope_id(&self) -> Option<ScopeId> {
|
||||||
self.scope_stack.borrow().last().copied()
|
self.scope_stack.borrow().last().copied()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get the context for any scope given its ID
|
/// Get the context for any scope given its ID
|
||||||
///
|
///
|
||||||
/// This is useful for inserting or removing contexts from a scope, or rendering out its root node
|
/// This is useful for inserting or removing contexts from a scope, or rendering out its root node
|
||||||
pub fn get_context(&self, id: ScopeId) -> Option<Ref<'_, ScopeContext>> {
|
pub(crate) fn get_context(&self, id: ScopeId) -> Option<Ref<'_, ScopeContext>> {
|
||||||
Ref::filter_map(self.scope_contexts.borrow(), |contexts| {
|
Ref::filter_map(self.scope_contexts.borrow(), |contexts| {
|
||||||
contexts.get(id.0).and_then(|f| f.as_ref())
|
contexts.get(id.0).and_then(|f| f.as_ref())
|
||||||
})
|
})
|
||||||
|
|
|
@ -662,6 +662,11 @@ impl VirtualDom {
|
||||||
fn finalize(&mut self) -> Mutations {
|
fn finalize(&mut self) -> Mutations {
|
||||||
std::mem::take(&mut self.mutations)
|
std::mem::take(&mut self.mutations)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Get the current runtime
|
||||||
|
pub fn runtime(&self) -> Rc<Runtime> {
|
||||||
|
self.runtime.clone()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Drop for VirtualDom {
|
impl Drop for VirtualDom {
|
||||||
|
|
Loading…
Reference in a new issue