dioxus/src/lib.rs

173 lines
6 KiB
Rust
Raw Normal View History

2021-07-28 14:52:38 +00:00
use anyhow::Result;
2022-01-01 04:53:37 +00:00
use crossterm::{
2022-01-01 14:49:08 +00:00
event::{DisableMouseCapture, EnableMouseCapture, Event as TermEvent, KeyCode, KeyEvent},
2022-01-01 04:53:37 +00:00
execute,
terminal::{disable_raw_mode, enable_raw_mode, EnterAlternateScreen, LeaveAlternateScreen},
};
2022-01-01 14:49:08 +00:00
use dioxus::core::exports::futures_channel::mpsc::unbounded;
2021-07-28 14:52:38 +00:00
use dioxus::core::*;
2022-01-01 14:49:08 +00:00
use futures::{future::Either, pin_mut, StreamExt};
2022-01-01 06:08:31 +00:00
use std::{
collections::HashMap,
io,
time::{Duration, Instant},
};
2022-01-01 04:53:37 +00:00
use stretch2::{prelude::Size, Stretch};
use tui::{backend::CrosstermBackend, style::Style as TuiStyle, Terminal};
mod attributes;
mod layout;
mod render;
pub use attributes::*;
pub use layout::*;
pub use render::*;
pub struct TuiNode<'a> {
pub layout: stretch2::node::Node,
pub block_style: TuiStyle,
pub node: &'a VNode<'a>,
2021-07-28 14:52:38 +00:00
}
2022-01-01 14:49:08 +00:00
pub fn render_vdom(vdom: &mut VirtualDom) -> Result<()> {
2022-01-01 06:08:31 +00:00
// Setup input handling
2022-01-01 14:49:08 +00:00
let (tx, mut rx) = unbounded();
2022-01-01 06:08:31 +00:00
std::thread::spawn(move || {
let tick_rate = Duration::from_millis(100);
let mut last_tick = Instant::now();
loop {
// poll for tick rate duration, if no events, sent tick event.
let timeout = tick_rate
.checked_sub(last_tick.elapsed())
.unwrap_or_else(|| Duration::from_secs(0));
2022-01-01 14:49:08 +00:00
if crossterm::event::poll(timeout).unwrap() {
let evt = crossterm::event::read().unwrap();
tx.unbounded_send(InputEvent::UserInput(evt)).unwrap();
2022-01-01 06:08:31 +00:00
}
if last_tick.elapsed() >= tick_rate {
2022-01-01 14:49:08 +00:00
tx.unbounded_send(InputEvent::Tick).unwrap();
2022-01-01 06:08:31 +00:00
last_tick = Instant::now();
}
}
});
2022-01-01 14:49:08 +00:00
tokio::runtime::Builder::new_current_thread()
.enable_all()
.build()?
.block_on(async {
/*
Get the terminal to calcualte the layout from
*/
enable_raw_mode().unwrap();
let mut stdout = std::io::stdout();
execute!(stdout, EnterAlternateScreen, EnableMouseCapture).unwrap();
let backend = CrosstermBackend::new(io::stdout());
let mut terminal = Terminal::new(backend).unwrap();
terminal.clear().unwrap();
loop {
let dims = terminal.size().unwrap();
let width = dims.width;
let height = dims.height;
/*
-> collect all the nodes with their layout
-> solve their layout
-> render the nodes in the right place with tui/crosstream
-> while rendering, apply styling
use simd to compare lines for diffing?
todo: reuse the layout and node objects.
our work_with_deadline method can tell us which nodes are dirty.
*/
let mut layout = Stretch::new();
let mut nodes = HashMap::new();
let root_node = vdom.base_scope().root_node();
layout::collect_layout(&mut layout, &mut nodes, vdom, root_node);
/*
Compute the layout given th terminal size
*/
let node_id = root_node.try_mounted_id().unwrap();
let root_layout = nodes[&node_id].layout;
layout.compute_layout(
root_layout,
Size {
width: stretch2::prelude::Number::Defined(width as f32),
height: stretch2::prelude::Number::Defined(height as f32),
},
)?;
terminal.draw(|frame| {
//
render::render_vnode(frame, &layout, &mut nodes, vdom, root_node);
assert!(nodes.is_empty());
})?;
use futures::future::{select, Either};
{
let wait = vdom.wait_for_work();
pin_mut!(wait);
match select(wait, rx.next()).await {
Either::Left((a, b)) => {
// println!("work");
}
Either::Right((evt, o)) => {
// println!("event");
match evt.unwrap() {
InputEvent::UserInput(event) => match event {
TermEvent::Key(key) => {
match key.code {
KeyCode::Char('q') => {
disable_raw_mode()?;
execute!(
terminal.backend_mut(),
LeaveAlternateScreen,
DisableMouseCapture
)?;
terminal.show_cursor()?;
break;
}
_ => {} // handle event
}
}
TermEvent::Mouse(_) => {}
TermEvent::Resize(_, _) => {}
},
InputEvent::Tick => {} // tick
InputEvent::Close => {
break;
}
};
}
}
2022-01-01 06:08:31 +00:00
}
2022-01-01 14:49:08 +00:00
vdom.work_with_deadline(|| false);
2022-01-01 06:08:31 +00:00
}
2022-01-01 14:49:08 +00:00
disable_raw_mode()?;
execute!(
terminal.backend_mut(),
LeaveAlternateScreen,
DisableMouseCapture
)?;
terminal.show_cursor()?;
Ok(())
})
2021-07-28 14:52:38 +00:00
}
2022-01-01 06:08:31 +00:00
enum InputEvent {
2022-01-01 14:49:08 +00:00
UserInput(TermEvent),
2022-01-01 06:08:31 +00:00
Close,
Tick,
}