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;
|
|
|
|
}
|
2022-02-04 13:24:02 +00:00
|
|
|
|
|
|
|
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);
|
|
|
|
|
2022-02-05 02:19:17 +00:00
|
|
|
// 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);
|
|
|
|
|
2022-02-05 02:19:17 +00:00
|
|
|
// 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!(),
|
|
|
|
}
|
|
|
|
}
|