【UWP】通過 MarkupExtension 實現 ValueConveter 的依賴註入

来源:https://www.cnblogs.com/h82258652/archive/2020/04/03/12625081.html
-Advertisement-
Play Games

最近是真的比較閑,花了點時間算是把我自己的微博庫的 nuget 包的坑填上了(https://github.com/h82258652/HN.Social.Weibo 歡迎大佬來 Star)。dino 大佬也一直忽悠我弄動畫,可惜我沒啥藝術細胞而且 Composition API 也不太熟悉,就只能 ...


最近是真的比較閑,花了點時間算是把我自己的微博庫的 nuget 包的坑填上了(https://github.com/h82258652/HN.Social.Weibo 歡迎大佬來 Star)。dino 大佬也一直忽悠我弄動畫,可惜我沒啥藝術細胞而且 Composition API 也不太熟悉,就只能逃了(哈哈哈 )。閑著無事就刷刷 Github,看到 wpf repo 的一個 issue(https://github.com/dotnet/wpf/issues/499),確實目前的 XAML 跟控制反轉這塊幾乎都沒啥結合。控制項層面由於要求無參構造函數,所以目前來看難以實現了。但 ValueConverter 這玩意,想了下,好像可以耶,於是做了下實驗,成功並且寫下了這篇 blog。


UWP 的 MarkupExtension 是在 16299 版本引入的,所以我們的項目必須要 target 16299 或以上。

以一般 MVVM 模式為例,創建 ViewModelLocator.cs,這裡 IoC 容器我就使用最常用的 Autofac 好了,引用 Autofac.Extras.CommonServiceLocator 包。

public class ViewModelLocator
{
    static ViewModelLocator()
    {
        var autofacServiceLocator = new AutofacServiceLocator(CreateAutofacContainer());
        ServiceLocator.SetLocatorProvider(() => autofacServiceLocator);
    }

    private static IContainer CreateAutofacContainer()
    {
        var containerBuilder = new ContainerBuilder();

        // TODO Register services

        return containerBuilder.Build();
    }
}

並修改 App.xaml

<Application x:Class="ConverterIocDemo.App"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:viewModels="using:ConverterIocDemo.ViewModels">
    <Application.Resources>
        <ResourceDictionary>
            <viewModels:ViewModelLocator x:Key="Locator" />
        </ResourceDictionary>
    </Application.Resources>
</Application>


接下來添加一些測試代碼吧。

namespace ConverterIocDemo.Models
{
    public class Person
    {
        public string Name { get; set; }

        public int Age { get; set; }
    }
}
using ConverterIocDemo.Models;

namespace ConverterIocDemo.Services
{
    public interface IPersonService
    {
        string GetHello(Person person);
    }
}
using System;
using ConverterIocDemo.Models;

namespace ConverterIocDemo.Services
{
    public class PersonService : IPersonService
    {
        public string GetHello(Person person)
        {
            if (person == null)
            {
                throw new ArgumentNullException(nameof(person));
            }

            var now = DateTime.Now;
            if (now.Hour >= 9 && now.Hour <= 21 && now.DayOfWeek != DayOfWeek.Sunday)
            {
                return $"大家好,我叫 {person.Name},今年 {person.Age} 歲";
            }
            else
            {
                return "996 大法好(mmp)";
            }
        }
    }
}
using ConverterIocDemo.Models;

namespace ConverterIocDemo.ViewModels
{
    public class MainViewModel
    {
        public MainViewModel()
        {
            Person = new Person
            {
                Name = "justin liu",
                Age = 18
            };
        }

        public Person Person { get; }
    }
}
using ConverterIocDemo.Services;
using System;
using Windows.UI.Xaml.Data;
using ConverterIocDemo.Models;

namespace ConverterIocDemo.Converters
{
    public class PersonSayHelloConverter : IValueConverter
    {
        private readonly IPersonService _personService;

        public PersonSayHelloConverter(IPersonService personService)
        {
            _personService = personService;
        }

        public object Convert(object value, Type targetType, object parameter, string language)
        {
            return _personService.GetHello((Person)value);
        }

        public object ConvertBack(object value, Type targetType, object parameter, string language)
        {
            throw new NotImplementedException();
        }
    }
}

修改一下 ViewModelLocator,把這堆玩意註冊上去。

using Autofac;
using Autofac.Extras.CommonServiceLocator;
using CommonServiceLocator;
using ConverterIocDemo.Converters;
using ConverterIocDemo.Services;

namespace ConverterIocDemo.ViewModels
{
    public class ViewModelLocator
    {
        static ViewModelLocator()
        {
            var autofacServiceLocator = new AutofacServiceLocator(CreateAutofacContainer());
            ServiceLocator.SetLocatorProvider(() => autofacServiceLocator);
        }

        public MainViewModel Main => ServiceLocator.Current.GetInstance<MainViewModel>();

        private static IContainer CreateAutofacContainer()
        {
            var containerBuilder = new ContainerBuilder();

            containerBuilder.RegisterType<PersonService>().As<IPersonService>();
            containerBuilder.RegisterType<MainViewModel>();
            containerBuilder.RegisterType<PersonSayHelloConverter>().SingleInstance();

            return containerBuilder.Build();
        }
    }
}

接下來就是本文關鍵,通過 MarkupExtension 消費這個 PersonSayHelloConveter 了。這裡我就叫 ConverterProviderExtension。

using CommonServiceLocator;
using System;
using Windows.UI.Xaml.Data;
using Windows.UI.Xaml.Markup;

namespace ConverterIocDemo.Converters
{
    [MarkupExtensionReturnType(ReturnType = typeof(IValueConverter))]
    public class ConverterProviderExtension : MarkupExtension
    {
        public Type ConverterType { get; set; }

        protected override object ProvideValue()
        {
            if (ConverterType == null)
            {
                throw new ArgumentException("轉換器類型沒有設置");
            }

            return ServiceLocator.Current.GetInstance(ConverterType);
        }
    }
}

接下來修改 MainPage 看看效果了

<Page x:Class="ConverterIocDemo.MainPage"
      xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
      xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
      xmlns:converters="using:ConverterIocDemo.Converters"
      xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
      xmlns:local="using:ConverterIocDemo"
      xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
      Background="{ThemeResource ApplicationPageBackgroundThemeBrush}"
      DataContext="{Binding Source={StaticResource Locator}, Path=Main}"
      mc:Ignorable="d">
    <Grid>
        <TextBlock HorizontalAlignment="Center"
                   VerticalAlignment="Center"
                   Text="{Binding Path=Person, Converter={converters:ConverterProvider ConverterType=converters:PersonSayHelloConverter}}" />
    </Grid>
</Page>

運行起來:

Snipaste_2020-04-03_10-31-24

改個時間再跑起來:

Snipaste_2020-04-03_23-36-27

還行。


理論上可以修改 MarkupExtensionReturnTypeAttribute 的 ReturnType 為 typeof(object) 然後從 IoC 容器獲取任意已經註冊了的東西就是了。但寫完 blog 發現好像滿滿的偽需求的樣子。ε=ε=ε=┏(゜ロ゜;)┛


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • .NET Core 概述 .NET Core是一個免費的、開源的、跨平臺的、廣泛使用的Web框架;它是由微軟維護的。社區廣泛參與支持的一個框架。.NET Core可以運行在:Windows、MacOS以及Linux操作系統上。 .Net Core可以用來開發各種不同的應用程式,例如:移動端、桌面端、 ...
  • 一、環境搭建 1.1、由於RabbitMQ是使用Erlang語言開發的,因此要安裝Erlang運行時環境,下載地址:Erlang官網下載 CSDN分享下載 1.2、去RabbitMQ官網下載RabbitMQ Server服務端程式,選擇合適的平臺版本下載並安裝。 RabbitMQ安裝時,會自動在Wi ...
  • 我在面試別人的時候,經常會問對方,如何設計一個秒殺系統?回答的好的同學並不多,這裡我簡要說一下考察這個問題的目的.秒殺系統,那麼顧名思義就是搶購,庫存有限情況下的競爭問題,其實就是一個高併發的處理. 首先我們模擬不做併發處理的情況: 比如我們用戶一個庫存表 stock,庫存數量5 我們對外提供了一個 ...
  • 本文介紹通過C# 編程如何在PPT幻燈片中添加超鏈接的方法,添加鏈接時,可給文本或者圖片添加超鏈接,鏈接對象可指向網頁地址、郵件地址、指定幻燈片等,此外,也可以參考文中編輯、刪除幻燈片中已有超鏈接的方法。 程式使用類庫:Free Spire.Presentation for .NET (免費版) d ...
  • 如何處理幾十萬條併發數據? 答:用存儲過程或事務。取得最大標識的時候同時更新..註意主鍵不是自增量方式這種方法併發的時候是不會有重覆主鍵的..取得最大標識要有一個存儲過程來獲取. 2.寫出一條Sql語句,取出表A中第31到第40記錄(SQLServer,以自動增長的ID作為主鍵,註意:數據不是連續的 ...
  • 有時我們臨時需要一個 JSON 字元串,直接拼接肯定不是好方法,但又懶得去定義一個類,這是用 就會非常的方便。 但是在 中添加數組卻經常被坑。 輸出結果: 非常正確,但如果把 換成 就不對了。 這麼寫會報: Could not determine JSON object type for type ...
  • 0、概述 先瞭解下https是個啥: https://www.bilibili.com/video/BV1j7411H7vV so!只要給我們的web伺服器配置一個證書就行了,證書可以買,也可以用免費的Let's Encrypt,此證書提供商是多個牛X大公司為了推進全球https化搞出來的,所以不用 ...
  • HTTP Method 較為簡單,我們常用的習慣如下: 一般查詢我們都會使用 GET 方法, 創建新的記錄使用 POST 方法 更新已有數據使用 PUT 方法 更新已有數據部分屬性使用 PATCH 方法 刪除已有數據使用 DELETE 方法 下麵來詳細介紹一下常用的 HTTP 狀態碼 1xx 1xx ...
一周排行
    -Advertisement-
    Play Games
  • Dapr Outbox 是1.12中的功能。 本文只介紹Dapr Outbox 執行流程,Dapr Outbox基本用法請閱讀官方文檔 。本文中appID=order-processor,topic=orders 本文前提知識:熟悉Dapr狀態管理、Dapr發佈訂閱和Outbox 模式。 Outbo ...
  • 引言 在前幾章我們深度講解了單元測試和集成測試的基礎知識,這一章我們來講解一下代碼覆蓋率,代碼覆蓋率是單元測試運行的度量值,覆蓋率通常以百分比表示,用於衡量代碼被測試覆蓋的程度,幫助開發人員評估測試用例的質量和代碼的健壯性。常見的覆蓋率包括語句覆蓋率(Line Coverage)、分支覆蓋率(Bra ...
  • 前言 本文介紹瞭如何使用S7.NET庫實現對西門子PLC DB塊數據的讀寫,記錄了使用電腦模擬,模擬PLC,自至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1.Windows環境下鏈路層網路訪問的行業標準工具(WinPcap_4_1_3.exe)下載鏈接:http ...
  • 從依賴倒置原則(Dependency Inversion Principle, DIP)到控制反轉(Inversion of Control, IoC)再到依賴註入(Dependency Injection, DI)的演進過程,我們可以理解為一種逐步抽象和解耦的設計思想。這種思想在C#等面向對象的編 ...
  • 關於Python中的私有屬性和私有方法 Python對於類的成員沒有嚴格的訪問控制限制,這與其他面相對對象語言有區別。關於私有屬性和私有方法,有如下要點: 1、通常我們約定,兩個下劃線開頭的屬性是私有的(private)。其他為公共的(public); 2、類內部可以訪問私有屬性(方法); 3、類外 ...
  • C++ 訪問說明符 訪問說明符是 C++ 中控制類成員(屬性和方法)可訪問性的關鍵字。它們用於封裝類數據並保護其免受意外修改或濫用。 三種訪問說明符: public:允許從類外部的任何地方訪問成員。 private:僅允許在類內部訪問成員。 protected:允許在類內部及其派生類中訪問成員。 示 ...
  • 寫這個隨筆說一下C++的static_cast和dynamic_cast用在子類與父類的指針轉換時的一些事宜。首先,【static_cast,dynamic_cast】【父類指針,子類指針】,兩兩一組,共有4種組合:用 static_cast 父類轉子類、用 static_cast 子類轉父類、使用 ...
  • /******************************************************************************************************** * * * 設計雙向鏈表的介面 * * * * Copyright (c) 2023-2 ...
  • 相信接觸過spring做開發的小伙伴們一定使用過@ComponentScan註解 @ComponentScan("com.wangm.lifecycle") public class AppConfig { } @ComponentScan指定basePackage,將包下的類按照一定規則註冊成Be ...
  • 操作系統 :CentOS 7.6_x64 opensips版本: 2.4.9 python版本:2.7.5 python作為腳本語言,使用起來很方便,查了下opensips的文檔,支持使用python腳本寫邏輯代碼。今天整理下CentOS7環境下opensips2.4.9的python模塊筆記及使用 ...