A positional memoization runtime similar to Jetpack Compose Runtime.
Below example show how to build a declarative GUI using compose-rt
toml
[dependencies]
compose-rt = "0.2"
downcast-rs = "1.2.0"
log = "0.4"
env_logger = "0.6"
fake = "2.4"
```rust
use composert::{ComposeNode, Composer}; use downcastrs::{impl_downcast, Downcast}; use fake::{Fake, Faker}; use std::{ any::TypeId, cell::{RefCell, RefMut}, 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
pub fn MoviesScreen(cx: Context, movies: Vec
pub fn MovieOverview(cx: Context, movie: &Movie) { Column(cx, |cx| { Text(cx, &movie.name); Image(cx, &movie.img_url); RandomRenderObject(cx, &movie.name) }) }
fn main() { // Setup logging envlogger::Builder::fromdefaultenv() .filterlevel(log::LevelFilter::Trace) .init();
// define root compose
let root_fn = |cx: Context, movies| MoviesScreen(cx, movies);
let mut cx = Composer::new(10);
// first run
let movies = vec![Movie::new(1, "A", "IMG_A"), Movie::new(2, "B", "IMG_B")];
root_fn(&mut cx, movies);
println!("{:#?}", cx);
// reset composer cursor, etc. for recompose
cx = cx.finalize();
// 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"),
];
root_fn(&mut cx, movies);
println!("{:#?}", cx);
}
////////////////////////////////////////////////////////////////////////////
// Components - Usage of compose-rt
////////////////////////////////////////////////////////////////////////////
type Context<'a> = &'a mut Composer
pub fn Column
pub fn Text(cx: Context, text: impl AsRef
pub fn Image(cx: Context, url: impl AsRef
pub fn RandomRenderObject(cx: Context, text: impl AsRef
if ty_id == TypeId::of::<RenderLabel>() {
let mut label = RefMut::map(n, |x| x.downcast_mut::<RenderLabel>().unwrap());
label.0 = t.to_string();
} else if ty_id == TypeId::of::<RenderImage>() {
let mut img = RefMut::map(n, |x| x.downcast_mut::<RenderImage>().unwrap());
img.0 = t.to_string();
};
},
);
}
//////////////////////////////////////////////////////////////////////////// // Rendering backend - Not scope of compose-rt //////////////////////////////////////////////////////////////////////////// pub trait Node: Debug + Downcast + Unpin {} impl_downcast!(Node);
impl
impl
impl<'a> ComposeNode for &'a mut dyn Node {
fn castmut
pub trait RenderObject: Node {} impl_downcast!(RenderObject);
impl Into
pub struct RenderFlex {
children: Vec
impl Debug for RenderFlex { fn fmt(&self, f: &mut std::fmt::Formatter<'>) -> std::fmt::Result { // trim debug print f.debugstruct("RenderFlex") .field("children_count", &self.children.len()) .finish() } }
impl RenderFlex { pub fn new() -> Self { RenderFlex { children: Vec::new(), } } }
impl RenderObject for RenderFlex {}
pub struct RenderLabel(String); impl RenderObject for RenderLabel {}
pub struct RenderImage(String); impl RenderObject for RenderImage {} ```