Async component

Zero overhead reactive programming

The current goal is composing gui components easily without performance degrade in Rust.

Core concepts

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.

Example

See async_component/examples/example.rs for simple example.

See examples/gui-demo project for example using with gui(winit, raqote, pixels).

Code

```Rust use async_component::AsyncComponent;

[derive(Debug, 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: ...

Expanded

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 { 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;
    }

    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)
    }
}

} ```