Skip to content
This repository has been archived by the owner on Nov 16, 2023. It is now read-only.

Latest commit

 

History

History
155 lines (128 loc) · 6.11 KB

README.md

File metadata and controls

155 lines (128 loc) · 6.11 KB

WPFGenericHost

此方案已经过时,新的方案请参考:https://github.com/elvisw/WPFGenericHost2

在WPF应用中引入Microsoft.Extensions.Hosting,以便具备依赖注入、日志、配置等功能

参考:https://laurentkempe.com/2019/09/03/WPF-and-dotnet-Generic-Host-with-dotnet-Core-3-0/

依赖CommunityToolkit.MvvmMicrosoft.Extensions.Hosting这两个nuget包。

App.xaml

<Application x:Class="wpfGenericHost.App"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:local="clr-namespace:wpfGenericHost"
             Startup="Application_Startup"
             Exit="Application_Exit">
    <Application.Resources />         
</Application>

App.xaml.cs

    public partial class App : Application
    {
        private readonly IHost _host;
        public App()
        {
            _host = new HostBuilder()
                .ConfigureAppConfiguration((context, configurationBuilder) =>
                {
                    configurationBuilder.SetBasePath(context.HostingEnvironment.ContentRootPath);
                    configurationBuilder.AddJsonFile("appsettings.json", optional: false);
                })
                .ConfigureServices((context, services) =>
                {
                    //对于桌面应用,AddScoped似乎和AddSingleton区别不大,建议用AddSingleton。
                    //AddSingleton用于长期驻留内存的服务或窗口;AddTransient用于临时的窗口和服务。
                    //需要注意的是,如果窗口有关闭后再打开的情况,建议使用AddTransient。使用AddSingleton关闭再打开会导致报错:
                    //“关闭窗口后,无法设置可见性,也无法调用 Show、ShowDialogor 或 WindowInteropHelper.EnsureHandle。”
                    //此时建议重写该窗口的OnClosing()方法:
                    /*
                     protected override void OnClosing(CancelEventArgs e)
                        {
                            e.Cancel = true;  // cancels the window close    
                            this.Hide();      // Programmatically hides the window
                        }
                     */
                    services.Configure<Settings>(context.Configuration);
                    services.AddSingleton<ITextService, TextService>();
                    services.AddSingleton<MainViewModel>();
                    services.AddSingleton<MainWindow>();
                    services.AddTransient<Window1ViewModel>();
                    services.AddTransient<Window1>();
                })
                .ConfigureLogging(logging =>
                {
                    //可以修改成自己需要的日志服务
                    logging.AddDebug();
                })
                .Build();

            Services = _host.Services;

        }

        /// <summary>
        /// Gets the current <see cref="App"/> instance in use
        /// </summary>
        public new static App Current => (App)Application.Current;

        /// <summary>
        /// Gets the <see cref="IServiceProvider"/> instance to resolve application services.
        /// </summary>
        public IServiceProvider Services { get; }

        private async void Application_Startup(object sender, StartupEventArgs e)
        {
            await _host.StartAsync();
            var mainWindow = _host.Services.GetService<MainWindow>();
            mainWindow?.Show();
        }

        private async void Application_Exit(object sender, ExitEventArgs e)
        {
            using (_host)
            {
                await _host.StopAsync(TimeSpan.FromSeconds(5));
            }
        }
    }

问题:

问题1

视图模型使用了依赖注入后,构造函数里带有参数,无法在xaml中绑定DataContext,只能在隐藏代码里处理,这就导致了无法使用Visual Studio的xaml设计视图来处理绑定,只能手工编写xaml代码,并且xaml设计视图无法实时预览绑定数据,只能运行程序后看到效果。

解决:

问题1

  1. 为视图模型类型创建两个构造函数,无参构造函数用于设计时数据,另一个用于依赖注入。
public partial class MainViewModel : ObservableObject
    {
        private readonly ITextService _textService;

        [ObservableProperty]
        private string? helloText;
        /// <summary>
        /// 无参构造函数,为XAML设计器提供设计时支持
        /// </summary>
        public MainViewModel()
        {
            _textService = new TextServiceForDesigner();
            HelloText = _textService.GetText();
        }
        /// <summary>
        /// 用于依赖注入的构造函数
        /// </summary>
        /// <param name="textService">用于依赖注入</param>
        public MainViewModel(ITextService textService)
        {
            _textService = textService;
            HelloText = _textService.GetText();
        }
    }
  1. 使用 设计时属性 ,将视图模型在设计器中绑定,在XAML中,添加以下属性到<Window><UserControl>
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
mc:Ignorable="d"
d:DataContext="{d:DesignInstance Type=vm:MainViewModel, IsDesignTimeCreatable=True}"

为设计时编写的无参构造函数提供的服务不适合运行时的业务,好在设计时属性的绑定在运行时不起作用。因此我们可以将运行时的DataContext通过隐藏代码和注入的ViewModel绑定,此时的ViewModel已经通过含参构造函数被容器注入了运行时需要的依赖项,可以正常处理业务:

public MainWindow(MainViewModel mvm)
        {
            InitializeComponent();
            DataContext = mvm;
        }

参考:https://stackoverflow.com/questions/25366291/how-to-handle-dependency-injection-in-a-wpf-mvvm-application/25508012#25508012