# 文本模板 ## 介绍 ABP框架提供了一个简单有效的文本模板系统,文本模板用于动态渲染基于模板和模型(数据对象)内容: ***TEMPLATE + MODEL ==render==> RENDERED CONTENT*** 它非常类似于 ASP.NET Core Razor View (或 Page): *RAZOR VIEW (or PAGE) + MODEL ==render==> HTML CONTENT* 你可以将渲染的输出用于任何目的,例如发送电子邮件或准备一些报告. ### 示例 Here, a simple template: ```` Hello {%{{{model.name}}}%} :) ```` 你可以定义一个含有 `Name` 属性的类来渲染这个模板: ````csharp public class HelloModel { public string Name { get; set; } } ```` 如果你使用 `Name` 为 `John` 的 `HelloModel` 渲染模板,输出为: ```` Hello John :) ```` 模板渲染引擎非常强大; * 它基于 [Scriban](https://github.com/lunet-io/scriban) 库, 所以它支持 **条件逻辑**, **循环** 等. * 模板内容 **可以本地化**. * 你可以定义 **布局模板** 在渲染其他模板中用做布局. * 对于高级场景,你可以传递任何对象到模板上下文. ### 源码 这里是本文开发和引用的[示例应用程序源码](https://github.com/abpframework/abp-samples/tree/master/TextTemplateDemo). ## 安装 推荐使用 [ABP CLI](CLI.md) 安装包. ### 使用 ABP CLI 在项目目录(.csproj file)打开命令行窗口运行以下命令: ````bash abp add-package Volo.Abp.TextTemplating ```` ### 手动安装 如果你想要手动安装; 1. 添加 [Volo.Abp.TextTemplating](https://www.nuget.org/packages/Volo.Abp.TextTemplating) NuGet包到你的项目: ```` Install-Package Volo.Abp.TextTemplating ```` 2. 添加 `AbpTextTemplatingModule` 到你的模块依赖列表: ````csharp [DependsOn( //...other dependencies typeof(AbpTextTemplatingModule) //Add the new module dependency )] public class YourModule : AbpModule { } ```` ## 定义模板 在渲染模板之前,需要定义它. 创建一个继承自 `TemplateDefinitionProvider` 的类: ````csharp public class DemoTemplateDefinitionProvider : TemplateDefinitionProvider { public override void Define(ITemplateDefinitionContext context) { context.Add( new TemplateDefinition("Hello") //template name: "Hello" .WithVirtualFilePath( "/Demos/Hello/Hello.tpl", //template content path isInlineLocalized: true ) ); } } ```` * `context` 对象用于添加新模板或获取依赖模块定义的模板. 使用 `context.Add(...)` 定义新模板. * `TemplateDefinition` 是代表模板的类,每个模板必须有唯一的名称(在渲染模板时使用). * `/Demos/Hello/Hello.tpl` 是模板文件的路径. * `isInlineLocalized` 声明针对所有语言使用一个模板(`true` 还是针对每种语言使用不同的模板(`false`). 更多内容参阅下面的本地化部分. ### 模板内容 `WithVirtualFilePath` 表示我们使用[虚拟文件系统](Virtual-File-System.md)存储模板内容. 在项目内创建一个 `Hello.tpl` 文件,并在属性窗口中将其标记为"**嵌入式资源**": ![hello-template](images/hello-template.png) 示例 `Hello.tpl` 内容如下所示: ```` Hello {%{{{model.name}}}%} :) ```` [虚拟文件系统](Virtual-File-System.md) 需要在[模块](Module-Development-Basics.md)类的 `ConfigureServices` 方法添加你的文件: ````csharp Configure(options => { options.FileSets.AddEmbedded("TextTemplateDemo"); }); ```` * `TextTemplateDemoModule`是模块类. * `TextTemplateDemo` 是你的项目的根命名空间. ## 渲染模板 `ITemplateRenderer` 服务用于渲染模板内容. ### 示例: 渲染一个简单的模板 ````csharp public class HelloDemo : ITransientDependency { private readonly ITemplateRenderer _templateRenderer; public HelloDemo(ITemplateRenderer templateRenderer) { _templateRenderer = templateRenderer; } public async Task RunAsync() { var result = await _templateRenderer.RenderAsync( "Hello", //the template name new HelloModel { Name = "John" } ); Console.WriteLine(result); } } ```` * `HelloDemo` 是一个简单的类,在构造函数注入了 `ITemplateRenderer` 并在 `RunAsync` 方法中使用它. * `RenderAsync` 有两个基本参数: * `templateName`: 要渲染的模板名称 (本示例中是 `Hello`). * `model`: 在模板内部用做 `model` 的对象 (本示例中是 `HelloModel` 对象). 示例会返回以下结果: ````csharp Hello John :) ```` ### 匿名模型 虽然建议为模板创建模型类,但在简单情况下使用匿名对象也是可行的: ````csharp var result = await _templateRenderer.RenderAsync( "Hello", new { Name = "John" } ); ```` 示例中我们并没有创建模型类,但是创建了一个匿名对象模型. ### 大驼峰 与 小驼峰 PascalCase 属性名(如 `UserName`) 在模板中用做小驼峰(如 `userName`). ## 本地化 可以基于当前文化对模板内容进行本地化. 以下部分描述了两种类型的本地化选项. ### 内联本地化 内联本地化使用[本地化系统](Localization.md)本地化模板内的文本. #### 示例: 重置密码链接 假设你需要向用户发送电子邮件重置密码. 模板内容: ```` {%{{{L "ResetMyPassword" model.name}}}%} ```` `L` 函数用于根据当前用户的文化来定位给定的Key,你需要在本地化文件中定义 `ResetMyPassword` 键: ````json "ResetMyPasswordTitle": "Reset my password", "ResetMyPassword": "Hi {0}, Click here to reset your password" ```` 你还需要在模板定义提供程序类中声明要与此模板一起使用的本地化资源: ````csharp context.Add( new TemplateDefinition( "PasswordReset", //Template name typeof(DemoResource) //LOCALIZATION RESOURCE ).WithVirtualFilePath( "/Demos/PasswordReset/PasswordReset.tpl", //template content path isInlineLocalized: true ) ); ```` 当你这样渲染模板时: ````csharp var result = await _templateRenderer.RenderAsync( "PasswordReset", //the template name new PasswordResetModel { Name = "john", Link = "https://abp.io/example-link?userId=123&token=ABC" } ); ```` 你可以看到以下本地化结果: ````csharp Hi john, Click here to reset your password ```` > 如果你为应用程序定义了 [默认本地化资源](Localization.md), 则无需声明模板定义的资源类型. ### 多个内容本地化 你可能希望为每种语言创建不同的模板文件,而不是使用本地化系统本地化单个模板. 如果模板对于特定的文化(而不是简单的文本本地化)应该是完全不同的,则可能需要使用它. #### 示例: 欢迎电子邮件模板 假设你要发送电子邮件欢迎用户,但要定义基于用户的文化完全不同的模板. 首先创建一个文件夹,将模板放在里面,像 `en.tpl`, `tr.tpl` 每一个你支持的文化: ![multiple-file-template](images/multiple-file-template.png) 然后在模板定义提供程序类中添加模板定义: ````csharp context.Add( new TemplateDefinition( name: "WelcomeEmail", defaultCultureName: "en" ) .WithVirtualFilePath( "/Demos/WelcomeEmail/Templates", //template content folder isInlineLocalized: false ) ); ```` * 设置 **默认文化名称**, 当没有所需的文化模板,回退到缺省文化. * 指定 **模板文件夹** 而不是单个模板文件. * 设置 `isInlineLocalized` 为 `false`. 就这些,你可以渲染当前文化的模板: ````csharp var result = await _templateRenderer.RenderAsync("WelcomeEmail"); ```` > 为了简单我们跳过了模型,但是你可以使用前面所述的模型. ### 指定文化 `ITemplateRenderer` 服务如果没有指定则使用当前文化 (`CultureInfo.CurrentUICulture`). 如果你需要你可以使用 `cultureName` 参数指定文化. ````csharp var result = await _templateRenderer.RenderAsync( "WelcomeEmail", cultureName: "en" ); ```` ## 布局模板 布局模板用于在其他模板之间创建共享布局. 它类似于ASP.NET Core MVC / Razor Pages中的布局系统. ### 示例: 邮件HTML布局模板 例如,你想为所有电子邮件模板创建一个布局. 首先像之前一样创建一个模板文件: ````xml {%{{{content}}}%} ```` * 布局模板必须具有 **{%{{{content}}}%}** 部分作为渲染的子内容的占位符. 在模板定义提供程序中注册模板: ````csharp context.Add( new TemplateDefinition( "EmailLayout", isLayout: true //SET isLayout! ).WithVirtualFilePath( "/Demos/EmailLayout/EmailLayout.tpl", isInlineLocalized: true ) ); ```` 现在你可以将此模板用作任何其他模板的布局: ````csharp context.Add( new TemplateDefinition( name: "WelcomeEmail", defaultCultureName: "en", layout: "EmailLayout" //Set the LAYOUT ).WithVirtualFilePath( "/Demos/WelcomeEmail/Templates", isInlineLocalized: false ) ); ```` ## 全局上下文 ABP传递 `model`,可用于访问模板内的模型. 如果需要,可以传递更多的全局变量. 示例模板内容: ```` A global object value: {%{{{myGlobalObject}}}%} ```` 模板假定它渲染上下文中的 `myGlobalObject` 对象. 你可以如下所示提供它: ````csharp var result = await _templateRenderer.RenderAsync( "GlobalContextUsage", globalContext: new Dictionary { {"myGlobalObject", "TEST VALUE"} } ); ```` 渲染的结果将是: ```` A global object value: TEST VALUE ```` ## 高级功能 本节介绍文本模板系统的一些内部知识和高级用法. ### 模板内容Provider `TemplateRenderer` 用于渲染模板,这是大多数情况下所需的模板. 但是你可以使用 `ITemplateContentProvider` 获取原始(未渲染的)模板内容. > `ITemplateRenderer` 内部使用 `ITemplateContentProvider` 获取原始模板内容. 示例: ````csharp public class TemplateContentDemo : ITransientDependency { private readonly ITemplateContentProvider _templateContentProvider; public TemplateContentDemo(ITemplateContentProvider templateContentProvider) { _templateContentProvider = templateContentProvider; } public async Task RunAsync() { var result = await _templateContentProvider .GetContentOrNullAsync("Hello"); Console.WriteLine(result); } } ```` 结果是原始模板内容: ```` Hello {%{{{model.name}}}%} :) ```` * `GetContentOrNullAsync` 如果没有为请求的模板定义任何内容,则返回 `null`. * 它可以获取 `cultureName` 参数,如果模板针对不同的文化具有不同的文件,则可以使用该参数(请参见上面的"多内容本地化"部分). ### 模板内容贡献者 `ITemplateContentProvider` 服务使用 `ITemplateContentContributor` 实现来查找模板内容. 有一个预实现的内容贡献者 `VirtualFileTemplateContentContributor`,它从上面描述的虚拟文件系统中获取模板内容. 你可以实现 `ITemplateContentContributor` 从另一个源读取原始模板内容. 示例: ````csharp public class MyTemplateContentProvider : ITemplateContentContributor, ITransientDependency { public async Task GetOrNullAsync(TemplateContentContributorContext context) { var templateName = context.TemplateDefinition.Name; //TODO: Try to find content from another source return null; } } ```` 如果源无法找到内容, 则返回 `null`, `ITemplateContentProvider` 将回退到下一个贡献者. ### Template Definition Manager `ITemplateDefinitionManager` 服务可用于获取模板定义(由模板定义提供程序创建). ## 另请参阅 * 本文开发和引用的[应用程序示例源码](https://github.com/abpframework/abp-samples/tree/master/TextTemplateDemo). * [本地化系统](Localization.md). * [虚拟文件系统](Virtual-File-System.md).