//! //! //! //! //! This crate demonstrates how to implement a custom renderer for Dioxus VNodes via the `TextRenderer` renderer. //! The `TextRenderer` consumes a Dioxus Virtual DOM, progresses its event queue, and renders the VNodes to a String. //! //! While `VNode` supports "to_string" directly, it renders child components as the RSX! macro tokens. For custom components, //! an external renderer is needed to progress the component lifecycles. The `TextRenderer` shows how to use the Virtual DOM //! API to progress these lifecycle events to generate a fully-mounted Virtual DOM instance which can be renderer in the //! `render` method. use std::fmt::{Display, Formatter}; use dioxus_core::exports::bumpalo; use dioxus_core::nodes::IntoVNode; use dioxus_core::*; macro_rules! render_lazy { ($f:expr) => { $crate::SsrRenderer::new().render_lazy($f) }; } pub struct SsrRenderer { inner: bumpalo::Bump, } impl Default for SsrRenderer { fn default() -> Self { Self { inner: bumpalo::Bump::new(), } } } impl SsrRenderer { pub fn new() -> Self { SsrRenderer::default() } pub fn render_lazy<'a, F: FnOnce(NodeFactory<'a>) -> VNode<'a>>( &'a self, f: LazyNodes<'a, F>, ) -> String { let factory = NodeFactory::new(&self.inner); let root = f.into_vnode(factory); format!( "{:}", TextRenderer { cfg: SsrConfig::default(), root: &root, vdom: None } ) } } pub fn render_vnode(vnode: &VNode, string: &mut String) {} pub fn render_vdom(dom: &VirtualDom, cfg: impl FnOnce(SsrConfig) -> SsrConfig) -> String { format!( "{:}", TextRenderer::from_vdom(dom, cfg(SsrConfig::default())) ) } pub fn render_vdom_scope(vdom: &VirtualDom, scope: ScopeId) -> Option { Some(format!( "{:}", TextRenderer { cfg: SsrConfig::default(), root: vdom.get_scope(scope).unwrap().root_node(), vdom: Some(vdom) } )) } /// A configurable text renderer for the Dioxus VirtualDOM. /// /// /// ## Details /// /// This uses the `Formatter` infrastructure so you can write into anything that supports `write_fmt`. We can't accept /// any generic writer, so you need to "Display" the text renderer. This is done through `format!` or `format_args!` /// /// ## Example /// ```ignore /// static App: FC<()> = |(cx, props)|cx.render(rsx!(div { "hello world" })); /// let mut vdom = VirtualDom::new(App); /// vdom.rebuild(); /// /// let renderer = TextRenderer::new(&vdom); /// let output = format!("{}", renderer); /// assert_eq!(output, "
hello world
"); /// ``` pub struct TextRenderer<'a, 'b> { vdom: Option<&'a VirtualDom>, root: &'b VNode<'a>, cfg: SsrConfig, } impl Display for TextRenderer<'_, '_> { fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result { self.html_render(self.root, f, 0) } } impl<'a> TextRenderer<'a, '_> { pub fn from_vdom(vdom: &'a VirtualDom, cfg: SsrConfig) -> Self { Self { cfg, root: vdom.base_scope().root_node(), vdom: Some(vdom), } } fn html_render(&self, node: &VNode, f: &mut std::fmt::Formatter, il: u16) -> std::fmt::Result { match &node { VNode::Text(text) => { if self.cfg.indent { for _ in 0..il { write!(f, " ")?; } } write!(f, "{}", text.text)? } VNode::Anchor(anchor) => { // if self.cfg.indent { for _ in 0..il { write!(f, " ")?; } } write!(f, "")?; } VNode::Element(el) => { if self.cfg.indent { for _ in 0..il { write!(f, " ")?; } } write!(f, "<{}", el.tag_name)?; let mut attr_iter = el.attributes.iter().peekable(); while let Some(attr) = attr_iter.next() { match attr.namespace { None => write!(f, " {}=\"{}\"", attr.name, attr.value)?, Some(ns) => { // write the opening tag write!(f, " {}=\"", ns)?; let mut cur_ns_el = attr; 'ns_parse: loop { write!(f, "{}:{};", cur_ns_el.name, cur_ns_el.value)?; match attr_iter.peek() { Some(next_attr) if next_attr.namespace == Some(ns) => { cur_ns_el = attr_iter.next().unwrap(); } _ => break 'ns_parse, } } // write the closing tag write!(f, "\"")?; } } } // we write the element's id as a data attribute // // when the page is loaded, the `querySelectorAll` will be used to collect all the nodes, and then add // them interpreter's stack match (self.cfg.pre_render, node.try_mounted_id()) { (true, Some(id)) => { write!(f, " dio_el=\"{}\"", id)?; // for listener in el.listeners { // write the listeners } } _ => {} } match self.cfg.newline { true => write!(f, ">\n")?, false => write!(f, ">")?, } for child in el.children { self.html_render(child, f, il + 1)?; } if self.cfg.newline { write!(f, "\n")?; } if self.cfg.indent { for _ in 0..il { write!(f, " ")?; } } write!(f, "", el.tag_name)?; if self.cfg.newline { write!(f, "\n")?; } } VNode::Fragment(frag) => { for child in frag.children { self.html_render(child, f, il + 1)?; } } VNode::Component(vcomp) => { let idx = vcomp.associated_scope.get().unwrap(); match (self.vdom, self.cfg.skip_components) { (Some(vdom), false) => { let new_node = vdom.get_scope(idx).unwrap().root_node(); self.html_render(new_node, f, il + 1)?; } _ => { // render the component by name } } } VNode::Suspended { .. } => { // we can't do anything with suspended nodes } } Ok(()) } } pub struct SsrConfig { /// currently not supported - control if we indent the HTML output indent: bool, /// Control if elements are written onto a new line newline: bool, /// Choose to write ElementIDs into elements so the page can be re-hydrated later on pre_render: bool, // Currently not implemented // Don't proceed onto new components. Instead, put the name of the component. // TODO: components don't have names :( skip_components: bool, } impl Default for SsrConfig { fn default() -> Self { Self { indent: false, pre_render: false, newline: false, skip_components: false, } } } impl SsrConfig { pub fn indent(mut self, a: bool) -> Self { self.indent = a; self } pub fn newline(mut self, a: bool) -> Self { self.newline = a; self } pub fn pre_render(mut self, a: bool) -> Self { self.pre_render = a; self } pub fn skip_components(mut self, a: bool) -> Self { self.skip_components = a; self } } #[cfg(test)] mod tests { use super::*; use dioxus_core as dioxus; use dioxus_core::prelude::*; use dioxus_core_macro::*; use dioxus_html as dioxus_elements; static SIMPLE_APP: FC<()> = |(cx, props)| { cx.render(rsx!(div { "hello world!" })) }; static SLIGHTLY_MORE_COMPLEX: FC<()> = |(cx, props)| { cx.render(rsx! { div { title: "About W3Schools" {(0..20).map(|f| rsx!{ div { title: "About W3Schools" style: "color:blue;text-align:center" class: "About W3Schools" p { title: "About W3Schools" "Hello world!: {f}" } } })} } }) }; static NESTED_APP: FC<()> = |(cx, props)| { cx.render(rsx!( div { SIMPLE_APP {} } )) }; static FRAGMENT_APP: FC<()> = |(cx, props)| { cx.render(rsx!( div { "f1" } div { "f2" } div { "f3" } div { "f4" } )) }; #[test] fn to_string_works() { let mut dom = VirtualDom::new(SIMPLE_APP); dom.rebuild(); dbg!(render_vdom(&dom, |c| c)); } #[test] fn hydration() { let mut dom = VirtualDom::new(NESTED_APP); dom.rebuild(); dbg!(render_vdom(&dom, |c| c.pre_render(true))); } #[test] fn nested() { let mut dom = VirtualDom::new(NESTED_APP); dom.rebuild(); dbg!(render_vdom(&dom, |c| c)); } #[test] fn fragment_app() { let mut dom = VirtualDom::new(FRAGMENT_APP); dom.rebuild(); dbg!(render_vdom(&dom, |c| c)); } #[test] fn write_to_file() { use std::fs::File; use std::io::Write; let mut file = File::create("index.html").unwrap(); let mut dom = VirtualDom::new(SLIGHTLY_MORE_COMPLEX); dom.rebuild(); file.write_fmt(format_args!( "{}", TextRenderer::from_vdom(&dom, SsrConfig::default()) )) .unwrap(); } #[test] fn styles() { static STLYE_APP: FC<()> = |(cx, props)| { cx.render(rsx! { div { style: { color: "blue", font_size: "46px" } } }) }; let mut dom = VirtualDom::new(STLYE_APP); dom.rebuild(); dbg!(render_vdom(&dom, |c| c)); } #[test] fn lazy() { let p1 = SsrRenderer::new().render_lazy(rsx! { div { "ello" } }); let p2 = render_lazy!(rsx! { div { "ello" } }); assert_eq!(p1, p2); } }