2021年7月7日 星期三

Blazor Server 必會開發技能 - C# 程式碼設計方法

Blazor Server 必會開發技能 - C# 程式碼設計方法


Blazor Server 必會開發技能

Blazor Server 建立一個新的頁面

Blazor Server 元件生命週期事件 Component Life Cycle

Blazor Server C# 程式碼設計方法

Blazor Server 單向資料綁定 One Way Data Binding 與 重新轉譯 Binding

Blazor Server Hello 互動頁面與事件設計 Two Way Data Binding

Blazor Server 元件間的參數傳遞與回應事件 Component Parameter EventCallback

Blazor Server C# 與 JavaScript 互相呼叫 IJSRuntime

Blazor Server 表單和驗證 Form Validation 

在進行 Blazor 專案開發的時候,會需要透過 HTML / CSS 來宣告網頁要呈現的內容,對於商業邏輯處理的部分,將會需要透過 C# 程式語言來進行設計;然而,在進行 Razor Component 元件 ( 也可以稱為 Blazor 元件) 設計的時候,對於 HTML 與 C# 這兩個語言的設計,將會有五種方式可以選擇,在這篇文章中將會針對這五種模式來進行開發體驗與說明。

這裡說明的範例專案原始碼位於 BS03

建立 Blazor Server-Side 的專案

  • 打開 Visual Studio 2019

  • 點選右下方的 [建立新的專案] 按鈕

  • [建立新專案] 對話窗將會顯示在螢幕上

  • 從[建立新專案] 對話窗的中間區域,找到 [Blazor 應用程式] 這個專案樣板選項,並且選擇這個項目

  • 點選右下角的 [下一步] 按鈕

  • 現在 [設定新的專案] 對話窗將會出現

  • 請在這個對話窗內,輸入適當的 [專案名稱] 、 [位置] 、 [解決方案名稱]

    在這裡請輸入 [專案名稱] 為 BS03

  • 完成後,請點選 [建立] 按鈕

  • 當出現 [建立新的 Blazor 應用程式] 對話窗的時候

  • 請選擇最新版本的 .NET Core 與 [Blazor 伺服器應用程式]

  • 完成後,請點選 [建立] 按鈕

    稍微等會一段時間,Blazor 專案將會建立起來

使用 Razor 元件 - Inline 方式來設計元件

  • 這樣的設計方式,可以直接參考這個專案範本所建立的 Counter 元件
  • 從 Blazor 專案內的 [Pages] 資料夾找到並且打開 [Counter.razor] 檔案
  • 這個檔案內容如下
@page "/counter"

<h3>Counter Razor 元件 - Inline</h3>

<p>Current count: @currentCount</p>

<button class="btn btn-primary" @onclick="IncrementCount">Click me</button>

@code {
    private int currentCount = 0;

    private void IncrementCount()
    {
        currentCount++;
    }
}

這樣的設計方式將會把 HTML / C# 全部都寫在 .razor 的 Razor元件 檔案內,其中大約可以將這些內容區分成為三大部分,最前面的 [@page] 為 Razor Directive 指示詞,在這裡將會相關指示詞來宣告這個 [Razor元件] 會用到的其他功能,例如,可以加入其他 .NET 命名空間、從相依性注入容器 DI / IoC Container 內注入一個物件、繼承或者實作介面等等。

第二個部分就是要顯示這個元件會用到的相關 HTML 宣告標記語言程式碼,在這裡會使用 Blazor 框架內建的 資料綁定/繫節 Data Binding 功能,做出動態網頁的能力。例如要顯示一個 C# 執行個體的內容,可以使用 @currentCount 這樣的方式,想要綁定一個委派事件方法,可以使用 @onclick="IncrementCount" 這樣的方式來做到。

第三個部分將會使用 @code{ ... } 這個指示詞,在大括號區塊內來使用 C# 程式語言進行設計相關商業邏輯程式碼。

這樣的元件設計方法是最為簡單的,不過,因為 UI 設計(使用 HTML)程式碼與 商業邏輯(使用 C#)程式語言都混和在一起,使得設計一個元件可以在單一一個檔案就可以完成,不過,卻會造成 UI 與 商業邏輯都混和再一起了。

使用這個元件

  • 打開 [Pages] 資料夾內的 [Index.razor] 檔案
  • 請使用底下程式碼替換到這個檔案內容
@page "/"

<h1>
    Hello, C# 程式碼設計方法!
</h1>

<div class="container-fluid">
    <div class="row">
        @*Razor 元件 - Inline*@
        <div class="col-4">
            <div class="card">
                <h5 class="card-header">Counter.razor</h5>
                <div class="card-body">
                    <Counter />
                </div>
            </div>
        </div>

        @*Counter - C# 類別*@

        @*Counter - 繼承 CodeBehind*@

        @*Razor 元件 Partial 類別*@

        @*Razor 元件 View/ViewModel*@

    </div>
</div>

要使用 [Razor 元件 - Inline 方式] 設計的元件,僅需要將這個元件檔案名稱加入到 HTML 標記內即可,例如: <Counter />

使用 C# 類別 來設計元件

  • 滑鼠右擊 Blazor 專案內的 [Pages] 資料夾
  • 選擇 [加入] > [類別]
  • 當 [新增項目 - BS03] 對話窗出現之後,請在下方名稱欄位內,輸入 CounterCsharp
  • 最後點選 [新增] 按鈕
  • 請依據底下程式碼替換到這個檔案內容
namespace BS03.Pages
{
    public partial class CounterCsharp : Microsoft.AspNetCore.Components.ComponentBase
    {
        /// <summary>
        /// 這裡將會重新產生出最新狀態的轉譯樹
        /// </summary>
        /// <param name="__builder"></param>
        protected override void BuildRenderTree(Microsoft.AspNetCore.Components.Rendering.RenderTreeBuilder __builder)
        {
            __builder.AddMarkupContent(0, "<h3>Counter - C# 類別</h3>\r\n\r\n");
            __builder.OpenElement(1, "p");
            __builder.AddContent(2, "Current count: ");
            __builder.AddContent(3,
                   currentCount
            );
            __builder.CloseElement();
            __builder.AddMarkupContent(4, "\r\n\r\n");
            __builder.OpenElement(5, "button");
            __builder.AddAttribute(6, "class", "btn btn-primary");
            __builder.AddAttribute(7, "onclick", Microsoft.AspNetCore.Components.EventCallback.Factory.Create<Microsoft.AspNetCore.Components.Web.MouseEventArgs>(this,
                                          IncrementCount
            ));
            __builder.AddContent(8, "Click me");
            __builder.CloseElement();
        }

        int currentCount { get; set; } = 0;

        private void IncrementCount()
        {
            currentCount++;
        }
    }
}

要設計一個 Blazor 元件,可以完全都使用 C# 程式語言就可以進行設計,首先先來設計一個類別,該類別名稱就是這個 Razor 元件名稱,並且這個類別要繼承 ComponentBase 類別。

接著,需要覆寫 [BuildRenderTree] 方法,在這個方法內要將會要設計要顯示在網頁上的 DOM 物件內容,如同前面使用 HTML 的方法來進行設計網頁要出現的 UI 一樣。

對於 商業邏輯 部分當然就像是一般設計 C# 程式語言一樣,在這個類別內設計出相關 C# 程式碼即可。

使用這個元件

  • 打開 [Pages] 資料夾內的 [Index.razor] 檔案
  • 找到 @*Counter - C# 類別*@ 關鍵字,在這個文字內加入底下的程式碼
<div class="col-4">
    <div class="card">
        <h5 class="card-header">CounterCsharp.cs</h5>
        <div class="card-body">
            <CounterCsharp />
        </div>
    </div>
</div>

當要使用一個 C# 類別設計出來的 [Razor 元件] ,只需要使用這個類別名稱當作為一個 HTML 標籤來使用即可,如同這樣使用方式 <CounterCsharp />

使用 繼承 來採用 CodeBehind 方式 來設計元件

  • 滑鼠右擊 Blazor 專案內的 [Pages] 資料夾
  • 選擇 [加入] > [類別]
  • 當 [新增項目 - BS03] 對話窗出現之後,請在下方名稱欄位內,輸入 CounterInherit.razor.cs
  • 最後點選 [新增] 按鈕
  • 請依據底下程式碼替換到這個檔案內容
using Microsoft.AspNetCore.Components;

namespace BS03.Pages
{
    public class CounterInheritBase : ComponentBase
    {
        public int currentCount = 0;

        public void IncrementCount()
        {
            currentCount++;
        }
    }
}
  • 滑鼠右擊 Blazor 專案內的 [Pages] 資料夾
  • 選擇 [加入] > [Razor元件]
  • 當 [新增項目 - BS03] 對話窗出現之後,請在下方名稱欄位內,輸入 CounterInherit.razor
  • 最後點選 [新增] 按鈕
  • 請依據底下程式碼替換到這個檔案內容
@inherits CounterInheritBase

<h3>Counter - 繼承 CodeBehind</h3>

<p>Current count: @currentCount</p>

<button class="btn btn-primary" @onclick="IncrementCount">Click me</button>

這是第一種將 UI 與 商業邏輯 拆開的做法,對於 UI 的部分,將會完全設計在副檔案名稱為 .razor 的檔案內 (CounterInherit.razor),在這裡將會僅會使用 HTML 的宣告式標記語言,而對於 商業邏輯 部分,將會設計在一個 C# 檔案內,副檔案名稱為 .razor.cs 的檔案內 (CounterInherit.razor.cs),在這裡將會僅使用 C# 程式語言來設計。

在 .razor.cs 檔案內,要使用 public class CounterInheritBase : ComponentBase 方式來建立一個類別,該類別要繼承 [ComponentBase] 類別,對於該類別的名稱將會 [元件名稱]+Base 組合。

在 .razor 檔案內,要使用 @inherits 指示詞 Directive 方式標示這個元件要繼承剛剛設計的 C# 類別,接著,就是要在這個檔案內來設計相關 HTML 宣告式標記語言。

這樣的設計特色將會是 UI 將會在 .razor 檔案內來設計,商業邏輯 將會在 .razor.cs 來設計。

使用這個元件

  • 打開 [Pages] 資料夾內的 [Index.razor] 檔案
  • 找到 @*Counter - 繼承 CodeBehind*@ 關鍵字,在這個文字內加入底下的程式碼
<div class="col-4">
    <div class="card">
        <h5 class="card-header">CounterInherit.razor</h5>
        <div class="card-body">
            <CounterInherit />
        </div>
    </div>
</div>

當要使用這樣的設計方式,也通稱為 Codebehind 設計模式,採用這樣方法設計出來的 [Razor 元件] ,只需要使用這個類別名稱當作為一個 HTML 標籤來使用即可,如同這樣使用方式 <CounterInherit />

使用 Partial 類別 來設計元件

  • 滑鼠右擊 Blazor 專案內的 [Pages] 資料夾
  • 選擇 [加入] > [Razor元件]
  • 當 [新增項目 - BS03] 對話窗出現之後,請在下方名稱欄位內,輸入 CounterPartialClass.razor
  • 最後點選 [新增] 按鈕
  • 請依據底下程式碼替換到這個檔案內容
<h3>Counter - Razor 元件 Partial 類別</h3>

<p>Current count: @currentCount</p>

<button class="btn btn-primary" @onclick="IncrementCount">Click me</button>
  • 滑鼠右擊 Blazor 專案內的 [Pages] 資料夾
  • 選擇 [加入] > [類別]
  • 當 [新增項目 - BS03] 對話窗出現之後,請在下方名稱欄位內,輸入 CounterPartialClass.cs
  • 最後點選 [新增] 按鈕
  • 請依據底下程式碼替換到這個檔案內容
namespace BS03.Pages
{
    public partial class CounterPartialClass
    {
        private int currentCount = 0;

        void IncrementCount()
        {
            currentCount++;
        }
    }
}

這是第二種方式來將 UI 與 商業邏輯 拆開的做法,對於 UI 的部分,將會完全設計在副檔案名稱為 .razor 的檔案內 (CounterPartialClass.razor),在這裡將會僅會使用 HTML 的宣告式標記語言,而對於 商業邏輯 部分,將會設計在一個 C# 類別,該類別名稱為一個 [partial] 部分類別,這裡將會設計在 .cs 的檔案內 (CounterPartialClass.cs),在這裡將會僅使用 C# 程式語言來設計。

這樣的設計特色將會是 UI 將會在 元件名稱.razor 檔案內來設計,商業邏輯 將會在 元件名稱.cs 來設計。

使用這個元件

  • 打開 [Pages] 資料夾內的 [Index.razor] 檔案
  • 找到 @*Razor 元件 Partial 類別*@ 關鍵字,在這個文字內加入底下的程式碼
<div class="col-4">
    <div class="card">
        <h5 class="card-header">CounterPartialClass.razor</h5>
        <div class="card-body">
            <CounterPartialClass />
        </div>
    </div>
</div>

當要採用這樣方法設計出來的 [Razor 元件] ,只需要使用這個元件名稱當作為一個 HTML 標籤來使用即可,如同這樣使用方式 <CounterPartialClass />

使用 View 與 ViewModel 來設計元件

  • 滑鼠右擊 Blazor 專案內的 [Pages] 資料夾
  • 選擇 [加入] > [Razor元件]
  • 當 [新增項目 - BS03] 對話窗出現之後,請在下方名稱欄位內,輸入 CounterView.razor
  • 最後點選 [新增] 按鈕
  • 請依據底下程式碼替換到這個檔案內容
@inject CounterViewModel CounterViewModel
<h3>Counter - Razor 元件 View/ViewModel</h3>

<p>Current count: @CounterViewModel.currentCount</p>

<button class="btn btn-primary" @onclick="CounterViewModel.IncrementCount">Click me</button>
  • 滑鼠右擊 Blazor 專案內的 [Pages] 資料夾
  • 選擇 [加入] > [類別]
  • 當 [新增項目 - BS03] 對話窗出現之後,請在下方名稱欄位內,輸入 CounterViewModel.cs
  • 最後點選 [新增] 按鈕
  • 請依據底下程式碼替換到這個檔案內容
namespace BS03.Pages
{
    public class CounterViewModel
    {
        public int currentCount { get; set; }

        public void IncrementCount()
        {
            currentCount++;
        }
    }
}
  • 最後在專案根目錄下找到並且打開 [Startup.cs] 檔案
  • 在 [Startup] 類別內找到 [ConfigureServices] 方法
  • 在這個方法內加入底下的敘述
#region 宣告要注入的 ViewModel 類別
services.AddTransient<CounterViewModel>();
#endregion

這是第三種方式來將 UI 與 商業邏輯 拆開的做法,對於 UI 的部分,將會完全設計在副檔案名稱為 .razor 的檔案內 (CounterView.razor),在這裡將會僅會使用 HTML 的宣告式標記語言,而對於 商業邏輯 部分,將會設計在一個 C# 類別,這裡將會設計在 .cs 的檔案內 (CounterViewModel.cs),在這裡將會僅使用 C# 程式語言來設計。

這兩個檔案的結合是透過 @inject CounterViewModel CounterViewModel 指示詞來透過相依性注入容器來注入這個 CounterViewModel 類別的執行個體,如此,在 View 中要綁定的 資料 物件或者 事件 委派方法,都會設計在這個 ViewModel 內。

這樣的設計特色將會是 UI 將會在 元件名稱View.razor 檔案內來設計,商業邏輯 將會在 ViewModel.cs 來設計。

使用這個元件

  • 打開 [Pages] 資料夾內的 [Index.razor] 檔案
  • 找到 @*Razor 元件 View/ViewModel*@ 關鍵字,在這個文字內加入底下的程式碼
<div class="col-4">
    <div class="card">
        <h5 class="card-header">CounterView.razor</h5>
        <div class="card-body">
            <CounterView />
        </div>
    </div>
</div>

當要採用這樣方法設計出來的 [Razor 元件] ,只需要使用這個元件名稱當作為一個 HTML 標籤來使用即可,如同這樣使用方式 <CounterView />

執行這個專案

  • 按下 [F5] 按鍵,開始執行這個 Blazor 專案

  • 一旦啟動完成,就會自動開以瀏覽器

 














2021年7月6日 星期二

Blazor 開發經驗分享 - 了解 Blazor 參數傳入過程

Blazor 開發經驗分享 - 了解 Blazor 參數傳入過程

在進行 Blazor 專案開發的時候,會設計出許多的 Razor Component 元件 ( 也可以稱為 Blazor 元件),並且將這些元件組合起來,便可以設計出相當優秀的 Web 頁面專案。

這些 [Razor 元件] 可以設計具有參數傳遞機制,讓使用這個元件 Parent 元件,可以透過傳遞或者變更參數的方式,將不同物件值傳遞到子元件內。

這裡說明的範例專案原始碼位於 bzSetParametersAsyncOnParametersSet

建立 Blazor Server-Side 的專案

  • 打開 Visual Studio 2019

  • 點選右下方的 [建立新的專案] 按鈕

  • [建立新專案] 對話窗將會顯示在螢幕上

  • 從[建立新專案] 對話窗的中間區域,找到 [Blazor 應用程式] 這個專案樣板選項,並且選擇這個項目

  • 點選右下角的 [下一步] 按鈕

  • 現在 [設定新的專案] 對話窗將會出現

  • 請在這個對話窗內,輸入適當的 [專案名稱] 、 [位置] 、 [解決方案名稱]

    在這裡請輸入 [專案名稱] 為 bzSetParametersAsyncOnParametersSet

  • 完成後,請點選 [建立] 按鈕

  • 當出現 [建立新的 Blazor 應用程式] 對話窗的時候

  • 請選擇最新版本的 .NET Core 與 [Blazor 伺服器應用程式]

  • 完成後,請點選 [建立] 按鈕

    稍微等會一段時間,Blazor 專案將會建立起來

建立 Razor 元件

  • 滑鼠右擊 Blazor 專案內的 [Pages] 資料夾
  • 選擇 [加入] > [Razor 元件]
  • 當 [新增項目 - bzSetParametersAsyncOnParametersSet] 對話窗出現之後,請在下方名稱欄位內,輸入 ChildComponent
  • 最後點選 [新增] 按鈕
  • 請依據底下程式碼替換到這個檔案內容
@code {
    [Parameter]
    public int IntParameter { get; set; }
    [Parameter]
    public string StringParameter { get; set; }
    [Parameter]
    public DateTime DateTimeParameter { get; set; }

    public override Task SetParametersAsync(ParameterView parameters)
    {
        ShowCurrentParameter();
        foreach (var param in parameters)
        {
            Console.WriteLine($"    SetParametersAsync : {param.Name}, Value: {param.Value.ToString()}");
        }
        return base.SetParametersAsync(parameters);
    }

    protected override void OnParametersSet()
    {
        base.OnParametersSet();
    }

    void ShowCurrentParameter()
    {
        Console.WriteLine($"Current IntParameter : {IntParameter}");
        Console.WriteLine($"Current StringParameter : {StringParameter}");
        Console.WriteLine($"Current DateTimeParameter : {DateTimeParameter}");
    }
}

這個子元件內使用 [Parameter] C# Attribute 來宣告了三個屬性,而且綁定了 SetParametersAsync & OnParametersSet 兩個元件生命週期事件,用來觀察何時觸發這些事件,並且了解到這些參數的變化情況。

在這個 SetParametersAsync 方法內,將會有 ParameterView parameters 參數,因此,可以透過 parameters 參數物件來讀取到要傳入的參數數值是多少?

使用這個元件

  • 打開 [Pages] 資料夾內的 [Index.razor] 檔案
  • 請使用底下程式碼替換到這個檔案內容
@page "/"

<h1>Hello, SetParametersAsync 與 OnParametersSet!</h1>

<ChildComponent IntParameter="@IntArgument"
                StringParameter="@StringArgument"
                DateTimeParameter="@DateTimeArgument" />
<div>
    <button class="btn btn-primary" @onclick="BtnChangeArgument">變更引數</button>
</div>

@code{
    public int IntArgument { get; set; } = 99;
    public string StringArgument { get; set; } = "Initialization";
    public DateTime DateTimeArgument { get; set; } = DateTime.Now;

    void BtnChangeArgument()
    {
        Console.WriteLine();
        Console.WriteLine($"變更傳入引數");
        IntArgument = 168;
        StringArgument = "Change String Value";
        DateTimeArgument = DateTimeArgument.AddDays(1);
    }
}

在這個使用 [ChildComponent] 元件的首頁元件內,使用了 <ChildComponent IntParameter="@IntArgument" StringParameter="@StringArgument" DateTimeParameter="@DateTimeArgument" /> 標記宣告的方式來使用這個元件,這裡將會傳遞三個引數到這個元件內。

另外,宣告了一個按鈕與綁定 OnClick() 按鈕事件,一旦按下這個按鈕之後,將會變更這些引數物件值,現在來觀察看看執行結果。

執行這個專案

  • 使用 Kestrel 的方式來執行這個專案

  • 按下 [F5] 按鍵,開始執行這個 Blazor 專案

  • 一旦啟動完成,就會自動開以瀏覽器

  • 此時,從 [命令提示字元視窗] 內將會看到底下的內容輸出

Current IntParameter : 0
Current StringParameter :
Current DateTimeParameter : 0001/1/1 上午 12:00:00
    SetParametersAsync : IntParameter, Value: 99
    SetParametersAsync : StringParameter, Value: Initialization
    SetParametersAsync : DateTimeParameter, Value: 2021/6/24 下午 XX:10:59
Current IntParameter : 0
Current StringParameter :
Current DateTimeParameter : 0001/1/1 上午 12:00:00
    SetParametersAsync : IntParameter, Value: 99
    SetParametersAsync : StringParameter, Value: Initialization
    SetParametersAsync : DateTimeParameter, Value: 2021/6/24 下午 XX:11:00

前面六筆紀錄為 Blazor 預先轉譯過程中所產生的輸出紀錄,而最後面六筆紀錄則為實際要轉譯內容輸出到網頁上的輸出紀錄;從這裡可以看到,在呼叫 [SetParametersAsync] 的時候,實際上當時在子元件內的參數變數尚未發生變化,而最新狀態值的參數值將會在這個 [SetParametersAsync] 方法內的參數 ParameterView parameters 內。

當按下 [變更引數] 按鈕之後,將會看到底下的輸出紀錄

變更傳入引數
Current IntParameter : 99
Current StringParameter : Initialization
Current DateTimeParameter : 2021/6/24 下午 02:11:00
    SetParametersAsync : IntParameter, Value: 168
    SetParametersAsync : StringParameter, Value: Initialization
    SetParametersAsync : DateTimeParameter, Value: 2021/6/24 下午 XX:11:00