CSLA Blazor 02 28 PDF
CSLA Blazor 02 28 PDF
CSLA Blazor 02 28 PDF
Rockford Lhotka
Using CSLA 5: Blazor and WebAssembly
Copyright © 2020 by Marimer LLC
All rights reserved. No part of this work may be reproduced or transmitted in any form or by any means, electronic
or mechanical, including photocopying, recording, or by any information storage or retrieval system, without the
prior written permission of the copyright owner.
Trademarked names may appear in this book. Rather than use a trademark symbol with every occurrence of a
trademarked name, we use the names only in an editorial fashion and to the benefit of the trademark owner, with
no intention of infringement of the trademark.
The information in this book is distributed on an “as is” basis, without warranty. Although every precaution has been
taken in the preparation of this work, the author shall not have any liability to any person or entity with respect to
any loss or damage caused or alleged to be caused directly or indirectly by the information contained in this work.
The source code for this book (CSLA .NET version 5.1.0) is available at http:/cslanet.com.
Errata or other comments about this book should be emailed to errata@lhotka.net.
Revision: 200228
Page 1
Acknowledgments
Magenic
Neither this book, nor CSLA .NET, would have been possible without support from Magenic. Magenic helps you get
your digital products to market faster.
You can reach Magenic at https://magenic.com
Blazor Community
Blazor also has a very active community, especially in the Blazor gitter channel. Numerous people from this channel
have been helpful in getting CSLA .NET working well with Blazor, and in writing this book.
I would like to specifically acknowledge @chrissainty, @tleylan, @shawty, and @chucker for their thoughtful and
timely help solving challenging problems.
Page 2
About the Author
Rockford Lhotka is the author of more than 20 books on developing software using the Microsoft platform and
technologies. He is a member of the Microsoft Regional Director and Microsoft MVP programs. Rockford speaks at
many conferences and user groups around the world. He is the CTO at Magenic, a company that specializes getting
your digital products to market faster.
Page 3
Organization of the Book
This book is part of the Using CSLA 2019 book series. The content in this book is written with the assumption that
you have read the first four books in the series:
1. Using CSLA 2019: CSLA .NET Overview
2. Using CSLA 2019: Creating Business Objects
3. Using CSLA 2019: Data Access
4. Using CSLA 2019: Data Portal Configuration
This book demonstrates how to create basic web applications using Blazor, WebAssembly, and CSLA .NET. The book
uses Visual Studio 2019 with .NET Core 3.1, along with Blazor tooling and various NuGet packages.
Blazor supports server-side and client-side development. This book will cover both, but the primary focus will be on
client-side Blazor with WebAssembly, enabling the creation of true smart client applications that can run in any
modern browser on any operating system.
By the end of the book you will understand how to build basic server-side and client-side Blazor applications that
use business domain objects created with CSLA .NET as the application's model.
Page 4
Chapters
1. Introduction to Blazor and WebAssembly
2. Server-Side Blazor
3. Client-Side Blazor
4. Components, Data Binding, and Other Features
5. Authentication and Authorization
6. Multi-Headed Blazor Solutions
7. Blazor and CSLA .NET
8. ProjectTracker UI Using Blazor
Page 5
Chapter 1: Introduction to Blazor and WebAssembly
Blazor is a UI framework from Microsoft, designed to enable productivity in web app development. You can build
server-side web apps using Blazor, but the real power of Blazor is that it can be used to create smart client
applications that run in any modern browser on any operating system.
These client-side capabilities are possible thanks to an underlying technology called WebAssembly, combined with a
version of .NET that runs in WebAssembly to allow us to run C# code natively in any modern browser.
It is important to lay the groundwork regarding WebAssembly before describing Blazor itself.
About WebAssembly
In the late 1990's, as the web started to become mainstream, there were two languages competing to become the
standard for scripting client-side behaviors in the browser: JavaScript and VBscript. As time went on, JavaScript
became the defacto standard scripting language.
In subsequent years, JavaScript has become more than just a scripting language, and is used as a first class
application development language when building smart client apps (often called single page apps or SPAs) that run
in any browser.
Not everyone finds JavaScript to be an ideal language for software development. As a result, many other languages
have been created that transpile into JavaScript. Because browsers used to only support JavaScript, it was
impossible to compile most languages for use in the browser, and only languages that can transpile into JavaScript
have been available.
To overcome this limitation, the WebAssembly initiative brings the ability for a browser to execute JavaScript and
also execute a type of assembly language code.
It is important to understand that WebAssembly is not a plug-in to any browser. WebAssembly is native to every
modern browser, exactly like JavaScript. As a result, this technology is entirely unlike Flash or Silverlight or other
technologies based on a plug-in architecture.
Another important side-effect of WebAssembly being a peer to JavaScript is that your code runs in the same
sandbox as JavaScript, and so has exactly the same security profile. If you are comfortable running JavaScript in a
Page 6
browser you should be just as comfortable running WebAssembly in a browser.
All modern "evergreen" browsers currently support WebAssembly as well as JavaScript. These include:
FireFox
Chromium (Chrome and Edge)
Safari
Edge (pre-Chromium)
The power of WebAssembly is that high level languages like C, C++, and most others have always compiled into
assembly language. When you compile a C program, the C compiler generates assembly language for the target
hardware architecture, which is then compiled to a native binary for the computer.
Now your C code can be compiled to wasm, the WebAssembly assembly language. All browsers understand how to
execute that wasm code. In fact, the browsers download the wasm code, then compile it to be native for the client
device hardware, whether that is Intel, AMD, or ARM.
The result is that your high level language code ends up running as a native binary on the client device!
Numerous languages currently compile to wasm, including:
C and C++
C#
GoLang
Java
PHP
Python
Rust
Swift
and many others.
WebAssembly has the potential to broaden our ability to develop software for the browser by eliminating the
single-language mono-culture enforced by JavaScript. This is a very exciting technology that will almost certainly
lead to innovation and substantial change around client-side development into the future.
Remember that WebAssembly is a peer to JavaScript; both are nothing more than a way to execute code in a
browser. Neither are a UI framework, which is where technologies like Angular, React, Vue, and Blazor come into the
picture.
Before discussing the Blazor UI framework however, it is important to understand how .NET runs in the browser via
WebAssembly.
Page 7
Microsoft has indicated that .NET 5 will bring .NET Core and mono together, so it is possible that future .NET
WebAssembly apps may run on .NET 5 instead of mono. In the near future however, you should expect that your
code will run on mono. This should not be a concern, as .NET code in Xamarin runs on mono for iOS, Android, and
other platforms. The mono runtime is widely used and stable.
The mono runtime implements .NET Standard 2.1. As a result, the majority of .NET Standard 2.0 and 2.1 assemblies,
projects, and NuGet packages are available for your use when building WebAssembly-based apps. This includes
CSLA .NET, because the Csla NuGet package includes a .NET Standard 2.0 implementation that runs in mono on
WebAssembly.
About Blazor
Blazor is a .NET based UI framework designed to help you build rich, interactive web experiences for your users. It
runs on top of ASP.NET Core on the server, or on top of mono on WebAssembly on the client.
A Blazor UI is created using HTML, CSS, and C#. Your components are created using a variant of the Razor syntax
that has been in use with ASP.NET MVC for many years. In modern ASP.NET MVC, and the new Razor Pages model,
pages are defined using cshtml files. In the Blazor Razor Components model, components are defined using razor
files. This is because the Razor syntax used by MVC and Blazor are very similar, but not identical.
It is important to understand that Blazor is designed to create highly interactive user experiences, similar to those
you might have created in the past with WPF, UWP, Angular, or React. As a result, your .NET code runs in the same
context as your components, and can react instantly to UI events that occur due to user interaction. For example,
when the user clicks or taps a button, your code is immediately invoked to handle that click event.
Contrast this to a web site model where user interactions in the browser can only be processed by your code after a
postback transfers control from the browser to the web server. Blazor doesn't follow this model, and instead follows
a model where your code is invoked immediately for all events.
If you are coming from a web development background, Blazor creates what is commonly known as a Single Page
Application (SPA). Instead of using Angular, React, or a similar JavaScript-based UI framework, Blazor provides a
.NET-based UI framework.
If you are coming from a Windows smart client background, Blazor creates a user experience very comparable to
Windows Forms, WPF, or UWP. Instead of forms or XAML pages, you use Blazor to create Razor Components:
components or pages with which the user interacts.
Regardless of your previous experience, Blazor provides a UI framework that allows you to build rich, interactive
user experiences based on .NET, HTML, and CSS.
You can choose whether to run your code on the server or on the client. If your users have low-end client devices
you may choose to run the code on the server so the browser is little more than a terminal used to render HTML and
CSS. If your users have more powerful client devices you may choose to run the code on the client to offload
processing power from your server, thus improving scalability and harnessing the substantial compute power
Page 8
available on modern PCs, laptops, tablets, and phones.
Regardless of whether you run your code on the server or client, what is running are your Razor components. The
server-side and client-side WebAssembly Blazor environments are basically hosts for Razor Components.
Although Razor Components provides a virtually identical user experience between server-side and client-side
Blazor, as a developer you should understand the underlying architecture supporting both scenarios.
Server-Side Blazor
Server-side Blazor rests on top of ASP.NET Core, and all your code executes on the web server. To provide rich
client-side interactivity for the end user, a messaging protocol is used to send events back and forth between the
browser and the web server.
When the user clicks a button, leaves an input field, or otherwise interacts with the app in the browser, events
corresponding to those actions are immediately sent to the web server and your code is invoked to handle the
event. These are not traditional postback operations, they are granular events that flow between browser and web
server. Any changes your code makes to the page result in messages being immediately sent to the browser so the
DOM (browser Document Object Model) can be updated.
To the user, it feels like your code is running on the client, even though it is actually running on the web server.
For those who've been in this industry long enough, you'll recognize this model as being very similar to
mainframe (3270 terminals) and minicomputer (VT terminals) software architectures that were mainstream
before client/server and web development became popular. The app code runs on the server, and the
browser (in this case) is used as a very capable terminal.
Server-side Blazor relies on SignalR to as a transport for the near-realtime messaging that flows between the
browser client and the web server.
As a developer, you don't directly see or affect the messages used by Blazor to coordinate between the client and
server, nor do you write the client-side code that manages the user experience. You write server-side Blazor code
and allow the Blazor runtime to do all the work.
The end result is that the user enjoys a highly interactive user experience comparable to a smart client, but all the
code runs on the server. This does imply that there's a constant stream of message traffic flowing back and forth
Page 9
between the client and server, which is a factor to keep in mind when you decide between server-side and client-
side Blazor deployments for your app.
You should also be aware that server-side Blazor maintains state in memory on the web server. Some state is kept in
memory for each user actively using the app. As as a result there are limits on how scalable a server-side Blazor app
will be. The amount of memory and server resources consumed per-user varies greatly depending on how you
implement your app and what the app is doing, so you'll need to do your own benchmarking to determine how
many users can be supported by a web server.
Client-Side Blazor
Client-side Blazor runs entirely in the browser, with the Blazor UI framework running on top of mono on
WebAssembly. In this regard, client-side Blazor is very similar to Angular or React, in that it is a UI framework that
runs in the browser. The primary difference is that your code will be C# instead of JavaScript or TypeScript.
Because all the UI code is running in the browser, there's no messaging needed between the browser and web
server. In fact, there's no real need for a traditional web server at all, beyond deploying the html, css, and wasm files
to the browser.
You should be aware that platform or UI specific code will not run in WebAssembly; for example no
Windows Forms, WPF, or Xamarin code. Similarly there is no provision for direct access to databases, so no
use of System.Data to interact with your databases. If you need access to databases you will need to invoke
services exposed from an application server.
Although most of the Blazor framework runs on WebAssembly, it does have the ability to interact with JavaScript
that is also running in the browser.
As with any SPA, mobile, or smart client technology, if your code is running on a user's device it is
vulnerable to hacking. This is an issue Windows developers have dealt with since the early 1990's, and
mobile and SPA web developers have also dealt with for years. You should be aware of the consequences of
running code on a client device, in terms of improved scalability and potentially reduced security.
When you create an app with client-side Blazor you are creating a smart client app, very comparable to Angular,
React, WPF, Windows Forms, Xamarin, and other smart client development technologies. Well designed smart client
apps that interact with services on an app server can scale far higher with less cost than traditional web sites. This is
one key reason architectures based on smart clients are such a great approach.
Conclusion
It has become clear over the past several years that the browser is now the primary development target for client-
side app development. WebAssembly breaks us free from the JavaScript mono-culture, allowing us to use many
other programming languages, and their corresponding frameworks and tools, to develop software that runs in the
browser.
One language and runtime that supports WebAssembly is C# with .NET. The majority of code that targets .NET
Standard 2.0 and later will just work in WebAssembly, like it does on Windows, Mac, iOS, Android, and other
operating environments.
Blazor is a UI framework built on top of .NET that leverages HTML and CSS for markup and layout, replacing the use
of JavaScript or TypeScript with C#. It uses a slight variant of the Razor syntax that has been in use by ASP.NET MVC
and Razor Pages developers on the server.
Page 11
Chapter 2: Server-Side Blazor
Starting with the basic understanding of Blazor from Chapter 1, I will walk through the basic features of server-side
Blazor.
The released version of Visual Studio 2019, as of the end of 2019, directly supports the creation of server-side
Blazor projects. Microsoft provides full support for server-side Blazor and Razor Components, alongside ASP.NET
MVC, ASP.NET Razor Pages, and other ASP.NET features.
Name the new project BlazorServerExample and choose an appropriate location for the solution and project.
Page 12
By default Visual Studio will offer to create a .NET Core 3.0 project, but .NET Core 3.1 is the LTS (long-term support)
version of .NET Core, and so should be your preferred choice.
Page 13
The Authentication option allows you to select between various authentication options supported by the Microsoft
project template.
I will discuss the authentication options later in this chapter. For now leave this at the default No Authentication
value.
You will have the option to Configure for HTTPS, and the default is checked. Leave this alone so your web site is
configured for HTTPS.
If you have Docker Desktop installed you will see the option to Enable Docker Support. In this example I will assume
this option is not checked, so if it is available to you make sure to uncheck it.
Click the Create button to create the solution and project.
The result is a solution with the BlazorServerExample project.
Page 14
Exploring the Project
The Blazor project template creates a fully functional app that demonstrates error handling, basic user interaction,
and data retrieval and display.
<Project Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup>
<TargetFramework>netcoreapp3.1</TargetFramework>
</PropertyGroup>
</Project>
Notice how the project uses the Microsoft.NET.Sdk.Web namespace to bring in all the necessary references for
ASP.NET. This includes support for server-side Blazor, as well as MVC and Razor Pages.
Because you chose to target .NET Core 3.1, the target framework for this project is netcoreapp3.1.
wwwroot Folder
The wwwroot folder in the project contains client-side content such as the web site's icon and CSS resources.
Page 15
Blazor itself does not require the use of bootstrap, but the templates are created around those styles.
These are not required if you change the HTML in each page to use a different CSS framework and assets.
You can change or add CSS assets in this folder to control the layout and other aspects of your web UI.
Data Folder
The Data folder in the project contains sample data, and a mock service that creates the data. In a real app this
service would most likely call a remote service via HTTP.
In a CSLA .NET based application there is not normally a Data folder, because the app will be built to use
a formal business domain layer.
WeatherForecast Class
The WeatherForecast class defines a DTO (data transfer object) that is intended to store strongly typed data.
The only behavior in this class beyond storing data, is that it calculates the Fahrenheit temperature from the Celsius
temperature.
WeatherForecastService Class
The WeatherforecastService class is a mock service that creates sample data.
Page 16
public class WeatherForecastService
{
private static readonly string[] Summaries = new[]
{
"Freezing", "Bracing", "Chilly", "Cool", "Mild", "Warm", "Balmy", "Hot", "Sweltering",
"Scorching"
};
You can imagine how the GetForecastAsync method could make a call to a remote server endpoint to retrieve the
data. The app consuming the data wouldn't be affected by such a change.
Pages Folder
The Pages folder contains most of the UI displayed to your users. The pages and components in this folder represent
web pages, but you should think of them as smart-client "forms" or "pages" much like you would in a Windows
Forms, WPF, Xamarin, UWP, Angular, or React app. Although these pages are created using HTML and Razor
markup, the Blazor UI framework makes them feel like smart client pages, not traditional server-side web pages.
The pages in the Pages folder are rendered for the user in the browser by combining the HTML and Razor markup in
each page, plus the CSS defined in the wwwroot folder, plus templates and components defined in the Shared folder
I will discuss later in this chapter.
Page 17
Blazor and Razor Components use a container-based, or nested, approach to constructing the content shown to the
user. You can see that the primary content, in this diagram Index.razor, is contained within the MainLayout.razor
component, replacing the @body tag. The MainLayout component is contained within an App.razor component,
which is contained within the _Host.cshtml page.
The _Host and Index pages are found in the Pages folder, the other components come from the Shared folder.
_Host Page
The ASP.NET Blazor framework is based on the same ASP.NET Core technologies as MVC and Razor Pages. A server-
side Blazor project is a specialized type of web site. As a result, it relies on ASP.NET Core Razor Pages to launch the
web site and start up the Blazor framework on the client.
Remember that server-side Blazor uses SignalR to provide near-realtime communication between the
browser and web server. This implies that there is JavaScript code running in the browser to act as a SignalR
client and to manage this messaging.
The _Host.cshtml file contains this Razor Pages bootstrap code, including some important elements for you as a
Blazor developer.
As I will discuss in Chapter 3, this is comparable to the index.html page in a client-side Blazor project.
<title>BlazorServerExample</title>
Next it controls how error text is displayed to the user in the case that the Blazor UI framework doesn't start
properly. Once Blazor is running errors are displayed using the Error Razor page.
<div id="blazor-error-ui">
<environment include="Staging,Production">
An error has occurred. This application may no longer respond until reloaded.
</environment>
<environment include="Development">
An unhandled exception has occurred. See browser dev tools for details.
</environment>
<a href="" class="reload">Reload</a>
<a class="dismiss"> </a>
</div>
Finally, the Blazor client-side JavaScript framework is loaded to handle the SignalR messaging with the server.
<script src="_framework/blazor.server.js"></script>
In most cases you will only edit this file to update the web site title, improve SEO by changing meta tags, alter the
CSS framework or files used by the app, and possibly alter or enhance the error text displayed to the user when the
Blazor UI framework can't be loaded during app startup.
The other components end in a .razor postfix because they are handled by the Blazor UI framework, not by Razor
Page 18
Pages. If the Blazor framework successfully loads during app startup everything from that point forward is handled
by Blazor.
Index Component
Each page displayed to the user by Blazor is a combination of the page plus content from the Shared folder. I will
discuss the Shared folder later. Right now it is important for you to understand that each page in the Pages folder
represents only the specific page content, not other content such as headers, footers, navigation, or anything else
outside the page itself.
The first page displayed to the user is the Index page.
@page "/"
<h1>Hello, world!</h1>
Notice the @page directive at the top of the code. This directive is part of the routing mechanism used by Blazor. In
more advanced pages, later in this book, you will see that a page can have multiple @page directives if it handles
more than one route. The Index page handles one route, the default, and so has one directive.
The rest of the page is HTML displaying content to the user. This page could have more advanced Razor markup as
well, but the basic template simply displays text.
Error Component
Page 19
The Error page is displayed to users when an unhandled error occurs within the app. Like all pages, it starts with a
@page directive for routing.
@page "/error"
The rest of the content in the page is default text explaining how to get more detailed information by enabling
developer mode. You will want to edit this content to provide meaningful and useful text for your users.
Counter Component
The Counter page implements functionality that allows the user to interact with the app. Specifically, it provides a
button that the user can click on, and each click increments a numeric value.
@page "/counter"
<h1>Counter</h1>
@code {
private int currentCount = 0;
There are two new concepts in this page: the properties on the button element, and the @code block.
The button element displays a standard HTML button, but the @onclick property is a special Razor markup item.
This property causes Blazor to invoke the IncrementCount method each time the button is clicked.
Remember that the button is rendered in the browser, but all code runs on the web server. This means that each
time the user clicks the button the Blazor JavaScript framework running in the browser sends a message to the
server so the server knows that the IncrementCount method should be invoked.
The @code block is defining C# code that runs on the web server, and is invoked when the Blazor framework sends a
message from the browser via SignalR.
You should be aware that server-side Blazor does maintain state on the web server for each active user
of the app. As a result, the state necessary to handle events from the client is available in memory and so
event handling is fast and responsive. The potentially negative consequences to this architecture include
memory consumption on the server for each user, and loss of in-memory user information if the web server
goes down.
When the user clicks the button in the browser, Blazor sends a message to the server. On the server Blazor invokes
the IncrementCount method in the @code block. That method increments the currentCount field.
Blazor automatically monitors the values of fields in a page, so it is immediately aware that currentCount has
changed. The result is that Blazor sends a message to the browser using SignalR, so the Blazor framework code
running in the browser can immediately update the value displayed to the user.
From the user's perspective this is all nearly instantaneous. The user will feel like this is a smart client app
experience, with all the rich, interactive UI they'd expect from an Angular, React, Windows, or mobile app.
Page 20
However, behind the scenes you, as a developer, need to remember that all code is running on the server, with
Blazor providing the seamless illusion that everything is running on the client.
It is important to remember that client-side Blazor really does run everything on the client!
FetchData Component
The FetchData page demonstrates how to have UI code invoke a service to retrieve data, and then how to display
that data to the user. It also provides an opportunity for me to discuss how Blazor leverages the dependency
injection (DI) service provided by .NET Core.
Near the top of the page there are a couple new directives.
@using BlazorServerExample.Data
@inject WeatherForecastService ForecastService
The @using directive works like the using statement in C#, bringing a namespace into scope for use within the page.
The @inject directive indicates that Blazor should use the .NET Core DI framework to inject an instance of the
WeatherForecastService type with the variable name ForecastService. You can use the ForecastService variable
throughout the page to interact with the object provided by dependency injection.
I will show you the code where the WeatherForecastService type is mapped to a specific instance later in this
chapter when I discuss the Startup.cs file.
This service is used in the @code block for the page.
@code {
private WeatherForecast[] forecasts;
As I discussed with the Counter page, fields declared within the page are available for data binding to the UI. In this
case, the forecasts field is loaded with data from a remote service, so the array of forecast data can be displayed to
the user.
You can look at the HTML and Razor markup in the page to see how the forecasts field is used.
It is important to understand that the page might render before data is loaded, and so the forecasts field will be
null. Blazor will fail, rather ungracefully, to render a page if data binding tries to bind to a null value, so the markup
in the page checks to see if the value is null before rendering.
If the value is null a message is shown to the user to indicate that data is being loaded in the background, otherwise
the value is not null and a foreach loop is used to build UI for each row of data in the array.
Page 21
@foreach (var forecast in forecasts)
{
<tr>
<td>@forecast.Date.ToShortDateString()</td>
<td>@forecast.TemperatureC</td>
<td>@forecast.TemperatureF</td>
<td>@forecast.Summary</td>
</tr>
}
Because this is read-only data, the "data binding" used in this case are simple @ directives to retrieve data values
from each item in the row. Razor markup understands that it should apply ToString to any standalone value so that
value can be rendered in the browser as text. This means that @forecast.TemperatureC, a numeric value, is
converted to text and rendered in the browser for the user.
If you have used Razor markup in ASP.NET MVC or Razor Pages this will be very familiar to you.
Although there are Blazor-specific variations for some Razor markup, this use of Razor markup has remained
unchanged for many years.
Later in this book I will discuss data binding in much more depth. Data binding can be used to not only display
values, but to enable rich two-way bindings between UI elements and fields (or object properties) when creating
data entry forms or other types of UI where the user provides input to the app.
Shared Folder
Earlier in this chapter I discussed the Index page and how what the user sees in the browser is a combination of
each page plus content from the Shared folder. When you look at a page, all the content in the left-hand navigation
area, the title and the header with About all come from the files in the Shared folder.
Page 22
The default Blazor template provides the user with a responsive web experience. For example, when the browser
window is small enough, the navigation area is moved to a hamburger menu so the user's focus remains on the
content of each page.
Page 23
You can change all this content by editing the files in the Shared folder and the CSS from the wwwroot folder. The
Shared folder contains MainLayout.razor and NavMenu.razor files you will edit to control the overall appearance
and navigation for the app.
MainLayout Page
The MainLayout page defines the overall layout of all pages in the app.
Page 24
@inherits LayoutComponentBase
<div class="sidebar">
<NavMenu />
</div>
<div class="main">
<div class="top-row px-4">
<a href="https://docs.microsoft.com/aspnet/" target="_blank">About</a>
</div>
You can see how this page specifies where the NavMenu and @body content will be displayed, as well as any global
header content such as the About link.
Although I will discuss this in more detail in Chapter 4, you should be aware that nearly any page can be displayed
as a component of another page, just like this page is displaying the content of the NavMenu page. The ability to
compose pages by using other page/component content is one of the most important features of the Blazor UI
framework.
NavMenu Page
The NavMenu page not only displays the navigation menu, it also implements events and code to handle expanding
the menu when the user clicks the hamburger menu button when it is visible.
The page relies on the navbar-toggler CSS style to control whether the hamburger menu button is visible.
If the button is visible, then when the user clicks on the button the ToggleNavMenu method is invoked from the @code
block on the server.
@code {
private bool collapseNavMenu = true;
This code toggles the collapseNavMenu value between true and false. More importantly, you can see how this
value is used to change the NavMenuCssClass value between collapse and null. The collapse CSS style is used to
alter the appearance of the navigation menu content in the page.
If the text of the NavMenuCssClass field is collapse then CSS controls whether the navigation menu content is visible
to the user. If the value is null then the content is visible.
The collapse CSS is in the wwwroot folder in the site.css file. It is within a @media element so it is only expressed
Page 25
for certain screen sizes.
When the width is 786px or greater then the sidebar will not collapse, otherwise it collapses.
.sidebar .collapse {
/* Never collapse the sidebar for wide screens */
display: block;
}
The interaction between the NavMenu HTML and Razor, the CSS styles, and the C# code works together to provide
the user with a responsive experience for the navigation menu.
There are also a number of files at the project level. These files contain markup or code that affects the overall Blazor
app.
_Imports Page
The _Imports page defines @using directives that apply to all pages in the app.
@using System.Net.Http
@using Microsoft.AspNetCore.Authorization
@using Microsoft.AspNetCore.Components.Authorization
@using Microsoft.AspNetCore.Components.Forms
@using Microsoft.AspNetCore.Components.Routing
@using Microsoft.AspNetCore.Components.Web
@using Microsoft.JSInterop
@using BlazorServerExample
@using BlazorServerExample.Shared
If you consistently use a namespace in your pages and want it to be globally available then you can add it to this file.
App Page
The App page manages routing for the app, and controls what the user sees if they navigate to an invalid URL.
<Router AppAssembly="@typeof(Program).Assembly">
<Found Context="routeData">
<RouteView RouteData="@routeData" DefaultLayout="@typeof(MainLayout)" />
</Found>
<NotFound>
<LayoutView Layout="@typeof(MainLayout)">
<p>Sorry, there's nothing at this address.</p>
</LayoutView>
</NotFound>
</Router>
You may edit this page to alter routing, alter the text displayed to the user if a route doesn't exist, and to implement
some authentication and authorization features. I'll discuss authentication and authorization in Chapter 5.
Program.cs
A server-side Blazor app, like any ASP.NET Core web site, can be hosted by IIS, the .NET Core Kestrel web host, or in
other hosting environments. In all cases, when the web site is started on the web server the first code that runs is the
Main method in Program.cs.
Page 26
public class Program
{
public static void Main(string[] args)
{
CreateHostBuilder(args).Build().Run();
}
You will probably never edit this code, as it exists to properly bootstrap ASP.NET. If you want to affect how the app
starts up or is configured, you will edit the code in Startup.cs.
Because this is a server-side Blazor web site, the AddRazorPages and AddServerSideBlazor methods are required.
The AddSingleton method is used to tell the DI framework about the WeatherForecastService type. Remember that
this type is injected into the FetchData page. All that is necessary is for an instance of the type to be created and
provided to any pages or code that requests the type, and this AddSingleton method call indicates that the DI
framework should provide a single instance of the type to any requestors.
Later in this book you will see more advanced uses of dependency injection and configuration.
Configure Method
The Configure method runs after the ConfigureServices method, and provides an opportunity to provide other
app configuration. Most of this code is standard for any ASP.NET Core web site.
Page 27
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
if (env.IsDevelopment())
{
app.UseDeveloperExceptionPage();
}
else
{
app.UseExceptionHandler("/Error");
// The default HSTS value is 30 days. You may want to change this for production scenarios, see
https://aka.ms/aspnetcore-hsts.
app.UseHsts();
}
app.UseHttpsRedirection();
app.UseStaticFiles();
app.UseRouting();
app.UseEndpoints(endpoints =>
{
endpoints.MapBlazorHub();
endpoints.MapFallbackToPage("/_Host");
});
}
The app is configured with the error page route, and to use HTTPS. It also is configured to serve up static files, and to
use routing.
The endpoints.MapBlazorHub and endpoints.MapFallbackToPage("/_Host") lines are unique to server-side Blazor,
and are required for Blazor to operate.
The MapBlazorHub call sets up the server-side endpoint for SignalR that is used by the client-side JavaScript Blazor
code.
The MapFallbackToPage is triggered any time the user requests a route to a URL that doesn't exist as a physical file.
Any time the user asks for content not represented by a physical file on the server they are redirected to the Blazor
app. The MapFallbacktoPage call sets up the use of the _Host.cshtml page so it can bootstrap Blazor as I discussed
earlier in this chapter.
Conclusion
At this point you should understand how to create a server-side Blazor project in Visual Studio, and all the folders
and files that are provided in the default project template. Let's move on to discuss client-side Blazor projects.
Page 28
Chapter 3: Client-Side Blazor
In Chapter 2 you learned how to create a server-side Blazor project, and I discussed all the files and functionality
included in the default project template. In this chapter I will go through the same process, focusing on client-side
Blazor.
Client-side Blazor WebAssembly projects are not finalized at this time. Microsoft continues to rapidly
evolve the code and project templates. Because of this, the specific code, solution and project templates, and
other content in this chapter may not be exactly the same as the latest preview release provided with Visual
Studio 2019 preview you are using.
The project template is virtually identical between server-side and client-side Blazor. The only differences relate to
how the app starts up and is configured, so there are differences in the Program.cs and Startup.cs files.
The way the app is hosted and executes is also very different. A server-side Blazor app runs all .NET code on the web
server, relying on a Blazor JavaScript framework and SignalR messaging to provide the user with a rich, interactive
experience. A client-side Blazor app runs entirely on the client, in the browser. All the .NET code in your app runs in
the browser, and any web server is responsible only for deploying the static files containing your code to the
browser.
Your web server might also expose REST or other service endpoints, but from the perspective of the
client-side Blazor app, the web server is nothing more than a way to deploy the app to the client device.
You must be using Visual Studio 2019 preview to create a client-side Blazor project.
Page 29
Name the new project BlazorClientExample and choose an appropriate location for the solution and project.
If .NET Core 3.1 is not selected, make sure to select that version. This is the LTS (long-term support) version of .NET
Core, and you must select version 3.1 to create a client-side Blazor project.
If you are running the release version of Visual Studio 2019 you may only see one option at this point. However, if
you are running Visual Studio 2019 preview then you will see two options as shown here.
Page 30
Make sure to select the Blazor WebAssembly App option.
Notice that the Authentication option is not available. Client-side Blazor requires that you use some server-side
authentication mechanism, most likely leveraging ASP.NET Core features.
If you have Docker Desktop installed you will see the option to Enable Docker Support, but it is always disabled. This
is because you are creating an app that will run on the client, not on the server, and so Docker containers are not
relevant.
There is an option for the project to be ASP.NET Core Hosted. The choice you make here will change the generated
solution in important ways.
You should check this option, as I will use the ASP.NET Core Hosted approach in this chapter.
Page 31
This is similar to the server-side project created in Chapter 2, though there are differences because it is a smart client
project, not an ASP.NET Core web project.
When you press ctl-F5 to launch this project, the app is compiled and loaded into your browser. There actually is a
tiny web server created for you behind the scenes, but it exists only to deploy the static files for your app onto the
browser, and to provide access to other static files such as the sample data for the FetchData page.
Page 32
The web server project in this solution has two roles. First, it is responsible for deploying the static files containing
your app to the browser. Second, it exposes a REST service so the client-side Blazor app can make an HttpClient call
to retrieve the sample data in the FetchData page.
When you launch this solution the web server project is started, just like any ASP.NET Core web project. When the
browser loads, the web server will provide it with the static files containing the Blazor app. From that point forward
the app is completely running in the browser. The only time the web server is used after the app is loaded, is if the
user opens the FetchData page, triggering a call to the web server's REST endpoint.
When you run the a standalone project a small web server is used to serve up static files to the browser, including
the app itself, along with all the files in the wwwroot folder.
I'll discuss dependency injection and related configuration later in this chapter. For now it is enough to understand
that the Http variable contains a reference to a properly initialized HttpClient object.
This page has a @code block containing C# code that is responsible for retrieving the sample data.
The GetJsonAsync method of the Http object is used to make a web call to retrieve the static weather.json file using
a relative URL.
Now that you have seen the two small areas that are different in the standalone template, I will focus the rest of the
chapter on exploring the ASP.NET Core Hosted solution.
Page 34
If you are familiar with CSLA .NET, the idea of sharing a common assembly between client and server is
familiar. This is an approach I have been advocating since CSLA .NET 1.0 was released in 2002.
if (env.IsDevelopment())
{
app.UseDeveloperExceptionPage();
app.UseBlazorDebugging();
}
app.UseStaticFiles();
app.UseClientSideBlazorFiles<Client.Program>();
app.UseRouting();
app.UseEndpoints(endpoints =>
{
endpoints.MapDefaultControllerRoute();
endpoints.MapFallbackToClientSideBlazor<Client.Program>("index.html");
});
}
The UseBlazorDebugging method configures support for debugging Blazor code in the browser. You should
understand that the debugging experience for client-side Blazor code is very primitive compared to server-side
Blazor debugging. In Chapter 6 I will discuss a way to build a Blazor solution that can run as a server-side or client-
side app. One benefit of that approach is that you can make use of the powerful server-side debugging during
development, and still deploy and test your app using WebAssembly.
The UseClientSideBlazorFiles method abstracts the steps necessary for the web server to properly tell the
browser to load and launch the client-side Blazor app.
Page 35
Finally, the MapFallbackToClientSideBlazor method creates a redirect so if a user navigates to the web server they
are redirected to the client-side Blazor app.
The goal of this configuration is to ensure that the client-side Blazor app is properly launched in the browser during
debugging or when a user attempts to navigate to a URL on the web server.
WeatherForecast Controller
In the Controllers folder the WeatherForecastController class implements a REST service endpoint that creates
sample data and returns it to the calling code.
[ApiController]
[Route("[controller]")]
public class WeatherForecastController : ControllerBase
{
private static readonly string[] Summaries = new[]
{
"Freezing", "Bracing", "Chilly", "Cool", "Mild", "Warm", "Balmy", "Hot", "Sweltering",
"Scorching"
};
[HttpGet]
public IEnumerable<WeatherForecast> Get()
{
var rng = new Random();
return Enumerable.Range(1, 5).Select(index => new WeatherForecast
{
Date = DateTime.Now.AddDays(index),
TemperatureC = rng.Next(-20, 55),
Summary = Summaries[rng.Next(Summaries.Length)]
})
.ToArray();
}
}
This is the same code to create random sample data that was used in the server-side Blazor project template in
Chapter 2. In this case, the code is hosted in a server-side web service that can be called from the client app using
HTTP.
Shared Project
The shared project is a .NET Standard 2.1 class library that implements code used by the client app and the web
server. The solution template includes just one class in this project, but you can put any code into this project that is
necessary for the client app and server to operate, including CSLA .NET business domain classes.
The WeatherForecast class is the same type used in all the Blazor project templates.
Page 36
This class defines strongly typed properties to store data, and implements the conversion from Celsius to Fahrenheit
temperatures. It is used by the WeatherForecastController in the web server, and in the FetchData page in the
client app.
<Project Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup>
<TargetFramework>netstandard2.1</TargetFramework>
<RazorLangVersion>3.0</RazorLangVersion>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Microsoft.AspNetCore.Blazor" Version="3.2.0-preview1.20073.1" />
<PackageReference Include="Microsoft.AspNetCore.Blazor.Build" Version="3.2.0-preview1.20073.1"
PrivateAssets="all" />
<PackageReference Include="Microsoft.AspNetCore.Blazor.DevServer" Version="3.2.0-preview1.20073.1"
PrivateAssets="all" />
<PackageReference Include="Microsoft.AspNetCore.Blazor.HttpClient" Version="3.2.0-preview1.20073.1" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\Shared\BlazorClientExample.Shared.csproj" />
</ItemGroup>
</Project>
The project uses the Microsoft.NET.Sdk.Web namespace so it has access to ASP.NET functionality.
Notice that it targets the .NET Standard 2.1 framework instead of netcoreapp3.1. This is because the code will be
running on top of the mono implementation of .NET, which is running on top of WebAssembly in the browser as I
discussed in Chapter 1.
The .NET Framework, .NET Core, and mono all support .NET Standard 2.0 and 2.1, where only .NET Core 3.1 supports
netcoreapp3.1. This means that any code that runs on mono should target .NET Standard 2.x instead of .NET Core
3.1.
The project also references a number of NuGet packages that provide support for client-side Blazor. Notice that all
of these are preview packages, because client-side Blazor is not yet a released product.
Finally, you can see where the project references the BlazorClientExample.Shared project, so the types in that
project are available for use in the client-side app.
CSS Assets
The wwwroot folder in the project contains client-side content such as CSS resources. It also contains the start page
for the app.
Page 37
Blazor itself does not require the use of bootstrap, but the templates are created around those styles.
These are not required if you change the HTML in each page to use a different CSS framework and assets.
You can change or add CSS assets in this folder to control the layout and other aspects of your web UI.
Index HTML Page
The index.html page is the start page for the app. This is the page that is first loaded by the browser, and it hosts
the Blazor WebAssembly framework and your app.
In Chapter 2 I describe how Blazor and Razor Components use a container-based model for composing the content
shown to the user. The index.html page is the outer-most container, and it hosts all the Blazor content.
Your page content, such as Index.razor, is contained within the MainLayout.razor component, which is contained
within the App.razor component.
Here's the markup for the index.html page.
Page 38
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width" />
<title>BlazorClientExample</title>
<base href="/" />
<link href="css/bootstrap/bootstrap.min.css" rel="stylesheet" />
<link href="css/site.css" rel="stylesheet" />
</head>
<body>
<app>Loading...</app>
<div id="blazor-error-ui">
An unhandled error has occurred.
<a href="" class="reload">Reload</a>
<a class="dismiss"> </a>
</div>
<script src="_framework/blazor.webassembly.js"></script>
</body>
</html>
<title>BlazorClientExample</title>
Next it displays some content shown to the user while the app is being loaded into the browser. Because this is a
"single page app" (SPA), it can take some time to download the app's assets, resources, and code before the app can
launch. You can think of this content as the "splash page" for your app.
If an error occurs during app download or startup, the index page controls how error text is displayed to the user.
<div id="blazor-error-ui">
An unhandled error has occurred.
<a href="" class="reload">Reload</a>
<a class="dismiss"> </a>
</div>
<script src="_framework/blazor.webassembly.js"></script>
Although most of Blazor runs on mono in WebAssembly, it does use JavaScript code to bridge between the
WebAssembly and JavaScript engines in the browser. I discussed the need for this in Chapter 1.
Pages Folder
The Pages folder contains most of the UI displayed to your users. The pages in this folder represent web pages, but
you should think of them as smart-client "forms" or "pages" much like you would in a Windows Forms, WPF,
Xamarin, UWP, Angular, or React app. Although these pages are created using HTML and Razor markup, the Blazor
UI framework makes them feel like smart client pages, not traditional server-side web pages.
Page 39
The pages in the Pages folder are rendered for the user in the browser by combining the HTML and Razor markup in
each page, plus the CSS defined in the wwwroot folder, plus templates and components defined in the Shared folder
I will discuss later in this chapter.
Notice that some pages from the server-side Blazor project are not needed in a client-side Blazor project.
The _Host page is replaced by the index page in the wwwroot folder, as I discussed earlier in this chapter.
The primary need for an error page in a web site is due to the potential for unforeseen server errors. Because a
client-side Blazor app runs entirely on the client device, there is no such thing as a server error, and so there is no
need for the Error page.
The Index, Counter, and FetchData pages are very similar to their server-side counterparts.
Index Page
Each page displayed to the user by Blazor is a combination of the page plus content from the Shared folder. I will
discuss the Shared folder later. Right now it is important for you to understand that each page in the Pages folder
represents only the specific page content, not other content such as headers, footers, navigation, or anything else
outside the page itself.
The first page displayed to the user is the Index page.
Page 40
@page "/"
<h1>Hello, world!</h1>
Notice the @page directive at the top of the code. This directive is part of the routing mechanism used by Blazor. In
more advanced pages, later in this book, you will see that a page can have multiple @page directives if it handles
more than one route. The Index page handles one route, the default, and so has one directive.
Most of the page is HTML displaying content to the user. This page could have more advanced Razor markup as
well, but the basic template simply displays text.
The SurveyPrompt element is referencing something called a Blazor component, which is a reusable bit of UI. I'll
discuss the SurveyPrompt component later in this chapter, as it is located in the Shared folder of the project. And I'll
discuss how you can create and use Blazor components in more depth in Chapter 4.
Counter Page
The Counter page implements functionality that allows the user to interact with the app. Specifically, it provides a
button that the user can click on, and each click increments a numeric value.
@page "/counter"
<h1>Counter</h1>
@code {
private int currentCount = 0;
There are two new concepts in this page: the properties on the button element, and the @code block.
The button element displays a standard HTML button, but the @onclick property is a special Razor markup item.
This property causes Blazor to invoke the IncrementCount method each time the button is clicked.
Remember that all rendering and code executes on the client in the browser. This means that each time the user
clicks the button the IncrementCount method is directly invoked on the client. The method increments the
currentCount field.
Blazor automatically monitors the values of fields in a page, so it is immediately aware that currentCount has
changed. Because the value has changed, Blazor updates the display so the user sees the change.
FetchData Page
The FetchData page demonstrates how to have UI code invoke a service to retrieve data, and then how to display
that data to the user. It also provides an opportunity for me to discuss how Blazor leverages the dependency
injection (DI) service provided by .NET Core.
Near the top of the page there are a couple new directives.
Page 41
@page "/fetchdata"
@using BlazorClientExample.Shared
@inject HttpClient Http
The @using directive works like the using statement in C#, bringing a namespace into scope for use within the page.
The @inject directive indicates that Blazor should use the .NET Core DI framework to inject an instance of the
HttpClient type with the variable name Http. You can use the Http variable throughout the page to interact with
the object provided by dependency injection.
I will show you the code where the HttpClient type is mapped to a specific instance later in this chapter when I
discuss the Startup.cs file.
The HttpClient instance is used in the @code block in the page.
@code {
private WeatherForecast[] forecasts;
You can see how the GetJsonAsync method is used to call the code in the ASP.NET Core web site's
WeatherForecastController. Remember that controller generates sample data and returns it as json.
As I discussed with the Counter page, fields declared within the page are available for data binding to the UI. In this
case, the forecasts field is loaded with data from a remote service, so the array of forecast data can be displayed to
the user.
You can look at the HTML and Razor markup in the page to see how the forecasts field is used. A foreach loop is
used to build UI for each row of data in the array.
Because this is read-only data, the "data binding" used in this case are simple @ directives to retrieve data values
from each item in the row. Razor markup understands that it should apply ToString to any standalone value so that
value can be rendered in the browser as text. This means that @forecast.TemperatureC, a numeric value, is
converted to text and rendered in the browser for the user.
If you have used Razor markup in ASP.NET MVC or Razor Pages this will be very familiar to you.
Although there are Blazor-specific variations for some Razor markup, this use of Razor markup has remained
unchanged for many years.
Later in this book I will discuss data binding in much more depth. Data binding can be used to not only display
values, but to enable rich two-way bindings between UI elements and fields (or object properties) when creating
data entry forms or other types of UI where the user provides input to the app.
Page 42
Shared Folder
Earlier in this chapter I discussed the Index page and how what the user sees in the browser is a combination of
each page plus content from the Shared folder. When you look at a page, all the content in the left-hand navigation
area, the title and the header with About all come from the files in the Shared folder.
The default Blazor template provides the user with a responsive web experience. For example, when the browser
window is small enough, the navigation area is moved to a hamburger menu so the user's focus remains on the
content of each page.
Page 43
You can change all this content by editing the files in the Shared folder and the CSS from the wwwroot folder. The
Shared folder contains MainLayout.razor and NavMenu.razor files you will edit to control the overall appearance
and navigation for the app.
The Shared folder also contains the SurveyPrompt component used by the Index page.
MainLayout Page
The MainLayout page defines the overall layout of all pages in the app.
Page 44
@inherits LayoutComponentBase
<div class="sidebar">
<NavMenu />
</div>
<div class="main">
<div class="top-row px-4">
<a href="http://blazor.net" target="_blank" class="ml-md-auto">About</a>
</div>
You can see how this page specifies where the NavMenu and @body content will be displayed, as well as any global
header content such as the About link.
Although I will discuss this in more detail later in the chapter, you should be aware that nearly any page can be
displayed as a component of another page, just like this page is displaying the content of the NavMenu page. The
ability to compose pages by using other page/component content is one of the most important features of the
Blazor UI framework.
NavMenu Page
The NavMenu page not only displays the navigation menu, it also implements events and code to handle expanding
the menu when the user clicks the hamburger menu button when it is visible.
The page relies on the navbar-toggler CSS style to control whether the hamburger menu button is visible.
If the button is visible, then when the user clicks on the button the ToggleNavMenu method is invoked from the @code
block.
@code {
private bool collapseNavMenu = true;
This code toggles the collapseNavMenu value between true and false. More importantly, you can see how this
value is used to change the NavMenuCssClass value between collapse and null. The collapse CSS style is used to
alter the appearance of the navigation menu content in the page.
If the text of the NavMenuCssClass field is collapse then CSS controls whether the navigation menu content is visible
to the user. If the value is null then the content is visible.
The collapse CSS is in the wwwroot folder in the site.css file. It is within a @media element so it is only expressed
Page 45
for certain screen sizes.
When the width is 786px or greater then the sidebar will not collapse, otherwise it collapses.
.sidebar .collapse {
/* Never collapse the sidebar for wide screens */
display: block;
}
The interaction between the NavMenu HTML and Razor, the CSS styles, and the C# code works together to provide
the user with a responsive experience for the navigation menu.
SurveyPrompt Component
I will discuss Blazor components in more detail in Chapter 4, but the SurveyPrompt component will give you a
glimpse at what is possible. The component is basically a Blazor page, containing HTML, Razor markup, and code.
<span class="text-nowrap">
Please take our
<a target="_blank" class="font-weight-bold" href="https://go.microsoft.com/fwlink/?
linkid=2109206">brief survey</a>
</span>
and tell us what you think.
</div>
@code {
// Demonstrates how a parent component can supply parameters
[Parameter]
public string Title { get; set; }
}
A component is a self-contained bit of UI functionality that can be placed into any other page or component. You
might remember that the Index page makes use of this component.
You can see how the component name (based on the file name) becomes an element name when used in another
page. And you can see how the Title parameter in the component is decorated with a Parameter attribute so Blazor
understands that this is a parameter value that is provided by the page when the component is used.
Again, I'll discuss these concepts in more depth in Chapter 4.
There are also a number of files at the project level. These files contain markup or code that affects the overall Blazor
app.
_Imports Page
The _Imports page defines @using directives that apply to all pages in the app.
@using System.Net.Http
@using Microsoft.AspNetCore.Components.Forms
@using Microsoft.AspNetCore.Components.Routing
@using Microsoft.AspNetCore.Components.Web
@using Microsoft.JSInterop
@using BlazorClientExample.Client
@using BlazorClientExample.Client.Shared
Page 46
If you consistently use a namespace in your pages and want it to be globally available then you can add it to this file.
App Page
The App page manages routing for the app, and controls what the user sees if they navigate to an invalid URL.
<Router AppAssembly="@typeof(Program).Assembly">
<Found Context="routeData">
<RouteView RouteData="@routeData" DefaultLayout="@typeof(MainLayout)" />
</Found>
<NotFound>
<LayoutView Layout="@typeof(MainLayout)">
<p>Sorry, there's nothing at this address.</p>
</LayoutView>
</NotFound>
</Router>
You may edit this page to alter routing, alter the text displayed to the user if a route doesn't exist, and to implement
some authentication and authorization features. I'll discuss authentication and authorization in Chapter 5.
Program.cs
A client-side Blazor app, like most modern .NET Core apps, has Main method that acts as a single entry point for the
app. This is the first code executed when the app is started.
This is also where you implement any configuration code for the app, such as adding services for use with
dependency injection, or configuring services.
await builder.Build().RunAsync();
}
}
The builder field has a Services property that allows you to configure the .NET Core dependency injection
framework, providing the DI framework with type mappings for any types that will be injected into your pages or
other code.
Blazor automatically registers some types for DI behind the scenes, including the HttpClient type used in the
FetchData page. You can register other types in this method as needed by your app code. Later in this book you will
see how to configure the types required by CSLA .NET.
You can implement other configuration code in this method as needed by your app code. Later in this book you will
see how to configure CSLA .NET by adding code to the Main method.
Page 47
Conclusion
At this point you should understand how to create a client-side Blazor project in Visual Studio, including a
standalone app, or one hosted in ASP.NET Core. You should also understand all the folders and files that are
provided in the default project templates. Let's move on to discuss Blazor components and how to create a Blazor
class library that contains reusable UI elements.
Page 48
Chapter 4: Blazor Features
In Chapter 2 and Chapter 3 you were exposed to some concepts that I glossed over, specifically:
1. Blazor components
2. Page and component lifecycle
3. Routing and navigation
4. Data binding
In this chapter I will discuss these Blazor features in more depth. In many ways these features are the centerpiece of
what makes the Blazor UI framework so compelling and powerful.
Blazor Components
The Blazor UI framework is designed to support composition of a page from HTML, Razor markup, and Blazor
components.
A Blazor component is a reusable unit of UI that can be composed into pages or other components.
It turns out that a page is also a component, so it is possible to treat a page as either a page that the user can
navigate to directly, or as a component that can be embedded into another page.
<span class="text-nowrap">
Please take our
<a target="_blank" class="font-weight-bold" href="https://go.microsoft.com/fwlink/?
linkid=2109206">brief survey</a>
</span>
and tell us what you think.
</div>
@code {
// Demonstrates how a parent component can supply parameters
[Parameter]
public string Title { get; set; }
}
This defines some UI and behavior, and is contained in a file named SurveyPrompt.razor. The file name defines the
component name, and the contents of the file represent the UI that can be composed into pages or other
components. For example, it is used in the Index page.
You can use the SurveyPrompt component in any page or component where you want to leverage its preexisting UI
and behaviors.
Component Parameters
Components can accept parameters. The SurveyPrompt component accepts a Title parameter.
When you create a component that accepts a parameter you will use the Parameter attribute to declare any fields
Page 49
that should be treated as parameters. These fields must be public in scope.
[Parameter]
public string Title { get; set; }
This attribute indicates that when the component is used in another component or page that a value can be
provided using the field name. So when the SurveyPrompt component is used in the Index page, the Index page
provides a value for the Title parameter.
A component can accept multiple parameters by declaring multiple fields, each with the Parameter attribute.
When a component is used in a page there is no requirement that a value be provided to any of the parameters.
They are all optional. When you create a component you should provide default values or otherwise handle the case
that no value was provided for parameters.
Cascading Parameters
Because you can nest components inside other components there will be times when you need to pass a parameter
from the page that's hosting the first component into the nested component.
Passing parameters through one component into a nested component is supported by using the concept of
cascading parameters. This is supported by a Razor element called CascadingValue and a .NET attribute called
CascadingParameter.
You should know that there are many variations on how you can declare and use cascading parameters.
I am showing you just one approach in this chapter. In my opinion, this approach provides the best
readability and maintainability, but there are cases where other options may be useful.
Add a Parent component to the BlazorClientExample.Client project from Chapter 3. Add it to the Shared folder
by right-clicking on the Shared folder in Solution Explorer and choosing the Add | New Item option. In the New Item
dialog choose the Razor Component item.
Don't worry if you are not using Visual Studio 2019 preview and can only work with server-side Blazor.
Page 50
You can follow these instructions in the BlazorServerExample project from Chapter 2 and everything will
work the same.
Edit the resulting component to accept and display a cascading parameter value.
<h3>Parent</h3>
@code {
[CascadingParameter(Name = "Title")] string Title { get; set; }
}
Notice the use of the CascadingParameter attribute. It has a Name parameter that defines the name of the cascading
parameter value that should be mapped into this component's Title field. Also notice that this field does not have
to be public like a regular parameter, and in fact you should prefer to make these fields private so their value only
flows through the cascading parameter subsystem.
Now edit the Index page to define the cascading parameter value and to display the Parent component.
Page 51
@page "/"
<h1>Hello, world!</h1>
@code {
string Title = "42";
}
You can see how the CascadingValue element is used to define the context for the cascading value.
The Value parameter defines the expression that is used to retrieve the actual value. You can see how the Title field
is set to a real value in the @code block.
The Name parameter defines the name by which this cascading parameter value is referenced in child components.
This value matches the value used in the CascadingParameter attribute in the Parent component.
This probably seems like a lot of work to pass a parameter to the Parent component. The advantage of this
approach is that you can nest another component inside Parent.
Add another component to the Shared folder. Name this one Child, and update its code to also use the cascading
parameter value.
<h3>Child</h3>
@code {
[CascadingParameter(Name = "Title")] string Title { get; set; }
}
You can see it uses the same Name value in the CascadingParameter attribute. This will work because all components
loaded within the context of the top-level CascadingView element from the Index page will have access to this
cascading parameter value.
Now update the Parent component to include the new Child component.
<h3>Parent</h3>
@code {
[CascadingParameter(Name = "Title")] string Title { get; set; }
}
When you run the app you'll see how the cascading parameter value is displayed from the Parent and Child
components.
Page 52
The cascading parameter concept in Blazor enables powerful composition scenarios for building your UI from many
components, allowing you to nest components without having a parent component needing to be aware of the
parameters required by child components.
Pages as Components
Any Blazor page can be treated as a component. The only real difference between a "page" and a "component" is
that pages have one or more @page directives so they support routing.
When used as a component, any @page directives in a page are ignored.
For example, in the BlazorClientExample.Client project there is a Counter page.
@page "/counter"
<h1>Counter</h1>
@code {
private int currentCount = 0;
Because it has a @page directive the user can directly navigate to this page. You can also use this as a component in
other pages. For example, you can compose the Counter component into the Index page.
Page 53
@page "/"
<h1>Hello, world!</h1>
<Counter />
The result is that the UI and behavior of the Counter page is included in the Index page.
The original Counter page is entirely unaffected. The Counter component in the Index page is a totally separate
instance of the page/component.
It is literally a separate instance. Each time a component or page is loaded, Blazor creates an instance of
that component with its own private fields and state. Exactly what happens when you create instances of your
own classes.
Because pages are also components, I will refer to everything as a component in this part of the chapter.
Initialization
Page 54
When a component instance is created you have the opportunity to initialize the instance.
OnInitialized
OnInitializedAsync
If your initialization is synchronous override the OnInitialized method, and if your initialization is asynchronous
you must override the OnInitializedAsync method.
It is important to understand that initialization occurs exactly one time per component. You may be tempted to use
these methods to load data values into your component, but I recommend you avoid that and instead load any data
values after the parameters have been set.
Setting Parameters
After a component has been initialized, any parameters provided by the parent (or routing) will be set. You can
override a method to alter how parameter values are set within your component.
SetParametersAsync
By default parameter values are set based on Parameter and CascasdingParameter attributes in your component.
Once the parameters have been set you can override methods to initialize your component based on the now-
available parameter values.
OnParametersSet
OnParametersSetAsync
Choose the appropriate method based on whether you will implement sync or async code.
I recommend using one of these methods to initialize your component, especially if you are loading the component
with data from a service or other external source.
These methods are invoked before the component renders.
Component Rendering
Before your component is rendered Blazor invokes a method.
ShouldRender
You can override ShouldRender and return a bool value indicating whether rendering should occur. If you return
false then rendering will not occur.
After Blazor has rendered your component you can override methods.
OnAfterRender
OnAfterRenderAsync
Choose the appropriate method based on whether you will implement sync or async code.
These methods accept a bool parameter indicating whether this is the first time the component has been rendered.
You can use this parameter to do first-time initialization of your component.
@page "/"
This route indicates that the Index page is the default, root level, page.
The Counter page also has a directive, even though you have seen how this page can also be used as a component.
@page "/counter"
It is possible for a page to have multiple routes. Most commonly these alternate routes will be used to provide
parameters to the page through the URL. Edit the Counter page to have an alternate route with a parameter.
@page "/counter"
@page "/counter/{start:int}"
This indicates that the user can navigate to /counter or something like /counter/42.
Now add a start field with a Parameter attribute, and an OnParametersSet method override in the @code block.
[Parameter]
public int start { get; set; }
The start field is a parameter, and its name corresponds to the name used in the @page directive. This is how Blazor
knows to map the parameter value from the route into the field.
Notice that the field is an int value, matching the constraint applied to the value in the @page directive. If you don't
Page 56
apply a constraint the default string type is used.
The result is that when the user navigates to the Counter page using the new route, the counter value will start at the
provided value instead of zero.
<a href="/counter/42">Counter</a>
This is a simple href link using the relative URL and passing a parameter value to the target page.
You can then add a button element to the page that runs some code.
<button @onclick="GoToCounter">Counter</button>
Finally, you can implement the GoToCounter method in the @code block.
This code doesn't do anything differently from the href link used in the last section, but it does allow you to do the
navigation from code instead of markup.
Data Binding
Creating data-focused user experiences requires getting data out of your objects or data model so it can be
displayed in the UI. And it requires getting data out of the UI after being edited by the user, so the values can be put
back into your objects or data model.
Although this can be done by hand, writing code to move the data back and forth between the UI view and your
model, data binding technologies have been automating this process for more than two decades. It is also the case
that nearly every data binding framework it better than its predecessors, and Blazor is no exception. The data
binding model provided by Blazor is powerful and yet easy to use.
Blazor supports binding to read-only values, read-write values, and provides an EditForm component to make it
easy to create simple data entry forms.
For the code in this section of the chapter, create a new page in the Pages folder named Binding.
Page 57
Add a @page directive for routing.
@page "/binding"
<h3>Binding</h3>
@code {
Also add a PersonEdit class at the top level of the project, along side Startup.cs.
using System;
namespace BlazorClientExample.Client
{
public class PersonEdit
{
public string Name { get; set; }
public DateTime Birthday { get; set; }
public int Age { get => (DateTime.Now - Birthday).Days / 365; }
}
}
I will use these types as the start point for the rest of this section of the chapter.
To make it easy to access the Binding page, edit the NavMenu page in the Shared folder and add an element to the
existing ul list.
Page 58
<li class="nav-item px-3">
<NavLink class="nav-link" href="binding">
<span class="oi oi-list-rich" aria-hidden="true"></span> Binding
</NavLink>
</li>
This will add an item to the app's navigation menu so you can easily open the Binding page.
Displaying Values
You have already seen examples of one-way data binding to display values to the user in the Counter and FetchData
pages.
One-way binding is where values flow from your fields, data model, or domain objects to the UI for display.
Edit the Binding component to declare a field based on the PersonEdit class and to create an instance of the class.
Also add markup to display the properties of the person field.
@page "/binding"
<h3>One-Way Binding</h3>
<p>
@(person.Name)
<br />
@(person.Birthday)
<br />
@(person.Age)
</p>
@code {
private PersonEdit person;
In this example I am using the @() syntax for the binding expression instead of the simpler @ syntax. For simple types
like PersonEdit you can use either approach. These lines are equivalent.
@person.Name
@(person.Name)
Later in this book I will be using generic types, and then you must use the @() syntax. If you use the simpler syntax
then the Razor parser will get confused by the < and > characters because they are part of HTML as well as being
how C# designates generic type values.
When you run the app and navigate to the Binding page you should see the values displayed.
Page 59
The output of the Birthdate property is not ideal. Fortunately that's easily addressed with a small change to the
binding expression in the page.
@(person.Birthday.ToShortDateString())
This is possible because the way one-way data binding works is that Blazor executes the C# expression after the @
and returns its ToString value. If a value is displayed in a format that is undesirable, you can use standard C#
formatting techniques to change how the value is displayed.
Input Binding
Displaying values using one-way binding leverages the same Razor markup that's been in use for many years. The
two-way binding for input controls is new to Blazor, and is designed around providing interactive user experiences.
Basic Input Binding
Edit the Binding page and add an input element so the user can edit the Name property.
<h3>Two-Way Binding</h3>
<p>
<input @bind="person.Name" />
</p>
When you run the app you will see the Name property value in both sections of the page. When you edit the name
value and tab out of the input element you can see that the original one-way binding immediately updates.
This is because both UI elements are data bound to the same object property, and when that property is changed it
Page 60
automatically updates all data bound UI elements.
Controlling Input Binding
Depending on your user requirements you may need binding to happen when the user tabs out of the field, or on
each key press.
Edit the Binding page and add another input element.
When you run the app you can experiment with typing in the first input element and tabbing out to see the change.
And you can type in this new input element and see how the Name property value (and all UI elements data bound to
the property) update as you press each keystroke.
This applies not only to text input elements, but to other types of input element and other HTML elements that
allow for user input, such as a dropdown list or checkbox.
EditForm Binding
Binding to individual properties is powerful. Blazor adds to this by providing an EditForm component that makes it
easier to create data entry forms with a common display for any validation errors that might occur within the form.
By default only data attribute validation is supported. In Chapter 7 I will discuss how you can use
EditForm along with the richer set of business rules supported by CSLA .NET.
The EditForm component relies on a set of other input components, all of which inherit from InputBase<T>. These
components include:
InputText - allows user to enter a single text field
InputTextArea - allows user to enter multiple lines of text
InputNumber - allows user to enter a number
InputDate - allows user to enter a date
InputCheckBox - allows user to click a checkbox
InputSelect - allows user to select a value from a dropdown list
You can use standard input tags in an EditForm too, but these input components abstract common behaviors
required in most cases, and so they offer improved readability of code, as well as enhanced maintainability.
You can also create your own input components by inheriting from InputBase. That topic is outside the scope of this
book.
Now let's create a data entry form using the EditForm component.
Creating a Basic Form
Add a SimpleForm Razor Component to the Pages folder in the project, and update its contents as shown here.
Page 61
@page "/simpleform"
<h3>EditForm Example</h3>
@code {
private PersonEdit person;
The @code block is identical to the binding page you created earlier in this chapter. The markup is different however,
because it sets up an EditForm component that is bound to the person field.
It is important to understand that the person field could be null, and that the EditForm component will throw an
exception when attempting to bind to a null value. Because of this, you must always wrap any EditForm component
in an if statement to handle the possible null value of the the Model property.
Within the form is a button so the user can submit the form values (after you add input components).
Also add markdown to the NavMenu component in the Shared folder so it is easy to navigate to this new page.
With that done, let's discuss how a form is submitted by the user.
Handling Form Submission
When the user clicks the button to submit the form, the form's contents are routed to the @code block. There are two
ways to handle the form data.
First, you can handle the EditForm component's OnSubmit event. This is done by binding to the event.
To see the results of a simulated save operation, add a div tag above the EditForm component.
<div>@statusText</div>
Page 62
Then implement the event handler method in the @code block.
In this FormSubmitted method you can use the EditContext parameter to trigger validation and see the results of
validation. You can also access the form's Model property, making it possible to save the data from the form.
The second approach is to handle two events instead of a single event. These two events come from the EditForm
component and allow you to implement code for valid input, and separate code if there's a validation error in the
form.
1. OnValidSubmit - the form data is valid
2. OnInvalidSubmit - the form data is invalid due to a validation rule
To do this, start by binding the two events.
<EditForm Model="@person"
OnValidSubmit="@FormValid"
OnInvalidSubmit="@FormInvalid">
In this case validation occurs automatically, and the appropriate event is raised depending on whether the form data
is valid or invalid. Notice that you still have access to the EditContext parameter, so you can interact with the
validation and model data.
You must choose to use OnSubmit or to use the other two events, you can't mix the two models
together.
In the rest of this chapter I will be using the OnSubmit event, with the EditForm component declared as shown.
Page 63
<EditForm Model="@person" OnSubmit="@FormSubmitted">
<InputText @bind-Value=person.Name />
<InputDate @bind-Value=person.Birthday
ParsingErrorMessage="Birthday must be a date" />
<div>@person.Name is @person.Age years old</div>
<input type="submit" value="Save" class="btn btn-primary" />
</EditForm>
Running the app at this point will allow you to see and interact with the page.
Notice how the InputText component appears as a simple input element, and the InputDate component appears as
a nice date input element in the browser. And the Age property is displayed using one-way binding in a div tag.
You can also see how the Age property updates in the object and display as the Birthdate property is changed. This
demonstrates how Blazor is typically able to automatically detect any changes to the model and update the display
accordingly.
This automatic update the UI works because changing the Age property is as a result of a UI event
(changing the Birthdate property). If a property on the model changes because of some background task
(not a UI event) then the display may not be automatically updated.
At this point the form works to enter and submit data. In most apps a data entry form will at least implement some
basic validation rules, so let's discuss how that works.
Page 64
Running and Displaying Validation Rules
Blazor interacts with the data annotation attributes from the System.ComponentModel.DataAnnotations namespace.
These attributes provide basic validation rules, including:
Required - a string property is required
StringLength - a string property has a max length
Range - a property has min and max values
RegEx - a property must match a regular expression
Edit the PersonEdit class and use the Required attribute to make the Name property required.
[Required]
public string Name { get; set; }
Then edit the SimpleForm page and add a DataAnnotationsValidator component inside the EditForm component.
This new component will cause Blazor to run all the data annotations rules when the form is submitted.
If you are using the OnValidSubmit and OnInvalidSubmit events on the EditForm component, the appropriate event
will now be raised based on the validation rules.
If you are using the OnSubmit event, as in this chapter, you need to check the validation results in your event handler
method.
The EditContext parameter provides access to all the validation messages generated by the data annotations
attributes. If there are no messages, then no rules have been broken.
Run the app, blank the Name field, and click the button to see the result.
Page 65
You could write code to display those messages to the user, but you don't need to do that work thanks to the
ValidationSummary and Validation components.
Now run the app, blank the Name field, and click the button.
The bullet list is generated from the ValidationSummary component, and the single line message under the Name
Page 66
field is generated by the Validation component.
You can use either or both of these components depending on the type of UI you are creating for your users.
Notice that the OnSubmit event was not raised in this case. When you use any ValidationSummary or Validation
components within an EditForm the OnSubmit event will only be raised if no validation rules are broken.
At this point you should understand how you can create pages to display and edit data using one-way or two-way
data binding, or using the EditForm component.
Event Binding
Binding applies to events and methods much like it does to fields and properties with data. You have already seen
examples of event binding to handle the click of a button element and events on the EditForm component.
Binding to a Method
For example, in the Counter page there's a button that binds to a method.
The method is often referred to as an event handler because it executes in response to the button element's onclick
event.
When this element is clicked it doesn't call a named method, it directly runs the lambda expression. This expression
decrements the currentCount value.
This can be a powerful technique when some simple state or flag needs to be altered in response to the user
interacting with the page. For example, a button click might toggle a bool value which is data bound so it hides or
displays a section of the page.
Passing Parameters to a Method
Another variation on this lambda concept is to pass parameter values to a method. The simple approach of binding
an event to a method call doesn't allow for parameters to be passed to the method, but you can overcome that
limitation using a lambda.
Add a new button element to pass a parameter.
Page 67
private void IncrementCount(int increment)
{
currentCount += increment;
}
Data binding in Blazor enables display and formatting of data, input of data, and binding code to events. These
combine to enable very powerful and responsive UI scenarios with relatively little code.
Conclusion
In this chapter I discussed Blazor components and pages, the component lifecycle, routing and navigation, and data
binding. These features will be used throughout the rest of the book as I talk about how CSLA .NET and Blazor work
together, and finally walk through a complete app UI built using Blazor that leverages a CSLA based business
domain layer. Before getting into CSLA .NET it is important to cover two more topics: authentication and
authorization, and building multi-headed Blazor apps.
Page 68
Chapter 5: Authentication and Authorization
Authentication is the process of having a user prove their identity. Authorization uses the now-known user identity
to apply business rules about what the user can and can't do within the app.
The .NET platform supports role-based and claims-based authorization. In ASP.NET Core, including Blazor, the user's
identity is maintained in a ClaimsPrincipal and ClaimsIdentity that supports both models. Outside of ASP.NET
Core it is possible to use the older IPrincipal and IIdentity types, which only support role-based authorization.
No Authentication
This is the default option, and selecting this option means the project template will not add any authentication
scaffolding into the project.
If you are implementing custom authentication this is the option you will choose.
Page 69
The dialog gives you the option to use a local or cloud-based SQL database to store the user information.
You will likely choose this option if you do not already have an authentication process in your organization, as this
option provides a turnkey solution for user management. If the user is authenticated your app will get a user identity
object representing the roles defined for the user in the SQL database tables.
Work or school authentication delegates the authentication process to Microsoft Azure. If the user is authenticated
your app will get a user identity object representing the AAD groups the user belongs to in your organization.
Windows Authentication
Before AAD became the defacto standard, many organizations maintained their own private Windows Active
Directory (AD) in their data centers. This authentication option in Blazor allows your users to log into your app using
Page 70
their regular AD credentials, probably the same user id and password they use to log into Windows or other intranet
resources.
If your organization still uses AD to authenticate users this may be an attractive option.
At this point you should understand the basic capabilities of the built-in authentication options provided by server-
side Blazor. If your oganization uses AAD or AD you should prefer to use those options as they provide single sign-
on for your users. If you have no existing authentication model at all, the individual user account option may make
sense.
If you do have an existing authentication mechanism in your organization it is possible to implement custom
authentication.
Select No Authentication
When you get to the Create a new Blazor app dialog, notice the Change link under Authentication. You don't need to
click it, because the default is No Authentication, but feel free to click the link and explore the options.
Page 71
Once the solution is open in Visual Studio you can proceed to the next step.
Here is the complete code for the page content. I'll discuss the important parts.
Page 72
@page
@model BlazorAuthServer.Account.Pages.LoginModel
@addTagHelper *, Microsoft.AspNetCore.Mvc.TagHelpers
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<title>Blazor Custom Authentication</title>
<base href="~/" />
<link rel="stylesheet" href="~/css/bootstrap/bootstrap.min.css" />
<link rel="stylesheet" href="~/css/bootstrap4-toggle.min.css" />
<link rel="stylesheet" href="~/css/site.css" />
</head>
<body>
<app>
<div class="sidebar">
</div>
<div class="main">
<div class="top-row px-4">
<a href="https://docs.microsoft.com/aspnet/" target="_blank">About</a>
</div>
<div class="content px-4">
@if (!string.IsNullOrWhiteSpace(Model.ErrorText))
{
<div class="alert alert-danger">@Model.ErrorText</div>
}
<div>
<form method="post">
<div class="form-group">
<label asp-for="Username"></label>
<input asp-for="Username" class="form-control">
<div class="invalid-feedback"></div>
</div>
<div class="form-group">
<label asp-for="Password"></label>
<input asp-for="Password" class="form-control" type="password">
<div class="invalid-feedback"></div>
</div>
The page displays Model.ErrorText to provide the user with any error information.
Within the form element the page allows the user to enter username and password values using standard Razor
markup data binding. The Login button triggers a postback to the server that is handled by the OnPostAsync method
in the page's code.
Here's the code behind the page.
Page 73
[AllowAnonymous]
public class LoginModel : PageModel
{
[BindProperty]
public string Username { get; set; }
[BindProperty]
public string Password { get; set; }
[BindProperty]
public string ErrorText { get; set; }
if (!string.IsNullOrWhiteSpace(Username) &&
Users.TryGetValue(Username.ToLower(), out string pw))
{
if (pw == Password)
{
var identity = new ClaimsIdentity("password");
identity.AddClaim(new Claim(ClaimTypes.Name, Username.ToLower()));
var principal = new ClaimsPrincipal(identity);
var authProperties = new AuthenticationProperties();
await HttpContext.SignInAsync(
CookieAuthenticationDefaults.AuthenticationScheme,
principal,
authProperties);
return LocalRedirect(Url.Content("~/"));
}
}
ErrorText = "Invalid credentials";
return Page();
}
}
Notice the use of the AllowAnonymous attribute. This is important because the user won't be authenticated when they
login, so they are an anonymous user. The Login page won't allow an anonymous user to do a postback, so this
attribute is necessary for the page to operate.
Also notice that the "security database" in this sample is a static dictionary of hard-coded username/password
values. In a real app you would be interacting with a database, LDAP server, or some other location where the user's
identity can be verified.
The BindProperty attribute is used to indicate the fields available for data binding in the page. These fields
correspond to the data binding expressions in the Razor markup.
The OnPostAsync method is invoked when the user triggers a postback by clicking the Login button on the page. If
the user entered a value for the Username field, that value is used to look up the required password from the static
dictionary.
Again, in a real app this is where you would make any appropriate database or service call to validate the user's
credentials to ensure they are valid.
If the user's credentials are valid, the next step is to create a ClaimsIdentity object.
Page 74
var identity = new ClaimsIdentity("password");
identity.AddClaim(new Claim(ClaimTypes.Name, Username.ToLower()));
At a minimum, this identity object will contain the type of authentication used (in this case password) and the
username as a claim of type ClaimTypes.Name. You will probably add other claims as well depending on how your
authorization rules will use those claims.
Claims might be a role, a group, a department, or any other identifying fact about the user that will be
used to determine if or how the user will be allowed to interact with the app.
Once you have an identity object, the next step is to create a ClaimsPrincipal object to contain the identity.
Within .NET the user principal is always the primary object used to manage the user's identity, roles, and claims.
Finally, it is necessary to create a browser cookie to maintain the user's identity over time. The use of a cookie for
this purpose is a time-tested standard within the web server world, because each request from the browser to the
web server always includes the web site's cookies.
This means that each time the browser sends a request to the web server this cookie can be used to determine the
user's identity.
Now that the user is logged in, the page redirects to the Blazor app start page.
return LocalRedirect(Url.Content("~/"));
Once routing and configuration are complete, this page will allow the user to log into the app.
Page 75
Notice that this is a Razor View item, not a Razor Page. This is because there's no need for any UI, just some code to
clear out the cookie.
@page
@using Microsoft.AspNetCore.Authentication
@using Microsoft.AspNetCore.Authentication.Cookies
@attribute [IgnoreAntiforgeryToken]
@functions {
public async Task<IActionResult> OnPost()
{
if (User.Identity.IsAuthenticated)
{
await HttpContext.SignOutAsync(
CookieAuthenticationDefaults.AuthenticationScheme);
}
return Redirect("~/");
}
}
await HttpContext.SignOutAsync(
CookieAuthenticationDefaults.AuthenticationScheme);
return Redirect("~/");
The result is that the user is logged out and is treated as an anonymous user because there's no longer an
authentication cookie.
The user identity needs to be made available to all Blazor components in the app before it can be used.
Page 76
App Content
As I discussed in Chapter 4, Blazor uses cascading parameters to provide values to UI elements, and that is true for
the user identity as well.
To make the user's identity available to all UI elements in the app, edit the App page and wrap the contents in a
CascadingAuthenticationState element. This element provides the user identity as a cascading parameter to all
Blazor components in the app.
Also change the RouteView element to an AuthorizeRouteView element.
<CascadingAuthenticationState>
<Router AppAssembly="@typeof(Program).Assembly">
<Found Context="routeData">
<AuthorizeRouteView RouteData="@routeData" DefaultLayout="@typeof(MainLayout)" />
</Found>
<NotFound>
<LayoutView Layout="@typeof(MainLayout)">
<p>Sorry, there's nothing at this address.</p>
</LayoutView>
</NotFound>
</Router>
</CascadingAuthenticationState>
The AuthorizeRouteView element changes the behavior of Blazor routing so users are not allowed to see pages that
require authorization.
At this point pages exist to allow the user to log in and out of the app, but there's no routing or navigation for the
user to reach these pages.
MainLayout Content
Open the MainLayout page in the Shared folder and edit it to match the following.
@inherits LayoutComponentBase
<div class="sidebar">
<NavMenu />
</div>
<div class="main">
<div class="top-row px-4">
<AuthorizeView>
<Authorized>
Hello, @context.User.Identity.Name
<form method="post" action="/Account/Logout">
<button type="submit" class="nav-link btn btn-link">Log out</button>
</form>
</Authorized>
<NotAuthorized>
<a href="/Account/Login">Log in</a>
</NotAuthorized>
</AuthorizeView>
<a href="https://docs.microsoft.com/aspnet/" target="_blank">About</a>
</div>
I will discuss the AuthorizeView, Authorized, and NotAuthorized elements later in this chapter. You can infer what
they do however, and understand that the Log out and Log in links will appear in the app header depending on
whether the user is currently logged in.
Page 77
Startup Configuration
The final step before the custom authentication implementation will work is to do some configuration as the web
site starts up. This is done in the Startup class.
In the ConfigureServices method, the AddAuthentication method is used to add and configure authentication.
services.AddAuthentication(options =>
{
options.DefaultSignInScheme = CookieAuthenticationDefaults.AuthenticationScheme;
options.DefaultAuthenticateScheme = CookieAuthenticationDefaults.AuthenticationScheme;
options.DefaultChallengeScheme = CookieAuthenticationDefaults.AuthenticationScheme;
})
.AddCookie();
This must occur before adding the Razor Pages and server-side Blazor services.
Then in the Configure method the web site needs to be configured to use authentication and authorization.
app.UseAuthentication();
app.UseAuthorization();
These methods must be called after the UseRouting method is called, and before setting up the endpoints.
At this point you should be able to run the app and successfully log in and out by using the links in upper-right
corner of the UI.
Code running on a client device is always vulnerable to being compromised by a malicious actor. The
user principal and identity can be used on the client to provide a rich user experience, but you can not count
on these values from a security perspective.
Because ASP.NET and Blazor authorization assume the use of the ClaimsPrincipal and ClaimsIdentity types, the
goal of any client-side authentication implementation should be to create a principal and identity in the client app by
using those types.
There are a number of steps required to implement client-side authentication in Blazor, including:
1. Define message types shared between client and server
2. Implement server-side authentication controller (service)
3. Implement client-side types to manage user identity
4. Implement client-side login and logout pages
5. Add client-side navigation to login and logout pages
I will walk you through a very basic example to explain each of these steps, highlighting where you would write
actual user authentication code in a real app.
Page 78
Open Visual Studio 2019 and create a new ASP.NET Core hosted client-side Blazor project named BlazorAuthClient
as described in Chapter 3.
The client app will get the user's credentials and send them to the service for validation.
UserIdentity Class
If the user's credentials are valid, the service should return all user information necessary for the client to create a
ClaimsIdentity object. Create a UserIdentity class in the BlazorAuthClient.Shared project.
Mostly what needs to be returned is a list of claims, represented by the ClaimInfo class. To create a ClaimsIdentity
class you also need to provide the authentication type, and as a convenience the message will also return a bool
indicating whether the user was successfully authenticated.
Controller Implementation
Using the message types you just defined, it is now possible to implement the AuthenticationController. Add a
class to the BlazorAuthClient.Server project in the Controllers folder.
Page 79
[ApiController]
[Route("[controller]")]
public class AuthenticationController
{
private static Dictionary<string, string> Users = new Dictionary<string, string>
{
{ "rocky", "mypassword" },
{ "andrew", "otherpassword" }
};
[HttpPost]
public UserIdentity Post(UserCredentials credentials)
{
var result = new UserIdentity();
if (!string.IsNullOrWhiteSpace(credentials.Username) &&
Users.TryGetValue(credentials.Username.ToLower(), out string pw) &&
pw == credentials.Password)
{
result.IsAuthenticated = true;
result.AuthenticationType = "password";
result.Name = credentials.Username.ToLower();
result.Claims.Add(new ClaimInfo
{ ClaimType = ClaimTypes.Name, Claim = credentials.Username.ToLower() });
}
return result;
}
}
The implementation of this service is virtually identical to the Login page implementation from the server-side
example. You can see how the username and password are validated. If they are valid a claim is added for the
username. As always, this is the minimum claim necessary for a valid user identity.
I am using a dictionary of hard-coded values as a "user database". In a real app you would access a database or
some other service to validate the user's credentials and retrieve the user's claims or roles.
This service returns a UserIdentity message regardless of whether the user was authenticated or not. Both are
"valid" responses from the service, and it is the responsibility of the calling code to examine the IsAuthenticated
value to determine if the result represents a valid user identity.
Now let's move to the client-side Blazor project and implement the authentication that relies on this service.
With that done, edit the _Imports.razor file in the client-side project to add the following lines.
@using Microsoft.AspNetCore.Authorization
@using Microsoft.AspNetCore.Components.Authorization
Page 80
These namespaces contain the types necessary for authentication and authorization in the client-side components.
public CurrentUserService()
{
CurrentUser = new ClaimsPrincipal(new ClaimsIdentity());
}
This service keeps a reference to the current ClaimsPrincipal object that represents the user.
It also exposes an event you can use to know when the current user has changed, which will be useful in keeping the
UI up to date as the user logs in and out of the app.
AuthenticationStateProvider Implementation
Client-side Blazor relies on a AuthenticationStateProvider service to access the current user identity when
necessary. All the built-in authorization elements and attributes make use of this service.
Page 81
You need to create a subclass of this type to provide the current user identity upon request. Add a
CustomAuthenticationStateProvider class to the top level of the client-side project.
The important part of this class is the GetAuthenticationStateAsync method. This is the method called by Blazor
any time the framework needs access to the current user identity.
Startup Configuration
Now that you have types to keep the user's identity in memory, and to provide the identity to Blazor on-demand, it
is necessary to configure the app to use these types.
In a server-side Blazor app edit the Startup class and add these lines to the ConfigureServices method.
services.AddAuthorizationCore();
services.AddSingleton<
AuthenticationStateProvider, CustomAuthenticationStateProvider>();
services.AddSingleton<CurrentUserService>();
In a client-side Blazor app edit the Program class and add these lines to the Main method.
builder.Services.AddAuthorizationCore();
builder.Services.AddSingleton<
AuthenticationStateProvider, CustomAuthenticationStateProvider>();
builder.Services.AddSingleton<CurrentUserService>();
The AddAuthorizationCore method tells Blazor that it can access the user identity as necessary.
The next line adds a singleton object, so one instance for the lifetime of the app, so Blazor understands to use the
CustomAuthenticationStateProvider to retrieve the current user identity when necessary.
And finally, a singleton for CurrentUserService is added, because that is required by the
CustomAuthenticationStateProvider type. The CurrentUserService instance will be used by pages within the app
as well.
At this point the basic plumbing is in place so Blazor understands that it can use authorization, and has the type
information necessary to retrieve the user identity. Let's implement the pages and navigation so the user can log in
and out of the app.
Login Page
Add a Login Blazor Component to the Pages folder. It will need a @page directive along with some @using and
@inject directives.
Page 82
@page "/login"
@using BlazorAuthClient.Shared
@using System.Security.Claims
@inject HttpClient Http
@inject NavigationManager NavigationManager
@inject CurrentUserService CurrentUserService
Notice how the CurrentUserService singleton is injected, allowing the Login page to change the current user
identity if appropriate.
The Razor markup in the page needs to show any error text, and allow the user to enter their credentials.
<h3>Login</h3>
@if (!string.IsNullOrWhiteSpace(ErrorText))
{
<div class="alert-danger">@ErrorText</div>
}
<table class="table">
<thead>
<tr>
<th></th>
<th></th>
</tr>
</thead>
<tbody>
<tr>
<td>Username</td>
<td><input @bind="Credentials.Username" /></td>
</tr>
<tr>
<td>Password</td>
<td><input @bind="Credentials.Password" type="password" /></td>
</tr>
</tbody>
</table>
Finally, the @code block needs to expose fields for data binding and implement the LoginUser method to handle the
button element's click event.
Page 83
private UserCredentials Credentials { get; set; } = new UserCredentials();
private string ErrorText { get; set; }
The LoginUser method sends the credentials to the Authentication service in the web site.
The Credentials field will have the username and password values because it is data bound to the UI elements, and
the PostJsonAsync method automatically converts the data to JSON and sends it to the service. The service returns a
UserIdentity object indicating whether the credentials were valid.
If the credentials were valid the IsAuthenticated property will be true and the code can create a ClaimsIdentity
based on the information returned from the service.
Using this new ClaimsIdentity, the next step is to create a ClaimsPrincipal and update the current user identity
for the app by using the CurrentUserService.
CurrentUserService.CurrentUser =
new System.Security.Claims.ClaimsPrincipal(identity);
Finally, the NavigationManager instance injected at the top of the page is used to navigate to the start page for the
app.
NavigationManager.NavigateTo("/");
If the user is not successfully logged in the ErrorText field is updated with any exception message or other
information so the user can tell why their credentials weren't accepted.
Note the explicit StateHasChanged method call at the bottom of the method. This will only run if the user was not
Page 84
logged in and ErrorText was updated. This call is necessary because Blazor is not able to automatically detect that
the ErrorText value changed. The StateHasChanged method call ensures that the user will immediately see the error
message.
Logout Page
Similar to the server-side example, the client-side Logout page has no UI and is just code. Add a Logout Blazor
Component to the Pages folder.
@page "/logout"
@using System.Security.Claims
@inject NavigationManager NavigationManager
@inject CurrentUserService CurrentUserService
@code {
protected override void OnInitialized()
{
var identity = new ClaimsIdentity();
CurrentUserService.CurrentUser =
new System.Security.Claims.ClaimsPrincipal(identity);
base.OnInitialized();
NavigationManager.NavigateTo("/");
}
}
This page gets the CurrentUserService singleton and sets the current user identity to an empty (unauthenticated)
ClaimsPrincipal.
App Content
The Blazor AuthenticationStateProvider makes the current user identity available to UI elements. As I discussed in
Chapter 4, Blazor uses cascading parameters to provide this sort of value to UI elements, and that is true for user
state as well.
To make the user's identity available to all UI elements in the app, edit the App page and wrap the contents in a
CascadingAuthenticationState element. This element provides the user identity as a cascading parameter to all
Blazor components in the app.
Also change the RouteView element to an AuthorizeRouteView element.
<CascadingAuthenticationState>
<Router AppAssembly="@typeof(Program).Assembly">
<Found Context="routeData">
<AuthorizeRouteView RouteData="@routeData" DefaultLayout="@typeof(MainLayout)" />
</Found>
<NotFound>
<LayoutView Layout="@typeof(MainLayout)">
<p>Sorry, there's nothing at this address.</p>
</LayoutView>
</NotFound>
</Router>
</CascadingAuthenticationState>
The AuthorizeRouteView element changes the behavior of Blazor routing so users are not allowed to see pages that
require authorization.
Page 85
MainLayout Content
The last step is to update the MainLayout page so the user can navigate to the Login and Logout pages, and to
ensure that the current username is displayed when the user is logged into the app.
Add an @inject directive to the page.
And add a @code block that makes use of the CurrentUserService singleton to refresh the username display any
time the user's identity is changed.
Remember that the OnInitialized method is invoked one time as the component is loaded. The MainLayout
component is loaded just once during the lifetime of the app, so this event handler will be set up just one time.
The event handler calls the StateHasChanged method so Blazor understands that the MainLayout state needs to be
rendered any time the user identity changes.
In the Razor markup of the page, the AuthorizeView, Authorized, NotAuthorized, and Authorizing elements are
used to update the display.
<AuthorizeView>
<Authorized>
Hello, @context.User.Identity.Name
<a href="Logout">Log out</a>
</Authorized>
<NotAuthorized>
<a href="Login">Log in</a>
</NotAuthorized>
<Authorizing>
Authorizing...
</Authorizing>
</AuthorizeView>
I will discuss these elements in more detail later in this chapter. For now it is sufficient to understand that they
control the content shown to the user based on whether the user is currently authorized.
You should now be able to run the app and log in and out by using the hard-coded credentials in the
AuthorizationController class.
Blazor Authorization
Authorization within Blazor, and .NET in general, is built on the assumption that the user's identity, claims, roles, and
other information is available through the current principal object. The user must be logged into the app for the
principal to be available.
The Blazor authorization behaviors are the same for server-side and client-side Blazor. In this section of the chapter I
will use the BlazorAuthServer project to demonstrate how authorization works, but the same attributes and
elements function in client-side Blazor as well unless noted otherwise.
App Configuration
Page 86
Before authorization will work within Blazor, you need to have configured the app for authorization. This is different
for server-side and client-side Blazor.
Server-Side Configuration
Before using authorization in server-side Blazor there are some requirements.
1. Authentication must be configured and implemented
2. Authorization must be configured
3. The App page must be correct
I have already discussed server-side authentication and how to edit the App page to enable authorization.
The only step I want to call out here is that in the Startup class, in the Configure method, you must call the
UserAuthorization method to enable authorization.
app.UseAuthorization();
This, combined with the other requirements, will enable authorization within your components and code.
Client-Side Configuration
As with server-side Blazor, when using authorization in server-side Blazor there are requirements that must be met.
1. Authentication must be configured and implemented
2. Authorization must be configured
3. The App page must be correct
I have already discussed client-side authentication and how to edit the App page to enable authorization.
You also must configure authorization in the Program class. The Main method needs to add authorization as a
service.
builder.Services.AddAuthorizationCore();
This, combined with the other requirements, will enable authorization within your components and code.
AuthorizeRouteView Element
You saw earlier in the chapter how the App page uses the AuthorizeRouteView element so the user is prevented
from viewing pages if they aren't authorized. This element works together with the Authorize attribute I will discuss
later in this chapter.
It is possible to globally control what the user sees in place of a page when they are not authorized, or when
authorization is still occurring. This is handled by the NotAuthorized and Authorizing child elements.
In the current version of Blazor these child attributes only work in client-side Blazor. Microsoft has
indicated that these attributes will work in a future version of server-side Blazor.
In client-side Blazor you can edit the App page to use these elements within the AuthorizeRouteView element.
Page 87
<AuthorizeRouteView RouteData="@routeData" DefaultLayout="@typeof(MainLayout)">
<NotAuthorized>
Hey, you shouldn't be here!
</NotAuthorized>
<Authorizing>
Authorizing...
</Authorizing>
</AuthorizeRouteView>
Perhaps the most important is the NotAuthorized element, because it allows you to override the default Not
authorized text that is otherwise displayed.
Authorize Attribute
The default behavior for Blazor components and pages is to allow any user to view the content. You can use the
Authorize attribute to prevent anonymous access to a page, or to require a specific role or policy.
This attribute is applied to a component using the @attribute directive, and controls whether the user can view the
entire page or component.
I will discuss the AuthorizeView element later in this chapter. It allows you more control over
authorization within the content of a page or component.
@attribute [Authorize]
Run the app, make sure you are logged out, and try to navigate to the FetchData page. You should be prevented
from seeing the page.
If you log in and navigate to the page the content will appear. This is because the user is authenticated, and thus
meets the basic requirement to be authorized.
Require Roles
The Authorize attribute can also be used to require that the user be a member of a role. Edit the FetchData page
and change the Authorize attribute as follows.
Page 88
@attribute [Authorize(Roles = "admin")]
The string value for the Roles parameter is a comma-delimited list of roles. If the user is a member of any of the
roles they will be authorized. Role names are case-sensitive.
Because Blazor uses the ClaimsIdentity type, roles are a type of claim. When creating a ClaimsIdentity with roles,
you will add roles like this.
Edit the code for the Login page in the Areas\Account\Pages folder and add this line of code right after adding the
ClaimTypes.Role claim.
You should now be able to run the app, log in, and view the FetchData page.
Require Policy
Role-based authorization has been in use for decades, and is well understood. However, it is a little inflexible, in that
the role names need to be hard-coded throughout the pages of the app.
Policies provide a useful level of abstraction to roles, but go well beyond just roles. For example, you can define a
policy that requires or allows anonymous users, requires specific email addresses, looks at a user's country of origin,
and any other claims associated with the user's identity.
The Authorize attribute to use a policy looks like this.
Policies themselves are registered with Blazor in the Program class by adding services in the Main method. For
example, here are three possible policies.
builder.Services.AddAuthorization(config =>
{
config.AddPolicy("IsAuthenticated",
policy => policy.RequireAuthenticatedUser());
config.AddPolicy("IsAdmin",
policy => policy.RequireRole("admin", "supervisor", "manager"));
config.AddPolicy("SpainOnly",
policy => policy.RequireClaim(ClaimTypes.Country, "es"));
});
Policies provide a more elegant and maintainable way to define authorization rules. It is as easy to apply a policy
using the Authorize attribute as it is a role, but the long-term benefits mean you should prefer to use policies.
The primary drawback to the Authorize attribute is that it applies to entire pages or components. In many apps it is
far more common to allow users to see part of a page or interact with certain elements of data, rather than blocking
the user entirely.
AuthorizeView Element
The AuthorizeView element and its child Authorized, NotAuthorized, and Authorizing elements provide more
granular control over how authorization is applied within a page or component.
Detecting Anonymous Users
At a basic level the AuthorizeView element detects whether the current user is logged in or not.
Edit the Index page to add these elements.
Page 89
@page "/"
<h1>Hello, world!</h1>
<AuthorizeView>
<Authorized>
User is authorized
</Authorized>
<NotAuthorized>
User is not authorized
</NotAuthorized>
<Authorizing>
Authorization is occurring
</Authorizing>
</AuthorizeView>
Now run the app and notice how the main page switches its display depending on whether you are logged in or not.
As you would expect, when the user is logged in the content in the Authorized block is displayed, and when they are
not then the content in the NotAuthorized block is displayed.
If asynchronous authorization is implemented at some point (a topic outside the scope of this book), the content in
the Authorizing block would be displayed.
The AuthorizeView element allows you to provide Roles and Policy properties, much like the Authorize attribute.
Role Based Authorization
For example, you can specify a comma-delimited list of roles to the AuthorizeView element.
Direct use of hard-coded role values has the same limitations as with the Authorize attribute.
Policy Based Authorization
The AuthorizeView element also allows you to provide a policy name.
<AuthorizeView Policy="IsAdmin">
This technique has the same maintainability and abstraction benefits as using the Policy property with the
Authorize attribute.
You can use the AuthorizeView element in pages and components to control what content is rendered to the
current user depending on whether they are authorized.
This technique will not work on client-side Blazor, because it will prevent the user from accessing the
Login page. The server-side Login page is a Razor Page, not a Blazor page, so this technique works on the
server.
Page 90
<div class="content px-4">
<AuthorizeView>
<Authorized>
@Body
</Authorized>
<NotAuthorized>
Hey, this isn't for you!
</NotAuthorized>
<Authorizing>
Authorizing...
</Authorizing>
</AuthorizeView>
</div>
This will prevent all Blazor pages from being rendered until the user has logged into the app.
At this point you should understand the requirements and configuration necessary for server-side and client-side
authorization, along with the attributes and elements you can use to control what is rendered for the user.
Conclusion
In this chapter I discussed Blazor authentication for server-side and client-side scenarios. You have seen how
authentication is quite different between these two scenarios. Once the user has been authenticated and the current
user's identity is available to the app you should now understand how authorization can be applied through the use
of roles and other claims.
Chapter 6 will cover the creation of a Blazor app that has both server-side and client-side experiences, with a
common UI shared between both apps.
Page 91
Chapter 6: Multi-Headed Blazor Solutions
It is possible to create a Blazor solution that includes multiple "heads". In this context a head is a particular UI
platform, such as server-side or client-side hosting of your code. The goal is to create a solution where most of the
app code is common, and is leveraged by multiple UI platforms.
With Blazor the common scenario is to have a server-side Blazor project (head), a client-side Blazor project (head),
and a Blazor Class Library containing the common UI implementation. This allows you to maintain most of the code
once, but allow users to interact with your app using server-side or client-side Blazor deployments.
A similar concept can be found in Xamarin, where a solution often contains an iOS head, and Android
head, a Windows UWP head, and the class library with all the common UI implementation for the app.
The resulting solution will have the standard three projects: Blazor client, ASP.NET Core host, and the shared code.
Page 93
This will be the start point for creating a multi-headed solution.
Page 94
Name the new project BlazorMultiHead.Ui and make sure to edit the Location path to be something like
E:\src\rdl\BlazorMultiHead\BlazorMultiHead.
The default path will put the new project one level above the other projects in the solution. To avoid complexity, it is
better if the new project is at the same level in the folder hierarchy.
The final step in the new project wizard is to confirm that the project will target .NET Core 3.1 and to check the
Support pages and views option on the right side of the dialog.
Page 95
Click Create to add the project to the solution.
With this project added the solution now contains four projects.
Page 96
As I mentioned, the new BlazorMultiHead.Ui project isn't quite ready yet, but it is close. Let's make a few changes
so it can contain the Blazor components.
Remove Template Files
The Razor Class Library project template adds an Areas folder with some starting content. This is not necessary, so
use Solution Explorer to delete that folder and remove it from the project.
The result is a totally empty project.
Set the Target Frameworks
Next, double-click on the project node in Solution Explorer to open the project file.
Notice how the first line references the Razor SDK.
<Project Sdk="Microsoft.NET.Sdk.Razor">
Page 97
<PropertyGroup>
<TargetFrameworks>netcoreapp3.1;netstandard2.1</TargetFrameworks>
<LangVersion>7.3</LangVersion>
<RazorLangVersion>3.0</RazorLangVersion>
</PropertyGroup>
The TargetFramework element is now TargetFrameworks (plural), and lists the original netcoreapp3.1 target as well
as a new netstandard2.1 target.
Remember that server-side Blazor runs on .NET Core 3.1, while client-side Blazor projects target .NET Standard 2.1.
This change to the TargetFramework element means that the BlazorMultiHead.Ui project will now compile twice,
once for .NET Core 3.1 and then for .NET Standard 2.1. As a result, this project can be referenced by the .NET Core
3.1 server-side Blazor project and by the .NET Standard 2.1 client-side Blazor project.
Update Dependency References
Because this project will be compiled two times for different targets, you need to specify the NuGet package
references that will be used for each target.
Delete the ItemGroup referencing the Microsoft.AspNetCore.App package and replace it with the following.
Notice the Condition property applied to each ItemGroup element. This property is used to restrict each XML block
so it applies only to a specific build target.
You can see the different references for netcoreapp3.1 and netstandard2.1. These are the same package references
used by server-side and client-side Blazor projects.
Finally, add another ItemGroup to the file to reference the shared code project.
<ItemGroup>
<ProjectReference Include="..\Shared\BlazorMultiHead.Shared.csproj" />
</ItemGroup>
The EditForm page, for example, uses types from the BlazorMultiHead.Shared project. You will be moving all the
Blazor components, including the EditForm page, into this new shared UI project, and so a reference to the shared
code project will be necessary.
You can save and close the project file.
Build the project to confirm that all the changes are correct. The Visual Studio Output window should show
something similar to this:
Page 98
1>------ Rebuild All started: Project: BlazorMultiHead.Shared,
Configuration: Debug Any CPU ------
1>BlazorMultiHead.Shared -> E:\src\rdl\BlazorMultiHead\BlazorMultiHead\Shared\bin\
Debug\netstandard2.1\BlazorMultiHead.Shared.dll
2>------ Rebuild All started: Project: BlazorMultiHead.Ui,
Configuration: Debug Any CPU ------
2>BlazorMultiHead.Ui -> E:\src\rdl\BlazorMultiHead\BlazorMultiHead\BlazorMultiHead.Ui\bin\
Debug\netcoreapp3.1\BlazorMultiHead.Ui.dll
2>BlazorMultiHead.Ui -> E:\src\rdl\BlazorMultiHead\BlazorMultiHead\BlazorMultiHead.Ui\bin\
Debug\netstandard2.1\BlazorMultiHead.Ui.dll
========== Rebuild All: 2 succeeded, 0 failed, 0 skipped ==========
It is important to make sure that the BlazorMultiHead.Ui project builds twice, once for each build target.
Move Pages and Components to Project
Now that the BlazorMultiHead.Ui project is set up to host Blazor UI components it is time to move the UI
components from the BlazorMultiHead.Client project to their new location.
You can use Solution Explorer to drag-and-drop the content from the client project to the new shared UI project.
Once the files have been copied to the shared UI project you need to delete them from the original client project.
Page 99
The project will not build yet, but it is close.
Update Moved Code
There are just a few changes you need to make to the code to change namespaces and now-missing types.
The _Imports.razor file uses namespaces from the old client project.
@using BlazorMultiHead.Client
@using BlazorMultiHead.Client.Shared
@using BlazorMultiHead.Ui
@using BlazorMultiHead.Ui.Shared
Also, the App page references the Program type, which is no longer available.
<Router AppAssembly="@typeof(Program).Assembly">
The Program type is being used to find the name of the assembly that contains the Blazor components. Any type that
is guaranteed to exist, and which is in the new shared UI project will work. The obvious type that always exists is the
App type itself, so change the code to use that type.
<Router AppAssembly="@typeof(App).Assembly">
The project still won't build, but now it is because the EditForm page is using an extension method of the
HttpClient type that isn't available because the shared UI project doesn't reference the
Microsoft.AspNetCore.Components package.
Page 100
If you think back to how the EditForm page is implemented in the server-side Blazor project from Chapter 2, it is
quite different from this client-side implementation. I will discuss a technique to reconcile this difference later in this
chapter.
For now please comment out the offending line of code in the EditForm page.
This will cause compiler warnings, but the new shared UI project should now build.
Reference Shared UI Project
At this point you have moved the Blazor UI components from the old client project to the new shared UI project. As a
result the old client project has no UI components to display. Let's fix that.
Return to the BlazorMultiHead.Client project and add a reference to the BlazorMultiHead.Ui project.
You can do this with the Add References dialog, or by editing the csproj file.
<ItemGroup>
<ProjectReference Include="..\BlazorMultiHead.Ui\BlazorMultiHead.Ui.csproj" />
<ProjectReference Include="..\Ui\BlazorMultiHead.Ui.csproj" />
</ItemGroup>
The updated BlazorMultiHead.Client project file should now include an ItemGroup block similar to what I am
showing here.
The client project will not build yet, because the Startup class can't find the App type now that it is in a new
namespace. Add a using statement at the top of the client project's Startup class.
using Microsoft.AspNetCore.Components.Builder;
using Microsoft.Extensions.DependencyInjection;
using BlazorMultiHead.Ui;
As a general rule I recommend doing a rebuild with a multi-headed solution because I have observed
that Visual Studio will often not build the BlazorMultiHead.Ui project even when it has been changed. I
suspect this is a bug with the preview support for Blazor projects.
You can now run the solution and interact with the Index and Counter pages. The FetchData page will not work
because the line of code that loads the data has been commented out. Let's get that working again.
Xamarin.Forms developers are very familiar with this issue and the technique I am using in this chapter.
Page 101
It is very common for the Android and iOS heads of a mobile app to have different implementations for
certain functionality.
The easiest way to provide per-head implementations of functionality is to define an interface that can be
implemented for each head. The shared UI code will use that interface, trusting that it will be provided the
appropriate implementation at runtime.
Define Service Interface
Add a Services folder to the BlazorMultiHead.Ui project, and add an IForecastService interface to that folder.
If you remember the server-side Blazor implementation of the FetchData page in Chapter 2, this is the method
signature used by that code.
Update the FetchData Page
You can now update the code in the FetchData page to make use of this interface, and to require that an object
implementing this interface be injected into the page.
At the top of the page add an @inject directive.
@page "/fetchdata"
@using BlazorMultiHead.Shared
@using BlazorMultiHead.Ui.Services
@inject IForecastService ForecastService
I have also added some @using directives, and (importantly) I have removed the @inject directive for the
HttpClient instance. There's no need to inject an HttpClient instance because it is not needed in this page any
longer.
Now update the OnInitializeAsync code.
The code now relies on the injected service and invokes the GetForecastAsync method defined by the
IForecastService interface. It is now up to each head to provide its own implementation of the IForecastService
interface.
Create Client-Side Service Implementation
Add a Services folder to the BlazorMultiHead.Client project, and add a ForecastService class in that folder.
Page 102
public class ForecastService : IForecastService
{
private HttpClient Http;
public ForecastService(HttpClient httpClient)
{
Http = httpClient;
}
This class implements the IForecastService interface, and so implements the GetForecastAsync method.
The class relies on dependency injection to get an instance of the HttpClient object necessary to call the
GetJsonAsync method.
Now when the FetchData page requests an instance of IForecastService it will be provided with a
ForecastService object.
You can now run the project and the FetchData page will now return data.
At this point you are back where you started, with a fully functional client-side Blazor project that calls a service from
the ASP.NET Core web site. However, the Blazor UI components are all separated out into a shared UI project, so you
can also use them from a server-side Blazor project.
<ItemGroup>
<ProjectReference Include="..\Client\BlazorMultiHead.Client.csproj" />
<ProjectReference Include="..\Shared\BlazorMultiHead.Shared.csproj" />
<ProjectReference Include="..\BlazorMultiHead.Ui\BlazorMultiHead.Ui.csproj" />
</ItemGroup>
Server-side Blazor projects have some content unique to the environment that needs to be added. This includes the
Page 103
_Host and Error pages, along with the styles and assets from the wwwroot folder.
The content of this page doesn't vary much from app to app, just the namespace and title usually change. Here's the
entire content, and I will highlight the important changes below.
Page 104
@page "/"
@namespace BlazorMultiHead.Server.Pages
@using BlazorMultiHead.Ui
@addTagHelper *, Microsoft.AspNetCore.Mvc.TagHelpers
@{
Layout = null;
}
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>BlazorMultiHead</title>
<base href="~/" />
<link rel="stylesheet" href="css/bootstrap/bootstrap.min.css" />
<link href="css/site.css" rel="stylesheet" />
</head>
<body>
<app>
<component type="typeof(App)" render-mode="ServerPrerendered" />
</app>
<div id="blazor-error-ui">
<environment include="Staging,Production">
An error has occurred. This application may
no longer respond until reloaded.
</environment>
<environment include="Development">
An unhandled exception has occurred. See browser
dev tools for details.
</environment>
<a href="" class="reload">Reload</a>
<a class="dismiss"> </a>
</div>
<script src="_framework/blazor.server.js"></script>
</body>
</html>
At the top there are directives to set the namespace for this view, and to bring in the BlazorMultiHead.Ui
namespace for use.
@namespace BlazorMultiHead.Server.Pages
@using BlazorMultiHead.Ui
The _Hosts page uses the App type from the Blazor UI, which is now in the shared UI project. The @using directive
makes that available.
You will want to make sure the title element corresponds to your app.
<title>BlazorMultiHead</title>
The rest of the content in this page is unchanged from the server-side Blazor project in Chapter 2, or any other
server-side Blazor project you might create.
Create Error Page
Another set of content required by server-side Blazor is the Error page. This is an actual Blazor page, and so it needs
to go into the Blazor UI project, not the server project.
Add a Razor Component to the Pages folder in the BlazorMultiHead.Ui project with the name Error.razor.
Page 105
Here is the complete code.
@page "/error"
<h1 class="text-danger">Error.</h1>
<h2 class="text-danger">An error occurred while processing your request.</h2>
<h3>Development Mode</h3>
<p>
Swapping to <strong>Development</strong> environment will display
more detailed information about the error that occurred.
</p>
<p>
<strong>The Development environment shouldn't be enabled for deployed
applications.</strong>
It can result in displaying sensitive information from exceptions to
end users.
For local debugging, enable the <strong>Development</strong> environment
by setting the <strong>ASPNETCORE_ENVIRONMENT</strong> environment
variable to <strong>Development</strong>
and restarting the app.
</p>
This is the exact same content provided by default in the server-side Blazor project template, so you might choose to
copy it from the project in Chapter 2.
Move wwwroot Content
The final content required by server-side Blazor is all the CSS and other assets in the wwwroot folder.
Right now the solution has one wwwroot folder, in the client-side Blazor project. The easiest way to make this
content available to server-side Blazor is to move the folder to the server project.
Page 106
If you have a wwwroot folder in the ASP.NET Core host web site and also in the client-side Blazor
project, only the server-side folder will be used. The client-side wwwroot folder contents will be ignored. You
should avoid having a wwwroot folder in a client-side Blazor project and in the hosting ASP.NET Core web
site.
Use Solution Explorer to drag-and-drop the wwwroot folder from the client project to the server project, then delete
the folder from the client project.
You may also want to right-click on the wwwroot folder and choose the Add | Existing Item option to add the
favicon.ico file from the server-side Blazor project created in Chapter 2. This is optional, but is a good idea.
The result is that the wwwroot folder is now in the server project and has all the CSS and assets necessary to
support client-side and server-side Blazor projects.
At this point all the content required by server-side Blazor exists in the server and shared UI projects. Only two steps
remain: provide an implementation of the IForecastService service, and to configure the app for Blazor.
Create Service Implementation for Weather Data
The shared UI project requires that each head provide an implementation of the IForecastService interface, so the
server project needs to include an implementation.
Add a Services folder to the BlazorMultiHead.Server project, and add a ForecastService class to the folder.
Page 107
public class ForecastService : IForecastService
{
private static readonly string[] Summaries = new[]
{
"Freezing", "Bracing", "Chilly", "Cool",
"Mild", "Warm", "Balmy", "Hot", "Sweltering", "Scorching"
};
This is the implementation from the server-side Blazor project template in Chapter 2, but in a class that implements
the IForecastService interface defined in the BlazorMultiHead.Ui project.
An instance of this type will be provided on demand, once dependency injection is configured in the Startup class.
Update Startup Configuration
Server-side Blazor requires configuration to function.
Open the Startup class in the server project.
Edit the ConfigureServices method to match the following.
The AddMvc service has been removed because it isn't necessary. The AddRazorPages and AddServerSideBlazor
services have been added as they are required.
Also, a singleton has been added to register the ForecastService type so when the FetchData page requests an
instance of the IForecastService interface it will get the server-side object.
There are more changes required in the Configure method. I will show the method with the code to configure client-
side Blazor commented out for easy comparison.
Page 108
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
app.UseResponseCompression();
if (env.IsDevelopment())
{
app.UseDeveloperExceptionPage();
app.UseBlazorDebugging();
}
else
{
app.UseExceptionHandler("/Error");
}
app.UseStaticFiles();
//app.UseClientSideBlazorFiles<Client.Startup>();
//app.UseRouting();
//app.UseEndpoints(endpoints =>
//{
// endpoints.MapDefaultControllerRoute();
// endpoints.MapFallbackToClientSideBlazor<Client.Startup>("index.html");
//});
app.UseRouting();
app.UseEndpoints(endpoints =>
{
endpoints.MapBlazorHub();
endpoints.MapFallbackToPage("/_Host");
});
app.UseExceptionHandler("/Error");
app.UseRouting();
app.UseEndpoints(endpoints =>
{
endpoints.MapBlazorHub();
endpoints.MapFallbackToPage("/_Host");
});
You can see how the client-side and server-side configurations are similar, but different in important ways.
At this point you can run the app. It won't look or act differently, but this is now a server-side Blazor environment
instead of client-side with WebAssembly.
As a final step, let's make it relatively easy to switch between the WebAssembly and ASP.NET Core heads.
Page 109
This is reflected as a new PropertyGroup element in the project file.
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|AnyCPU'">
<DefineConstants>TRACE;WASM</DefineConstants>
</PropertyGroup>
It is now possible to use compiler directives in code so only some code is compiled when the WASM constant is set,
and other code will only compile if the constant is not set.
Use Conditional Compilation in Startup
Edit the ConfigureServices method in the server project's Startup class.
If the WASM constant is set then services for server-side Blazor should not be configured, otherwise they should be
configured. The #if and #endif directives are used to wrap a block of code that should only compile if the WASM
constant is not set.
Also edit the Configure method.
Page 110
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
app.UseResponseCompression();
if (env.IsDevelopment())
{
app.UseDeveloperExceptionPage();
app.UseBlazorDebugging();
}
#if !WASM
else
{
app.UseExceptionHandler("/Error");
}
#endif
app.UseStaticFiles();
#if WASM
app.UseClientSideBlazorFiles<Client.Startup>();
app.UseRouting();
app.UseEndpoints(endpoints =>
{
endpoints.MapDefaultControllerRoute();
endpoints.MapFallbackToClientSideBlazor<Client.Startup>("index.html");
});
#else
app.UseRouting();
app.UseEndpoints(endpoints =>
{
endpoints.MapBlazorHub();
endpoints.MapFallbackToPage("/_Host");
});
#endif
}
Notice how the client-side Blazor configuration that was commented out is now uncommented and is in a #if block
so it only compiles if the WASM constant is set.
Other blocks of code only compile if the constant is not set.
At this point you can run the project and the app will run in WebAssembly in the browser because the WASM constant
is set.
You can close the app and edit the server project (using the project properties window or the project file) to remove
the WASM constant. Then run the project again, and it will run in ASP.NET Core on the server.
Conclusion
Server-side Blazor provides superior debugging, and so is an attractive environment for app development. In many
cases you will want to deploy to users using WebAssembly and client-side Blazor to take full advantage of the
hardware and resources on client devices.
Being able to easily switch between server-side and client-side Blazor to run the exact same Blazor UI code enables
easy debugging and flexible deployment. In the future Blazor may support other heads as well, so having your
solution designed to support multiple heads is not only beneficial today, but also in the future.
The last two chapters of this book will focus on how CSLA .NET supports Blazor, and how you can use Blazor and
CSLA .NET together to create powerful apps that leverage the best features of both frameworks.
Page 111
Chapter 7: Blazor and CSLA .NET
CSLA .NET is an open source framework that provides a home for your business logic. Just like Blazor provides a
first-class framework for creating a user interface and Entity Framework is a first-class framework for data access,
CSLA .NET is a first-class framework for your business logic. You can learn about CSLA .NET from the Introduction to
CSLA .NET book.
Starting with version 5.1, CSLA .NET supports Blazor on both the server and client. In this chapter you will learn how
to create server-side and client-side Blazor apps that leverage a reusable business logic layer created using CSLA
.NET.
That same CSLA-based business layer can be used to create apps using Xamarin, Windows technologies, ASP.NET
web sites, services, and more. The intent is that you create your business logic once, and it can be reused anywhere
.NET can run.
NuGet Packages
CSLA .NET provides "helper packages" for all supported UI frameworks, including Blazor, ASP.NET Core, UWP, WPF,
and more. The Blazor NuGet package is named Csla.Blazor, and provides numerous types to help streamline the
use of CSLA when building a Blazor project.
When building server-side Blazor projects it is also necessary to use the Csla.AspNetCore package, because server-
side Blazor is hosted in ASP.NET Core.
Both of these packages depend on the core Csla NuGet package.
In a Blazor solution you may have the following project types, each with their own package dependency
requirements.
Page 112
1. Server-side Blazor project
Reference the Csla.AspNetCore package
Reference the Csla.Blazor package
2. Client-side Blazor project
Reference the Csla.Blazor package
3. Shared Blazor UI (Razor Components) project
Reference the Csla.Blazor package
4. Business library project (containing only domain business types)
Reference the Csla package
5. Data access projects (implementing any data access logic)
Reference the Csla package
Notice that the business and data access projects do not reference any UI-specific helper packages. This is because
your business domain types and data access types should be entirely platform neutral. Those types should work
behind any UI framework supported by .NET.
Although the Csla.AspNetCore package is required for a server-side Blazor project, you won't directly use types
from that package in your Blazor code. If you write any MVC or server-side Razor pages then those types are
valuable. I discuss how CSLA .NET supports ASP.NET Core app development in the Using CSLA: ASP.NET Core book.
In this chapter I will focus on the types provided by the Csla.Blazor package and how you can use them when
building Blazor apps.
using Csla.Configuration;
In the ConfigureServices method it is necessary to add the CSLA services for use in dependency injection. This is
done by adding the following line of code after configuring things like MVC, Razor Pages, and Blazor.
services.AddCsla().WithBlazorServerSupport();
In the Configure method it is necessary to configure the CSLA services and other configuration. Add this line of code
after configuring MVC, Razor Page, and Blazor.
app.UseCsla();
The UseCsla extension method has an overload you can optionally use to provide additional configuration for CSLA
.NET. One of the most commmon scenarios is configuring the data portal to call an app server instead of running the
logical server-side code on the web server.
Page 113
Many other aspects of CSLA .NET can be configured as well, but most of the configuration should default correctly
for use within your server-side Blazor app.
Client-Side Blazor Configuration
Client-side Blazor projects have configuration in the Program class.
Add a using statement to the top of the file.
using Csla.Configuration;
Configuration is performed in the Main method and relies on the builder variable created by the template code.
Configure CSLA before the RunAsync method is invoked.
builder.Services.AddOptions();
builder.Services.AddAuthorizationCore();
builder.UseCsla();
The UseCsla extension method adds all required services for use by dependency injection, and performs other
configuration necessary for CSLA to work properly in a Blazor WebAssembly app.
CSLA .NET supports authorization, and it is necessary to call AddOptions and AddAuthorizationCore before using
any Blazor authorization features. If you don't call these methods your app won't run because CSLA will be unable to
initialize.
The method also has an overload you can optionally use to provide additional configuration for CSLA .NET. One of
the most common scenarios is configuring the data portal to call an app server instead of running the logical server-
side code on the web server.
Many other aspects of CSLA .NET can be configured as well, but most of the configuration should default correctly
for use within your client-side Blazor app.
Page 114
In this section of the chapter I use diagrams indicating the use of a ViewModel type from the MVVM
design pattern. You do not need to use MVVM, and later in the chapter I will discuss alternatives. However, if
you don't use MVVM then something will fill the role of the Interface Control layer in each diagram.
If the data portal is configured to communicate with a remote server, you are also able to choose the network
transport protocol that should be used. The most common choice is to use HTTP, but you can also use gRPC, WCF,
or other network communication technologies.
When working with client-side Blazor, only HTTP is supported as a network transport protocol.
Let's discuss each scenario, assuming the use of the encapsulated invocation data access model as discussed in the
Using CSLA: Data Access book.
Normally in a client-side Blazor app data is stored using browser local storage to save data on the device that is
running the browser. There are numerous NuGet packages available that provide ways to access browser local
storage, and the use of those packages is outside the scope of this book.
The data portal abstracts the concept of domain business object persistence, allowing you to use
whatever package or library you choose to interact with browser local storage.
The default data portal configuration is to run the logical server-side code locally, in the same location as your UI
code. This means that the default is for your logical server-side code to run in the browser on the client device.
Because your logical server-side code will be running on the client device, this means the Data Access Layer (DAL)
will be deployed to the client along with the rest of the app.
The DAL code will normally define an interface for persistence. To keep this example as simple as possible I will
focus only on retrieving data.
Page 115
public interface IPersonDal
{
PersonInfo Get(int id);
}
This is where you must implement code to interact with browser local storage. Again, there are numerous NuGet
packages available, and the use of those packages is outside the scope of this book.
Now that the DAL exists, the client-side Blazor app needs to be configured to use those types. This is done in the
Main method of the Program class.
builder.UseCsla();
builder.Services.AddTransient<IPersonDal, PersonDal>();
await builder.Build().RunAsync();
}
You can see how the AddTransient method is used to tell dependency injection to return an instance of the
PersonDal type any time the IPersonDal type is requested.
Finally, in the business library you will implement a PersonEdit domain type that makes use of the DAL.
Page 116
[Serializable]
public class PersonEdit : BusinessBase<PersonEdit>
{
public static readonly PropertyInfo<int> IdProperty = RegisterProperty<int>(nameof(Id));
public int Id
{
get => GetProperty(IdProperty);
private set => LoadProperty(IdProperty, value);
}
[Fetch]
private void Fetch(int id, [Inject] IPersonDal dal)
{
var data = dal.Get(id);
using (BypassPropertyChecks)
{
Id = data.Id;
Name = data.Name;
}
}
}
Notice that the Fetch method relies on dependency injection to get an instance of the IPersonDal type, and then
uses that object to get the persisted data so the domain object can be loaded with the information.
In this implementation, all the code runs on the client device, relying on the DAL code to persist any data locally on
the device.
Page 117
Either way, some configuration is needed on the client, and the app server must expose a service endpoint to host
the data portal.
Implementing the Data Access Types
You should keep in mind that with this deployment model, the DAL assembly does not need to be deployed to the
client device. The DAL assembly is only invoked on the app server, and should only be deployed to the app server.
Because the DAL code is running on an app server, it is very likely that the DAL implementation will be interacting
with a database instead of some local file system. This will affect the implementation of your concrete DAL types.
In most cases you will use dependency injection to get access to your underlying data access object, such as an
Entity Framework ObjectContext instance, and then you will use that type to interact with the database.
Page 118
public class PersonDal : IPersonDal
{
public ObjectContext ctx { get; set; }
public PersonDal(ObjectContext context)
{
ctx = context;
}
Notice how the PersonDal type from earlier in the chapter has been enhanced to accept an EF type from
dependency injection, and then to use that type when implementing the Get method. Also notice how the same
PersonInfo DTO type is used to pass the data to the Fetch method in the business class. This means the business
class is totally unaffected by the use of an EF-based DAL vs the earlier DAL implementation that relied on browser
local storage.
HTTP Data Portal Channel
Before I discuss the specific configuration details for client and server, you should be aware that the only network
transport protocol supported is the HTTP protocol via the HttpProxy in the Csla.DataPortalClient namespace.
When working with client-side Blazor, only HTTP is supported as a network transport protocol.
Also, there is a limitation within the .NET (mono) runtime in WebAssembly that restricts data being passed over the
.NET HttpClient object to be text only. No binary data is allowed. This restriction may be lifted at some point in the
future, but today you need to configure the server-side data portal endpoint to understand that all data will be text-
based, not binary.
Client-side App Configuration
In your client-side Blazor app it is necessary to configure the data portal so it uses an HTTP data portal channel to
communicate with a remote app server. This is done in the Main method of the Program class, while configuring the
app to use CSLA .NET.
In this example the DefaultProxy method is invoked to indicate that the HttpProxy type should be used to
communicate with the app server, and a URL to the app server endpoint is provided.
The UseCsla method automatically configures the HttpProxy type to send and receive text-based data instead of
binary data. The data transferred is binary, it is just base64 encoded into text so it can be passed over the network
via the HttpClient object provided by Blazor.
App Server Configuration
Page 119
You must also set up the app server with an endpoint and some configuration. In most cases the app server will be
hosted by ASP.NET Core, whether this be the web server used to deploy the client-side Blazor app, or a dedicated
app server.
The data portal endpoint is a controller in the Controllers directory of the ASP.NET Core project. Because this
endpoint will send and receive text-based data instead of binary data, it must not be the same as the data portal
endpoint controller used to support other types of client (such as Xarmarin, UWP, ASP.NET Core, etc).
[Route("api/[controller]")]
[ApiController]
public class DataPortalTextController : Csla.Server.Hosts.HttpPortalController
{
public DataPortalTextController()
{
UseTextSerialization = true;
}
}
A "regular" data portal endpoint is the exact same code as shown here, except that you don't set the
UseTextSerialization property to true. By default the HttpPortalController base class sends and receives binary
data, but when the UseTextSerialization property is set, it will send and receive base64 encoded binary data
instead.
You must also configure the app server by adding code to the Startup class. In the ConfigureServices method it is
necessary to configure the CSLA .NET services, and also to configure the web server to allow the use of synchronous
methods.
// If using Kestrel:
services.Configure<KestrelServerOptions>(options =>
{
options.AllowSynchronousIO = true;
});
// If using IIS:
services.Configure<IISServerOptions>(options =>
{
options.AllowSynchronousIO = true;
});
services.AddCsla().WithBlazorServerSupport();
services.AddTransient<IPersonDal, PersonDal>();
The code that sets AllowSynchronousIO to true is required because the data portal currently uses some
synchronous methods to serialize and deserialize data. By default, ASP.NET Core 3.0 or higher prevents the use of
synchronous methods, and so it is necessary to enable their use.
The AddCsla method adds the necessary CSLA .NET types to the collection of services available for dependency
injection, allowing the data portal to function on the app server.
Notice that the IPersonDal type is added so dependency injection can properly inject the correct DAL object for use
in your business class's Fetch method.
Because the updated PersonDal type makes use of Entity Framework, you will also need to configure EF and add all
relevant EF types to the services collection.
There is no requirement by CSLA .NET to use EF. You can use raw ADO.NET, Dapper, EF, or any other
Page 120
technology you choose for interacting with the data store.
app.UseCsla();
This performs other configuration steps necessary for CSLA .NET to function properly within an ASP.NET Core app.
At this point you should understand how to configure a client-side Blazor app for use with a local or remote data
portal, and in the case of a remote data portal, you now know how to configure a data portal endpoint for use by
Blazor apps.
Now let's discuss how to set up and configure the data portal for server-side Blazor apps.
The data portal defaults to running in "local mode", so no explicit configuration is required for the data portal to run
locally. However, you do need to configure your DAL types on the web server for use by dependency injection.
Page 121
Because the logical server-side code will be running on the web server, you need to deploy the DAL assembly to the
web server, along with the Blazor app and the business library assembly.
Based on the PersonEdit, PersonDal, and IPersonDal types I discussed earlier in this chapter, you can configure the
app for server-side Blazor deployment by altering the methods in the Startup class of the Blazor project.
In the ConfigureServices method you need to add the CSLA .NET types and the DAL type.
services.AddCsla().WithBlazorServerSupport();
services.AddTransient<IPersonDal, PersonDal>();
The AddCsla method adds the necessary CSLA .NET types to the collection of services available for dependency
injection, and the IPersonDal type is added so dependency injection can properly inject the correct DAL object for
use in your business class's Fetch method. Also, the updated PersonDal type makes use of Entity Framework, you
will also need to configure EF and add all relevant EF types to the services collection.
There is no requirement by CSLA .NET to use EF. You can use raw ADO.NET, Dapper, EF, or any other
technology you choose for interacting with the data store.
In the Configure method you need to indicate the use of CSLA .NET.
app.UseCsla();
At this point you should have a server-side Blazor app where the presentation, business, and data access layers are
all deployed to, and run on, the web server.
Page 122
If you need this type of deployment it is easy to set up by configuring the web server and app server.
Web Server Configuration
The web server runs the Blazor app, and so must contain your Blazor UI components and a reference to the
assembly containing your business domain types. As with any server-side Blazor app that uses CSLA .NET, the
Startup class needs some configuration code.
In this deployment the DAL assembly should not be deployed to the web server, as it is only used on the app server.
In the ConfigureServices method you must configure CSLA .NET.
services.AddCsla().WithBlazorServerSupport();
The Configure method needs not just the CSLA configuration, but also some data portal configuration to indicate
Page 123
that a remote app server should be used.
This sample code shows using HttpProxy, but you can use any network transport that is supported by
.NET Core, including GrpcProxy and other data portal channels that may be created in the future.
This configures the app to use CSLA .NET, and to run all logical server-side logic on the remote app server.
App Server Configuration
The app server needs a data portal endpoint and some configuration. The most common scenario is for the app
server to be hosted by ASP.NET Core, so setting up a data portal endpoint is done by adding a controller to the
Controllers directory in the web project.
[Route("api/[controller]")]
[ApiController]
public class DataPortalController : Csla.Server.Hosts.HttpPortalController
{ }
The HttpPortalController type completely abstracts the endpoint implementation, so you don't need to put any
custom code here at all.
In the app server project's Startup class add the following to the ConfigureServices method.
// If using Kestrel:
services.Configure<KestrelServerOptions>(options =>
{
options.AllowSynchronousIO = true;
});
// If using IIS:
services.Configure<IISServerOptions>(options =>
{
options.AllowSynchronousIO = true;
});
services.AddCsla().WithBlazorServerSupport();
services.AddTransient<IPersonDal, PersonDal>();
This is the same configuration used earlier in this chapter for configuring the Blazor web server as an app server.
The AddCsla method adds the CSLA types, a resolution for the IPersonDal type is added, and you need to add any
necessary types for Entity Framework.
There is no requirement by CSLA .NET to use EF. You can use raw ADO.NET, Dapper, EF, or any other
technology you choose for interacting with the data store.
app.UseCsla();
Page 124
At this point you have configured the Blazor web server and an app server so they work together via the CSLA data
portal so logical client-side work runs on the web server and logical server-side work runs on the app server.
You should now have an understanding of the four primary deployment models for Blazor apps supported by CSLA
.NET and the data portal. Let's move on and discuss how business domain types created using CSLA .NET fit into a
Blazor app architecture.
Read-Only Values
When dealing with read-only properties, Blazor data binding is typically straightforward. For example.
<div>@person.Name</div>
The @person.Name expression will render the value of the Name property in the browser.
You should be aware that the CSLA rules engine supports authorization rules that run when a property getter is
invoked. In other words, the current user may not be authorized to view the Name property.
At one level this is automatic, because the property getter will return the default value for the type if the current user
isn't authorized to see the value. But you may want to provide a better experience, and later in this chapter I will
discuss how you can use the CSLA authorization API to enhance the user experience to hide or disable UI elements if
the current user isn't authorized to view or edit specific properties.
When you are creating data entry pages where data binding is used against read-write properties you have some
choices. The Blazor UI framework supports two different ways to create a data entry form.
1. The Blazor EditForm component
2. The MVVM (model-view-viewmodel) design pattern
The EditForm component is a great way to create simple data entry forms. The Csla.Blazor namespaces includes
helper types to integrate CSLA .NET rules into the UI experience.
The MVVM design pattern offers all the power provided by the CSLA .NET rules engine with the potential for a great
deal of UI code reuse.
Although I will discuss both options in this chapter, my general recommendation is to use the MVVM design pattern,
and that's what I'll use in Chapter 8.
This code assumes that person and statusText fields have been declared in the @code block, similar to the coding
approach used in Chapter 4.
The SaveAndMergeAsync method is asynchronous, so the FormSubmit method is marked with the async keyword,
enabling the use of the await keyword.
However, there's a timing issue here that will cause Blazor data binding to miss any updates to the UI that occur
after the await statement. The FormSubmit method is called synchronously by .NET, and the method will return
immediately when the SaveAndMergeAsync method is invoked. It will not wait until the method completes.
The Blazor UI framework will update the UI based on any changes made when the FormSubmit method completes,
and will be entirely unaware of any changes made when the SaveAndMergeAsync method completes, or after that
line of code in the FormSubmit method.
To ensure that the UI is updated to reflect any changes to objects or fields (such as statusText) it is necessary to call
the Blazor StateHasChanged method. This method tells the Blazor UI framework that state has changed, so it updates
the UI display based on the new values.
Validation
You might recall from Chapter 4 that the EditForm component supports data annotations attributes. However, CSLA
business rules can be much more complex than simple data annotations, and so the Csla.Blazor namespace
includes helper types to enhance the display of broken rules to include those from CSLA .NET.
Page 126
These components are direct replacements for their data annotation equivalents.
CslaValidator should be used in place of DataAnnontationValidator
CslaValidationMessages should be used in place of ValidationMessages
As a result, the markup within the EditForm component for editing a CSLA-based PersonEdit type looks like this.
Notice the use of the IsSavable metastate property from the CSLA business object. This property is data bound to
the submit button element's disabled property. As a result, the button is only available for the user to click when the
underlying business object can be saved.
As a refresher, a CSLA object can only be saved when:
The object has no broken validation rules (Error severity)
Data in the object has been changed, or there's reason to believe the data doesn't match what is in the data
store
The current user is authorized to edit the object
The Using CSLA: Creating Business Objects book has full details on CSLA metastate properties. I am using the
IsSavable property here to demonstrate how easy and powerful it can be to use data binding against those
properties to manipulate the UI and provide a great user experience.
When this code is run and the form is submitted with a blank Name field, the result is that the validation messages
are displayed.
Page 127
Although this appears to be the same behavior as with the data annotations attributes in Chapter 4, it is important to
understand that all CSLA validation rules are handled and their results displayed. That includes data annotations
attributes, because CSLA honors those too, but it also includes any custom rules you've created within your code.
For example, the PersonEdit class in the code for this chapter includes a custom rule named NoSingleName that
generates warning text if it appears that a full name was not entered.
Page 128
The ValidationSummary component only displays broken rules with an Error severity, and will not display Warning
or Information severity rule messages.
EditContext Extensions
If you are writing code to implement validation by invoking methods on the EditContext parameter passed to the
OnSubmit, OnValidSubmit, and OnInvalidSubmit event handlers from an EditForm component, the Csla.Blazor
namespace provides the AddCslaValidation extension method.
The AddCslaValidation method is analogous to the AddDataAnnotationsValidation method, and is used to add
CSLA validation to the context.
In most cases however, you will use the CslaValidator component as shown in this chapter.
At this point you should understand how the EditForm component works with CSLA-based domain types, and how
you can use the helper types from the Csla.Blazor namespace to create data entry forms.
Although that approach can be useful if the model is poorly designed, when using CSLA .NET you should strive to
design your domain business types based on good object-oriented design, focused on the business domain and
user scenario. As a result, your domain types should be well designed and ready to bind directly to the view.
To this end, the ViewModel type provided by CSLA exposes a Model property so the view can bind directly to the
business object without a lot of wasted coding effort.
In both cases the viewmodel object handles events raised by the view and implements behavior to properly handle
those events. The ViewModel type provided by CSLA includes implementations for a number of standard behaviors,
and you can add your own if necessary.
ViewModel Class
The Csla.Blazor namespace includes a ViewModel type that provides standard viewmodel behaviors necessary for
most Blazor pages. This type is designed to work with dependency injection, and is automatically available to your
code if you've configured CSLA in your project as discussed earlier in this chapter.
In a Razor Component you get access to a ViewModel instance by using the @inject directive. For example, in the
sample project's Page folder the PersonEditMvvm page contains this code.
Page 130
@page "/personeditmvvm"
@page "/personeditmvvm/{id}"
@using Data
@using Csla.Blazor
@inject ViewModel<PersonEdit> vm
The @inject statement ensures that a ViewModel<PersonEdit> instance is available to the component. This object
will help manage the lifetime of the business object.
Initializing the Component
In the @code block for the page an override of the Initialize method is used to set up some basic event handling.
Handling the Saved event is optional, but in most cases you will find it useful to know when the user has submitted
the form and the object has been saved. In this example I am updating some text on the page, but as you'll see in
Chapter 8, this is also an ideal location to implement navigation to another page after saving the object.
The ModelPropertyChanged event must be handled, and when that event occurs the StateHasChanged method needs
to be invoked. Because background tasks might update the state of the business object, the InvokeAsync method is
used to ensure that the StateHasChanged method is invoked in the Blazor UI context.
It is quite realistic for a CSLA-based domain object to be updated by background tasks. The data portal often runs in
the background, as do any async rules you implement as part of your business logic. Handling the
ModelPropertyChanged event ensures that the user's display always matches the properties of the underlying
business domain object.
You can also handle the viewmodel's ModelChanging and ModelChanged events in the Initialize method. These
events are raised before and after the Model property changes, allowing you to perform any necessary work in those
scenarios.
Getting the Model Object
As always, the OnParametersSetAsync method is the event you should use to initialize the component's data. In this
case that means using the ViewModel object to create or fetch the business object.
Page 131
vm.Model = await Csla.DataPortal.FetchAsync<PersonEdit>(id);
The end result is the same. The RefreshAsync method is intended to be a useful abstraction for most scenarios, and
its use is optional.
It is possible for an exception to occur during the RefreshAsync method. In this case you will probably want to
display that information to the user. This is easily done by binding a UI element to the ViewModelErrorText property
of the ViewModel object.
<p class="text-danger">@vm.ViewModelErrorText</p>
For more detailed debugging information the ViewModel type also implements an Exception property that will
reference any Exception that caused the viewmodel to fail to get or save the model.
Saving the Model Object
The ViewModel type implements a SaveAsync method that saves the business object. It is intended for use via data
binding from a save or submit button in the page. For example:
If the save operation is successful the Saved event is raised. Remember that this event is handled in the component's
Initialize method.
It is possible for an exception to occur during the save process. As with the RefreshAsync method, any error
information is available from the viewmodel's ViewModelErrorText property. Also, the Exception property
references any Exception that caused the viewmodel to fail to get or save the model.
No other code is required, the rest of the Blazor page relies on data binding to interact with the model and
viewmodel.
GetPropertyInfo Method
A CSLA domain object typically has properties, and each property has a set of metastate associated with that
property. You can write code to directly interact with the low-level CSLA APIs to access that metastate, but CSLA
provides helper functionality for all supported UI frameworks to simplify the process.
The ViewModel type implements a GetPropertyInfo method that gives you access to the value and metastate values
for a property. In C# code you can call the method like this:
The PropertyInfo type from the Csla.Blazor namespace provides access to the property value and metastate,
including:
Value - read-write access to the property value
FriendlyName - gets the friendly display name for the property
IsBusy - returns true if an async rule is running for this property
ErrorText - gets any Error severity broken rule messages
WarningText - gets any Warning severity broken rule messages
InformationText - gets any Information severity broken rule messages
CanRead - returns true if the current user is authorized to read the property
CanWrite - returns true if the current user is authorized to alter the property
In markup you can access the same information. For example, this code displays the ErrorText value for a Name
property:
Page 132
<div class="alert-danger">@vm.GetPropertyInfo("Name").ErrorText</div>
The GetPropertyInfo method and all the information available from the PropertyInfo type are designed to allow
you to create reusable Blazor components based on your UI requirements. I will discuss how to create these
components later in this chapter.
ViewModel and Authorization
As I discussed in Chapter 5, Blazor has component-level authorization features that are quite useful, if somewhat
coarse-grained. The rules engine in CSLA provides for per-type and per-property authorization rules.
The CanRead and CanWrite properties of the PropertyInfo type allow you to alter the UI based on whether the
current user is allowed to read or write to a specific business object property. I will show how those properties can
be used to create reusable UI components later in this chapter.
At a per-type level you can use properties provided by the ViewModel type.
CanGetObject - returns true if the user is allowed to fetch and view the model
CanEditObject - returns true if the user is allowed to edit the model
CanDeleteObject - returns true if the user is allowed to delete the model's data
You can use these properties to alter the UI so it is clear to the user what they are and are not allowed to do on any
given page or component in your app.
At this point you should have a high level understanding of the features of the ViewModel type and how it supports
building pages and components. I will now show how to create reusable UI components based on these features.
@Property.FriendlyName
@code {
[Parameter]
public Csla.Blazor.IPropertyInfo Property { get; set; }
}
This component requires a Property parameter of type IPropertyInfo. I listed the properties avaiable from the
IPropertyInfo type earlier in this chapter, and you can see here how the FriendlyName property is used to display
the friendly display name for the property.
The PersonEditMvvm page in the Pages folder uses this component to display a label.
Page 133
<div>
<LabelText Property="vm.GetPropertyInfo(nameof(vm.Model.Id))" />
is @vm.Model.Id
</div>
This is preferable to hard-coding label names in each page or component, because CSLA supports concepts such as
localization of the friendly name based on the user's current UI culture. Additionally this approach avoids repeating
the label text through the app, making the app easier to maintain over time.
@if (Property.CanRead)
{
<tr>
<td><LabelText Property="@Property" /></td>
<td>@Property.Value</td>
</tr>
}
@code {
[Parameter]
public Csla.Blazor.IPropertyInfo Property { get; set; }
}
This layout assumes that the first table column is the label and the second column is the value to be displayed.
The component requires a parameter of type IPropertyInfo so it has access to the property value and metastate.
Notice how the LabelText component is reused to display the friendly name, passing the Property value through to
the sub-component.
The CanRead metastate property is used to determine whether the current user is allowed to see this property value.
If the user isn't authorized to see the value then no row is rendered into the table.
You might choose to display something different in the case that the user isn't authorized, but the benefit of having
this DisplayRow component is that it provides a central location to change such UI behavior across the entire app.
The DisplayRow component is used in the PersonEditMvvm page.
<table>
<thead>
<tr>
<th></th>
<th></th>
</tr>
</thead>
<tbody>
<DisplayRow Property="vm.GetPropertyInfo(nameof(vm.Model.Id))" />
</tbody>
</table>
This is a requirement because the DisplayRow component renders content that goes inside a table. If the
component is used outside a table the result will be a runtime rendering error.
Page 134
Building an Input Component
Most data forms support data entry. The TextInput component in the Shared folder implements basic text entry
functionality.
<div>
<input @bind-value="TextValue" @bind-value:event="oninput" type="@InputType"
disabled="@(!Property.CanWrite)" /><br />
<span class="text-danger">@Property.ErrorText</span>
<span class="text-warning">@Property.WarningText</span>
<span class="text-info">@Property.InformationText</span>
</div>
@code {
[Parameter]
public Csla.Blazor.IPropertyInfo Property { get; set; }
[Parameter]
public string InputType { get; set; } = "text";
This component is a little more complex because it not only supports text input, but also CSLA validation messaging
for all severities.
The text input is handled by rendering a standard input element.
The @bind-value property binds the element's text to the text value of the business object property. A TextValue
property is used within the component to cast the property value to a string, so properties of many types can be
edited by the user.
The @bind-value:event="oninput" property ensures that data binding occurs on each keystroke, instead of the
default behavior of binding when the user leaves the field in the browser. This provides for maximum interactivity in
the user experience.
The type="@InputType" property binds the input element's type to a dynamic parameter value. The default value is
text, but having this be a parameter to the component allows the UI developer to use this component for things like
password input.
Finally, the disabled property is bound to the business object property's CanWrite metastate value. This ensures
that the user will only be allowed to edit the value if they are authorized.
Next the three severities of validation messages are displayed, each with the appropriate style.
<span class="text-danger">@Property.ErrorText</span>
<span class="text-warning">@Property.WarningText</span>
<span class="text-info">@Property.InformationText</span>
This component is not directly used in any page, but it is used to create the TextInputRow component.
@code {
[Parameter]
public Csla.Blazor.IPropertyInfo Property { get; set; }
[Parameter]
public string InputType { get; set; } = "text";
}
This component is not very complex, because much of the work has already been implemented in the LabelText
and TextInput components.
This component is used in the PersonEditMvvm page.
<table>
<thead>
<tr>
<th></th>
<th></th>
</tr>
</thead>
<tbody>
<DisplayRow Property="vm.GetPropertyInfo(nameof(vm.Model.Id))" />
<TextInputRow Property="vm.GetPropertyInfo(nameof(vm.Model.Name))" />
<DateInputRow Property="vm.GetPropertyInfo(nameof(vm.Model.Birthdate))" />
<DisplayRow Property="vm.GetPropertyInfo(nameof(vm.Model.Age))" />
</tbody>
</table>
http://localhost:59771/personeditmvvm/123
Page 136
You can see how the validation messages are displayed, and how the save button is disabled because the model
object is not currently valid.
At this point you should understand how to create reusable UI components based on the features of the ViewModel
type. I will use this MVVM technique extensively in Chapter 8.
Next, it is important to understand how CSLA supports user authentication with Blazor.
Authentication
I discussed Blazor authentication in Chapter 5. However, if you are basing your app on a set of domain business
types, your object model may also support authentication. CSLA .NET has always provided support for
authentication, including extending user identity information to smart client platforms when necessary.
In the case of client-side Blazor it is necessary to extend the user identity to the client, otherwise it is not possible to
run authorization business rules on the client device. In the case of server-side Blazor you may also need to extend
the user's identity from the app server to the web server, though it is more likely that you will rely on the underlying
operating system for such impersonation.
In Chapter 5 I talked about how server-side Blazor exclusively uses the ClaimsPrincipal and ClaimsIdentity types,
and how client-side Blazor has no inherent concept of authentication.
This User property adapts to the underlying .NET implementation and platform, providing a common way to get at
the user identity regardless of whether your code is running in Windows Forms, ASP.NET Core, Blazor, Xamarin, or
some other environment.
Page 137
When writing apps using CSLA .NET you should always rely on this User property to gain access to the current user
identity. This ensures that your code is portable across operating systems, .NET implementations, and platforms.
It is a truism that any code running on a client device is more vulnerable to hacking than server-side
code. You need to evaluate whether running any code on a client device fits within your risk profile. Keep in
mind however, that the risk profile for client-side Blazor is the same as Angular, React, Windows Forms, WPF,
UWP, Xamarin, and any other smart-client deployment target.
CSLA .NET makes client-side Blazor authentication easier thanks to built-in support for the concept. The data portal,
when using the default MobileFormatter for serialization, is able to automatically serialize a copy of the server-side
ClaimsPrincipal to the Blazor client running in the browser.
Page 138
builder.Services.AddOptions();
builder.Services.AddAuthorizationCore();
builder.Services.AddSingleton
<AuthenticationStateProvider, CslaAuthenticationStateProvider>();
builder.Services.AddSingleton<CslaUserService>();
The standard AddOptions and AddAuthorizationCore methods configure Blazor for authorization, and are required
for CSLA .NET to properly initialize.
The next line adds a singleton object, so one instance for the lifetime of the app, so Blazor understands to use the
CslaAuthenticationStateProvider to retrieve the current user identity when necessary. This type is in the
Csla.Blazor namespace, and integrates with CSLA to properly manage the current user identity.
And finally, a singleton for CslaUserService is added, because that is required by the
CslaAuthenticationStateProvider type. The CslaUserService instance may be used by pages within the app as
well.
Make sure to add a line to the _Imports.razor component in the Shared folder as well.
@using Microsoft.AspNetCore.Components.Authorization
This brings the namespace into scope for use in Blazor components.
Finally, the content in App.razor needs to be wrapped with a CascadingAuthenticationState component, and the
RouteView must be replaced with an AuthorizeRouteView as discussed in Chapter 5.
<CascadingAuthenticationState>
<Router AppAssembly="@typeof(Program).Assembly">
<Found Context="routeData">
<AuthorizeRouteView RouteData="@routeData" DefaultLayout="@typeof(MainLayout)" />
</Found>
<NotFound>
<LayoutView Layout="@typeof(MainLayout)">
<p>Sorry, there's nothing at this address.</p>
</LayoutView>
</NotFound>
</Router>
</CascadingAuthenticationState>
This completes the basic app configuration necessary to use authentication and authorization in a client-side Blazor
app.
Implementing a Custom Identity Type
It is necessary to implement a custom identity type because this is where you will validate the credentials supplied
by the user. There are many ways you might validate credentials, and in this example the code does no actual
validation. In Chapter 8 you will see a similar class that validates the credentials against a database.
In the Data folder look at the CustomIdentity class.
Page 139
[Serializable]
public class CustomIdentity : Csla.Security.CslaIdentityBase<CustomIdentity>
{
[Fetch]
private void Fetch(UserCredentials credentials)
{
// validate credentials here
if (string.IsNullOrWhiteSpace(credentials.Username))
{
Name = credentials.Username;
IsAuthenticated = true;
AuthenticationType = "Custom";
Roles = new Csla.Core.MobileList<string>();
Roles.Add("StandardUser");
}
else
{
Name = string.Empty;
IsAuthenticated = false;
AuthenticationType = string.Empty;
Roles = new Csla.Core.MobileList<string>();
}
}
}
This class inherits from CslaIdentityBase, a type designed to simplify the creation of custom identity classes. It is
similar to ReadOnlyBase, but also implements the behaviors required by .NET to act as an custom IIdentity type.
Because the base class already implements all necessary identity behaviors, the only code required in the class is to
validate the user credentials and to load the identity object with the user's name and roles.
For example, if the credentials are valid then the object is loaded with appropriate values:
Name = credentials.Username;
IsAuthenticated = true;
AuthenticationType = "Custom";
Roles = new Csla.Core.MobileList<string>();
Roles.Add("StandardUser");
The standard .NET ClaimsPrincipal class understands how to interact with any type that implements IIdentity,
treating each role as a "role claim".
Implementing a Login Page
A login page is typically just a way for the user to enter their username and password, or other credentials. As such,
this is just another type of data entry form that can be built using a domain type.
Before creating the page itself, a domain type to collect the credentials is required. In the sample app's Data folder
this is the UserCredentials class:
Page 140
[Serializable]
public class UserCredentials : BusinessBase<UserCredentials>
{
public static readonly PropertyInfo<string> UsernameProperty = RegisterProperty<string>
(nameof(Username));
[Required]
public string Username
{
get => GetProperty(UsernameProperty);
set => SetProperty(UsernameProperty, value);
}
[Create]
[RunLocal]
private void Create()
{ }
}
This is a standard business class with two required string properties. Because this type will only be used to bind to
a form, it'll never be saved, so there's no need for fetch or update data portal methods.
In the Pages folder, the Login page uses the MVVM pattern I discussed earlier in this chapter to provide the user
with an edit experience over the UserCredentials type. I will discuss only the parts of this page that are unique.
This includes reuse of the TextInputRow UI component. Notice how the password field has an input type of
password.
<TextInputRow Property="vm.GetPropertyInfo(nameof(vm.Model.Password))"
InputType="password" />
The page also relies on NavigationManager and CslaUserService instances, which it gets via injection.
The biggest difference from what I discussed earlier in the chapter is that the submit button doesn't ask the
viewmodel object to save the model, and instead it raises an event to a standard event handler method.
The onclick event is handled by a VerifyCredentials method that does the login operation.
Page 141
private async void VerifyCredentials()
{
var identity = await DataPortal.FetchAsync<CustomIdentity>(vm.Model);
var baseidentity = new ClaimsIdentity(identity.AuthenticationType);
baseidentity.AddClaim(new Claim(ClaimTypes.Name, identity.Name));
if (identity.Roles != null)
foreach (var item in identity.Roles)
baseidentity.AddClaim(new Claim(ClaimTypes.Role, item));
var principal = new System.Security.Claims.ClaimsPrincipal(baseidentity);
userService.CurrentUser = principal;
StateHasChanged();
nav.NavigateTo("/");
}
The data portal is used to fetch a CustomIdentity instance. As I discussed earlier in this chapter, the resulting object
will contain information about whether the credentials are or are not valid.
Modern .NET, and Blazor, are designed with the assumption that the user identity is a ClaimsPrincipal containing a
ClaimsIdentity. The challenge is that ClaimsIdentity is not serializable, so the server-side code is creating an
instance of CustomIdentity and the VerifyCredentials method needs to translate the identity information into a
new ClaimsIdentity. That is the purpose of this block of code:
Using that new ClaimsIdentity instance, a .NET ClaimsPrincipal is created. Then that principal object is set to
represent the current user by setting the CurrentUser property value of the CslaUserService object.
Because the data portal is used to asynchronously validate the credentials on the server, the StateHasChanged
method must be called so Blazor realizes it needs to update the UI.
Finally, the Blazor navigation manager is used to navigate to the root page for the app.
Implementing a Logout Page
The Logout page, in the Pages folder, is similar to the Login page.
@page "/logout"
@using System.Security.Claims
@using Csla.Blazor.Client.Authentication
@inject CslaUserService userService
@inject NavigationManager nav
<h3>Logout</h3>
@code {
private void LogoutUser()
{
userService.CurrentUser =
new ClaimsPrincipal(new ClaimsIdentity());
nav.NavigateTo("/");
}
}
This implementation acts as a confirmation page. You can also create this page without any visible UI elements and
just log out the user as the page loads.
Page 142
@page "/logout"
@using System.Security.Claims
@using Csla.Blazor.Client.Authentication
@inject CslaUserService userService
@inject NavigationManager nav
@code {
protected override void Initialize()
{
userService.CurrentUser =
new ClaimsPrincipal(new ClaimsIdentity());
nav.NavigateTo("/");
}
}
Either way, the page sets the CurrentUser property of the CslaUserService object to an unauthenticated
ClaimsPrincipal and navigates to the home page for the app.
Because this code has no async server calls, it is not necessary to call StateHasChanged to update the UI.
At this point you should understand how CSLA .NET integrates with server-side and client-side Blazor to enable
authentication of a user. Once a user has been authenticated it is possible to implement authorization rules to
control what the user can and can not do within your app.
Authorization
Authorization relies on the user being authenticated, so there is a ClaimsPrincipal available from
Csla.ApplicationContext.User on server or client. This enables the use of the helper types in the Csla.Blazor
namespace.
Authorization is generally per-component. If you want more fine-grained authorization you should use the features
from the Csla.Blazor.ViewModel type or the Csla.Rules namespace.
I discussed the per-component Blazor authorization functionality in Chapter 5. That behavior relies on the user being
authenticated, and is unaffected by whether you use your own custom authentication or the CSLA-based
authentication approach discussed in this chapter.
If you want more fine-grained authorization behaviors that rely on the business rules in your CSLA-based domain
business layer you should use the helper types from the Csla.Blazor namespace. These include:
CslaPermissionsHandler - Blazor permissions handler automatically configured by CSLA
CslaPermissionPolicy - Blazor permission policy automatically configured by CSLA
CslaPermissionRequirement - Permission definition used by CslaPolicy and HasPermission
CslaPolicy - Blazor authorization policy used to check per-type CSLA authorization rules
HasPermission - attribute used on a Blazor component to check per-type CSLA authorization rules
<AuthorizeView Policy="MyPolicy">
Page 143
The Blazor authorization system relies on a list of policy handlers to enforce policies. The authorization subsystem
loops through the list of handlers, asking each if it can handle the requested policy. If a handler can handle the
policy, it provides a true/false answer.
CslaPolicy
CSLA authorization rules are relatively complex, at least compared to a simple string-based model. A per-type
authorization rule in CSLA has a action and target elements.
The action element indicates whether the current user can:
Create an instance of the target
Get/view the target
Edit/save the target
Delete the target
The target is typically a business domain type.
The CslaPolicy type exposes a GetPolicy method that accepts the action and target and returns a string value that
encodes both values. This string value is then used as a policy name anywhere the Blazor authorization subsystem
requires such a name.
The CslaPermissionsHandler type, automatically configured by CSLA, handles these encoded policies, providing the
appropriate response based on the specified action and target. It does this by invoking the CSLA API, asking the
business layer whether the current user is authorized to perform the requested action on the indicated target.
The BlazorCslaAuthentication sample demonstrates this functionality. In the Index page the following markup
uses CslaPolicy to create a policy name.
<AuthorizeView
Policy="@(CslaPolicy.GetPolicy(AuthorizationActions.CreateObject,
typeof(PersonEdit)))">
<Authorized>
User can create a person object
</Authorized>
<NotAuthorized>
User can NOT create a person object
</NotAuthorized>
</AuthorizeView>
The project includes a PersonEdit class that defines per-type rules for the PersonEdit class.
[ObjectAuthorizationRules]
private static void PerTypeRules()
{
Csla.Rules.BusinessRules.AddRule(
typeof(PersonEdit),
new Csla.Rules.CommonRules.IsInRole(
Csla.Rules.AuthorizationActions.CreateObject,
"PersonCreator"));
}
This is standard CSLA business layer code, indicating that only users who are members of the "PersonCreator" role
are allowed to create new instances of the PersonEdit type.
You can use the AuthorizeView element to display different UI content to the user based on whether the user is
allowed to create, view, edit, or delete business domain types.
HasPermission
Page 144
The HasPermission attribute operates at a component level. Adding this attribute to the top of a component uses
your business domain layer's authorization rules to grant or deny access to the component as a whole.
This is comparable to the default Blazor behavior, but doesn't require hard-coding the roles or authorization policies
into your UI layers. Instead this approach takes advantage of the rules already encoded into your business layer.
Conclusion
CSLA .NET and Blazor are a perfect match! Blazor is a UI framework built directly on top of the core architectural
concepts I've been championing with CSLA since 1996.
In Chapter 8 I will discuss the ProjectTracker reference app, and how it implements server-side and client-side Blazor
apps based on everything discussed in Chapters 1 through 7.
Page 145
Chapter 8: ProjectTracker UI Using Blazor
In Chapters 1-6 I discussed the basic use of Blazor, including data binding, multi-headed deployments,
authentication, and authorization. In Chapter 7 I discussed how CSLA .NET supports Blazor UI development with the
helper types in the Csla.Blazor NuGet package and namespace.
In this chapter I will use the ProjectTracker reference app as an example of how to build server-side and client-side
Blazor apps, with a common UI codebase, that makes use of a CSLA-based business domain layer.
The ProjectTracker app includes a data access layer, a business layer, and numerous apps for various types of UI
technologies. All the apps make use of those common business and data layers, and the existing ProjectTracker app
server that exposes CSLA data portal endpoints.
The solution, with the Blazor UI implementation, exists in GitHub. I will describe how the Blazor UI projects were
created, configured, and implemented.
<Project Sdk="Microsoft.NET.Sdk.Razor">
<PropertyGroup>
<TargetFrameworks>netcoreapp3.1;netstandard2.1</TargetFrameworks>
<LangVersion>7.3</LangVersion>
<RazorLangVersion>3.0</RazorLangVersion>
</PropertyGroup>
Notice the TargetFrameworks element and how it specifies the two targets for compilation, as discussed in Chapter
6.
The netstandard2.1 target is required by the client-side Blazor app, and the netcoreapp3.1 target is required by the
server-side Blazor app.
The project relies on types from the Csla.Blazor NuGet package, and so the csproj file references that package and
its dependencies:
Page 146
<ItemGroup>
<PackageReference Include="Csla.Blazor" Version="5.1.0" />
<PackageReference Include="Microsoft.AspNetCore.Components.Authorization" Version="3.1.1" />
</ItemGroup>
Because this project will implement the Blazor UI, it needs access to the existing business domain layer in the
ProjectTracker.BusinessLibrary.NetStandard project. It also needs access to the abstract types from the data
access layer in the ProjectTracker.Dal project. Here is the relevant code from the csproj file:
<ItemGroup>
<ProjectReference Include="..\ProjectTracker.BusinessLibrary.Netstandard
\ProjectTracker.BusinessLibrary.Netstandard.csproj" />
<ProjectReference Include="..\ProjectTracker.Dal\ProjectTracker.Dal.csproj" />
</ItemGroup>
I won't discuss the business or data access layers in this book, focusing instead on how to leverage the business
domain types when creating a Blazor UI.
Let's talk about the client-side and server-side app projects.
This means that the project can be directly launched from Visual Studio, or published and deployed to nearly any
static web server host.
Removing Template Items
Page 147
The resulting project contains Blazor components and other elements that are not needed because this project will
get its UI elements from the shared project.
Remove the
Pages folder
Shared folder
_Imports.razor file
App.Razor file
All that is required is the Program.cs file for startup configurtation, and the wwwroot folder that contains static CSS
and other resources required by the UI.
Project References
The project references the Csla.Blazor NuGet package and other standard client-side Blazor packages. And it
references the shared UI project. All this is in the csproj file:
<Project Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup>
<TargetFramework>netstandard2.1</TargetFramework>
<RazorLangVersion>3.0</RazorLangVersion>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Csla.Blazor" Version="5.1.0" />
<PackageReference Include="Microsoft.AspNetCore.Blazor" Version="3.1.0-preview4.19579.2" />
<PackageReference Include="Microsoft.AspNetCore.Blazor.Build" Version="3.1.0-preview4.19579.2"
PrivateAssets="all" />
<PackageReference Include="Microsoft.AspNetCore.Blazor.HttpClient" Version="3.1.0-preview4.19579.2" />
<PackageReference Include="Microsoft.AspNetCore.Blazor.DevServer" Version="3.1.0-preview4.19579.2"
PrivateAssets="all" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\ProjectTracker.Ui.Blazor\ProjectTracker.Ui.Blazor.csproj" />
</ItemGroup>
</Project>
Because this project targets .NET Standard 2.1, the reference to ProjectTracker.Ui.Blazor will use that project's
.NET Standard 2.1 compile target.
App Configuration
Page 148
The Program class contains the startup configuration for the client-side Blazor app. The configuration is the same as
discussed in Chapter 7.
builder.Services.AddOptions();
builder.Services.AddAuthorizationCore();
builder.Services.AddSingleton<AuthenticationStateProvider,
CslaAuthenticationStateProvider>();
builder.Services.AddSingleton<CslaUserService>();
builder.UseCsla(c => c
.DataPortal()
.DefaultProxy(typeof(Csla.DataPortalClient.HttpProxy),
"http://localhost:8040/api/dataportaltext/"));
The AddOptions and AddAuthorizationCore methods are required to configure Blazor authentication and
authorization. The CslaAuthenticationStateProvider and CslaUserService are necessary services for
authentication to work as discussed in Chapter 7.
The UseCsla method configures CSLA .NET to work properly in a client-side Blazor app, and this code also
configures the data portal to communicate with the existing ProjectTracker app server.
At this point the client-side Blazor project is complete, but there's one more important detail to cover: configuring
the app server.
Configuring the App Server
The ProjectTracker app server is in the ProjectTracker.AppServerCore project. This is an ASP.NET Core project that
exposes data portal endpoints for use by all the different UI apps in the ProjectTracker solution.
There are two changes necessary to use this app server from client-side Blazor: a text-based data portal endpoint,
and CORS (cross-origin resource sharing) configuration.
Text-Based Data Portal Endpoint
Remember from Chapter 7 that Blazor WebAssembly can only transfer text-based data via HTTP. This means any
app server must expose a data portal endpoint that sends and receives text-encoded data.
In the project's Controllers folder the DataPortalTextController class implements this endpoint.
[Route("api/[controller]")]
[ApiController]
public class DataPortalTextController : Csla.Server.Hosts.HttpPortalController
{
public DataPortalTextController()
{
UseTextSerialization = true;
}
}
When running the ProjectTracker solution in debug mode on your local computer, the URL for this endpoint is
http://localhost:8040/api/dataportaltext/.
Contrast this to the DataPortalController that uses the default binary behavior.
[Route("api/[controller]")]
[ApiController]
public class DataPortalController : Csla.Server.Hosts.HttpPortalController
{
}
When running the ProjectTracker solution in debug mode on your local computer, the URL for this endpoint is
http://localhost:8040/api/dataportal/.
Page 149
Most client apps can transfer data in binary format and so use this second endpoint. The Blazor WebAssembly app
will use the first endpoint to transfer the same data, but text-encoded.
CORS Configuration
By default, ASP.NET Core prevents the use of cross-origin resource sharing (CORS). This means that, by default,
browser-hosted apps are not allowed to directly call the service endpoints exposes by the app server. That is true for
any JavaScript-based technologies such as Angular and React, as well as WebAssembly-based technologies such as
Blazor.
Enabling CORS in ASP.NET Core requires editing the Startup class and adding code to the ConfigureServices
method:
services.AddCors(options =>
{
options.AddPolicy(BlazorClientPolicy,
builder =>
{
builder
.AllowAnyOrigin()
.AllowAnyHeader()
.AllowAnyMethod();
});
});
app.UseCors(BlazorClientPolicy);
Page 150
The following are removed:
Data folder
Counter.razor from the Pages folder
Error.razor from the Pages folder
FetchData.razor from the Pages folder
Index.razor from the Pages folder
_Imports.razor file
App.razor file
These are not needed because the Blazor UI will come from the shared UI project.
Project References
The server-side project references the shared UI project in its csproj file:
<ItemGroup>
<ProjectReference Include="..\ProjectTracker.Ui.Blazor\ProjectTracker.Ui.Blazor.csproj" />
</ItemGroup>
Because this is a .NET Core 3.1 project, the reference to ProjectTracker.Ui.Blazor will pull in the .NET Core 3.1 DLL
from that project.
The server-side project also references the Csla.AspNetCore and Csla.Blazor packages along with other required
packages:
Page 151
<ItemGroup>
<PackageReference Include="Csla.AspNetCore" Version="5.1.0" />
<PackageReference Include="Csla.Blazor" Version="5.1.0" />
<PackageReference Include="Microsoft.Extensions.Logging.Debug" Version="3.1.1" />
<PackageReference Include="Microsoft.VisualStudio.Web.CodeGeneration.Design" Version="3.1.0" />
</ItemGroup>
As I discussed in Chapter 6, when creating a server-side Blazor project with CSLA .NET both the Csla.AspNetCore
and Csla.Blazor packages are required. This is because the app is both ASP.NET Core and also Blazor at the same
time.
App Configuration
Server-side Blazor app configuration occurs in the Startup class.
The ConfigureServices method needs two new lines of code:
services.AddHttpContextAccessor();
services.AddCsla().WithBlazorServerSupport();
The AddHttpContextAccessor method allows the use of HttpContext throughout the app. This is required because
CSLA .NET uses the HttpContext instance.
The AddCsla method adds services required by CSLA .NET to function.
And the Configure method needs to configure for authentication and authorization:
app.UseAuthentication();
app.UseAuthorization();
app.UseCsla(c => c
.DataPortal()
.DefaultProxy(typeof(Csla.DataPortalClient.HttpProxy),
"http://localhost:8040/api/dataportal/"));
The UseCsla method configures CSLA .NET to function in an ASP.NET Core Blazor app, and also configures the data
portal to interact with the app server.
Notice that the data portal is configured to call the binary data portal endpoint. Because the Blazor code is running in
ASP.NET Core it can use binary data transfers. Only client-side Blazor running in WebAssembly is restricted to using
text-based encoding.
At this point the server-side Blazor app is configured. The next step is to implement the Blazor UI itself, in the shared
UI project.
There are many ways to design and implement any user experience. The choices I've made in the
design of the ProjectTracker UI are just one option, and you may choose other navigation, display, or
interaction patterns. That's fine, and you should be able to adapt the concepts from this chapter to many
other types of Blazor UI.
Let's start by discussing one of the most common user scenarios: displaying a read-only list.
Read-Only Lists
One of the most common user scenarios is for the user to view a list of items, and possible select an item from the
list with which to interact. That interaction might be to view details, edit, delete, or otherwise affect the selected item
from the list.
CSLA .NET supports this concept at the business layer with the read-only list stereotype, relying on the
ReadOnlyList base type to create business objects for use by the UI app.
In the ProjectTracker.Library namespace, the ProjectList and ResourceList classes are examples of the read-
only list stereotype. Each of them relies on a child type, ProjectInfo and ResourceInfo respectively, to contain the
necessary information for each item in the list.
In the ProjectTracker.Ui.Blazor project these types are exposed to the user in the Projects and Resources pages
in the Pages folder. These pages use the same implementation, and so I will only walk through the Projects
component.
Getting to the Page
The NavMenu component in the Shared folder inludes a link so the user can access the Projects page.
@page "/projects"
@using Csla.Blazor
@using Csla.Rules
@inject ViewModel<ProjectTracker.Library.ProjectList> vm
As discussed in Chapter 7, in this chapter I am using the MVVM design pattern that relies on the ViewModel type
from the Csla.Blazor namespace. In this case the component requires access to a viewmodel object representing
the ProjectList business type.
Displaying Page Level Errors
The page has markup to display the list of items to the user. Near the top of the markup is a line to display any
page-level errors that might occur.
Page 153
<p class="text-danger">@vm.ViewModelErrorText</p>
This line binds to the viewmodel object's ViewModelErrorText property, which will contain any error messages
encountered by the ViewModel type as it attempts to retrieve or save the model object.
Handling a Null Model
As the page first loads the Model property of the viewmodel object will be null. As I discussed in Chapter 4, Blazor
data binding can't handle a null binding target, so it is necessary to check the value and display alternate content
until the value is no longer null:
This way, as the page first appears and is waiting to get any data from the app server or database, the user will see
"Loading..." on the page.
The rest of the markup is in an else clause and is visible when the Model property is not null.
Allow User to Add New Item
Users may or may not be authorized to add new items to the list. This business rule is encoded in the business
domain layer as a per-type rule associated with the ProjectEdit type. The UI should reflect this rule by only
showing the user a "new item" link if they are authorized to add a new item:
<AuthorizeView Policy="@(CslaPolicy.GetPolicy(AuthorizationActions.CreateObject,
typeof(ProjectTracker.Library.ProjectEdit)))">
<p><a href="projectedit">Add project</a></p>
</AuthorizeView>
The Blazor AuthorizeView component is used, along with the CslaPolicy type discussed in Chapter 7, to determine
whether the current user is authorized to create new projects, and whether to display the "Add project" link on the
page.
If the user clicks the "Add project" link the app will navigate to the ProjectEdit page. I will discuss that page later in
this chapter.
Displaying the List of Items
The next block of markup defines a table with three columns:
1. Display item Id property
2. Display item Name property
3. Provide the user with any actions allowed on the item
Displaying the list of Id and Name values is straightforward:
A foreach loop is used, and each item in the list is rendered as a row in the table.
The third column is more complex however, because authorization business rules must be honored. The current
user may or may not be allowed to edit or delete each item in the list.
Page 154
<td>
<AuthorizeView Policy="@(CslaPolicy.GetPolicy(
AuthorizationActions.EditObject,
typeof(ProjectTracker.Library.ProjectEdit)))">
<a href="projectedit/@item.Id">Edit</a>
<span> | </span>
</AuthorizeView>
<AuthorizeView Policy="@(CslaPolicy.GetPolicy(
AuthorizationActions.DeleteObject,
typeof(ProjectTracker.Library.ProjectEdit)))">
<a href="javascript: void(0);"
@onclick="() => SelectForDelete(item.Id)">Delete</a>
</AuthorizeView>
</td>
The AuthorizeView component is used with the CslaPolicy type to determine whether the business rules encoded
in the business layer allow the current user to edit or delete each row.
If the user is authorized to edit an item they will see an "Edit" link. Clicking that link will navigate the user to the
Projectedit page. I will discuss that page later in this chapter.
If the user is authorized to delete an item they will see a "Delete" link. Clicking that link will call a SelectForDelete
method in the page's code.
Implementing Delete Confirmation
When the user indicates they want to delete an item by clicking the "Delete" link on a row I want to confirm that they
really mean to delete the item.
There are many ways to implement such a confirmation step, and in this page I have chosen to use an in-place
confirmation model.
Page 155
<AuthorizeView Policy="@(CslaPolicy.GetPolicy(
AuthorizationActions.DeleteObject,
typeof(ProjectTracker.Library.ProjectEdit)))">
<a href="javascript: void(0);"
@onclick="() => SelectForDelete(item.Id)">Delete</a>
</AuthorizeView>
Notice that the @onclick event is handled by a lambda expression that invokes the SelectForDelete method. This is
necessary because the method requires a paramter, and a lambda expression allows passing in a parameter to a
method.
The SelectForDelete records the Id property value of the item selected:
The itemSelectedForDeletion field is used in the page to determine which markup should be rendered for the
selected row:
If the current row matches the selected row then a delete confirmation is rendered:
Page 156
private async void Delete(int id)
{
itemSelectedForDeletion = -1;
vm.Model = null;
await ProjectTracker.Library.ProjectEdit.DeleteProjectAsync(id);
await vm.RefreshAsync();
StateHasChanged();
}
This method uses the existing business layer API to call the DeleteProjectAsync method of the ProjectEdit domain
type. That method implements the behavior necessary to delete a project from the system.
Once the project has been deleted it is necessary to reload the Model object so the list reflects the change. This is
handled by calling the RefreshAsync method on the viewmodel object.
Because the Delete method calls asynchronous methods, it is necessary to tell Blazor to refresh the display by
calling the StateHasChanged method.
If the user clicks the "Cancel" link the implementation is simpler. All that is necessary is to set the
itemSelectedForDeletion field to an invalid value such as -1. This ensures that it will never match any row in the
ProjectList collection and so the confirmation markup will never appear to the user.
The Projects and Resources pages demonstrate how to display read-only lists to the user, and to add actions for
the user to take based on authorization rules from your business layer.
<AuthorizeView Policy="@(CslaPolicy.GetPolicy(
AuthorizationActions.EditObject,
typeof(ProjectTracker.Library.Admin.RoleEditList)))">
<li class="nav-item px-3">
<NavLink class="nav-link" href="roles">
<span class="oi oi-plus" aria-hidden="true"></span> Roles
</NavLink>
</li>
</AuthorizeView>
This ensures that only authorized users will see the "Roles" link.
Page 157
When the user clicks on the "Roles" link they are presented with the Roles page.
The Roles page has an authorization attribute to prevent unauthorized users from navigating directly to the URL:
In both cases the business layer API is used to determine if the current user is authorized to edit an object of type
RoleEditList so there are no explicit rules in the UI, and yet the UI is able to alter the user experience based on the
rules.
Page Layout
The page also relies on a ViewModel instance for the RoleEditList type:
@inject ViewModel<RoleEditList> vm
As with all pages that dynamically load content for data binding, any viewmodel errors are displayed and the Model
property is check for a null value:
<p class="text-danger">@vm.ViewModelErrorText</p>
If the Model value is not null then the user might be presented with a link to add new items.
<AuthorizeView Policy="@(CslaPolicy.GetPolicy(
AuthorizationActions.CreateObject,
typeof(ProjectTracker.Library.Admin.RoleEditList)))">
<p><a href="javascript: void(0);" @onclick="AddRole">Add role</a></p>
</AuthorizeView>
Page 158
The AuthorizeView component is used to determine whether to display the link to the current user, based on
authorization rules implemented in the business layer.
If the user clicks the "Add role" link an AddRole method is called. I will discuss this method and how a new item is
added to the list later in this chapter.
As with the Projects page, a table is used to define three columns for the Id and Name properties, and for potential
user actions.
<table class="table">
<thead>
<tr>
<th>Id</th>
<th>Name</th>
<th></th>
</tr>
</thead>
The rest of the page markup is a little more complex than the Projects page. This is because each row of content
can be displayed in one of several modes:
Display the item plus actions
In-place editing of the item
In-place confirmation of item deletion
A foreach loop is used to display each item in the list, and page-level itemSelectedForEdit and
itemSelectedForDeletion fields are used to indicate the rows that should be rendered for edit or confirmation of
delete.
Let's look at the markup and code for each mode.
Display Item
If an item is not selected for edit or delete, its data is displayed to the user, along with links for authorized actions.
<td>@item.Id</td>
<td>@item.Name</td>
<AuthorizeView Policy="@(CslaPolicy.GetPolicy(
AuthorizationActions.EditObject,
typeof(ProjectTracker.Library.Admin.RoleEditList)))">
<Authorized>
<td>
<a href="javascript: void(0);"
@onclick="() => SelectForEdit(item.Id)">Edit</a>
<span> | </span>
<a href="javascript: void(0);"
@onclick="() => SelectForDelete(item.Id)">Delete</a>
</td>
</Authorized>
<NotAuthorized>
<td></td>
</NotAuthorized>
</AuthorizeView>
The AuthorizeView component is used to check the authorization rules for the current user. The "Edit" and "Delete"
links are only visible to authorized users.
Page 159
If the user clicks the "Edit" or "Delete" links the code calls SelectForEdit or SelectForDelete methods respectively,
passing the current item's Id property as a parameter.
I will discuss these two methods in the context of their specific UI.
Delete Item
The SelectForDelete method is very similar to the one in the Projects page:
The difference is that this code makes sure any current add or edit operation is stopped before marking an item for
deletion.
When the list of items is rendered, any item marked for deletion will render with a confirmation link instead of the
default actions.
This is the same implementation as in the Projects page earlier in this chapter.
Page 160
The Delete method is invoked if the user clicks the "Confirm" link.
This method finds the selected item in the list and removes it from the list. Remember that the RoleEditList type is
an editable business list and so its items are manipulated in memory, and then the list is saved.
The SaveAndRefresh method is called to save the list and refresh the UI.
The SaveAsync method saves any changes in the list by calling the data portal. The RefreshAsync method reloads
the list from the app server so it reflects any changes to the underlying database.
Finally, because these are asynchronous method calls, the StateHasChanged method is called so Blazor understands
that it must refresh all data binding.
Edit Item
Editing an item is a little more complex because changes made to an item occur in memory (potentially in a
browser) on the client, and if the user chooses to cancel edits made to a row there must be some way to reset the
item to its original state.
This can be done by reloading the list with a data portal call, but it is usually more effective to just restore the
original values without calling a remote service. CSLA .NET supports this concept using a concept called n-level undo
as discussed in the Using CSLA: Creating Business Objects book.
The idea is that a snapshot of the row object's state is taken before allowing the user to edit any values. If the user
accepts their edits then the snapshot is discarded. If the user cancels their edits then that original snapshot is used to
restore the object to its original values.
Page 161
When the user clicks the "Edit" link on an item in the list a SelectForEdit method is invoked:
This method makes sure no new item is being added or existing item being edited or deleted, then it sets the
itemSelectedForEdit field to indicate that this row is being edited.
Finally the item is found in the collection and a BeginEdit method is called. This method takes a snapshot of the
object's current state for later use.
In the markup, the itemSelectedForEdit field is used to render the selected row for editing:
This markup renders input controls for each editable property of the business object, along with "Confirm" and
"Cancel" action links.
If the user clicks the "Cancel" link the EditCancelled method is invoked:
Page 162
private void EditCancelled()
{
if (itemSelectedForEdit > -1)
{
var item = vm.Model
.Where(r => r.Id == itemSelectedForEdit)
.FirstOrDefault();
if (item != null)
item.CancelEdit();
itemSelectedForEdit = -1;
}
}
This method finds the item currently being edited and calls its CancelEdit method. This method restores the object's
original state from the snapshot created by calling the BeginEdit method earlier.
If the user clicks the "Confirm" button the EditAccepted method is called:
This method finds the item being edited and calls its ApplyEdit method. This method discards the snapshot taken by
the BeginEdit method, effectively committing the user's changes in memory.
The method then calls the SaveAndRefresh method to save the list and refresh the UI. This is the same method called
in the delete operation.
Whether the user clicks "Confirm" or "Cancel" or chooses to edit or delete another row, this row is no longer
selected for editing.
Add a New Item
The last action available to the user in this page is to click an "Add role" link to add a new item to the list. When the
user clicks that link an AddRole method is called:
This method ensures no existing item is being edited or deleted, then it creates a new child object for the list:
newRole = Csla.DataPortal.CreateChild<RoleEdit>();
It is important to understand that this new RoleEdit item has not been added to the list, it is referenced by the
newRole field in the page.
The addingRole field is set so Blazor renders UI for the user to add a new item. Here is the markdown:
Page 163
@if (addingRole)
{
<tr>
<td>new</td>
<td>
<TextInput Property=
"@(new PropertyInfo(newRole, nameof(newRole.Name)))" />
</td>
<td>
<a class="alert-warning" href="javascript: void(0);"
@onclick="AddAccepted">Confirm</a>
<span> | </span>
<a href="javascript: void(0);"
@onclick="AddCancelled">Cancel</a>
</td>
</tr>
}
This markup is above the foreach loop that displays the existing items from the list. That means when the user clicks
the "New role" link that the new item appears at the top of the list.
As with other edit pages, the TextInput component is used to provide the user with a rich editing experience,
including the display of validation messages from the business layer. In this case though, the Proeprty property is
set in a new way:
<TextInput Property=
"@(new PropertyInfo(newRole, nameof(newRole.Name)))" />
Remember that this newRole object is not in the list, and so it is not in the viewmodel object's Model object. That
means that the viewmodel object can't provide a pre-created PropertyInfo representing the newRole object's Name
property.
The @(new PropertyInfo(newRole, nameof(newRole.Name))) clause creates a PropertyInfo object that does
represent this object's Name property and all its metastate. That allows the TextInput component to work as
expected.
After editing the row, the user can click on the "Cancel" button, calling the AddCancelled method:
Page 164
private void AddCancelled()
{
addingRole = false;
newRole = null;
}
This method discards the new object without adding it to the list or calling the data portal, and it sets the addingRole
field so Blazor stops rendering the UI for adding a new item.
The user can click on the "Confirm" link to run the AddAccepted method:
This method sets the addingRole field to disable the add new item UI. Then it adds the newRole object to the
collection. Finally, it calls the SaveAndRefresh method to save the list through the data portal and to then refresh the
UI with any changes to the underlying data.
At this point you should understand how to display a read-only list with actions, and how to create an in-place
editing experience for an editable collection of simple data. Now I'll discuss how to create more complex edit pages,
including master-detail relationships.
Master-Detail Pages
The ProjectEdit and ResourceEdit pages in the ProjectTracker app implement master-child edit experiences. They
are the same, and so I will focus on the ProjectEdit page.
I've chosen to implement the ProjectEdit page to use an explicit "Save" button. This means that all edits made by
the user occur in memory and are not sent to the server until the user clicks that "Save" button. Also, the "Save"
button will not be available for the user to click until the business object graph is in a saveable state.
This page makes use of the ProjectEdit business domain type from the business layer. That type encapsulates the
business rules and behaviors necessary to edit a project, and is the root object for an object graph that allows the
user to view, add, edit, and remove resources from the project.
Page Layout
The ProjectEdit page has several areas of UI functionality:
Page header
Edit the ProjectEdit object properties
Display list of assigned resources, with associated actions
Page 165
Page Header
The page header markup is similar to the previous pages:
@page "/projectedit"
@page "/projectedit/{id}"
@inject Csla.Blazor.ViewModel<ProjectTracker.Library.ProjectEdit> vm
@inject NavigationManager NavigationManager
<h1>Edit Project</h1>
<p class="text-danger">@vm.ViewModelErrorText</p>
<p>
<a href="projects">Project list</a>
</p>
Key differences from the earlier pages in this chapter are that this page makes use of the Blazor NavigationManager
type, and provides a "Project list" page that navigates to the Projects page so the user can see a list of projects.
Edit Project Properties
The page uses a table to create a two-column edit experience for the properties of the ProjectEdit business object.
The first column is the friendly name of the property, and the second column allows the user to edit the property
value. These are all created using the reusable UI components discussed in Chapter 7.
Page 166
<table class="table">
<thead>
<tr>
<th></th>
<th></th>
</tr>
</thead>
<tbody>
<LabelRow Property="vm.GetPropertyInfo(nameof(vm.Model.Id))" />
<TextInputRow Property="vm.GetPropertyInfo(nameof(vm.Model.Name))" />
<TextAreaRow rows="5" Property="@(vm.GetPropertyInfo(nameof(vm.Model.Description)))" />
<DateInputRow Property="@(vm.GetPropertyInfo(nameof(vm.Model.Started)))" />
<DateInputRow Property="@(vm.GetPropertyInfo(nameof(vm.Model.Ended)))" EmptyIsMin="false" />
</tbody>
</table>
The UI components used here understand how to interact with CSLA-based authorization and validation rules,
adjusting the page content to reflect what the user can and can't view or edit.
I do want to call your attention to the Ended property.
The Started and Ended properties of the ProjectEdit business class use the CSLA .NET SmartDate data type, which
understands the concept of an "empty date". For comparison purposes, when a SmartDate is "empty" the value is
either smaller than, or larger than, the smallest or largest possible date. By default an "empty" value is the smallest
value.
In the case of the Ended date it is important that an "empty" date be the largest possible date, because there is a
business rule that says a project must start before it ends. If the user enters a start date with an empty end date, that
rule should pass.
The DateInputRow component relies on a DateInput component, and that component understands how to work
with a SmartDate value. In particular, the DateInput component has code to convert the business object's property
value into and out of a text value:
Notice how the EmptyIsMin value is passed through to the SmartDate type so it behaves as required.
Display Assigned Resources
Page 167
The light gray block of content on the page is also a table, with each row displaying a resource assigned to the
project. The ProjectEdit business class has an AssignedResources property that provides access to the list of
assigned resources.
The assigned resources display area of the page will change, depending on the user's actions. I am using a technique
similar to the in-place delete and edit implementations shown earlier in this chapter. In this case the region of the UI
will be one of:
Displaying the list of assigned resources
Show a list of resources that could be assigned to the project
Allow the user to edit an assigned resource
I chose to use an enum to define these three UI modes:
In the ProjectEdit page there is a field used to control which view is rendered.
Page 168
The user can see the first name, last name, and role of the resource, along with actions available for the item.
As with the previous pages in this chapter, the action links invoke methods in the page's code, changing the mode of
the display to allow the user to add, edit, or remove the item.
I will discuss the EditResource and RemoveResource methods later in this chapter. First I will walk through the
process of assigning a new resource to the project.
Assign a Resource
At the top of the default display is a link so the user can assign a resource to the project.
This method changes the display mode to Select and it loads a list of resources from the data portal. The Where
clause is used so the list won't include resources that are already assigned to the project.
Here is the markup for the Select UI mode:
The resulting UI has a "Cancel assignment" link and a two column table listing the resource name and a "Select"
Page 169
action link.
If the user clicks the "Cancel assignment" link the ShowDefaultView method is called:
This resets the view to the default mode, so the user goes back to seeing the list of resources already assigned to the
project. No new assignment occurs, and the CancelEdit method ensures that any changes the user might have
made to this object's state are reset to a snapshot value.
If the user clicks the "Select" link on a listed resource, the AssignRole method is invoked, passing the selected item's
Id property value as a parameter:
The AssignRole method uses the ProjectResourceEditCreator type from the business library. This type
implements a command that creates a ProjectResourceEdit object for a specific resource. This new object is the
correct type to become a child object in the ProjectEdit object graph, but you can see that the code doesn't add it
to the list of assigned resources.
Instead, the code changes the UI display mode to Details so the user can select the new resource's role on the
project, and also choose to confirm assigning the resource (or not).
First though, it calls the BeginEdit method on the business object to take a snapshot of its current state. This is
necessary because the user can choose to cancel the assignment operation later in the UI workflow.
Edit Resource Assignment Details
There are two ways the UI display mode can be set to Details. One is the result of the user selecting a resource to
assign to the project. The other is if the user clicks the "Edit" action link on a resource that is already assigned to the
project.
Clicking that "Edit" button invokes the EditResource method:
Page 170
private void EditResource(int resourceId)
{
selectedResource = vm.Model.Resources
.Where(r => r.ResourceId == resourceId).FirstOrDefault();
if (selectedResource != null)
{
selectedResource.BeginEdit();
viewMode = SubViewModes.Details;
}
}
Either way, the page's selectedResource field will be set to the ProjectResourceEdit object representing the
resource the user selected, and that object will have had BeginEdit called to take a snapshot of the object's previous
state. The UI will display the details about that assignment:
In this display mode the user is shown the resource name, and is allowed to edit the role the resource will play on
the project. There is also a "Cancel assignment" link to cancel any changes, and an "Assign" action link to confirm the
changes.
The business layer includes a read-only collection RoleList type that is used to get the list of possible roles. This list
is normally cached in memory in the app, with the assumption being that the list of roles changes rarely.
The list of roles is used to create a dropdown combobox UI element by using the select and option elements.
If the user clicks the "Cancel assignment" link, the ShowDefaultView method is called. I discussed this method earlier,
talking about how it changes the UI display mode back to Default without saving any changes made by the user.
In this case remember that the user could be in Details mode because they are adding or editing a resource
Page 171
assignment. In both cases, reverting to the Default UI mode without explicitly saving changes to the business object
graph means that any changes are discarded.
If the user clicks the "Assign" link the AddResource method is invoked:
This method calls the ApplyEdit method to accept any changes to the object's state made by the user.
It then checks to see if the list of assigned resources already contains the selectedResource item. If it does not, then
the item is added to the collection.
Finally, the ShowDefaultView method is called to display the list of assigned resources to the user.
Remove a Resource Assignment
The final action the user can take with an assigned resource is to remove the assignment. In the Default UI mode
there is a "Remove" action link on each row, and if the user clicks that link it calls a RemoveResource method:
I chose not to implement any confirmation in this UI, so this method directly removes the selected item from the list
of resources assigned to the project.
I didn't confirm removal in this case, because no changes to the ProjectEdit object are committed to
the database until the user clicks the "Save" button to save all changes.
At this point you should understand how the user is able to edit details about a project, and add, edit, or remove
resources assigned to the project. All that remains now is the "Save" button so the user can save their changes to the
server.
Save Button
The "Save" button is like the save or submit buttons I discussed in Chapter 7:
<button @onclick="vm.SaveAsync"
disabled="@(!vm.Model.IsSavable)">Save</button>
The @onclick event is handled by the viewmodel object's SaveAsync method. Any errors that occur during that
process are reflected by the viewmodel object's ViewModelErrorText property and so are displayed to the user at
the top of the page.
The disabled property is data bound to the IsSaveable property of the Model object. This ensures that the "Save"
button is only enabled if the ProjectEdit business object graph has no broken validation or authorization rules.
You should now be able to create pages for displaying read-only lists, editing simple editable business lists, and
editing more complex master-child object graphs.
In the last section of this chapter I will discuss the implementation of authentication in the ProjectTracker apps.
Page 172
Authentication
In Chapters 5 and 7 I discussed Blazor and CSLA .NET authentication.
As required by server-side and client-side Blazor, the App.razor component in the shared UI project wraps all
content in a CascadingAuthenticationState element, and replaces the RouteView component with the
AuthorizeRouteView component:
<CascadingAuthenticationState>
<Router AppAssembly="@typeof(App).Assembly">
<Found Context="routeData">
<AuthorizeRouteView RouteData="@routeData" DefaultLayout="@typeof(MainLayout)" />
</Found>
<NotFound>
<LayoutView Layout="@typeof(MainLayout)">
<p>Sorry, there's nothing at this address.</p>
</LayoutView>
</NotFound>
</Router>
</CascadingAuthenticationState>
@code {
public static bool IsServerSide { get; set; } = true;
}
This property will be set as the client-side app starts up so it is possible, within the code in the shared UI project, to
know whether the code is hosted by ASP.NET Core or WebAssembly.
In the Shared folder of the shared UI project, the MainLayout component also includes authentication-related
markup:
This markup uses the AuthorizeView component to display UI to allow the user to log in if they aren't logged in, or
to log out if they are logged in.
Notice the use of the IsServerSide property in the App.razor component. The markup for server-side
authentication is different from client-side authentication markup, and this property allows Blazor to render the
appropriate content for each environment.
Because this markup is in the main layout, in the header section of the UI, it is available to the user on all pages
Page 173
within the app.
The configuration and pages necessary to implement authentication are different for client-side and server-side
Blazor.
Client-side Authentication
The client-side Blazor ProjectTracker app follows the patterns from Chapters 5 and 7.
The Main method in the Program class includes code to set the IsServerSide property defined in the App.razor
component, and to configure Blazor and CSLA .NET for authentication:
ProjectTracker.Ui.Blazor.App.IsServerSide = false;
builder.Services.AddOptions();
builder.Services.AddAuthorizationCore();
builder.UseCsla();
The IsServerSide property is used in the MainLayout component to render the correct authentication markup as
appropriate for client-side authentication. Specifically, the logout functionality needs to route to the Blazor Logout
page:
The shared UI project includes an Account folder in the Pages folder where the Login and Logout pages are located.
These pages are ignored for server-side Blazor, but are used to implement the authentication process for client-side
Blazor.
Login Page
The Login page makes use of a locally defined editable business class to collect the user credentials. In this case a
username and password:
Page 174
[Serializable]
public class LoginData : BusinessBase<LoginData>
{
public static readonly PropertyInfo<string> UsernameProperty = RegisterProperty<string>
(nameof(Username));
public string Username
{
get => GetProperty(UsernameProperty);
set => SetProperty(UsernameProperty, value);
}
[Create]
[RunLocal]
private void Create()
{ }
}
The page binds to this type using the same MVVM pattern I've been using throughout this chapter:
<table class="table">
<thead>
<tr>
<th></th>
<th></th>
</tr>
</thead>
<tbody>
<TextInputRow Property="vm.GetPropertyInfo(nameof(vm.Model.Username))" />
<TextInputRow Property="vm.GetPropertyInfo(nameof(vm.Model.Password))"
InputType="password" />
</tbody>
</table>
<button @onclick="LoginUser"
disabled="@(!vm.Model.IsSavable)">Login</button>
When the user clicks the "Login" button the LoginUser method is called:
Page 175
private async void LoginUser()
{
var identity =
await Library.Security.PTIdentity.GetPTIdentityAsync(
vm.Model.Username,
vm.Model.Password);
if (identity.IsAuthenticated)
{
userService.CurrentUser = principal;
NavigationManager.NavigateTo("/");
}
else
{
ErrorText = "Invalid credentials";
StateHasChanged();
}
}
As I discussed in Chapter 7, this method calls the data portal to get a custom identity object. The PTIdentity type is
implemented in the business layer and is used by all the ProjectTracker UI apps.
Blazor only understands the ClaimsIdentity type however, so the information from the custom identity type is used
to create a new ClaimsIdentity instance. That identity is then used to create a ClaimsPrincipal object, and that
principal is set as the current user identity.
Logout Page
The Logout page has no visible UI at all. When the app navigates to this page the user is logged out and the current
user principal is set to an unauthenticated ClaimsPrincipal object:
@page "/account/logout"
@inject Csla.Blazor.Client.Authentication.CslaUserService userService
@inject NavigationManager NavigationManager
@code {
protected override void OnInitialized()
{
userService.CurrentUser =
new System.Security.Claims.ClaimsPrincipal();
NavigationManager.NavigateTo("/");
}
}
Once the current user is set to an unauthenticated identity, the app navigates to the home page.
Server-side Authentication
The server-side Blazor ProjectTracker app also follows the patterns from Chapters 5 and 7.
The server-side app is configured for authentication. In the ConfigureServices method of the Startup class, before
any other methods are called, the AddAuthorization method is invoked:
services.AddAuthentication(CookieAuthenticationDefaults.AuthenticationScheme)
.AddCookie();
Then in the Configure method of the Startup class, before the UseEndpoints method, authentication and
authorization are configured:
Page 176
app.UseAuthentication();
app.UseAuthorization();
Server-side Blazor relies on ASP.NET Core Razor Pages on the server to authenticate the user. These pages are in the
server-side Blazor UI project in the Areas\Account folder.
These pages follow the same model from Chapter 5, so I will discuss only highlights in this chapter.
Login Page
The Login page is bound to a simple LoginData type that collects the username and password credentials:
<form method="post">
<div class="form-group">
<label asp-for="loginData.UserName"></label>
<input asp-for="loginData.UserName" class="form-control">
<div class="invalid-feedback"></div>
</div>
<div class="form-group">
<label asp-for="loginData.Password"></label>
<input asp-for="loginData.Password" class="form-control" type="password">
<div class="invalid-feedback"></div>
</div>
<button type="submit" class="btn btn-primary">Login</button>
</form>
When the "Login" button is clicked the page's postback method is called:
Page 177
public async Task<IActionResult> OnPostAsync()
{
if (!ModelState.IsValid)
{
AlertMessage = "Form has validation errors.";
return Page();
}
if (User.Identity.IsAuthenticated)
{
await HttpContext.SignOutAsync(CookieAuthenticationDefaults.AuthenticationScheme);
}
Csla.ApplicationContext.User = principal;
If a user was already logged into the app, they are logged out:
if (User.Identity.IsAuthenticated)
{
await HttpContext.SignOutAsync(CookieAuthenticationDefaults.AuthenticationScheme);
}
The next step is to use the data portal to get a custom PTIdentity instance:
Notice that this is the same type used by client-side authentication in the ProjectTracker Blazor app. The PTIdentity
type is used by all ProjectTracker UI apps to verify the user's credentials.
ASP.NET Core, like client-side Blazor, only understands the ClaimsPrincipal and ClaimsIdentity types. So the next
step in the process is to use the information in the PTIdentity object to create a new ClaimsIdentity instance. That
instance is used to create a new ClaimsPrincipal, which is then set as the current user:
Page 178
Csla.ApplicationContext.User = principal;
if (Csla.ApplicationContext.User.Identity.IsAuthenticated)
{
AuthenticationProperties authProperties = new AuthenticationProperties();
await HttpContext.SignInAsync(
CookieAuthenticationDefaults.AuthenticationScheme,
principal,
authProperties);
When running in an ASP.NET Core environment, the User property of Csla.ApplicationContext relies on the
ASP.NET Core HttpContext object to maintain the user identity. Setting the User property merely sets the underlying
HttpContext principal.
If the new user identity is authenticated, the SignInAsync method of HttpContext is used to create a cookie to
maintain the user identity between page requests.
The web site then navigates to the home page of the Blazor app.
Logout Page
As with the client-side Blazor implementation, the server-side Logout page has no visible display elements. When
the app navigates to this page, the current user identity is set to an unauthenticated value:
@page
@using Microsoft.AspNetCore.Authentication
@using Microsoft.AspNetCore.Authentication.Cookies
@attribute [IgnoreAntiforgeryToken]
@functions {
public async Task<IActionResult> OnPost()
{
if (User.Identity.IsAuthenticated)
{
await HttpContext.SignOutAsync(CookieAuthenticationDefaults.AuthenticationScheme);
}
return Redirect("~/");
}
}
The SignOutAsync method of the HttpContext object is used to remove the authenticated user cookie, meaning that
the user is effectively logged out of the app.
The web site then navigates to the home page of the Blazor app.
At this point you should understand how to create a business app using CSLA .NET and Blazor.
Conclusion
In this chapter I have discussed how to create a business app using Blazor and CSLA .NET. This ProjectTracker
sample supports server-side and client-side Blazor with a shared UI project containing all the Blazor components for
both deployments.
The ProjectTracker app demonstrates how to create pages to display lists of data, edit simple lists of data, and to edit
more complex master-detail relationships. You can also see how client-side and server-side authentication are
implemented.
This book covers the use of Blazor to create server-side and client-side apps that make use of the Blazor UI
Page 179
framework, including its powerful data binding, event binding, and method binding mechanisms.
It also covers the use of the helper types in the Csla.Blazor namespace. These types are designed to make it easy to
build Blazor apps that take full advantage of the validation, authorization, and other business rules provided by the
CSLA .NET rules engine.
You have also seen how the CSLA .NET data portal provides a powerful abstraction, allowing you to deploy the same
Blazor app server-side or client-side, ranging from 1-tier to 2-tier to n-tier deployments.
Blazor is an exciting new technology that seems designed to take the best possible advantage of all CSLA .NET has to
offer!
Page 180
Page 181