Flexible and easy unidirectional store-pattern container for state management with Dependency Injection for Frontend app on .NET or JS/TS.
DEMO with React in Typescript
You can define stores inspired by MVU patterns such as Flux and Elm to observe state changes more detail.
Some are inspired by Elm and MVU. And Redux and Flux pattern are same too, but memento is not Redux and Flux.
- Less boilarplate and simple usage
- Is not flux or redux
- Observe detailed status with command patterns and makes it easier to monitor what happened within the application
- Immutable and Unidirectional data flow
- Multiple stores but manged by single provider, so can observe and manage as one state tree
- Less rules have been established
- Fragile because there are fewer established rules than Redux and Flux
Note the concept is a bit different from Flux and Redux
- State should always be read-only.
- To change state our app should Dispatch via Reducer in the action method
- Every Reducer that processes in the action will create new state to reflect the old state combined with the changes expected for the action.
- The UI then uses the new state to render its display.
This is an C# and Blazor example that implements counter.
Store
using Memento.Core;
using System.Collections.Immutable;
using static Blazor.Sample.Stores.AsyncCounterCommands;
namespace Blazor.Sample.Stores;
public record AsyncCounterState {
public int Count { get; init; } = 0;
public bool IsLoading { get; init; } = false;
public ImmutableArray<int> Histories { get; init; } = ImmutableArray.Create<int>();
}
public record AsyncCounterCommands: Command {
public record CountUp : AsyncCounterCommands;
public record Increment : AsyncCounterCommands;
public record SetCount(int Count) : AsyncCounterCommands;
public record BeginLoading : AsyncCounterCommands;
}
public class AsyncCounterStore : Store<AsyncCounterState, AsyncCounterCommands> {
public AsyncCounterStore() : base(() => new(), Reducer) { }
static AsyncCounterState Reducer(AsyncCounterState state, AsyncCounterCommands command) {
return command switch {
CountUp => state with {
Count = state.Count + 1,
IsLoading = false,
Histories = state.Histories.Add(state.Count + 1),
},
SetCount payload => state with {
Count = payload.Count,
},
Increment => state with {
Count = state.Count + 1,
},
BeginLoading => state with {
IsLoading = true,
},
_ => throw new CommandNotHandledException(command),
};
}
public async Task CountUpAsync() {
Dispatch(new BeginLoading());
await Task.Delay(800);
Dispatch(new CountUp());
}
public void CountUpManyTimes(int count) {
for (int i = 0; i < count; i++) {
Dispatch(new Increment());
}
}
public void SetCount(int c) {
Dispatch(new SetCount(c));
}
}
Razor view
@using Blazor.Sample.Stores
@using System.Text.Json
@page "/counter"
@inherits ObserverComponet
@inject AsyncCounterStore AsyncCounterStore
<PageTitle>Counter</PageTitle>
<h1>Async Counter</h1>
<p role="status">Current count: @AsyncCounterStore.State.Count</p>
<p role="status">Loading: @AsyncCounterStore.State.IsLoading</p>
<p role="status" class="mb-0">History</p>
<div class="d-flex">
[
@foreach (var item in string.Join(", ", AsyncCounterStore.State.Histories)) {
@item
}
]
</div>
<button class="mt-3 btn btn-primary" @onclick="IncrementCount">Count up</button>
<button class="mt-3 btn btn-primary" @onclick="CountupMany">Count up 10000 times</button>
@code {
async Task IncrementCount() {
await this.AsyncCounterStore.CountUpAsync();
}
void CountupMany() {
this.AsyncCounterStore.CountUpManyTimes(10000);
}
}
Lang | Framework |
---|---|
TS/JS | React |
C# | Blazor |
Package Name | Version | Lang | Platform | Package manager | Release Notes | Package provider |
---|---|---|---|---|---|---|
memento.core | 1.0.3 | TS/JS | node.js 14 or later | npm or yarn | Notes | npm |
memento.react | 1.0.4 | TS/JS | node.js 14 or later | npm or yarn | Notes | npm |
Memento.Core | 0.2.0 | C# | .NET 6 or later | Nuget | Notes | Nuget |
Memento.Blazor | 0.2.0 | Blazor | .NET 6 or later | Nuget | Notes | Nuget |
Basic Concept with Typescript/Javascript
Here is a demo site built with React in Typescript. DEMO
Designed with ♥ by le-nn. Licensed under the MIT License.