2021-07-29 01:46:53 +00:00
|
|
|
//!
|
|
|
|
//!
|
|
|
|
//!
|
|
|
|
//!
|
2021-01-21 07:25:44 +00:00
|
|
|
//! 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.
|
|
|
|
|
2021-07-11 21:24:47 +00:00
|
|
|
use std::fmt::{Display, Formatter};
|
2021-01-21 07:25:44 +00:00
|
|
|
|
2021-07-13 03:44:20 +00:00
|
|
|
use dioxus_core::*;
|
2021-07-11 18:49:52 +00:00
|
|
|
|
2021-07-18 07:54:42 +00:00
|
|
|
pub fn render_vnode(vnode: &VNode, string: &mut String) {}
|
|
|
|
|
2021-07-29 01:46:53 +00:00
|
|
|
pub fn render_vdom(dom: &VirtualDom, cfg: impl FnOnce(SsrConfig) -> SsrConfig) -> String {
|
|
|
|
format!(
|
|
|
|
"{:}",
|
|
|
|
TextRenderer::from_vdom(dom, cfg(SsrConfig::default()))
|
|
|
|
)
|
2021-07-18 07:54:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn render_vdom_scope(vdom: &VirtualDom, scope: ScopeId) -> Option<String> {
|
|
|
|
Some(format!(
|
|
|
|
"{:}",
|
|
|
|
TextRenderer {
|
|
|
|
cfg: SsrConfig::default(),
|
2021-07-24 04:29:23 +00:00
|
|
|
root: vdom.get_scope(scope).unwrap().root(),
|
2021-07-18 07:54:42 +00:00
|
|
|
vdom: Some(vdom)
|
|
|
|
}
|
|
|
|
))
|
2021-03-23 03:52:54 +00:00
|
|
|
}
|
|
|
|
|
2021-07-11 18:49:52 +00:00
|
|
|
/// A configurable text renderer for the Dioxus VirtualDOM.
|
|
|
|
///
|
2021-01-21 07:25:44 +00:00
|
|
|
///
|
2021-07-11 18:49:52 +00:00
|
|
|
/// ## Details
|
2021-01-21 07:25:44 +00:00
|
|
|
///
|
2021-07-11 18:49:52 +00:00
|
|
|
/// 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!`
|
2021-01-21 07:25:44 +00:00
|
|
|
///
|
2021-07-11 19:17:55 +00:00
|
|
|
/// ## Example
|
|
|
|
/// ```ignore
|
2021-07-29 01:46:53 +00:00
|
|
|
/// static App: FC<()> = |cx| cx.render(rsx!(div { "hello world" }));
|
2021-07-11 19:17:55 +00:00
|
|
|
/// let mut vdom = VirtualDom::new(App);
|
|
|
|
/// vdom.rebuild_in_place();
|
2021-07-11 18:49:52 +00:00
|
|
|
///
|
2021-07-11 19:17:55 +00:00
|
|
|
/// let renderer = TextRenderer::new(&vdom);
|
|
|
|
/// let output = format!("{}", renderer);
|
|
|
|
/// assert_eq!(output, "<div>hello world</div>");
|
|
|
|
/// ```
|
2021-07-11 18:49:52 +00:00
|
|
|
pub struct TextRenderer<'a> {
|
2021-07-18 07:54:42 +00:00
|
|
|
vdom: Option<&'a VirtualDom>,
|
|
|
|
root: &'a VNode<'a>,
|
2021-07-11 19:17:55 +00:00
|
|
|
cfg: SsrConfig,
|
2021-01-21 07:25:44 +00:00
|
|
|
}
|
|
|
|
|
2021-07-18 07:54:42 +00:00
|
|
|
impl Display for TextRenderer<'_> {
|
|
|
|
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
|
|
|
|
self.html_render(self.root, f, 0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-11 18:49:52 +00:00
|
|
|
impl<'a> TextRenderer<'a> {
|
2021-07-29 01:46:53 +00:00
|
|
|
pub fn from_vdom(vdom: &'a VirtualDom, cfg: SsrConfig) -> Self {
|
2021-07-11 19:17:55 +00:00
|
|
|
Self {
|
2021-07-29 01:46:53 +00:00
|
|
|
cfg,
|
2021-07-18 07:54:42 +00:00
|
|
|
root: vdom.base_scope().root(),
|
|
|
|
vdom: Some(vdom),
|
2021-07-11 19:17:55 +00:00
|
|
|
}
|
2021-01-21 07:25:44 +00:00
|
|
|
}
|
|
|
|
|
2021-07-11 23:31:07 +00:00
|
|
|
fn html_render(&self, node: &VNode, f: &mut std::fmt::Formatter, il: u16) -> std::fmt::Result {
|
2021-07-13 03:44:20 +00:00
|
|
|
match &node.kind {
|
2021-08-20 14:34:41 +00:00
|
|
|
VNode::Text(text) => {
|
2021-07-11 23:31:07 +00:00
|
|
|
if self.cfg.indent {
|
|
|
|
for _ in 0..il {
|
|
|
|
write!(f, " ")?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
write!(f, "{}", text.text)?
|
|
|
|
}
|
2021-08-20 14:34:41 +00:00
|
|
|
VNode::Anchor(anchor) => {
|
2021-07-30 21:04:04 +00:00
|
|
|
//
|
|
|
|
if self.cfg.indent {
|
|
|
|
for _ in 0..il {
|
|
|
|
write!(f, " ")?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
write!(f, "<!-- -->")?;
|
|
|
|
}
|
2021-08-20 14:34:41 +00:00
|
|
|
VNode::Element(el) => {
|
2021-07-11 23:31:07 +00:00
|
|
|
if self.cfg.indent {
|
|
|
|
for _ in 0..il {
|
|
|
|
write!(f, " ")?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-11 18:49:52 +00:00
|
|
|
write!(f, "<{}", el.tag_name)?;
|
2021-07-15 15:06:52 +00:00
|
|
|
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, "\"")?;
|
|
|
|
}
|
|
|
|
}
|
2021-07-11 18:49:52 +00:00
|
|
|
}
|
2021-07-15 15:06:52 +00:00
|
|
|
|
2021-07-29 01:46:53 +00:00
|
|
|
// 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
|
2021-07-30 21:04:04 +00:00
|
|
|
match (self.cfg.pre_render, node.try_direct_id()) {
|
2021-07-29 01:46:53 +00:00
|
|
|
(true, Some(id)) => {
|
|
|
|
write!(f, " dio_el=\"{}\"", id)?;
|
|
|
|
//
|
|
|
|
for listener in el.listeners {
|
|
|
|
// write the listeners
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
|
2021-07-11 19:17:55 +00:00
|
|
|
match self.cfg.newline {
|
|
|
|
true => write!(f, ">\n")?,
|
|
|
|
false => write!(f, ">")?,
|
|
|
|
}
|
|
|
|
|
2021-07-11 18:49:52 +00:00
|
|
|
for child in el.children {
|
2021-07-11 23:31:07 +00:00
|
|
|
self.html_render(child, f, il + 1)?;
|
2021-07-11 18:49:52 +00:00
|
|
|
}
|
2021-07-11 23:31:07 +00:00
|
|
|
|
|
|
|
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")?;
|
2021-07-11 19:17:55 +00:00
|
|
|
}
|
|
|
|
}
|
2021-08-20 14:34:41 +00:00
|
|
|
VNode::Fragment(frag) => {
|
2021-07-11 19:17:55 +00:00
|
|
|
for child in frag.children {
|
2021-07-11 23:31:07 +00:00
|
|
|
self.html_render(child, f, il + 1)?;
|
2021-07-11 19:17:55 +00:00
|
|
|
}
|
2021-07-11 18:49:52 +00:00
|
|
|
}
|
2021-08-20 14:34:41 +00:00
|
|
|
VNode::Component(vcomp) => {
|
2021-07-11 19:17:55 +00:00
|
|
|
let idx = vcomp.ass_scope.get().unwrap();
|
2021-07-29 01:46:53 +00:00
|
|
|
match (self.vdom, self.cfg.skip_components) {
|
|
|
|
(Some(vdom), false) => {
|
|
|
|
let new_node = vdom.get_scope(idx).unwrap().root();
|
|
|
|
self.html_render(new_node, f, il + 1)?;
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
// render the component by name
|
|
|
|
}
|
2021-07-18 07:54:42 +00:00
|
|
|
}
|
2021-07-11 18:49:52 +00:00
|
|
|
}
|
2021-08-20 14:34:41 +00:00
|
|
|
VNode::Suspended { .. } => {
|
2021-07-15 07:38:09 +00:00
|
|
|
// we can't do anything with suspended nodes
|
|
|
|
}
|
2021-07-11 18:49:52 +00:00
|
|
|
}
|
2021-07-11 19:17:55 +00:00
|
|
|
Ok(())
|
2021-01-21 07:25:44 +00:00
|
|
|
}
|
2021-07-11 18:49:52 +00:00
|
|
|
}
|
2021-01-21 07:25:44 +00:00
|
|
|
|
2021-07-29 01:46:53 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-11 18:49:52 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
2021-01-21 07:25:44 +00:00
|
|
|
|
2021-07-11 18:49:52 +00:00
|
|
|
use dioxus_core as dioxus;
|
2021-07-11 21:24:47 +00:00
|
|
|
use dioxus_core::prelude::*;
|
2021-07-11 18:49:52 +00:00
|
|
|
use dioxus_html as dioxus_elements;
|
2021-02-03 07:26:04 +00:00
|
|
|
|
2021-07-29 01:46:53 +00:00
|
|
|
static SIMPLE_APP: FC<()> = |cx| {
|
2021-07-11 18:49:52 +00:00
|
|
|
cx.render(rsx!(div {
|
|
|
|
"hello world!"
|
|
|
|
}))
|
|
|
|
};
|
2021-01-21 07:25:44 +00:00
|
|
|
|
2021-07-29 01:46:53 +00:00
|
|
|
static SLIGHTLY_MORE_COMPLEX: FC<()> = |cx| {
|
2021-07-11 18:49:52 +00:00
|
|
|
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}"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
};
|
|
|
|
|
2021-07-29 01:46:53 +00:00
|
|
|
static NESTED_APP: FC<()> = |cx| {
|
2021-07-11 21:24:47 +00:00
|
|
|
cx.render(rsx!(
|
|
|
|
div {
|
|
|
|
SIMPLE_APP {}
|
|
|
|
}
|
|
|
|
))
|
|
|
|
};
|
2021-07-29 01:46:53 +00:00
|
|
|
static FRAGMENT_APP: FC<()> = |cx| {
|
2021-07-11 21:24:47 +00:00
|
|
|
cx.render(rsx!(
|
|
|
|
div { "f1" }
|
|
|
|
div { "f2" }
|
|
|
|
div { "f3" }
|
|
|
|
div { "f4" }
|
|
|
|
))
|
|
|
|
};
|
|
|
|
|
2021-07-11 18:49:52 +00:00
|
|
|
#[test]
|
2021-07-11 21:24:47 +00:00
|
|
|
fn to_string_works() {
|
2021-07-11 18:49:52 +00:00
|
|
|
let mut dom = VirtualDom::new(SIMPLE_APP);
|
|
|
|
dom.rebuild_in_place().expect("failed to run virtualdom");
|
2021-07-29 01:46:53 +00:00
|
|
|
dbg!(render_vdom(&dom, |c| c));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn hydration() {
|
|
|
|
let mut dom = VirtualDom::new(NESTED_APP);
|
|
|
|
dom.rebuild_in_place().expect("failed to run virtualdom");
|
|
|
|
dbg!(render_vdom(&dom, |c| c.pre_render(true)));
|
2021-01-21 07:25:44 +00:00
|
|
|
}
|
|
|
|
|
2021-07-11 18:49:52 +00:00
|
|
|
#[test]
|
2021-07-11 21:24:47 +00:00
|
|
|
fn nested() {
|
|
|
|
let mut dom = VirtualDom::new(NESTED_APP);
|
|
|
|
dom.rebuild_in_place().expect("failed to run virtualdom");
|
2021-07-29 01:46:53 +00:00
|
|
|
dbg!(render_vdom(&dom, |c| c));
|
2021-07-11 21:24:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn fragment_app() {
|
|
|
|
let mut dom = VirtualDom::new(FRAGMENT_APP);
|
|
|
|
dom.rebuild_in_place().expect("failed to run virtualdom");
|
2021-07-29 01:46:53 +00:00
|
|
|
dbg!(render_vdom(&dom, |c| c));
|
2021-07-11 21:24:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn write_to_file() {
|
2021-07-11 18:49:52 +00:00
|
|
|
use std::fs::File;
|
2021-07-11 21:24:47 +00:00
|
|
|
use std::io::Write;
|
2021-07-11 18:49:52 +00:00
|
|
|
|
|
|
|
let mut file = File::create("index.html").unwrap();
|
|
|
|
|
2021-07-11 19:17:55 +00:00
|
|
|
let mut dom = VirtualDom::new(SLIGHTLY_MORE_COMPLEX);
|
2021-07-11 18:49:52 +00:00
|
|
|
dom.rebuild_in_place().expect("failed to run virtualdom");
|
|
|
|
|
2021-07-29 01:46:53 +00:00
|
|
|
file.write_fmt(format_args!(
|
|
|
|
"{}",
|
|
|
|
TextRenderer::from_vdom(&dom, SsrConfig::default())
|
|
|
|
))
|
|
|
|
.unwrap();
|
2021-01-16 04:25:29 +00:00
|
|
|
}
|
2021-07-15 15:06:52 +00:00
|
|
|
|
2021-07-29 01:46:53 +00:00
|
|
|
#[test]
|
|
|
|
fn styles() {
|
|
|
|
static STLYE_APP: FC<()> = |cx| {
|
|
|
|
cx.render(rsx! {
|
|
|
|
div { style: { color: "blue", font_size: "46px" } }
|
|
|
|
})
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut dom = VirtualDom::new(STLYE_APP);
|
|
|
|
dom.rebuild_in_place().expect("failed to run virtualdom");
|
|
|
|
dbg!(render_vdom(&dom, |c| c));
|
|
|
|
}
|
2021-01-16 04:25:29 +00:00
|
|
|
}
|