我有一个ComboBox,其ItemSource绑定到ObservableCollection,其SelectedItem属性绑定到T的属性。两个属性都存在于我的视图模型中。一切都很好,除了一个导致我撕掉头发的问题。当我在绑定到SelectedItem的对象上设置属性时,ComboBox绑定停止工作。 VS 2012输出窗口不报告任何绑定错误,也不会抛出任何异常。
XAML
<ComboBox ItemSource="{Binding Path=Connections, Mode=OneWay}"
SelectedItem="{Binding Path=SelectedConnection, Mode=TwoWay}"
DisplayMemberPath="Name"/>
<TextBlock Text="{Binding Path=ConnectionName, Mode=TwoWay}" />
<TextBlock Text="{Binding Path=ConnectionGroup, Mode=TwoWay}" />
查看模型
private void SaveChanges()
{
this.SelectedConnection.Name = this.ConnectionName;
this.SelectedConnection.Group = this.ConnectionGroup;
// -- Save the changes
}
现在所有这些都很好。但是一旦SaveChanges()完成,ComboBox开始表现得很有趣。看起来一切都很好,但如果你试图改变选择没有任何反应;选择相同的项目并且不激活“SelectedConnection”的setter。我尝试系统地在SaveChanges()中注释掉代码,如果我只留下一行在SelectedConnection上设置任何属性,我就会发现这种失败。
我还在控件后面的代码中添加了一个事件处理程序来处理ComboBox上的SelectionChanged事件。发生中断后,每当我尝试通过UI或在代码中更改值时,事件处理程序会立即连续触发两次。第一个事件args'RelectedItems设置了一个值,这是我刚刚选择和期望的值。但是事件会立即再次触发,事件args的RemovedItems集合中只有一个项目。该集合还包含我刚刚选择的项目。
我尝试将SelectedConnection属性设置为null,然后在方法完成后,设置回相关对象。没有运气。
我还看到一篇关于在我的XAML标记上的SelectedItem属性之后移动ItemSource属性的旧帖子,但也没有效果。
我正在努力避免创建一个影子列表,并尝试保持所有内容同步,因为这将是我的下一次尝试,但这样做只是变得如此简单。任何想法将不胜感激。
如果重要,这是构建到4.5框架的Visual Studio 2012。
修改
我现在也尝试在设置其属性后从ObservableCollection中完全删除修改后的对象。它仍然显示它好像是选定的值(它的名称在ComboBox中立即可见),但它不再显示在下拉列表中。即使它不再存在于ItemSource中,它仍然不会允许自己被“取消选择”。
**更新**
我将有问题的方法更改为完全取消绑定到ItemSource的ObservableCollection,然后在进程结束后将其恢复,如下所示:
private void SaveChanges()
{
var selected = this.SelectedConnection;
var name = this.ConnectionName; // -- bound to fields on the view
var group = this.ConnectionGroup;
this.Connections = null;
this.RaisePropertyChanged("Connections"); // -- SelectionChanged fires as expected
selected.Name = name;
selected.Group = group;
this._repository.SaveChanges();
this.Connections = new ObservableCollection<Connection>(this._repository.AllConnections);
this.RaisePropertyChanged("Connections"); // -- SelectionChanged event doesn't fire
this.SelectedConnection = selected; // -- SelectionChanged event doesn't fire
}
完成此操作后,ComboBox再次允许我按预期更改值。但是,所选项目实际上并未显示为已选中(即使它已在视图模型中正确设置)。所以是的,这是一个非常丑陋的解决方法,它也创造了自己的问题。
答案 0 :(得分:2)
刚刚遇到同样的问题,我想我已经找到了正在做的事情。
它只是在我做了一些最近的更改(代码分析建议)之后才出现为我有IComparable的类添加equals / hashcode / operator重载,并且在一个ObservableCollection中用于组合框。我根据对象的描述性字段制作了哈希码。但是,该字段可以由用户修改,并且在用户修改该值之后,组合框断开,无法再取消选择该项目。
鉴于此对象中没有不变的字段,因此无法根据对象的属性创建固定的哈希码。但是,如果我完全删除哈希码覆盖(或将其设置为固定的东西,如type.GetHashCode),一切都会再次起作用。
因此,请检查您的哈希码是否基于您正在更改的值。
答案 1 :(得分:0)
我放弃了我原先设想的方式。我没有直接修改ItemSource集合中的对象,而是将ItemSource集合作为键/值对的列表,并根据所选对的键从源列表中提取我需要的对象。我能够毫无问题地编辑该对象(如预期的那样)。它现在有效,我只是在那里添加了一层我希望避免的代码。用户界面还允许我更新对的价值方面,以反映基础模型的“名称”字段的更改,而不会给我以前的相同问题。我只把它算作一个小胜利并继续前进。