Mustate 1.0.1
dotnet tool install --global Mustate --version 1.0.1
dotnet new tool-manifest # if you are setting up this repo dotnet tool install --local Mustate --version 1.0.1
#tool dotnet:?package=Mustate&version=1.0.1
nuke :add-package Mustate --version 1.0.1
Mustate
Mustate (Mutable State) is a small dotnet library which purpose is to allow snapshoting the state of a model's mutable properties and detect state changes with minimum effort and high convenience.
Why
This arised from the requirement of form pages (e.g. when adding a product, editing person details or similar) to know when exactly the used model instance changes state on user input.
While C#9 introduced records and value comparisons are much easier now, one still needs to implement equality members in order to exclude certain properties.
This is not only error prone (did you forget to include a property in the check?) and increases maintenance times (how will you enforce that you don't forget to handle a newly introduced property) but also requires implementation for all affected models.
Furthermore, even value comparison is most of the time not enough, because usually when starting off with a null intitialised property (let's say a string), once the user inputs a value and clears it, the property might be an empty string (e.g. string.Empty or ""), which would mean the default value comparison detects a difference, while from the user's perspective nothing changed (the form field being empty is all (s)he sees).
Last but not least, not all code uses C#9 or records but instead needs to use classes (there are a bunch of reasons why this could be the case), so value comparison requires a mouthful of implementation, flooding the model with implementation code.
That's where Mustate can help avoid the aforementioned issues and reduce workload.
Basic usage
Mustate relies on a custom C# attribute, a marker interface and extension methods to semi-automate the process of detecting mutable state changes in a type safe manner.
Imagine you have a mobile app page with a form and some fields. This form is bound to a model, which has only three properties of interest for the form.
What we do is mark the model as IMutable
and only give those properties a Mutable
tag that are relevant for the form and/or which we want to detect state changes for.
public class Person : IMutable
{
public string Id { get; set; }
[Mutable]
public string First { get; set; }
[Mutable]
public string Last { get; set; }
public string FullName => $"{First} {Last}";
[Mutable]
public Person Sibling { get; set; }
public Car RentedCar { get; set; }
public List<string> Awards { get; set; }
public string LinkToNirvana { get; set; }
}
Now in your UI handling code (e.g. a view model) we need to let Mustate know the initial state of your model.
We do this via the extension method Snapshot
:
personInstance.Snapshot();
Once that is done, we can from now on check at any time if the state, i.e. any of the properties marked as Mutable
, has changed.
This is done just as easily using the extension method HasChanged
:
if (personInstance.HasChanged()) {
// Do something
}
Custom types
Custom types may require a special treatment when it comes to checking for equality. For that scenario we can register a user-defined equality checking method via the extension method RegisterMutableCheckForType
:
/*
x is a mutable property value from the initial snapshot state
y is a mutable property value from the current state
*/
personInstance.RegisterMutableCheckForType((x,y) => {
// Do a custom check...
if(x is List<string> xStringList) {
return xStringList.Count == (y as List<string>).Count;
}
return true;
});
Note: Mustate will always check for null equality before your custom check.
Dependencies
For the library:
- Mapster - to create a copy of the model instance
- net 6.0
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net6.0 is compatible. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. |
This package has no dependencies.
User-defined equality check function now registered on the IMutable type