为什么WPF ComboBox没有正确反映绑定值?

时间:2012-04-09 20:23:06

标签: c# wpf xaml data-binding

我有一个ComboBox,其属性ItemsSource和SelectedValue绑定到模型。有时,模型需要将所选项目调整为不同的项目,但是当我在模型中执行时,模型值不会反映在视图中,即使正确设置了SelectedValue(使用snoop和SelectionChanged进行检查)事件处理程序)。

为了说明问题,这里有一个简单的xaml:

<Window x:Class="WpfApplication1.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="MainWindow" Height="350" Width="525" DataContext="{Binding RelativeSource={RelativeSource Self}}">
    <Grid>
      <ComboBox Height="25" Width="120" SelectedValue="{Binding SelectedValue}" SelectedValuePath="Key" ItemsSource="{Binding PossibleValues}" DisplayMemberPath="Value"/>
   </Grid>
</Window>

以下是模型:

using System.Collections.Generic;
using System.Windows;
using System.ComponentModel;

namespace WpfApplication1
{
   public partial class MainWindow : Window, INotifyPropertyChanged
   {
      int m_selectedValue = 2;
      Dictionary<int, string> m_possibleValues = new Dictionary<int, string>() { { 1, "one" }, { 2, "two" }, { 3, "three" }, {4,"four"} };

      public int SelectedValue
      {
         get { return m_selectedValue; }
         set 
         {
            if (value == 3)
            {
               m_selectedValue = 1;
            }
            else
            {
               m_selectedValue = value;
            }
            PropertyChanged(this, new PropertyChangedEventArgs("SelectedValue"));
         }
      }
      public Dictionary<int, string> PossibleValues
      {
         get { return m_possibleValues; }
         set { m_possibleValues = value; }
      }


      public MainWindow()
      {
         InitializeComponent();
      }

      public event PropertyChangedEventHandler PropertyChanged;
   }
}

我预计行为如下:

  1. 最初,选择了两个
  2. 选择“一个” - &gt; ComboBox显示“一个”
  3. 选择“两个” - &gt; ComboBox显示“两个”
  4. 选择“三个” - &gt; ComboBox显示“一个
  5. 选择“四” - &gt; ComboBox显示“四个”
  6. 然而,在#4中,显示“三个”。为什么?模型中的值更改为1(“1”),但视图仍显示3(“3”)。

    我通过显式更新SelectionChanged事件处理程序中的绑定目标找到了一种解决方法,但这似乎是错误的。还有另一种方法可以达到这个目的吗?

2 个答案:

答案 0 :(得分:5)

当您选择一个项目时,绑定引擎将更新模型,并且会在调用属性设置器期间忽略它刚刚更改的属性的任何PropertyChanged。如果没有,事情可能会进入无限循环。

系统假定setter不会更改传递的值。这里的解决方法很简单:在绑定上使用Dispatcher.BeginInvoke()或设置IsAsync=True以获取您正在寻找的行为。但是,我个人强烈反对你这样做。不仅因为它引入了一大堆与时间相关的新问题,而且主要是因为它是一个不应该存在的问题的解决方法。

根本不要这样做。它不仅适用于WPF:更改setter的任何内容都可以预期它刚设置的值被正确设置也不会引发异常。更改setter内部的值是违反直觉的,可能会在以后咬你。此外,对于您的应用程序的用户而言,看到组合框忽略其选择可能会非常令人不安。

如果您不喜欢传递的值,请抛出异常并使用Binding.ValidatesOnExceptions告诉WPF它是否是预期的。如果您不希望用户选择此值,请不要将其放在列表中。如果由于其他业务规则而无法有条件地使用该项目,请动态过滤列表或应用触发器。

答案 1 :(得分:2)

在绑定到SelectedValue控件的SelectedItem和/或ComboBox属性时,我在控件的正确更新方面遇到了持续存在的问题。

要解决这个问题,我必须绑定到SelectedIndex属性。在我的ViewModel中处理'index'属性更麻烦,但它解决了ComboBox没有更新的问题。