WPF MVVM ComboBox SelectedItem 或 SelectedValue 不起作用
Posted
技术标签:
【中文标题】WPF MVVM ComboBox SelectedItem 或 SelectedValue 不起作用【英文标题】:WPF MVVM ComboBox SelectedItem or SelectedValue not working 【发布时间】:2010-10-14 09:56:41 【问题描述】:更新
经过一番调查。问题似乎是 SelectedValue/SelectedItem 在项目源完成加载之前发生。如果我坐在断点处等待几秒钟,它会按预期工作。不知道我将如何解决这个问题。
结束更新
我有一个在 WPF 中使用 MVVM 和 ComboBox 的应用程序。下面是 ViewModel 示例。我遇到的问题是,当我们离开页面并迁移回 ComboBox 时,没有选择当前选中的值。
查看模型
public class MyViewModel
private MyObject _selectedObject;
private Collection<Object2> _objects;
private IModel _model;
public MyViewModel(IModel model)
_model = model;
_objects = _model.GetObjects();
public Collection<MyObject> Objects
get
return _objects;
private set
_objects = value;
public MyObject SelectedObject
get
return _selectedObject;
set
_selectedObject = value;
为了这个例子,假设 MyObject 有两个属性(Text 和 Id)。我的 ComboBox 的 XAML 如下所示。
XAML
<ComboBox Name="MyComboBox" Height="23" Width="auto"
SelectedItem="Binding Path=SelectedObject,Mode=TwoWay"
ItemsSource="Binding Objects"
DisplayMemberPath="Text"
SelectedValuePath="Id">
无论我以哪种方式配置它,当我返回页面并重新组装对象时,ComboBox 都不会选择该值。该对象通过属性中的 get 返回正确的对象。
我不确定这是否只是 ComboBox 和 MVVM 模式工作方式的问题。我们正在做的文本框绑定工作正常。
【问题讨论】:
【参考方案1】:设置 IsSynchronizedWithCurrentItem="True"
对我有用!
【讨论】:
谢谢维姆。已经为此苦苦挣扎了 40 分钟。 一直在努力解决这个问题。这对我来说非常有效。谢谢:) 在这个小问题上花了太长时间,这对我有用!非常感谢! 简单而完美。花了大约4个小时才找到这个。谢谢!【参考方案2】:您是否尝试过在视图模型中实现INotifyPropertyChanged
,然后在设置SelectedItem
时引发PropertyChanged
事件?
如果这本身不能解决问题,那么您将能够在导航回页面时自己手动引发PropertyChanged
事件,这足以让 WPF 重绘自身并显示正确的选择项目。
【讨论】:
我们的实际实现继承自实现 INotifyPropertyChanged 的 ViewBase 类。我正在提高 PropertyChange,但组合框仍为空白。 如果我放入断点并稍等片刻,效果会更好。我认为我的 selecteditem 在组合框的绑定完成之前就已经设置好了。呃。 我忘记在导航回页面时引发 PorpertyChanged 事件。【参考方案3】:您需要将 ItemsSource 属性放在 SelectedItem 属性之前。几天前我看到一个博客提到了这个问题。
【讨论】:
非常感谢,我一直在努力找出我的 XAML 或视图模型做错了什么,而一直都是这样。 波什,很好的答案! 不适用于 VS 15.8.1,@rp7799 的回答完美。【参考方案4】:我遇到过类似的问题,通过确保我正确实施 IEquatable 解决了这个问题。当绑定发生时,它会尝试查看对象是否匹配,因此请确保您正确实施相等性检查。
【讨论】:
我现在也试过了。我在 MyObject 上实现了 IEquatable 并匹配 .id 字段并返回 true。那也没有效果。如果我添加一个绑定到属性的文本框,它会在返回页面时显示正确的信息。【参考方案5】:在这种情况下,selecteditem绑定不起作用,因为对象的hash id不同。
一种可能的解决方案是:
根据选中项id,恢复itemsource集合上的对象,并设置选中项属性为。
例子:
<ctrls:ComboBoxControlBase SelectedItem="Binding Path=SelectedProfile, Mode=TwoWay" ItemsSource="Binding Path=Profiles, Mode=OneWay" IsEditable="False" DisplayMemberPath="Name" />
绑定到 ItemSource 的属性是:
public ObservableCollection<Profile> Profiles
get return this.profiles;
private set profiles = value; RaisePropertyChanged("Profiles");
与SelectedItem绑定的属性是:
public Profile SelectedProfile
get return selectedProfile;
set
if (this.SelectedUser != null)
this.SelectedUser.Profile = value;
RaisePropertyChanged("SelectedProfile");
恢复码是:
[Command("SelectionChanged")]
public void SelectionChanged(User selectedUser)
if (selectedUser != null)
if (selectedUser is User)
if (selectedUser.Profile != null)
this.SelectedUser = selectedUser;
this.selectedProfile = this.Profiles.Where(p => p.Id == this.SelectedUser.Profile.Id).FirstOrDefault();
MessageBroker.Instance.NotifyColleagues("ShowItemDetails");
希望对你有帮助。 我花了很多时间寻找答案,但我找不到。
【讨论】:
【参考方案6】:离开当前页面时,与ComboBox
的ItemsSource
属性关联的CollectionView
将被清除。因为ComboBox
IsSyncronizedWithCurrent
属性默认为真,SelectedItem
和SelectedValue
属性被重置。
这似乎是绑定中的内部数据类型问题。正如上面其他人建议的那样,如果您使用SelectedValue
而不是通过绑定到视图模型上的 int 属性,它将起作用。
一个捷径是覆盖 MyObject 上的 Equals
运算符,以便在比较两个 MyObject 时,比较实际的 Id
属性。
另一个提示:如果您确实重组了您的视图模型并使用SelectedValue
,请仅在SelectedValuePath=Id
时使用它,其中Id
是int
。如果使用字符串键,则绑定到ComboBox
的Text
属性而不是SelectedValue
。
【讨论】:
我遇到了一个非常相似的问题。问题的原因是 ItemsSource 绑定到 RelativeSource,而 SelectedItem/SelectedValue 直接绑定到 DataContext。我能够解决它,但问题的原因尚不清楚。此处为遇到相同问题的任何人进行了描述:***.com/questions/33739513/…【参考方案7】:我之前也注意到了这种行为。我注意到 SelectedIndex 属性不会导致相同的错误。如果您可以重新构建 ViewModel 以公开所选项目的索引并绑定到该索引,那么您应该很高兴。
【讨论】:
【参考方案8】:我遇到了同样的问题。事情是。所选项目不知道它应该使用集合中的哪个对象。所以你必须对选中的项目说使用集合中的项目。
public MyObject SelectedObject
get
Objects.find(x => x.id == _selectedObject.id)
return _selectedObject;
set
_selectedObject = value;
我希望这会有所帮助。
【讨论】:
【参考方案9】:对于这个问题,我有一个非常简单的答案。首先将以下代码添加到视图 IsSynchronizedWithCurrentItem="True"。
接下来,当您将 ViewModel 中的新对象分配给该属性时,应将 SelectedObject 保存到该属性而不是私有成员。
viewmodel 属性应该如下所示
public Role SelectedObject
get return object;
set
if (value != null)
if (!object.Equals(value))
object = value;
OnPropertyChanged(() => SelectedObject );
这应该可以解决问题。
【讨论】:
【参考方案10】:我通过在 UserControl_Loaded 事件中添加调度程序解决了这个问题
Dispatcher.BeginInvoke(DispatcherPriority.Loaded, new Action(() =>
combobox.SelectedIndex = 0;
));
【讨论】:
【参考方案11】:IsSyncronizedWithCurrent=False 将使它工作。
【讨论】:
【参考方案12】:我与这个问题斗争了一段时间。在我的情况下,我使用复杂类型(列表)作为项目源,并使用 KeyType 作为选定值。在加载事件中,KeyType 被设置为 null。这导致一切都崩溃了。键更改时,不会更新任何子元素。事实证明,当我添加检查以确保 KeyType 的建议值不为空时,一切都按预期工作。
#region Property: SelectedKey
// s.Append(string.Format("SelectedKey : 0 " + Environment.NewLine, SelectedKey.ToString()));
private KeyType _SelectedKey = new KeyType();
public KeyType SelectedKey
get return _SelectedKey;
set
if(value != null )
if (!_SelectedKey.Equals(value))
_SelectedKey = value;
OnPropertyChanged("SelectedKey");
#endregion SelectedKey
【讨论】:
【参考方案13】:SelectedValuePath
和SelectedValue
的类型必须完全相同。
例如,如果SelectedValuePath
的类型是Int16
,而绑定到SelectedValue
的属性类型是int
,它将不起作用。
我花了好几个小时才找到它,这就是为什么我在问了这么久之后才在这里回答的原因。也许像我这样有同样问题的可怜人可以看到它。
【讨论】:
【参考方案14】:我遇到了显示颜色列表 ( List
修复是覆盖 ComboBox (Brush) 中选择的类型的 Equals(object obj) 方法,这并不简单,因为 Brush 是密封的。 所以我写了一个类 EqualityBrush 包含一个 Brush 并实现 Equals:
public class EqualityBrush
public SolidColorBrush Brush get; set;
public override bool Equals(object o)
if (o is EqualityBrush)
SolidColorBrush b = ((EqualityBrush)o).Brush;
return b.Color.R == this.Brush.Color.R && b.Color.G == this.Brush.Color.G && b.Color.B == this.Brush.Color.B;
else
return false;
使用我的新 EqualityBrush 类的列表而不是普通的 Brush 类解决了问题!
我的组合框 XAML 如下所示:
<ComboBox ItemsSource="Binding BuerkertBrushes" SelectedItem="Binding Brush, Mode=TwoWay" Width="40">
<ComboBox.Resources>
<DataTemplate DataType="x:Type tree:EqualityBrush">
<Rectangle Width="20" Height="12" Fill="Binding Brush"/>
</DataTemplate>
</ComboBox.Resources>
</ComboBox>
请记住,我在 ViewModel 中的“Brush”属性现在必须是 EqualityBrush 类型!
【讨论】:
【参考方案15】:这可能是您将 DataContext 应用于页面的方式。在 WPF 中,每次导航到页面时,所有内容都会重新初始化,调用构造函数,加载方法,等等。因此,如果您在 View 中设置 DataContext ,您无疑会吹走用户选择的 SelectedItem。为了避免这种情况,请使用页面的 KeepAlive 属性。
<Page KeepAlive="True" ...>
...
</Page>
这将导致在导航回您已经访问过的页面时仅触发 Loaded 事件。因此,您需要确保在 Initialize(外部或构造函数内)而不是 Load 上设置 DataContext。
但是,这仅适用于该页面的实例。如果您导航到该页面的新实例,它将再次调用它的构造函数。
【讨论】:
因为我使用的是 MVVM 模式,所以 XAML 页面没有隐藏代码。没有加载事件。一切都是通过数据绑定发生的。就此而言,模板是由 DataTemplateSelector 基于视图模型中的属性选择的。【参考方案16】:ComboBox.SelectionBoxItem.ToString()
【讨论】:
【参考方案17】:使用 Loaded 事件:
private void cmb_Loaded(object sender, RoutedEventArgs e)
if (cmb.Items.Count > 0) cmb.SelectedIndex = 0;
它对我有用。
【讨论】:
【参考方案18】:您还可以将 SelectedIndex 绑定到 ViewModel 中的属性并以这种方式操作 SelectedItem:
public int SelectedIndex
get return _selectedIndex;
set
_selectedIndex = value;
OnPropertyChanged();
在您的 XAML 中:
<ComboBox SelectedIndex="Binding SelectedIndex,Mode=TwoWay" ... >
【讨论】:
以上是关于WPF MVVM ComboBox SelectedItem 或 SelectedValue 不起作用的主要内容,如果未能解决你的问题,请参考以下文章
WPF MVVM 将 ComboBox 绑定到 Datagrid 选定项
MVVM WPF ComboBox SelectedValue 不是确切值
WPF MVVM ComboBox SelectedItem 或 SelectedValue 不起作用
WPF中ComboBox控件的SelectedItem和SelectedValue的MVVM绑定
WPF MVVM ComboBox(in DataGrid)触发事件SelectionChanged, 通知到ViewModel.