Zero overhead reactive programming
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 like structure, 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::AsyncComponent;
struct CounterComponent {
// State must be wrapped with StateCell
#[state(Self::oncounterupdate)]
counter: StateCell
// Stream
// It iterates every queued items in single poll to prevent slowdown.
// If the stream is immediate and resolves indefinitely, the task will fall to infinite loop. See expanded code below.
#[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 AsyncComponent
trait implementation for CounterComponent
like below.
```Rust
impl AsyncComponent for CounterComponent {
fn poll_next(self: Pin<&mut Self>, cx: &mut Context) -> Poll
if StateCell::poll_changed(
Pin::new(&mut self.counter),
cx
).is_ready() {
Self::on_counter_update(&mut self);
result |= ComponentPollFlags::STATE;
}
while 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(result)
}
}
} ```