ASP.NET Core (MVC / Razor Pages) 用户界面自定义指南

本文档解释了如何重写ASP.NET Core MVC / Razor Page 应用程序依赖应用模块的用户界面.

重写页面

本节介绍了Razor 页面开发,它是ASP.NET Core推荐的服务端渲染用户页面的方法. 预构建的模块通常使用Razor页面替代经典的MVC方式(下一节也介绍MVC模式).

你通过有三种重写页面的需求:

  • 重写页面模型(C#)端执行其他逻辑,不更改UI.
  • 重写Razor页面(.cshtml文件),不更改逻辑.
  • 完全重写 页面.

重写页面模型 (C#)

  1. using System.Threading.Tasks;
  2. using Microsoft.AspNetCore.Mvc;
  3. using Volo.Abp.DependencyInjection;
  4. using Volo.Abp.Identity;
  5. using Volo.Abp.Identity.Web.Pages.Identity.Users;
  6. namespace Acme.BookStore.Web.Pages.Identity.Users
  7. {
  8. [Dependency(ReplaceServices = true)]
  9. [ExposeServices(typeof(EditModalModel))]
  10. public class MyEditModalModel : EditModalModel
  11. {
  12. public MyEditModalModel(
  13. IIdentityUserAppService identityUserAppService,
  14. IIdentityRoleAppService identityRoleAppService
  15. ) : base(
  16. identityUserAppService,
  17. identityRoleAppService)
  18. {
  19. }
  20. public override async Task<IActionResult> OnPostAsync()
  21. {
  22. //TODO: Additional logic
  23. await base.OnPostAsync();
  24. //TODO: Additional logic
  25. }
  26. }
  27. }
  • 这个类继承并替换 EditModalModel ,重写了 OnPostAsync 方法在基类代码的前后执行附加逻辑
  • 它使用 ExposeServicesDependency attributes去替换这个类.

重写Razor页面 (.CSHTML)

同一路径下创建相同的.cshtml文件可以实现重写功能(razor page, razor view, view component… 等.)

示例

这个示例重写了账户模块定义的登录页面UI

账户模块在 Pages/Account 文件夹下定义了 Login.cshtml 文件. 所以你可以在同一路径下创建文件覆盖它: overriding-login-cshtml

通常你想要拷贝模块的 .cshtml 原文件,然后进行需要的更改. 你可以在这里找到源文件. 不要拷贝 Login.cshtml.cs 文件,它是隐藏razor页面的代码,我们不希望覆盖它(见下节).

这就够了,接下来你可以对文件内容做你想要的更改.

完全重写Razo页面

也许你想要完全重写页面,Razor和页面相关的C#文件.

在这种情况下;

  1. 像上面描述过的那术重写C#页面模型类,但不需要替换已存在的页面模型类.
  2. 像上面描述过的那样重写Razor页面,并且更改@model指向新的页面模型

示例

这个示例重写了账户模块定义的登录页面

创建一个继承自 LoginModel(定义在Volo.Abp.Account.Web.Pages.Account命名空间下)的页面模型类:

  1. public class MyLoginModel : LoginModel
  2. {
  3. public MyLoginModel(
  4. IAuthenticationSchemeProvider schemeProvider,
  5. IOptions<AbpAccountOptions> accountOptions
  6. ) : base(
  7. schemeProvider,
  8. accountOptions)
  9. {
  10. }
  11. public override Task<IActionResult> OnPostAsync(string action)
  12. {
  13. //TODO: Add logic
  14. return base.OnPostAsync(action);
  15. }
  16. //TODO: Add new methods and properties...
  17. }

如果需要,你可以重写任何方法或添加新的属性/方法

注意我们没有使用 [Dependency(ReplaceServices = true)][ExposeServices(typeof(LoginModel))],因为我们不想替换依赖注入中已存在的类,我们定义了一个新的.

拷贝 Login.cshtml 到你们解决方案,更改 @model 指定到 MyLoginModel:

  1. @page
  2. ...
  3. @model Acme.BookStore.Web.Pages.Account.MyLoginModel
  4. ...

这就够了,接下来你可以做任何想要更改.

不使用继承替换页面模型

你不需要继承源页面模型类(像之前的示例). 你可以完全重写实现你自己的页面. 在这种事情下你可以从 PageModel,AbpPageModel 或任何你需要的合适的基类派生.

重写视图组件

在ABP框架,预构建的模块和主题定义了一些可重用的视图组件. 这些视图组件可以像页面一样被替换.

示例

下面是应用程序启动模板自带的 基本主题 的截图.

bookstore-brand-area-highlighted

基本主题 为layout定义了一些视图组件. 例如上面带有红色矩形的突出显示区域称为 Brand组件, 你可能想添加自己的自己的应用程序logo来自定义此组件. 让我们来看看如何去做.

首先创建你的logo并且放到你的web应用程序文件夹中,我们使用 wwwroot/logos/bookstore-logo.png 路径. 然后在 Themes/Basic/Components/Brand 文件夹下复制Brand组件视图. 结果应该是类似下面的图片:

bookstore-added-brand-files

然后对 Default.cshtml 文件做你想要的更改. 例如内容可以是这样的:

  1. <a href="/">
  2. <img src="~/logos/bookstore-logo.png" width="250" height="60"/>
  3. </a>

现在你可以运行应用程序看到结果:

bookstore-added-logo

如果你需要,你也可以仅使用依赖注入系统替换组件背后的C#类代码

重写主题

正如上所解释的,你可以更改任何组件,layout或c#类. 参阅[主题文档]了解更多关于主题系统的信息.

重写静态资源

重写模块的静态资源(像JavaScript,Css或图片文件)是很简单的. 只需要在解决方案的相同路径创建文件,虚拟文件系统会自动处理它.

操作捆绑

捆绑 & 压缩 系统提供了动态可扩展的 系统去创建scriptstyle捆绑. 它允许你扩展和操作现有的包.

示例: 添加全局CSS文件

例如APP框架定义了一个全局样式捆绑添加到所有的页面(事实上由主题添加layout). 让我们添加一个自定义样式文件到这个捆绑文件的最后,我们可以覆盖任何全局样式.

创建在 wwwroot 文件夹下创建一个CSS文件

bookstore-global-css-file

在CSS文件中定义一些规则. 例如:

  1. .card-title {
  2. color: orange;
  3. font-size: 2em;
  4. text-decoration: underline;
  5. }
  6. .btn-primary {
  7. background-color: red;
  8. }

然后在你的模块 ConfigureServices 方法添加这个文件到标准的全局样式捆绑包:

  1. Configure<AbpBundlingOptions>(options =>
  2. {
  3. options.StyleBundles.Configure(
  4. StandardBundles.Styles.Global, //The bundle name!
  5. bundleConfiguration =>
  6. {
  7. bundleConfiguration.AddFiles("/styles/my-global-styles.css");
  8. }
  9. );
  10. });

全局脚本捆绑包

就像 StandardBundles.Styles.Global 一样,还有一个 StandardBundles.Scripts.Global,你可以添加文件或操作现有文件.

示例: 操作捆绑包文件

上面的示例中添加了新文件到捆绑包. 如果你创建 bundle contributor 类则可以做到更多. 示例:

  1. public class MyGlobalStyleBundleContributor : BundleContributor
  2. {
  3. public override void ConfigureBundle(BundleConfigurationContext context)
  4. {
  5. context.Files.Clear();
  6. context.Files.Add("/styles/my-global-styles.css");
  7. }
  8. }

然后你可以添加这个contributor到已存在的捆绑中:

  1. Configure<AbpBundlingOptions>(options =>
  2. {
  3. options.StyleBundles.Configure(
  4. StandardBundles.Styles.Global,
  5. bundleConfiguration =>
  6. {
  7. bundleConfiguration.AddContributors(typeof(MyGlobalStyleBundleContributor));
  8. }
  9. );
  10. });

示例中清除了所有的CSS文件,在现实中这并不是一个好主意,你可以找到某个特定的文件替换成你自己的文件.

示例: 为特定页面添加JavaScript文件

上面的示例将全局包添加到布局中. 如果要在依赖模块中为特定页面定义添加CSS/JavaScript文件(或替换文件)怎么做?

假设你想要用户进入身份模块的角色管理页面时运行JavaScript代码.

首先在 wwwroot, PagesViews 文件夹下创建一个标准的JavaScript文件(默认ABP支持这些文件夹下的静态文件). 根据约定我们推荐 Pages/Identity/Roles 文件夹:

bookstore-added-role-js-file

该文件的内容很简单:

  1. $(function() {
  2. abp.log.info('My custom role script file has been loaded!');
  3. });

然后将这个文件添加到角色管页面理捆绑包中:

  1. Configure<AbpBundlingOptions>(options =>
  2. {
  3. options.ScriptBundles
  4. .Configure(
  5. typeof(Volo.Abp.Identity.Web.Pages.Identity.Roles.IndexModel).FullName,
  6. bundleConfig =>
  7. {
  8. bundleConfig.AddFiles("/Pages/Identity/Roles/my-role-script.js");
  9. });
  10. });

typeof(Volo.Abp.Identity.Web.Pages.Identity.Roles.IndexModel).FullName 是获取角色管理页面捆绑包名称的安全方式:

请注意并非每个页面都定义了这个页面的捆绑包. 它们仅在需要时定义.

除了添加新的CSS/JavaScript文件到页面,你也可以以替换(通过捆绑包contributor)已存在.

布局定制

布局由主题(参阅主题)定义设计. 它们不包含在下载的应用程序解决方案中. 通过这种方式你可以轻松的更改主题并获取新的功能. 你不能直接更改应用程序中的布局代码,除非你用自己的布局替换它(在下一部分中说明).

有一些通用的方法可以自定义布局,将在下一节中介绍.

菜单贡献者

ABP框架定义了两个标准菜单:

bookstore-menus-highlighted

  • StandardMenus.Main: 应用程序的主菜单.
  • StandardMenus.User: 用户菜单 (通常在屏幕的右上方).

显示菜单是主题的责任,但菜单项由模板和你的应用程序代码决定. 只需要实现 IMenuContributor 接口并在 ConfigureMenuAsync 方法操作菜单项.

渲染菜单时需要执行菜单贡献者. 应用程序启动模板 已经定义了菜单贡献者,所以你可以使用它. 参阅导航菜单文档了解更多.

工具栏贡献者

工具栏系统用于在用户界面定义 工具栏 . 模块 (或你的应用程序)可以将 添加到工具栏, 随后主题将在布局上呈现工具栏.

只有一个 标准工具栏 (名称为 “Main” - 定义为常量: StandardToolbars.Main). 对于基本主题,按如下呈现:bookstore-toolbar-highlighted

在上面的屏幕快照中,主工具栏添加了两个项目:语言开关组件和用户菜单. 你可以在此处添加自己的项.

示例: 添加通知图标

在这个示例中,我们会添加一个通知(响铃)图标到语言切换项的左侧. 工具栏的项项目是一个视图组件. 所以,在你的项目中创建一个新的视图组件:

bookstore-notification-view-component

NotificationViewComponent.cs

  1. public class NotificationViewComponent : AbpViewComponent
  2. {
  3. public async Task<IViewComponentResult> InvokeAsync()
  4. {
  5. return View("/Pages/Shared/Components/Notification/Default.cshtml");
  6. }
  7. }

Default.cshtml

  1. <div id="MainNotificationIcon" style="color: white; margin: 8px;">
  2. <i class="far fa-bell"></i>
  3. </div>

现在,我们创建一个类实现 IToolbarContributor 接口:

  1. public class MyToolbarContributor : IToolbarContributor
  2. {
  3. public Task ConfigureToolbarAsync(IToolbarConfigurationContext context)
  4. {
  5. if (context.Toolbar.Name == StandardToolbars.Main)
  6. {
  7. context.Toolbar.Items
  8. .Insert(0, new ToolbarItem(typeof(NotificationViewComponent)));
  9. }
  10. return Task.CompletedTask;
  11. }
  12. }

这个类向 Main 工具栏的第一项添加了 NotificationViewComponent.

最后你需要将这个贡献者添加到 AbpToolbarOptions,在你模块类的 ConfigureServices 方法:

  1. Configure<AbpToolbarOptions>(options =>
  2. {
  3. options.Contributors.Add(new MyToolbarContributor());
  4. });

这就够了,当你运行应用程序后会看到工具栏上的通知图标:

bookstore-notification-icon-on-toolbar

示例中的 NotificationViewComponent 返回没有任何数据的视图. 在实际场景中,你可能想查询数据库(或调用HTTP API)获取通知并传递给视图. 如果需要可以将 JavaScriptCSS 文件添加到工具栏的全局捆绑包中(如前所述).

参阅工具栏文档了解更多关于工具栏系统.

布局钩子

布局钩子 系统允许你在布局页面的某些特定部分 添加代码 . 所有主题的所有布局都应该实现这些钩子. 然后你可以将视图组件添加到钩子.

示例: 添加谷歌统计

假设你想要添加谷歌统计脚本到布局(将适用所有的页面). 首先在你的项目中创建一个视图组件:

bookstore-google-analytics-view-component

NotificationViewComponent.cs

  1. public class GoogleAnalyticsViewComponent : AbpViewComponent
  2. {
  3. public IViewComponentResult Invoke()
  4. {
  5. return View("/Pages/Shared/Components/GoogleAnalytics/Default.cshtml");
  6. }
  7. }

Default.cshtml

  1. <script>
  2. (function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){
  3. (i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o),
  4. m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;m.parentNode.insertBefore(a,m)
  5. })(window,document,'script','//www.google-analytics.com/analytics.js','ga');
  6. ga('create', 'UA-xxxxxx-1', 'auto');
  7. ga('send', 'pageview');
  8. </script>

在你自己的代码中更改 UA-xxxxxx-1 .

然后你可以在你模块的 ConfigureServices 方法将这个组件添加到任何的钩子点:

  1. Configure<AbpLayoutHookOptions>(options =>
  2. {
  3. options.Add(
  4. LayoutHooks.Head.Last, //The hook name
  5. typeof(GoogleAnalyticsViewComponent) //The component to add
  6. );
  7. });

现在谷歌统计代码将在页面的 head 所为最后一项插入. 你(或你在使用的模块)可以将多个项添加到相同的钩子,它们都会添加到布局.

在上面我们添加 GoogleAnalyticsViewComponent 到所有的布局,你可能只想添加到指定的布局:

  1. Configure<AbpLayoutHookOptions>(options =>
  2. {
  3. options.Add(
  4. LayoutHooks.Head.Last,
  5. typeof(GoogleAnalyticsViewComponent),
  6. layout: StandardLayouts.Application //Set the layout to add
  7. );
  8. });

参阅下面的布局部分,以了解有关布局系统的更多信息.

布局

布局系统允许主题定义标准,命名布局并且允许任何页面选择使用合适的布局. 有三种预定义的布局:

  • Application“: 应用程序的主要(和默认)布局. 它通常包含页眉,菜单(侧栏),页脚,工具栏等.
  • Account“: 登录,注册和其他类似页面使用此布局. 默认它用于 /Pages/Account 文件夹下的页面.
  • Empty“: 空的最小的布局.

这些名称在 StandardLayouts 类定义为常量. 这是标准的布局名称,所有的主题开箱即用的实现. 你也可以创建自己的布局.

布局位置

你可以在这里找到基本主题的布局文件. 你可以将它们作用构建自己的布局的参考,也可以在必要时覆盖它们.

ITheme

ABP框架使用 ITheme 服务通过局部名称获取布局位置. 你可以替换此服务动态的选择布局位置.

IThemeManager

IThemeManager 用于获取当前主题,并得到了布局路径. 任何页面可以都决定自己的布局. 例:

  1. @using Volo.Abp.AspNetCore.Mvc.UI.Theming
  2. @inject IThemeManager ThemeManager
  3. @{
  4. Layout = ThemeManager.CurrentTheme.GetLayout(StandardLayouts.Empty);
  5. }

此页面将使用空白布局. 它使用 ThemeManager.CurrentTheme.GetEmptyLayout() 扩展方法.

如果你设置特定目录下所有页面的布局,可以在该文件夹下的 _ViewStart.cshtml 文件编写以上代码.