6

我有一个 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. 选择“一”-> ComboBox 显示“一”
  3. 选择“二”-> ComboBox 显示“二”
  4. 选择“三” -> ComboBox 显示“
  5. 选择“四”-> ComboBox 显示“四”

然而,在#4 中,显示“三”。为什么?模型中的值更改为 1(“一”),但视图仍显示 3(“三”)。

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

4

2 回答 2

6

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

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

根本不要这样做。它不仅适用于 WPF:任何更改 setter 的东西都可以期望它刚刚设置的值被正确设置,不会引发异常。更改设置器中的值是反直觉的,以后可能会咬你。另外,对于您的应用程序的用户来说,看到组合框忽略其选择可能会非常令人不安。

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

于 2012-04-09T21:58:13.177 回答
2

SelectedValue在绑定到控件的和/或SelectedItem属性时,我一直存在正确更新ComboBox控件的问题。

为了解决这个问题,我不得不绑定到SelectedIndex属性。在我的 ViewModel 中处理“索引”属性更麻烦,但它解决了ComboBox不更新的问题。

于 2012-04-09T21:30:50.410 回答