Home

Awesome

Memento

License: MIT

A simple client-side state management container for Blazor/.NET includes redo/undo and ReduxDevTools support.

<img src="./docs/Assets/ReduxDevTools.gif" width="800px" />

Japanese Link

使い方と紹介

Redux DevToolsについて

Basic Concept

We provides a Store that allows you to share state between components. All stores are managed by a single provider and can subscribe to state change notifications. Undirectional flow and immutable change of state provides a predictable architecture. In addition, we provide a store that easily implements Redo/Undo by managing in immutable states.

For patterns like Flux or MVU

Besides simple store pattern, we also provide patterns inspired by MVU patterns such as Flux and Elm. Since you should change the state via the Reducer, you can change the state based on stricter rules and observe the state in detail.

ReduxDevTools

Redux DevTools is supported. Redux DevTools is a tool for debugging application's state changes. State can be time traveled and history can be viewed in DevTools.

See docs for details of usage.

DEMO Page

https://le-nn.github.io/memento/

If you have ReduxDevTool installed, DevTool will launch automatically. You can do state history and time travel.

Documentation

https://le-nn.github.io/memento/docs/README.html

React or TS/JS bindings

Currently, moved to here https://github.com/le-nn/memento-js

Features

Concepts and Flow

<img width="800px" src="./Architecture.jpg"/>

Rules

For patterns like Flux

Store Overview

This is an C# and Blazor example that implements counter.

Simple Store Pattern

using Memento.Core;
using System.Collections.Immutable;

namespace Memento.Sample.Blazor;

public record AsyncCounterState {
    public int Count { get; init; } = 0;

    public bool IsLoading { get; init; } = false;

    public int[] Histories { get; init; } = [];
}

public class AsyncCounterStore() : Store<AsyncCounterState>(() => new()) {
    public async Task CountUpAsync() {
        Mutate(state => state with { IsLoading = true, });
        await Task.Delay(800);
        Mutate(state => state with {
            IsLoading = false,
            Count = state.Count + 1,
            Histories = [.. state.Histories, state.Count + 1],
        });
    }
}

Flux Store Pattern

public record AsyncCounterState {
    public int Count { get; init; } = 0;

    public bool IsLoading { get; init; } = false;

    public int[] Histories { get; init; } = [];
}

public record AsyncCounterCommands : Command {
    public record Increment : AsyncCounterCommands;
    public record BeginLoading : AsyncCounterCommands;
}

public class AsyncCounterStore() : FluxStore<AsyncCounterState, AsyncCounterCommands>(() => new(), Reducer) {
    static AsyncCounterState Reducer(AsyncCounterState state, AsyncCounterCommands command) {
        return command switch {
            AsyncCounterCommands.Increment => state with {
                Count = state.Count + 1,
                IsLoading = false,
                Histories = [.. state.Histories, state.Count + 1],
            },
            AsyncCounterCommands.BeginLoading => state with {
                IsLoading = true,
            },
            _ => throw new CommandNotHandledException<AsyncCounterCommands>(command),
        };
    }

    public async Task CountUpAsync() {
        Dispatch(new AsyncCounterCommands.BeginLoading());
        await Task.Delay(800);
        Dispatch(new AsyncCounterCommands.Increment());
    }
}

Blazor view in Razor

@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>

@code {
    async Task IncrementCount() {
        await AsyncCounterStore.CountUpAsync();
    }
}

License

Designed with ♥ by le-nn. Licensed under the MIT License.