首页 > 解决方案 > WPF 使用空字符串调用值转换器

问题描述

我有一个 ComboBox 绑定到一个值列表,为了这个例子,一个简单的ints 数组。该控件有一个自定义ItemTemplate分配来格式化它的元素,反过来,它使用一个IValueConverter进行一些操作。

有一个失败的特殊情况:如果ComboBox选择了一个元素,并且项目列表发生了变化,则Convert使用空字符串调用转换器,这绝对不是绑定到我的控件的值之一。(请注意,我不是在谈论ConvertBack方法。)

我的问题是:

  1. 当没有字符串绑定到控件时,为什么会以空字符串 ( )IValueConverter调用?value""
  2. 这个问题有哪些非hacky的解决方案?(我可以只放入if (value is "") return null;转换器,它似乎使错误消失,但我觉得它正在治疗症状,而不是原因。)

可以使用dotnet new wpf仅包含这 3 个文件的简单 WPF 项目 ( ) 重现该问题(该问题在 .NET 的 Framework 和 Core 版本中都存在):

MainWindow.xaml:

<Window x:Class="test.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:local="clr-namespace:test"
        mc:Ignorable="d"
        Title="MainWindow" Height="450" Width="800">
    <Grid>
        <ComboBox x:Name="Selector" VerticalAlignment="Center" HorizontalAlignment="Center" Width="100">
            <ComboBox.ItemTemplate>
                <DataTemplate>
                    <TextBlock Text="{Binding Converter={x:Static local:Converter.Instance}}" />
                </DataTemplate>
            </ComboBox.ItemTemplate>
        </ComboBox>
        <Button HorizontalAlignment="Right" VerticalAlignment="Bottom" Content="Replace" Click="Button_Click" />
    </Grid>
</Window>

MainWindow.xaml.cs:

using System;
using System.Windows;

namespace test
{
   public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();

            Selector.ItemsSource = new int[] { 1, 2, 3 };
        }

      private void Button_Click(object sender, RoutedEventArgs e)
      {
         Selector.ItemsSource = new int[] { -1, -2, -3 };
      }

      [STAThread]
      public static void Main(String[] args)
      {
         new MainWindow().ShowDialog();
      }
   }
}

最后是 Converter.cs:

using System;
using System.Diagnostics;
using System.Globalization;
using System.Windows.Data;

namespace test
{
    class Converter : IValueConverter
    {
        public static Converter Instance { get; } = new Converter();

        public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
        {
            Debug.Assert(value is int);

            return (2 * (int) value).ToString();
        }

        public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
        {
            throw new NotSupportedException();
        }
    }
}

这 3 个文件构成了展示此行为的最小示例。如果在没有先从 ComboBox 中选择任何内容的情况下单击“替换”按钮,则程序运行良好。但是,如果在单击按钮之前在 ComboBox 中选择了任何值,则转换器中的断言将失败,因为单击按钮时,转换器将传递一个value.""

标签: c#wpf

解决方案


当当前选定的项目不在新设置的项目源中时,我尝试了您的代码,应该更改选定的项目对吗?所以他们将所选项目更改为“”。那是您在应用程序运行的初始状态中看到的选定项。

事实上,如果您将所选项目保留在新项目源中,那么它将不会执行带有值“”的转换。考虑您的初始项目来源是

Selector.ItemsSource = new int[] { 1, 2, 3 };

现在您选择在组合框中显示为 4 的第二个元素,然后单击替换以运行它。

 Selector.ItemsSource = new int[]{ -1, 2, -3 };

然后它不会执行值为“”的转换器。

这里的问题是新项目源中不存在所选项目,因此 Combobox 正在选择其默认值,并且您的 Combobox 的行为将与您启动应用程序的确切方式一样。


推荐阅读