dotnetcore3.1 WPF 实现多语言
Intro
最近把 DbTool 从 WinForm 迁移到了 WPF,并更新到了 dotnet core 3.1,并实现了基于 Microsoft.Extensions.Localization
实现了基本的多语言支持。下面来分享一下如何来实现
服务注册
如果不熟悉如何在 WPF 中使用依赖注入,可以参考上一篇文章 dotnetcore3.1 WPF 中使用依赖注入
在应用启动前注册 Localization 服务,我这里使用的是自己自定义的基于 JSON 的多语言服务
services.AddJsonLocalization(options => options.ResourcesPathType = ResourcesPathType.CultureBased);
服务注册的最后使用了一个 ServiceLocator 模式的代码(DependencyResolver),保存了所有的注册服务,后面的 Localizer 扩展会用到
DependencyResolver.SetDependencyResolver(services);
代码文件实现方式
代码文件(如:MainWindow.xaml.cs) 中实现多语言较为简单,直接注入 IStringLocalizer
即可,获取对应的要实例化的,比如:
public partial class MainWindow: Window
{private readonly IStringLocalizer<MainWindow> _localizer;public MainWindow(IStringLocalizer<MainWindow> localizer){InitializeComponent();_localizer = localizer;}// ...{// ...MessageBox.Show(_localizer["Success"], _localizer["Tip"]);}
}
xaml 实现方式
LocaliazerExtension
xaml 文件中使用需要自定义一个扩展,定义如下,【实现源码】:
public class LocalizerExtension : MarkupExtension
{private readonly IStringLocalizerFactory _localizerFactory;public string Key { get; }public LocalizerExtension(string key){Key = key;_localizerFactory = DependencyResolver.Current.ResolveService<IStringLocalizerFactory>();}public override object ProvideValue(IServiceProvider serviceProvider){var targetRootType = serviceProvider.GetType().GetProperty("System.Xaml.IRootObjectProvider.RootObject", BindingFlags.Instance | BindingFlags.NonPublic)?.GetValue(serviceProvider)?.GetType();if (null == targetRootType){targetRootType = typeof(MainWindow);}var localizer = _localizerFactory.Create(targetRootType);var value = localizer[Key];return (string)value;}
}
这里使用到了上面提到的 ServiceLocator
模式的代码,从 DependencyResolver
获取注册的服务,感觉这里的实现需要优化,有更好想法的小伙伴还望一起交流一下,另外如果你的应用比较简单,我觉得上面代码里的通过反射获取 targetRootType
的代码可以直接使用某一个类型例如:typeof(MainWindow)
,这样会更高效
在 xaml 中使用
在
Window
标签中添加扩展对应的命令空间,例如:xmlns:loc="clr-namespace:DbTool.Localization"
在需要实现多语言的地方引用,例如:
<TextBlockMargin="0,0,4,0"Text="{loc:Localizer DbConnectionString}"></TextBlock>
在对应的资源文件中配置要使用多语言资源,如上面的
DbConnectionString
<Window x:Class="DbTool.MainWindow"xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"xmlns:d="http://schemas.microsoft.com/expression/blend/2008"xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"xmlns:loc="clr-namespace:DbTool.Localization"mc:Ignorable="d"Title="DbTool" Height="450" Width="800" FontSize="14"><Grid><TextBlock Margin="0,0,4,0" Text="{loc:Localizer DbConnectionString}"></TextBlock></Grid>
</Window>
语言切换
发生语言切换时或应用启动时设置默认语言时,要更新当前线程的 Culture 信息
// set current culture
var defaultCulture = "zh";
Thread.CurrentThread.CurrentCulture = CultureInfo.GetCultureInfo(defaultCulture);
Thread.CurrentThread.CurrentUICulture = CultureInfo.GetCultureInfo(defaultCulture);
实现效果
More
这种方式的实现,目前还需要重启之后界面的语言才会发生变化,可以进一步优化,实现动态多语言,修改语言之后界面就切换,目前不是太需要,暂时没做,有需要的可以先自己研究一下。
Reference
https://github.com/WeihanLi/WeihanLi.Extensions.Localization.Json
https://github.com/WeihanLi/DbTool
https://github.com/WeihanLi/DbTool/blob/wpf-dev/DbTool/MainWindow.xaml
https://github.com/WeihanLi/DbTool/blob/wpf-dev/DbTool/MainWindow.xaml.cs
https://github.com/WeihanLi/DbTool/blob/wpf-dev/DbTool/Localization/LocalizerExtension.cs