English | 简体中文

上一篇有简单介绍主工程的国际化,使用的资源字典(XAML)实现的。这几天我添加了几个Prism模块(Module),发现子模块使用资源字典的方式实现国际化和本地化不好做,没有找到比较好的参考文章,所以换了一种方式,使用资源文件实现了。

让WPF项目走上国际化路线

一、本文概述

子模块的国际化和本地化要求:

  • 各模块需要有自己单独的语言文件。

  • 在主工程中动态切换语言时,子模块也需要跟着切换。

  • 使用了Prism实现模块化框架,即要求主工程与各子模块不能有引用关系,即松耦合,不能直接在主工程中切换子模块的语言文件。

基于上面的要求,我尝试在各模块(Module)中也定义了语言文件(XAML),主窗体切换语言时,加载模块语言文件老是提示不存在对应的资源字典文件,我恼火呀,后面还是参考“Accelerider.Windows”国际化的方式,使用资源文件实现本地化和国际化了,不纠结Xaml的方式了,唉。

此路不通,我换条路走

下面是修改后的效果:

和上一版异同:

  • 标题栏国际化无变化,只是文字绑定换了种方式,实现效果一致。

  • 左侧添加了三个子模块(Home\Client\Server),使用Prism动态加载的,并且跟随主工程主窗体语言切换而切换语言。

下面简单介绍怎么创建模块,以及主窗体和模块国际化怎么做的,真的是很简单的介绍,具体的实现可以拉代码看看。

跌到,爬起来,继续趟坑

二、 添加三个Prism模块(Module)

可安装Prism模板,快速创建模块工程,当然手工创建.Net Core工程也是可以的,就是多了几个步骤而已(需要用Nuget安装Prism.Wpf包(7.2.0.1422)),我使用得的Prism模板快速创建的。

2.1 创建模块之前的准备工作

VS2019下载Prism模板

下载上图搜索到的Prism模板,重启VS,它会自动安装,新建项目时,就有Prism模块模板选择了:

Prism模板工程

注意要选择.NET Core 3的版本,因为我是使用.NET Core创建的WPF项目。

2.2 创建模块

下面是已经创建好的三个模块工程截图:

目前三个模块文件组织结构类似:

  • I18nResources:资源文件夹,放3个语言资源文件和一个T4模板文件(用于引用语言Key),其中T4模板文件在3个模块和主工程中定义是一样的,具体可从github下载源码查看。

  • Views放置视图文件,现在只使用到了主工程主窗体中显示的TabItem视图,即MainTabItem.xaml,继承自TabItem。

  • xxxxModule.cs:prism模板定义文件,prism发现模块使用。

三个模块关键点需要注意:

  • 1. 编辑模块工程文件,修改模块文件输出目录:

// 省略部分代码,下面这一行设置为False,代表输出目录不带.NET Core版本信息
<AppendTargetFrameworkToOutputPath>Flase</AppendTargetFrameworkToOutputPath>
// 省略部分代码,修改Debug与Release编译输出目录,方便主工程统一加载模块
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|AnyCPU'"><OutputPath>..\Build\Debug\Modules</OutputPath>
</PropertyGroup>
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|AnyCPU'"><OutputPath>..\Build\Release\Modules</OutputPath>
</PropertyGroup>
// 省略部分代码
  • 2. XXXModule中需要将资源文件的ResourceManager引用添加到另一个库中保存,待切换语言时需要使用,如在HomeModule的构造函数中添加代码如下,只添加这一句代码就好,模块的国际化及本地化就算完事了:

I18nManager.Instance.Add(TerminalMACS.Home.I18nResources.UiResource.ResourceManager);
  • 3. XXXModule的RegisterTypes方法中注册视图"MainTabItem"到"RegionNames.MainTabRegion",主窗体使用"RegionNames.MainTabRegion"关联模块视图显示加载。

_regionManager.RegisterViewWithRegion(RegionNames.MainTabRegion, typeof(MainTabItem));
  • 4. UI控件国际化文字绑定,其中markup使用的一个开源库命名空间,后面会给出链接,本项目直接将该库加载进了解决方案中;i18NResources:Language即T4模板文件生成的类,关联文字翻译的Key。绑定文字部分代码如下:

<TextBlock Grid.Row="2" Text="{markup:I18n {x:Static i18NResources:Language.MainTabItm_Header}}"

三、 主工程

主工程目录组织结构如下:

3.1 动态加载Prism模块

配置加载3个模块的关键代码在App.xaml.cs文件中,看上面的代码,我将三个模块输出到了Modules目录下,主工程直接加载此目录即可,其他加载方式还有使用配置文件等,可以参考Prism官方例子,文末给出链接:

protected override IModuleCatalog CreateModuleCatalog()
{string modulePath = @".\Modules";if (!Directory.Exists(modulePath)){Directory.CreateDirectory(modulePath);}return new DirectoryModuleCatalog() { ModulePath = modulePath };
}

主窗体显示子模块注册的TabItem视图,prism:RegionManager.RegionName即在各子模块中注册过的区域字符串,他与模块对应的TabItem视图关联,代码如下:

<TabControl Grid.ColumnSpan="2" SelectedIndex="0"Style="{StaticResource MainTabControlStyle}" ItemContainerStyle="{StaticResource MainTabItemStyle}"prism:RegionManager.RegionName="{x:Static ui:RegionNames.MainTabRegion}"/>

主窗体以TabControl的控件形式展示子模块视图:

子模块的TabItem视图

主工程要能正常加载子模块,主工程的工程文件也需要修改其输出目录:

// 省略部分代码,下面这一行设置为False,代表输出目录不带.NET Core版本信息
<AppendTargetFrameworkToOutputPath>Flase</AppendTargetFrameworkToOutputPath>
// 省略部分代码,修改Debug与Release编译输出目录,方便主工程统一加载模块
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|AnyCPU'"><OutputPath>..\Build\Debug</OutputPath>
</PropertyGroup>
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|AnyCPU'"><OutputPath>..\Build\Release</OutputPath>
</PropertyGroup>
// 省略部分代码

3.2 修改语言文件格式

删除了原有的XAML语言文件,替换为resx的资源文件,和三个模块的资源文件类型类似,下面是主工程的资源文件:

资源文件作为语言文件使用

替换成资源文件,编辑是要比XAML文件要方便点,起初是有考虑使用资源文件实现国际化的,作死想尝试XAML文件。

折腾是可以涨姿势的

3.3 语言切换核心代码

动态切换语言的关键代码改为:

public static void SetLanguage(string language = "")
{if (string.IsNullOrWhiteSpace(language)){language = ConfigHelper.ReadKey(KEY_OF_LANGUAGE);if (string.IsNullOrWhiteSpace(language)){language = System.Globalization.CultureInfo.CurrentCulture.ToString();}}ConfigHelper.SetKey(KEY_OF_LANGUAGE, language);_lastLanguage = language;var culture = new System.Globalization.CultureInfo(language);I18nManager.Instance.CurrentUICulture = culture;
}

核心的语言切换代码是最后一句,不详细说了,解决方案中有库、有源码:

I18nManager.Instance.CurrentUICulture = culture;

四. 源码

关于
  • 源码地址,欢迎star:https://github.com/dotnet9/TerminalMACS/tree/master/src/TerminalMACS.Manager/TerminalMACS.ManagerForWPF

  • 官方网站:https://terminalmacs.com

  • 合作网站:https://dotnet9.com

四. 参考资料

  • Prism Template Pack(Prism模板):https://marketplace.visualstudio.com/items?itemName=BrianLagunas.PrismTemplatePack

  • WPF国际化开源辅助库:https://github.com/DingpingZhang/WpfExtensions

  • Accelerider.Windows(子模块加载参考开源项目):https://github.com/Accelerider/Accelerider.Windows

  • Prism-Samples-Wpf(官方Demo):https://github.com/PrismLibrary/Prism-Samples-Wpf

怎样实现WPF Prism Module的国际化和本地化?相关推荐

  1. WPF PRISM开发入门一( 初始化PRISM WPF程序)

    原文:WPF PRISM开发入门一( 初始化PRISM WPF程序) 这篇博客将介绍在WPF项目中引入PRISM框架进行开发的一些基础知识.目前最新的PRISM的版本是Prism 6.1.0,可以在G ...

  2. C#/.Net Core/WPF框架初建(国际化、主题色)

    English | 简体中文 作为 TerminalMACS 的一个子进程模块 - WPF管理端,目前搭建框架部分功能:本地化.国际化.主题色修改等. 导航目录 1.框架已添加功能说明 1.1. 国际 ...

  3. c# contains方法_C#/.Net Core/WPF框架初建(国际化、主题色)

    C#/.Net Core/WPF框架初建(国际化.主题色) English | 简体中文 作为 TerminalMACS 的一个子进程模块 - WPF管理端,目前搭建框架部分功能:本地化.国际化.主题 ...

  4. WPF Prism框架

    Prism框架 1.关于Prism框架 ​ 官方地址:http://prismlibrary.com ​ 官方源码:https://github.com/PrismLibrary/Prism ​ 版本 ...

  5. WPF Prism(五)Navigation

    WPF Prsim(一)Region WPF Prism(二)Module WPF Prism(三)ViewModelLocator WPF Prism(四)MVVM WPF Prism(五)Navi ...

  6. WPF开发学生信息管理系统【WPF+Prism+MAH+WebApi】(一)

    最近通过WPF开发项目,为了对WPF知识点进行总结,所以利用业余时间,开发一个学生信息管理系统[Student Information Management System].本文主要简述如何通过WPF ...

  7. Flask 教程 第十三章:国际化和本地化

    本文转载自:https://www.jianshu.com/p/e2923f4042d6 这是Flask Mega-Tutorial系列的第十三部分,我将告诉你如何扩展Microblog应用以支持多种 ...

  8. web框架flask(12)——国际化和本地化

    2019独角兽企业重金招聘Python工程师标准>>> 国际化和本地化 今天的文章的主题是国际化和本地化,通常简称 I18n 和 L10n.我们想要我们的 microblog 应用程 ...

  9. Qt Quick的国际化和本地化

    Qt Quick的国际化和本地化 Qt Quick的国际化和本地化 国际化您的应用程序 1.对所有文字用户界面字符串使用qsTr() 2.为翻译器添加上下文 3.歧义相同的文本 4.用于%x将参数插入 ...

最新文章

  1. 用离散傅里叶变换来实现OFDM
  2. blank space in latex math environment
  3. Linux多线程实践(六)使用Posix条件变量解决生产者消费者问题
  4. php 查询数据库返回json数据
  5. ImageView相关
  6. NSString 中包含中文字符时转换为NSURL
  7. 在MacBook上Jupyter安装
  8. Rational Rose :从用例图开始
  9. Python 全栈开发二 python基础 字符串 字典 集合
  10. 教你如何使用automake生成Makefile文件
  11. ipad和iphone切图_如何在iPhone和iPad上使用触控板模式选择文本
  12. java制作魔方_CSS3 制作魔方 - 玩转魔方
  13. 软件开发工作量及费用量化评估方法在金融行业的应用
  14. 如何设置本电脑中的mysql让别人的电脑连接
  15. 分词 - 准确率评测
  16. CSharp代码示例每日一讲: 在GDI+中使用画笔和画刷
  17. Jmeter-PerfMon Metrics Collector监控解析
  18. 用JAVA写一款自己的小游戏
  19. php如何生成一年的日历表_html - PHP如何生成一个指定年份一整年的日历
  20. 千百倍的性能提升-Andrew大师谈数据库性能优化

热门文章

  1. 17、字符设备控制技术
  2. 【mongoDB运维篇③】replication set复制集
  3. AdBlock屏蔽网易的“我来挑错”和“转发至微博”
  4. Google:推荐几款好用的Chrome浏览器插件
  5. Apache并发处理模块
  6. Win2003中apache2整合tomcat5和iis6
  7. asp.net控件开发基础(20)
  8. 朱晔的互联网架构实践心得S1E7:三十种架构设计模式(上)
  9. keepalived 报错 Popt libraries is required
  10. 【linux磁盘分区--格式化】fdisk,parted,mkfs.ext3