dioxus/src/render.rs

96 lines
2.7 KiB
Rust
Raw Normal View History

2022-01-01 04:53:37 +00:00
use dioxus::core::*;
2022-02-04 22:57:00 +00:00
use std::{collections::HashMap, io::Stdout};
2022-01-01 04:53:37 +00:00
use stretch2::{
geometry::Point,
prelude::{Layout, Size},
Stretch,
};
use tui::{
backend::CrosstermBackend,
buffer::Buffer,
layout::Rect,
style::Style as TuiStyle,
2022-02-04 22:57:00 +00:00
widgets::{Block, Widget},
2022-01-01 04:53:37 +00:00
};
use crate::TuiNode;
pub fn render_vnode<'a>(
frame: &mut tui::Frame<CrosstermBackend<Stdout>>,
layout: &Stretch,
layouts: &mut HashMap<ElementId, TuiNode<'a>>,
vdom: &'a VirtualDom,
node: &'a VNode<'a>,
) {
match node {
VNode::Fragment(f) => {
for child in f.children {
render_vnode(frame, layout, layouts, vdom, child);
}
return;
}
VNode::Component(vcomp) => {
let idx = vcomp.scope.get().unwrap();
let new_node = vdom.get_scope(idx).unwrap().root_node();
render_vnode(frame, layout, layouts, vdom, new_node);
return;
}
2022-01-01 04:53:37 +00:00
2022-02-04 20:52:01 +00:00
VNode::Placeholder(_) => return,
VNode::Element(_) | VNode::Text(_) => {}
2022-01-01 04:53:37 +00:00
}
let id = node.try_mounted_id().unwrap();
let node = layouts.remove(&id).unwrap();
let Layout { location, size, .. } = layout.layout(node.layout).unwrap();
let Point { x, y } = location;
let Size { width, height } = size;
match node.node {
VNode::Text(t) => {
#[derive(Default)]
struct Label<'a> {
text: &'a str,
}
impl<'a> Widget for Label<'a> {
fn render(self, area: Rect, buf: &mut Buffer) {
buf.set_string(area.left(), area.top(), self.text, TuiStyle::default());
}
}
// let s = Span::raw(t.text);
// Block::default().
let label = Label { text: t.text };
let area = Rect::new(*x as u16, *y as u16, *width as u16, *height as u16);
// the renderer will panic if a node is rendered out of range even if the size is zero
if area.width > 0 && area.height > 0 {
frame.render_widget(label, area);
}
2022-01-01 04:53:37 +00:00
}
VNode::Element(el) => {
let block = Block::default().style(node.block_style);
let area = Rect::new(*x as u16, *y as u16, *width as u16, *height as u16);
// the renderer will panic if a node is rendered out of range even if the size is zero
if area.width > 0 && area.height > 0 {
frame.render_widget(block, area);
}
2022-01-01 04:53:37 +00:00
for el in el.children {
render_vnode(frame, layout, layouts, vdom, el);
}
}
VNode::Fragment(_) => todo!(),
VNode::Component(_) => todo!(),
VNode::Placeholder(_) => todo!(),
}
}