2021-02-17 14:18:21 +01:00
# Blazor
Blazor apps are based on *components* . A **component** in Blazor is an element of UI, such as a page, dialog, or data entry form.
Components are .NET C# classes built into .NET assemblies that:
- Define flexible UI rendering logic.
- Handle user events.
- Can be nested and reused.
- Can be shared and distributed as Razor class libraries or NuGet packages.
2023-03-13 11:33:51 +00:00


2021-02-17 14:18:21 +01:00
The component class is usually written in the form of a Razor markup page with a `.razor` file extension. Components in Blazor are formally referred to as *Razor components* .
## Components (`.razor`)
2021-04-30 19:21:53 +02:00
[Blazor Components ](https://docs.microsoft.com/en-us/aspnet/core/blazor/components/ )
2024-06-16 19:14:59 +02:00
```cs
2021-11-28 18:08:10 +01:00
@page "/route/{RouteParameter}" // make component accessible from a URL
@page "/route/{RouteParameter?}" // specify route parameter as optional
@page "/route/{RouteParameter:< type > }" // specify route parameter type
2021-04-30 19:21:53 +02:00
@namespace < Namespace > // set the component namespace
@using < Namespace > // using statement
@inherits BaseType // inheritance
@attribute [Attribute] // apply an attribute
@inject Type objectName // dependency injection
2021-02-17 14:18:21 +01:00
2021-04-30 19:21:53 +02:00
// html of the page here
2021-02-17 14:18:21 +01:00
2021-04-30 19:21:53 +02:00
< Namespace.ComponentFolder.Component /> // access component w/o @using
< Component Property = "value" / > // insert component into page, passing attributes
< Component @onclick =" @CallbackMethod " >
@ChildContent // segment of UI content
< / Component >
2021-02-17 14:18:21 +01:00
@code {
// component model (Properties, Methods, ...)
2021-04-30 19:21:53 +02:00
[Parameter] // capture attribute
public Type Property { get; set; } = defaultValue;
[Parameter] // capture route parameters
public type RouteParameter { get; set;}
[Parameter] // segment of UI content
public RenderFragment ChildContent { get; set;}
private void CallbackMethod() { }
2021-02-17 14:18:21 +01:00
}
```
2021-08-09 13:46:06 +02:00
## State Management
2022-11-08 17:53:20 +01:00
## Passing state with `NavigationManager`
It's now possible to pass state when navigating in Blazor apps using the `NavigationManager` .
2024-06-16 19:14:59 +02:00
```cs
2022-11-08 17:53:20 +01:00
navigationManager.NavigateTo("/< route > ", new NavigationOptions { HistoryEntryState = value });
```
This mechanism allows for simple communication between different pages. The specified state is pushed onto the browser’ s history stack so that it can be accessed later using either the `NavigationManager.HistoryEntryState` property or the `LocationChangedEventArgs.HistoryEntryState` property when listening for location changed events.
2021-08-09 13:46:06 +02:00
### Blazor WASM
2024-06-16 19:14:59 +02:00
```cs
2021-08-09 13:46:06 +02:00
// setup state singleton
builder.Services.AddSingleton< StateContainer > ();
```
2024-06-16 19:14:59 +02:00
```cs
2021-08-09 13:46:06 +02:00
// StateContainer singleton
using System;
public class StateContainer
{
private int _counter;
public string Property
{
get => _counter;
set
{
_counter = value;
NotifyStateChanged(); // will trigger StateHasChanged(), causing a render
}
}
public event Action OnChange;
private void NotifyStateChanged() => OnChange?.Invoke();
}
```
2024-06-16 19:14:59 +02:00
```cs
2021-08-09 13:46:06 +02:00
// component that changes the state
@inject StateContainer State
// Delegate event handlers automatically trigger a UI render
< button @onClick =" @HandleClick " >
Change State
< / button >
@code {
private void HandleClick()
{
State.Property += 1; // update state
}
}
```
2024-06-16 19:14:59 +02:00
```cs
2021-08-09 13:46:06 +02:00
// component that should be update on state change
@implements IDisposable
@inject StateContainer State
< p > Property: < b > @State .Property</ b ></ p >
@code {
// StateHasChanged notifies the component that its state has changed.
// When applicable, calling StateHasChanged causes the component to be rerendered.
protected override void OnInitialized()
{
State.OnChange += StateHasChanged;
}
public void Dispose()
{
State.OnChange -= StateHasChanged;
}
}
```
2021-10-15 12:04:57 +02:00
## Data Binding & Events
2024-06-16 19:14:59 +02:00
```cs
2021-10-15 12:04:57 +02:00
< p >
< button @on { DOM EVENT }="{ DELEGATE }" />
< button @on { DOM EVENT }="{ DELEGATE }" @on { DOM EVENT } :preventDefault /> // prevent default action
< button @on { DOM EVENT }="{ DELEGATE }" @on { DOM EVENT } :preventDefault = "{CONDITION}" /> // prevent default action if CONDITION is true
< button @on { DOM EVENT }="{ DELEGATE }" @on { DOM EVENT } :stopPropagation />
< button @on { DOM EVENT }="{ DELEGATE }" @on { DOM EVENT } :stopPropagation = "{CONDITION}" /> // stop event propagation if CONDITION is true
< button @on { DOM EVENT }="@( e = > Property = value )" /> // change internal state w/ lambda
< button @on { DOM EVENT }="@( e = > DelegateAsync(e, value ))" /> // invoke delegate w/ lambda
< input @ref =" elementReference " />
< input @bind ="{ PROPERTY }" /> // updates variable on ONCHANGE event (focus loss)
< input @bind ="{ PROPERTY }" @bind:event ="{ DOM EVENT }" /> // updates value on DOM EVENT
< input @bind ="{ PROPERTY }" @bind:format ="{ FORMAT STRING }" /> // use FORMAT STRING to display value
< ChildComponent @bind - { PROPERTY }="{ PROPERTY }" @bind - { PROPERTY } :event = "{EVENT}" /> // bind to child component {PROPERTY}
< ChildComponent @bind - { PROPERTY }="{ PROPERTY }" @bind - { PROPERTY } :event = "{PROPERTY}Changed" /> // bind to child component {PROPERTY}, listen for custom event
2022-11-08 17:53:20 +01:00
< input @bind ="{ PROPERTY }" @bind:after ="{ DELEGATE }" /> // run async logic after bind event completion
< input @bind:get ="{ PROPERTY }" @bind:set =" PropertyChanged " /> // two-way data binding
2021-10-15 12:04:57 +02:00
< / p >
@code {
private ElementReference elementReference;
public string Property { get; set; }
public EventCallback< Type > PropertyChanged { get; set; } // custom event {PROPERTY}Changed
// invoke custom event
public async Task DelegateAsync(EventArgs e, Type argument)
{
/* ... */
await PropertyChanged.InvokeAsync(e, argument); // notify parent bound prop has changed
await elementReference.FocusAsync(); // focus an element in code
}
2022-11-08 17:53:20 +01:00
[Parameter] public TValue Value { get; set; }
[Parameter] public EventCallback< TValue > ValueChanged { get; set; }
2021-10-15 12:04:57 +02:00
}
```
2022-08-06 10:48:24 +02:00
> **Note**: When a user provides an unparsable value to a data-bound element, the unparsable value is automatically reverted to its previous value when the bind event is triggered.
2022-11-08 17:53:20 +01:00
> **Note**: The `@bind:get` and `@bind:set` modifiers are always used together.
> `The @bind:get` modifier specifies the value to bind to and the `@bind:set` modifier specifies a callback that is called when the value changes
2021-04-30 19:21:53 +02:00
## Javascript/.NET Interop
2021-02-17 14:18:21 +01:00
[Call Javascript from .NET ](https://docs.microsoft.com/en-us/aspnet/core/blazor/call-javascript-from-dotnet )
2021-04-30 19:21:53 +02:00
[Call .NET from Javascript ](https://docs.microsoft.com/en-us/aspnet/core/blazor/call-dotnet-from-javascript )
2021-10-04 22:37:01 +02:00
2023-06-28 11:51:24 +02:00
### Render Blazor components from JavaScript
2021-10-04 22:37:01 +02:00
To render a Blazor component from JavaScript, first register it as a root component for JavaScript rendering and assign it an identifier:
2024-06-16 19:14:59 +02:00
```cs
2021-10-04 22:37:01 +02:00
// Blazor Server
builder.Services.AddServerSideBlazor(options =>
{
options.RootComponents.RegisterForJavaScript< Counter > (identifier: "counter");
});
// Blazor WebAssembly
builder.RootComponents.RegisterForJavaScript< Counter > (identifier: "counter");
```
Load Blazor into the JavaScript app (`blazor.server.js` or `blazor.webassembly.js` ) and then render the component from JavaScript into a container element using the registered identifier, passing component parameters as needed:
2024-06-16 19:14:59 +02:00
```js
2021-10-04 22:37:01 +02:00
let containerElement = document.getElementById('my-counter');
await Blazor.rootComponents.add(containerElement, 'counter', { incrementAmount: 10 });
```
2023-06-28 11:51:24 +02:00
### Blazor custom elements
2021-10-04 22:37:01 +02:00
Experimental support is also now available for building custom elements with Blazor using the Microsoft.AspNetCore.Components.CustomElements NuGet package.
Custom elements use standard HTML interfaces to implement custom HTML elements.
To create a custom element using Blazor, register a Blazor root component as custom elements like this:
2024-06-16 19:14:59 +02:00
```cs
2021-10-04 22:37:01 +02:00
options.RootComponents.RegisterAsCustomElement< Counter > ("my-counter");
```