myFavShrimp / leptos-tea

The Elm Architecture state management for leptos.

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

leptos_tea

The Elm Architecture for leptos.

This crate is a particular strategy for state management in leptos. It follows the Elm architecture, but not strictly so, which allows mixing and matching with other state management approaches.

First, let's look at an example.

Example

use leptos::*;
use leptos_tea::Cmd;

#[derive(Default, leptos_tea::Model)]
struct CounterModel {
  counter: usize,
}

#[derive(Default)]
enum Msg {
  Increment,
  Decrement,
  #[default]
  Init,
}

fn update(model: UpdateCounterModel, msg: &Msg, _: Cmd<Msg>) {
  match msg {
    Msg::Increment => model.counter.update(|c| *c += 1),
    Msg::Decrement => model.counter.update(|c| *c -= 1),
    Msg::Init => {}
  }
}

#[component]
fn Counter(cx: Scope) -> impl IntoView {
  let (model, msg_dispatcher) = CounterModel::default().init(cx, update);

  view! { cx,
    <h1>{model.counter}</h1>
   <button on:click=move |_| msg_dispatcher(Msg::Decrement)>"-"</button>
   <button on:click=move |_| msg_dispatcher(Msg::Increment)>"+"</button>
  }
}

In the above example, we're annotating CounterModel with leptos_tea::Model, which will derive a few important things:

// Original struct, stays as-is
struct CounterModel {
  counter: usize,
}

// Model passed to the update function
struct UpdateCounterModel {
  counter: RwSignal<bool>,
}

// model passed to the component when you call `.init()`
struct ViewCounterModel {
  counter: ReadSignal<bool>,
}

impl CounterModel {
  // Initializes everything and starts listening for messages.
  // Msg::default() will be send to the update function when
  // called
  fn init<Msg: Default + 'static>(
    self,
    cx: Scope,
    update_fn: impl Fn(UpdateCounterModel, &Msg, Cmd<Msg>),
  ) -> (ViewCounterModel, SignalSetter<Msg>) {
    /* ... */
  }
}

You first need to create your CounterModel, however you'd like. In this case, we're using Default. Then you call .init(), which will return a tuple containing the read-only model, as well as a SignalSetter, which allows you to do msg_dispatcher(Msg::Blah) on nightly, or msg_dispatcher.set(Msg::Blah) on stable.

And that's how this crate and state management approach works.

Model nesting

Models can be nested inside one another like thus:

#[derive(leptos_tea::Model)]
struct Model {
  #[model]
  inner_model: InnerModel,
}

#[derive(leptos_tea::Model)]
struct InnerModel(/* ... */);

Important Node: Although this can be done, it is not recommended, because it leads to nested .update()/.with calls for each level of nesting. Instead, try and break out each nested model into it's own independent model, view, update. Nevertheless, sometimes this isn't desired or worth it, so the option is there in case you need it.

Limitations

leptos_tea::Model currently only supports tuple and field structs. Support will be added soon.

License: MIT

About

The Elm Architecture state management for leptos.


Languages

Language:Rust 100.0%