compose-rt

Rust Docs Status Latest Version

A positional memoization runtime similar to Jetpack Compose Runtime.

example

Below example show how to build a declarative GUI using compose-rt

toml [dependencies] compose-rt = "0.7" downcast-rs = "1.2" log = "0.4" env_logger = "0.6" fake = "2.4"

```rust

![allow(nonsnakecase)]

use composert::{compose, ComposeNode, Composer, Recomposer}; use downcastrs::impl_downcast; use fake::{Fake, Faker}; use std::{cell::RefCell, fmt::Debug, rc::Rc};

//////////////////////////////////////////////////////////////////////////// // User application //////////////////////////////////////////////////////////////////////////// pub struct Movie { id: usize, name: String, imgurl: String, } impl Movie { pub fn new(id: usize, name: impl Into, imgurl: impl Into) -> Self { Movie { id, name: name.into(), imgurl: imgurl.into(), } } }

[compose]

pub fn MoviesScreen(movies: &Vec) { Column(cx, |cx| { for movie in movies { cx.tag(movie.id, |cx| MovieOverview(cx, &movie)); } }); }

[compose]

pub fn MovieOverview(movie: &Movie) { Column(cx, |cx| { Text(cx, &movie.name); Image(cx, &movie.img_url); RandomRenderObject(cx, &movie.name);

    let count = cx.remember(Rc::new(RefCell::new(0usize)));
    Text(cx, format!("compose count {}", count.borrow()));
    *count.borrow_mut() += 1;
});

}

fn main() { // Setup logging envlogger::Builder::fromdefaultenv() .filterlevel(log::LevelFilter::Trace) .init();

// define root compose
let root_fn = |cx: &mut Composer, movies| MoviesScreen(cx, movies);

let mut recomposer = Recomposer::new();

// first run
let movies = vec![Movie::new(1, "A", "IMG_A"), Movie::new(2, "B", "IMG_B")];
recomposer.compose(|cx| {
    root_fn(cx, &movies);
});

if let Some(root) = recomposer.root::<Rc<RefCell<RenderFlex>>>() {
    // call paint of render tree
    let mut context = PaintContext::new();
    root.borrow().paint(&mut context);
}

// rerun with new input
let movies = vec![
    Movie::new(1, "AA", "IMG_AA"),
    Movie::new(3, "C", "IMG_C"),
    Movie::new(2, "B", "IMG_B"),
];
recomposer.compose(|cx| {
    root_fn(cx, &movies);
});

// end compose, Recomposer allow you to access root
if let Some(root) = recomposer.root::<Rc<RefCell<RenderFlex>>>() {
    // call paint of render tree
    let mut context = PaintContext::new();
    root.borrow().paint(&mut context);
}

}

//////////////////////////////////////////////////////////////////////////// // Components - Usage of compose-rt ////////////////////////////////////////////////////////////////////////////

[compose(skipinjectcx = true)]

pub fn Column(cx: &mut Composer, content: C) where C: Fn(&mut Composer), { cx.groupusechildren( || Rc::new(RefCell::new(RenderFlex::new())), content, |node, children| { let mut flex = node.borrowmut(); flex.children.clear(); for child in children { if let Some(c) = child.downcastref::>>().cloned() { flex.children.push(c); } else if let Some(c) = child.downcastref::>>().cloned() { flex.children.push(c); } else if let Some(c) = child.downcastref::>>().cloned() { flex.children.push(c); } else if let Some(c) = child .downcastref::>>() .cloned() { flex.children.push(c); } } }, || false, || {}, |_| {}, ); }

[compose(skipinjectcx = true)]

pub fn Text(cx: &mut Composer, text: impl AsRef) { let text = text.asref(); cx.memo( || Rc::new(RefCell::new(RenderLabel(text.tostring()))), |n| n.borrow().0 == text, |n| { let mut n = n.borrowmut(); n.0 = text.tostring(); }, || {}, ); }

[compose(skipinjectcx = true)]

pub fn Image(cx: &mut Composer, url: impl AsRef) { let url = url.asref(); cx.memo( || Rc::new(RefCell::new(RenderImage(url.tostring()))), |n| n.borrow().0 == url, |n| { let mut n = n.borrowmut(); n.0 = url.tostring(); }, || {}, ); }

[compose(skipinjectcx = true)]

pub fn RandomRenderObject(cx: &mut Composer, text: impl AsRef) { let t = text.asref(); cx.memo( || { let obj: Rc> = if Faker.fake::() { let url = format!("http://image.com/{}.png", t); Rc::new(RefCell::new(RenderImage(url))) } else { Rc::new(RefCell::new(RenderLabel(t.tostring()))) }; obj }, || false, |n| { if let Some(label) = n.borrowmut().downcastmut::() { label.0 = t.tostring(); } if let Some(img) = n.borrowmut().downcastmut::() { let url = format!("http://image.com/{}.png", t); img.0 = url; } }, || {}, ); }

//////////////////////////////////////////////////////////////////////////// // Rendering backend - Not scope of compose-rt //////////////////////////////////////////////////////////////////////////// pub struct PaintContext { depth: usize, } impl PaintContext { pub fn new() -> Self { Self { depth: 0 } } }

pub trait RenderObject: Debug + ComposeNode { fn paint(&self, context: &mut PaintContext); } impl_downcast!(RenderObject);

[derive(Debug)]

pub struct RenderFlex { children: Vec>>, }

impl RenderFlex { pub fn new() -> Self { RenderFlex { children: Vec::new(), } } }

impl RenderObject for RenderFlex { fn paint(&self, context: &mut PaintContext) { println!( "{}", "\t".repeat(context.depth)); } }

[derive(Debug)]

pub struct RenderLabel(String); impl RenderObject for RenderLabel { fn paint(&self, context: &mut PaintContext) { println!("{}", "\t".repeat(context.depth), self.0); } }

[derive(Debug)]

pub struct RenderImage(String); impl RenderObject for RenderImage { fn paint(&self, context: &mut PaintContext) { println!("{}", "\t".repeat(context.depth), self.0); } } ```