当前位置:   article > 正文

dotnetcore3.1 WPF 实现多语言

dotnet 多语言支持

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 即可,获取对应的要实例化的,比如:

  1. public partial class MainWindow: Window
  2. {
  3. private readonly IStringLocalizer<MainWindow> _localizer;
  4. public MainWindow(
  5. IStringLocalizer<MainWindow> localizer)
  6. {
  7. InitializeComponent();
  8. _localizer = localizer;
  9. }
  10. // ...
  11. {
  12. // ...
  13. MessageBox.Show(_localizer["Success"], _localizer["Tip"]);
  14. }
  15. }

xaml 实现方式

LocaliazerExtension

xaml 文件中使用需要自定义一个扩展,定义如下,【实现源码】:

  1. public class LocalizerExtension : MarkupExtension
  2. {
  3. private readonly IStringLocalizerFactory _localizerFactory;
  4. public string Key { get; }
  5. public LocalizerExtension(string key)
  6. {
  7. Key = key;
  8. _localizerFactory = DependencyResolver.Current.
  9. ResolveService<IStringLocalizerFactory>();
  10. }
  11. public override object ProvideValue(IServiceProvider serviceProvider)
  12. {
  13. var targetRootType = serviceProvider.GetType()
  14. .GetProperty("System.Xaml.IRootObjectProvider.RootObject", BindingFlags.Instance | BindingFlags.NonPublic)
  15. ?.GetValue(serviceProvider)
  16. ?.GetType();
  17. if (null == targetRootType)
  18. {
  19. targetRootType = typeof(MainWindow);
  20. }
  21. var localizer = _localizerFactory.Create(targetRootType);
  22. var value = localizer[Key];
  23. return (string)value;
  24. }
  25. }

这里使用到了上面提到的 ServiceLocator 模式的代码,从 DependencyResolver 获取注册的服务,感觉这里的实现需要优化,有更好想法的小伙伴还望一起交流一下,另外如果你的应用比较简单,我觉得上面代码里的通过反射获取 targetRootType 的代码可以直接使用某一个类型例如:typeof(MainWindow) ,这样会更高效

在 xaml 中使用

  1. 在 Window 标签中添加扩展对应的命令空间,例如: xmlns:loc="clr-namespace:DbTool.Localization"

  2. 在需要实现多语言的地方引用,例如: <TextBlockMargin="0,0,4,0"Text="{loc:Localizer DbConnectionString}"></TextBlock>

  3. 在对应的资源文件中配置要使用多语言资源,如上面的 DbConnectionString

  1. <Window x:Class="DbTool.MainWindow"
  2. xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
  3. xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
  4. xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
  5. xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
  6. xmlns:loc="clr-namespace:DbTool.Localization"
  7. mc:Ignorable="d"
  8. Title="DbTool" Height="450" Width="800" FontSize="14">
  9. <Grid>
  10. <TextBlock Margin="0,0,4,0" Text="{loc:Localizer DbConnectionString}"></TextBlock>
  11. </Grid>
  12. </Window>

语言切换

发生语言切换时或应用启动时设置默认语言时,要更新当前线程的 Culture 信息

  1. // set current culture
  2. var defaultCulture = "zh";
  3. Thread.CurrentThread.CurrentCulture = CultureInfo.GetCultureInfo(defaultCulture);
  4. 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

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/代码探险家/article/detail/963701
推荐阅读
相关标签
  

闽ICP备14008679号