NopCommerce架构分析之八------多语言
系统支持的语言是有类:Language表示;
多语言资源对应的类为:LocalizedProperty;
当先选择某种语言存储在类中:GenericAttribute;
多语言可以导出为XML文件,当然也支持导出。
IWorkContext及其实体类WebWorkContext为当前运行上下文;用户的登录信息以及一些上下文环境设置都保存在此类中。
具体包括:当前用户信息:CurrentCustomer;当前用户Cookie;货币;语言;税的类型;供应商等;
展现多语言资源的方式有几种:
一、在自定义类WebViewPage<TModel>中放置了方法:T(),通过此方法,网页在展现时获取对应语言的文字。
其实T只是一个代理,代理的定义为:
- namespace Nop.Web.Framework.Localization
- {
- public delegate LocalizedString Localizer(string text, params object[] args);
- }
此代理返回值类型为LocalizedString,此类继承接口IHtmlString,以保证能正确显示本地化的文字资源。
IHtmlString的定义为:
- // 摘要:
- // 表示不应再次进行编码的 HTML 编码的字符串。
- public interface IHtmlString
- {
- // 摘要:
- // 返回 HTML 编码的字符串。
- //
- // 返回结果:
- // HTML 编码的字符串。
- string ToHtmlString();
- }
二、通过扩展HtmlHelper
类HtmlExtensions扩展了HtmlHelper类,
主要是对一些控件的封装,并支持多语言。
方法 LocalizedEditor<T, TLocalizedModelLocal>是对Telerik的TabStrip控件的封装(也就是多页签控件---Tab控件),的。系统同时支持有多种语言时,多为每种语言显示一个页签,当然仅当需要时才这么做。这里面用到了接口ILocalizedModel和接口ILocalizedModelLocal。接口ILocalizedModel用来标示某Model类支持这种多语言显示,其中里面包括多种语言数据列表Locales,实现接口ILocalizedModelLocal的类就是特定一种语言的数据。LocalizedEditor方法就是根据这些接口的配合实现了支持多种语言页签了。Admin项目使用此方法,Web项目没有使用。
- public static HelperResult LocalizedEditor<T, TLocalizedModelLocal>(this HtmlHelper<T> helper, string name,
- Func<int, HelperResult> localizedTemplate,
- Func<T, HelperResult> standardTemplate)
- where T : ILocalizedModel<TLocalizedModelLocal>
- where TLocalizedModelLocal : ILocalizedModelLocal
- {
- return new HelperResult(writer =>
- {
- if (helper.ViewData.Model.Locales.Count > 1)
- {
- var tabStrip = helper.Telerik().TabStrip().Name(name).Items(x =>
- {
- x.Add().Text("Standard").Content(standardTemplate(helper.ViewData.Model).ToHtmlString()).Selected(true);
- for (int i = 0; i < helper.ViewData.Model.Locales.Count; i++)
- {
- var locale = helper.ViewData.Model.Locales[i];
- var language = EngineContext.Current.Resolve<ILanguageService>().GetLanguageById(locale.LanguageId);
- x.Add().Text(language.Name)
- .Content(localizedTemplate
- (i).
- ToHtmlString
- ())
- .ImageUrl("~/Content/images/flags/" + language.FlagImageFileName);
- }
- }).ToHtmlString();
- writer.Write(tabStrip);
- }
- else
- {
- standardTemplate(helper.ViewData.Model).WriteTo(writer);
- }
- });
- }
扩展方法NopLabelFor<TModel, TValue>是另外一种多语言实现方式。
此方法主要是根据特性DisplayNameAttribute的子类NopResourceDisplayName实现对属性名称的描述。此特性是对Model属性的修饰,以指定属性的名称。
例如类AddNewsCommentModel的属性用NopResourceDisplayName特性指定:
- namespace Nop.Web.Models.News
- {
- public partial class AddNewsCommentModel : BaseNopModel
- {
- [NopResourceDisplayName("News.Comments.CommentTitle")]
- [AllowHtml]
- public string CommentTitle { get; set; }
- [NopResourceDisplayName("News.Comments.CommentText")]
- [AllowHtml]
- public string CommentText { get; set; }
- public bool DisplayCaptcha { get; set; }
- }
- }
HtmlHelper的扩展方法NopLabelFor的实现如下:
- public static MvcHtmlString NopLabelFor<TModel, TValue>(this HtmlHelper<TModel> helper, Expression<Func<TModel, TValue>> expression, bool displayHint = true)
- {
- var result = new StringBuilder();
- var metadata = ModelMetadata.FromLambdaExpression(expression, helper.ViewData);
- var hintResource = string.Empty;
- object value = null;
- if (metadata.AdditionalValues.TryGetValue("NopResourceDisplayName", out value))
- {
- var resourceDisplayName = value as NopResourceDisplayName;
- if (resourceDisplayName != null && displayHint)
- {
- var langId = EngineContext.Current.Resolve<IWorkContext>().WorkingLanguage.Id;
- hintResource =
- EngineContext.Current.Resolve<ILocalizationService>()
- .GetResource(resourceDisplayName.ResourceKey + ".Hint", langId);
- result.Append(helper.Hint(hintResource).ToHtmlString());
- }
- }
- result.Append(helper.LabelFor(expression, new { title = hintResource }));
- return MvcHtmlString.Create(result.ToString());
- }
NopCommerce架构分析之八------多语言的更多相关文章
- NopCommerce架构分析(转载)
原文 一,NopCommerce架构分析之开篇 NopCommerce是.net开源项目中比较成熟的一款业务应用框架,也是电子商务系统中的典范.所以很想多学习一下里面的设计和实现方式. 二,NopCo ...
- NopCommerce架构分析之六------自定义RazorViewEngine
系统中对Razor的支持包括两部分,其中之一就是自定义RazorViewEngine 一.自定义RazorViewEngine 在Global.asax.cs的Application_Start方法中 ...
- NopCommerce架构分析之三---数据库初试化及数据操作
系统启动时执行任务:IStartupTask,启动时执行的任务主要是数据库的初始化和加载. IStartupTask调用IEfDataProvider进行数据库的初始化. IEfDataProvide ...
- NopCommerce架构分析之四----插件机制
NopCommerce支持灵活的插件机制,所谓Web系统插件,其实也就是可以像原系统的一部分一样使用. Web系统的使用方式就是客户端发送一个请求,服务端进行解析.在asp.net MVC中对客户请求 ...
- NopCommerce架构分析之一----依赖类生成容器
NopCommerce为了实现松耦合的框架设计目的,使用了IOC框架:Autofac.据有人测试,Autofac是性能好的IOC工具. 1.在IOC中,组件首先需要在IOC中注册,有通过配置文件注册的 ...
- nopCommerce架构分析系列(二)数据Cache
原文(http://www.cnblogs.com/gusixing/archive/2012/04/12/2443799.html)非常感谢作者顾思行的分享! 序言 在很多访问量较大的系统中,尤其在 ...
- NopCommerce架构分析之五------Model绑定Action参数
asp.net MVC中Action参数不只是一些基本类型,也支持实体参数.那么从客户端传来的数据如何映射或转换成实体对象呢?就是通过实体绑定类ModelBinder.此系列类在请求转化为后台Cont ...
- NopCommerce架构分析之参考资料
http://www.cnblogs.com/RobbinHan/archive/2011/11/30/2269537.html 依赖注入框架Autofac的简单使用 http://www.cnblo ...
- Android架构分析之使用自定义硬件抽象层(HAL)模块
作者:刘昊昱 博客:http://blog.csdn.net/liuhaoyutz Android版本:2.3.7_r1 Linux内核版本:android-goldfish-2.6.29 在上一篇博 ...
随机推荐
- svn团队环境
1.安装VisualSVN Server VisualSVN-Server-3.3.1-x64.msi 下载,并安装标准版(免费) 2.下载TortoiseSVN TortoiseSVN-1.8.11 ...
- Fedora 17安装NFS
1.NFS概述 NFS(Network File System)是一种分布式文件系统,允许网络中的安装不同操作系统的计算机间共享文件和外设,所以它的通讯协定设计与主机及作业系统无关. 它是由SUN公司 ...
- grunt项目配置
安装完CLI,还要在项目安装Grunt npm install -g grunt-cli npm install grunt --save-dev 源码放在src下 package.json放在根目录 ...
- make问题:make[1] entering directory
执行make distclean命令.
- ASP.NET中POST数据并跳转页面
需求:先Post提交数据,然后跳转到目标页面 找了好久才发现这个神奇的类HttpHelper.原理很简单,利用html的from表单拼接,然后执行 使用方法: NameValueCollection ...
- Spring的配置文件
Web.xml将会配置Spring的配置文件位置: <servlet> <servlet-name>x</servlet-name> & ...
- bzoj 1318: [Spoj744] Longest Permutation 智商题
1318: [Spoj744] Longest Permutation Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 361 Solved: 215 ...
- 中国首个 SaaS 模式的云告警平台安卓版 APP 上线
今年一月底,国内首个 SaaS 模式的云告警平台 OneAlert 正式发布了 iOS 版 App 客户端,今天上午,安卓版 App 客户端也正式上线了!每个安卓用户,无需电脑,都可以通过手机全程跟踪 ...
- DJANGO的ORM的Q查询作多字段外键的模糊查询样码
工作中用到的,存照一下. from django.db.models import Q if self.kwargs.has_key('search_pk'): search_pk = self.kw ...
- 李洪强iOS开发之代理
如果A想让控制器B为他做事情 用代理的话 首先: 在A的.h文件中: 其次A的.m中 在控制器的.m文件中: 还是在控制器B的.m文件中 在A初始化的那一刻设置控制器B为A的代理 在B的.m中实现代 ...