Zero overhead component composition using async iterator (stream)
The current goal is composing gui components easily without performance degrade in Rust.
UI components are retained. It only need to recalculate some layout when properties are changed.
States are never updated unless some event occur from outside.
Using stream, we can poll for events from outside and apply update of changed value simultaneously.
See async_component/examples/example.rs
for simple example.
See examples/gui-demo
project for example using with gui(winit, raqote, pixels).
```Rust use async_component::Component;
struct CounterComponent {
// State must be wrapped with StateCell
#[state(Self::oncounterupdate)]
counter: StateCell
// Stream
#[stream(Self::on_counter_recv)]
counter_recv: Receiver<i32>,
}
impl CounterComponent { fn oncounterupdate(&mut self) { println!("Counter updated to: {}", *self.counter); }
fn on_counter_recv(&mut self, counter: i32) {
*self.sub_component.counter = counter;
}
} ```
Running this component stream will print initial value first and print changed value if new values are sent through channel.
Counter updated to: 0
Counter updated to: ...
Component
derive macro will generate Stream
trait implementation for CounterComponent
like below.
```Rust
impl Stream for CounterComponent {
type Item = ComponentPollFlags;
fn poll_next(self: Pin<&mut Self>, cx: &mut Context) -> Poll<Option<Self::Item>> {
let mut result = ComponentPollFlags::empty();
if StateCell::poll_changed(
Pin::new(&mut self.counter),
cx
).is_ready() {
Self::on_counter_update(&mut self);
result |= ComponentPollFlags::STATE;
}
if let Poll::Ready(Some(recv)) = Stream::poll_next(Pin::new(&mut self.counter_recv), cx) {
Self::on_counter_recv(&mut self, recv);
result |= ComponentPollFlags::STREAM;
}
if result.is_empty() {
Poll::Pending
} else {
Poll::Ready(Some(result))
}
}
} ```