Phork.Blazor.Reactivity is an unopinionated Blazor state management library that utilizes INotifyPropertyChanged and INotifyCollectionChanged .NET interfaces to automatically detect state changes in components.
Advantages of the library:
-
You can use reactive one-way and two-way (in combination with the
@bind
directive) bindings that can make the component re-render if anyINotifyPropertyChanged
instance in the binding path raisesPropertyChanged
event with the appropriate property name. -
You can make your components react to
CollectionChanged
notifications of objects implementingINotifyCollectionChanged
interface. -
You can optionally use converters with two-way bindings if the binding source and target have different types and/or additional logic is required in your bindings.
-
You don't need to worry about memory leaks and unnecessary re-renders as the library will take care of unsubscribing the events as soon as they get out of the render-tree.
-
The library is unopinionated:
- Usage of no MVx pattern is assumed, although it can greatly simplify implementing MVVM.
- Even though the library provides a base class for reactive components, components not inheriting from the provided base class can still take advantage of the reactivity system.
If you are already familiar with these two interfaces, this document will guide you through the steps required to setup the library and make your components automatically react to the notifications published through the interfaces.
There are also other documents that you may find useful:
- Phork.Blazor.Reactivity in Action: If you are new to these interfaces and/or you want to see the motivation behind the concepts of this library.
- Phork.Blazor.Reactivity vs the Alternatives: If you want to see how Phork.Blazor.Reactivity is different from the alternative libraries.
Install the NuGet package or use the package manager console:
Install-Package Phork.Blazor.Reactivity
Call AddPhorkBlazorReactivity()
extension method on the IServiceCollection
of your host builder. This is usually done in Program.cs located at the root of your Blazor project:
using Phork.Blazor;
// ...
var builder = ...;
// ...
builder.Services.AddPhorkBlazorReactivity();
Add the following line to Imports.razor file located at the root of your Blazor project:
namespace Phork.Blazor
In order to enable your components to take advantage of the library you must make them inherit from ReactiveComponentBase
.
Insert the following line at the start of the Razor file of your component:
@inherits ReactiveComponentBase
Note: If your component does not have a Razor file, simply make it inherit from
Phork.Blazor.ReactiveComponentBase
instead of the defaultComponentBase
.
Note: If your component has a direct base type other than the default
ComponentBase
, you can still take advantage of the library (as long asComponentBase
is still an indirect base type of the component). All you need to do, is to implementIReactiveComponent
in your component as instructed here.
When you intend to use a property of an object that implements INotifyPropertyChanged
in your Razor file and at the same time make your component re-render when the property changes, you can use observed values.
Assuming Person
is a parameter of the component and implements INotifyPropertyChanged
, instead of doing:
Name: @Person.Name
You can do the following in a reactive component:
Name: @Observed(() => Person.Name)
This way, not only the Observed
method returns the value of Person.Name
but also it subscribes to PropertyChanged
event of Person
and makes the component re-render (through calling StateHasChanged
) whenever it receives a change notification regarding the Name
property of the Person
object. (By checking e.PropertyName == "Name"
in its PropertyChanged
handler).
You can nest properties and let the observed value observe the changes to any of the intermediate properties to make your component re-render:
Dog Name: @Observed(() => Person.Dog.Name)
This way, if the person changes its dog object or the name of its existing dog (assuming the dog class implements INotifyPropertyChanged
) the component will re-render automatically.
Since the Observed
method returns the value of the expression, it can be mixed with your code seamlessly:
Dog Age Estimate: @(DateTime.Now.Year - Observed(() => Person.Dog.Birthdate).Year)
-----
<DogComponent Dog="Observed(() => Person.Dog)">
-----
@if (IsPalindrome(Observed(() => Person.Name)))
{
<text>Congrats!</text>
}
-----
@if (IsPalindrome(Observed(() => Person.Name)))
{
var dog = Observed(() => Person.Dog);
if(IsPalindrome(Observed(() => dog.Name)))
{
<text>Nested Congrats!</text>
}
}
----- Code -----
@code {
// Person parameter
private string IsPalindrome(string text)
{
// using System.Linq
return text.ToLower().SequenceEqual(text.ToLower().Reverse());
}
}
Read more:
You can use observed collections if you need your component to react to CollectionChanged
events of values implementing INotifyCollectionChanged
. Observed collections do react to property changes that happen in the property path leading to the collection in the same way as observed values do.
@foreach(var skill in ObservedCollection(() => Person.Skills))
{
<text>@Observed(() => skill.Name)</text>
}
By doing so, the ObservedCollection
method will return the value of Person.Skills
and will re-render the component when Skills
property of Person
object gets changes. In addition, if the Skills
collection implements INotifyCollectionChanged
, each CollectionChanged
event it fires will cause the component to re-render.
Read more:
While observed values are enough for one-way binding scenarios, they cannot be used with the @bind
syntax of Blazor. You can use observed bindings to create reactive two-way bindings:
<input type="text" @bind="Binding(() => Person.Name).Value" />
Here, not only any external change (outside of the Blazor events) to Person.Name
will make the component re-render to refresh the text-box, but also if the user changes the text inside the text-box, Person.Name
will get changed accordingly.
Since unlike observed values, observed bindings do not return the value of the expression directly, they cannot be mixed with your custom logic as easily as observed values could. Converters can be used if you want a conversion to occur between the source and the target value:
<input type="date" @bind-value="Binding(() => Person.Birthdate, ConvertToDateTime, ConvertToDateOnly).Value" />
@code {
// Person parameter (Person.Birthdate is DateOnly)
private DateTime ConvertToDateTime(DateOnly date)
{
return date.ToDateTime(TimeOnly.MinValue);
}
private DateOnly ConvertToDateOnly(DateTime dateTime)
{
return DateOnly.FromDateTime(dateTime);
}
}
The default @bind-value
on date input in Blazor currently does not work with DateOnly
struct. By using converters this way we can bind DateOnly
values to date inputs two-way.
Or:
<input type="text" @bind="Binding(() => Person.Age, AddFive, SubtractFive).Value" @bind:event="oninput" />
@code {
// Person parameter
private int AddFive(int number)
{
return number + 5;
}
private int SubtractFive(int number)
{
return number - 5;
}
}
Here, if the value of Person.Age
(the source value) gets changed externally, the text-box will show the new value plus five (as instructed by the AddFive
converter method) and if the user edits the number inside the text-box (the target value), Person.Age
will become the value inside the text-box minus five (as instructed by the SubtractFive
reverse-converter method).
Read more:
There might be situations where you need to create reactive bindings (observed values, observed collections, and rarely observed bindings) in code behind of your component, since the library has its own mechanisms to clean up inactive reactive bindings, it only expects reactive bindings to be created/consumed in a Razor file, or inside the ConfigureBindings
method of your component.
The library will call ConfigureBindings
method after the component renders and before the beginning of the clean-up process.
In a ReactiveComponentBase
component:
public partial class YourComponent : ReactiveComponentBase
{
...
protected override void ConfigureBindings()
{
base.ConfigureBindings();
Observed(() => Path.To.Property);
ObservedCollection(() => Path.To.Collection);
}
}
Here is a list of known limitations and considerations of the library:
- The
Observed
, theObservedCollection
, and theBinding
methods accept anExpression<Func<T>>
as thevalueAccessor
. Only a subset of allExpression<Func<T>>
s are valid value accessors as explained here. - Although the
Observed
, theObservedCollection
, and theBinding
methods are generic methods, you MUST NOT explicitly specify their generic type parameters (e.g., as a means of up-casting the returned value). Doing so will cease the library's ability to reuse reactivity elements, and in some cases may lead toInvalidOperationException
. Always let thevalueAccessor
(and converters in case of converted observed binding) define the generic type parameters. - To use converted observed bindings, the
converter
and thereverseConverter
delegates need to be inverse functions of each other for the binding logic to work.