目录
介绍
存储库和数据库
组件
RouteViews
表单
UI控件
UIBase
一些例子
UIButton
UIColumn
UILoader
UIContainer/UIRow/UIColumn
总结
介绍本文是构建Blazor数据库应用程序系列文章中的第四篇。本文着眼于我们在UI中使用的组件,然后重点介绍如何从HTML和CSS构建通用UI组件。
- 项目结构和框架——一些介绍。
- 服务——构建CRUD数据层。
- 查看组件——UI中的CRUD编辑和查看操作。
- UI组件——构建 HTML/CSS控件。
- 查看组件——UI中的CRUD列表操作。
文章的存储库已移至CEC.Blazor.SPA存储库。 CEC.Blazor GitHub存储库已过时,将被删除。
存储库中的/SQL中有一个用于构建数据库的SQL脚本。
您可以在同一站点上看到在此处运行的项目的Server和WASM版本。
组件要详细了解组件,请阅读我的文章深入了解 Blazor 组件。
Blazor UI中的所有内容(起始页除外)都是一个组件。是的,应用程序、路由器……它们都是组件。并非所有组件都发出Html。
您可以将组件分为四类:
- RouteViews ——这些是顶级组件。视图与布局相结合以形成显示窗口。
- Layouts——布局与视图结合构成显示窗口。
- Forms——表单是控件的逻辑集合。编辑表单、显示表单、列表表单、数据录入向导都是经典的表单。表单包含控件——不是HTML。
- Controls——控件要么显示某些内容——发出 HTML——要么执行某些工作单元。文本框、下拉菜单、按钮、网格都是经典的Hrtml发射控件。应用程序、路由器、验证是执行工作单元的控件。
RouteViews是特定于应用程序的,RouteView和Form之间的唯一区别是RouteView通过@Page指令声明一个或多个路由。在根App中声明的Router组件将设置为一个特定的代码程序集AppAssembly。这是Router在启动时搜索所有声明路由的程序集。
在应用程序中,RouteViews被声明在WASM应用程序库中。
天气预报查看器和列表视图如下所示。
// Blazor.Database/RouteViews/Weather/WeatherViewer.cs
@page "/weather/view/{ID:int}"
@code {
[Parameter] public int ID { get; set; }
[Inject] public NavigationManager NavManager { get; set; }
private void ExitToList()
=> this.NavManager.NavigateTo("/fetchdata");
}
// Blazor.Database/RouteViews/Weather/FetchData.cs
@page "/fetchdata"
表单
我们在上一篇文章中看到了表单。它们特定于应用程序。
下面的代码显示了天气查看器。都是UI控件,没有HTML标记。
// Blazor.Database/Components/Forms/WeatherForecastViewerForm.razor
@namespace Blazor.Database.Components
@inherits RecordFormBase
Weather Forecast Viewer
Date
Temperature °C
Temperature °f
Summary
Exit
页面背后的代码相对简单——复杂的是父类中的样板代码。它加载记录特定的控制器服务。
// Blazor.Database/Components/Forms/WeatherForecastViewerForm.razor.cs
public partial class WeatherForecastViewerForm : RecordFormBase
{
[Inject] private WeatherForecastControllerService ControllerService { get; set; }
protected async override Task OnInitializedAsync()
{
this.Service = this.ControllerService;
await base.OnInitializedAsync();
}
}
UI控件发出HTML和CSS标记。这里的所有控件都基于Bootstrap CSS框架。所有控件都继承自ComponentBase,UI控件继承自UIBase。
UIBaseUIBase继承自Component 。它构建了一个可以打开或关闭的HTML DIV块。
让我们详细看一些UIBase。
可以使用Tag参数设置HTML块标记。它只能由继承的类设置。
protected virtual string HtmlTag => "div";
控件CSS类是使用CssBuilder类构建的。继承类可以设置一个主要的Css值并添加他们想要的任意数量的次要值。添加CSS类可以通过AdditionalClasses参数或通过定义class属性来添加。
[Parameter] public virtual string AdditionalClasses { get; set; } = string.Empty;
protected virtual string PrimaryClass => string.Empty;
protected List SecondaryClass { get; private set; } = new List();
protected string CssClass
=> CSSBuilder.Class(this.PrimaryClass)
.AddClass(SecondaryClass)
.AddClass(AdditionalClasses)
.AddClassFromAttributes(this.UserAttributes)
.Build();
可以使用两个参数隐藏或禁用该控件。当Show为真时显示ChildContent。如果Show是假则显示HideContent,如果它不是null,否则什么也不显示。
[Parameter] public bool Show { get; set; } = true;
[Parameter] public bool Disabled { get; set; } = false;
[Parameter] public RenderFragment ChildContent { get; set; }
[Parameter] public RenderFragment HideContent { get; set; }
最后,控件捕获任何附加属性并将它们添加到标记元素中。
[Parameter(CaptureUnmatchedValues = true)] public IReadOnlyDictionary UserAttributes { get; set; } = new Dictionary();
该控件构建输入RenderTree代码。
protected override void BuildRenderTree(RenderTreeBuilder builder)
{
if (this.Show)
{
builder.OpenElement(0, this.HtmlTag);
if (!string.IsNullOrWhiteSpace(this.CssClass)) builder.AddAttribute(1, "class", this.CssClass);
if (Disabled) builder.AddAttribute(2, "disabled");
builder.AddMultipleAttributes(3, this.UserAttributes);
if (this.ChildContent != null) builder.AddContent(4, ChildContent);
else if (this.HideContent != null) builder.AddContent(5, HideContent);
builder.CloseElement();
}
}
本文的其余部分将更详细地介绍一些UI控件。
UIButton这是一个标准的Bootstrap按钮。
- Type设置按钮类型。
- PrimaryClass 设置。
- ButtonClick 处理按钮单击事件并调用EventCallback。
- Show和Disabled处理按钮状态。
// Blazor.SPA/Components/UIComponents/Base/UIButtons.cs
@namespace Blazor.SPA.Components
@inherits UIBase
@if (this.Show)
{
@this.ChildContent
}
@code {
[Parameter] public string Type { get; set; } = "button";
[Parameter] public EventCallback ClickEvent { get; set; }
protected override string PrimaryClass => "btn mr-1";
protected async Task ButtonClick(MouseEventArgs e) => await this.ClickEvent.InvokeAsync(e);
}
下面是一些显示正在使用的控件的代码。
Exit
这是一个标准的Bootstrap列。
- Cols 定义列数
- PrimaryCss是从Cols构建的。
- Base RenderTreeBuilder将控件构建为div。
// Blazor.SPA/Components/UIControls/Base/UIColumn.cs
public class UIColumn : UIBase
{
[Parameter] public virtual int Cols { get; set; } = 0;
protected override string PrimaryClass => this.Cols > 0 ? $"col-{this.Cols}" : $"col";
}
这是一个包装控件,旨在保存在子内容中实现错误检查。它仅在IsLoaded为 true时呈现其子内容。该控件节省了在子内容中实现大量错误检查。
@namespace Blazor.SPA.Components
@inherits UIBase
@if (this.Loaded)
{
@this.ChildContent
}
else
{
Loading....
}
@code {
[Parameter] public bool Loaded { get; set; }
}
您可以在“编辑”和“查看”表单中看到正在使用的控件。
UIContainer/UIRow/UIColumn这些控件通过使用正确的Css构建DIV来创建BootStrap网格系统——即容器、行和列。
public class UIContainer : UIBase
{
protected override string PrimaryClass => "container-fluid";
}
class UIRow : UIBase
{
protected override string PrimaryClass => "row";
}
public class UIColumn : UIBase
{
[Parameter] public virtual int Cols { get; set; } = 0;
protected override string PrimaryClass => this.Cols > 0 ? $"col-{this.Cols}" : $"col";
}
// CEC.Blazor/Components/UIControls/UIBootstrapContainer/UILabelColumn.cs
public class UILabelColumn : UIColumn
{
protected override string _BaseCss => $"col-{Columns} col-form-label";
}
下面是一些显示正在使用的控件的代码。
Date
............
..........
本文概述了如何使用组件构建UI控件,并详细研究了一些示例组件。你可以在GitHub Repository中看到所有的库UIControls
需要注意的一些关键点:
- UI控件使您可以从更高级别的组件(例如表单和视图)中抽象标记。
- UI控件为您提供控制权,并对HTML和CSS标记应用一些规则。
- View和Form组件更清晰、更易于查看。
- 使用尽可能少或尽可能多的抽象。
- 控件,例如UILoader,让生活更轻松!
如果您在未来阅读本文,请查看存储库中的自述文件以获取文章集的最新版本。
https://www.codeproject.com/Articles/5280090/Building-a-Database-Application-in-Blazor-Part-4-U