How to preserve TwoWay binding of CurrentItem when databinding to CollectionViewSource in ComboBox

前端 未结 4 1811
粉色の甜心
粉色の甜心 2021-01-04 03:07

Lets say we got a simple VM class

public class PersonViewModel : Observable
    {
        private Person m_Person= new Person(\"Mike\", \"Smith\");

                 


        
相关标签:
4条回答
  • 2021-01-04 03:48

    The TwoWay binding works as it should, but ComboBox doesn't update itself on the UI when you set SelectedItem or SelectedIndex from code. If you want this functionality, just extend ComboBox and listen to SelectionChanged inherited from Selector or if you only want to set an initial selection, do it on Loaded.

    0 讨论(0)
  • 2021-01-04 03:50

    I highly recommend the use of ComboBoxExtensions by Kyle McClellan of Microsoft, found here.

    You can declare a datasource for your ComboBox in XAML - and it's far more flexible and usable in async modes.

    Basically the solution, largely, is NOT to use CollectionViewSource for ComboBoxes. You can do the sorting on the server side query.

    0 讨论(0)
  • 2021-01-04 03:51

    There are 2 problems ganging up on you, but you have highlighted a real problem with using CollectionViewSource with a ComboBox. I am still looking for alternatives to fix this in a "better way", but your setter fix avoids the problem for good reason.

    I have reproduced your example in full detail to confirm the problem and a theory about the cause.

    ComboBox binding to CurrentPerson does not use the equals operator to find a match IF YOU USE SelectedValue INSTEAD OF SelectedItem. If you breakpoint your override bool Equals(object obj) you will see it is not hit when you change the selection.

    By changing your setter to the following, you are finding a specific matching object, using your Equals operator, so a subsequent value compare of 2 objects will work.

    set
    {
        if (m_AvailablePersons.Contains(value)) {
           m_Person = m_AvailablePersons.Where(p => p.Equals(value)).First();
        }
        else throw new ArgumentOutOfRangeException("value");
        NotifyPropertyChanged("CurrentPerson");
    
    }
    

    Now the really interesting result:

    Even if you change your code to use SelectedItem, it will work for a normal binding to the list but still fail for any binding to the sorted view!

    I added debug output to the Equals method and even though matches were found, they were ignored:

    public override bool Equals(object obj)
    {
        if (obj is Person)
        {
            Person other = obj as Person;
            if (other.Firstname == Firstname && other.Surname == Surname)
            {
                Debug.WriteLine(string.Format("{0} == {1}", other.ToString(), this.ToString()));
                return true;
            }
            else
            {
                Debug.WriteLine(string.Format("{0} <> {1}", other.ToString(), this.ToString()));
                return false;
            }
        }
        return base.Equals(obj);
    }
    

    My conclusion...

    ...is that behind the scenes the ComboBox is finding a match, but because of the presence of the CollectionViewSource between it and the raw data it is then ignoring the match and comparing objects instead (to decide which one was selected). From memory a CollectionViewSource manages its own current selected item, so if you do not get an exact object match it will never work using a CollectionViewSource with a ComboxBox.

    Basically your setter change works because it guarantees an object match on the CollectionViewSource, which then guarantees an object match on the ComboBox.

    Test code

    The full test code is below for those that want to play (sorry about the code-behind hacks, but this was just for testing and not MVVM).

    Just create a new Silverlight 4 application and add these files/changes:

    PersonViewModel.cs

    using System;
    using System.Collections.Generic;
    using System.Collections.ObjectModel;
    using System.ComponentModel;
    using System.Diagnostics;
    using System.Linq;
    namespace PersonTests
    {
        public class PersonViewModel : INotifyPropertyChanged
        {
            private Person m_Person = null;
    
            private readonly ObservableCollection<Person> m_AvailablePersons =
                new ObservableCollection<Person>(new List<Person> {
                   new Person("Mike", "Smith"),
                   new Person("Jake", "Jackson"),                                                               
                   new Person("Anne", "Aardvark"),                                                               
            });
    
            public ObservableCollection<Person> AvailablePersons
            {
                get { return m_AvailablePersons; }
            }
    
            public Person CurrentPerson
            {
                get { return m_Person; }
                set
                {
                    if (m_Person != value)
                    {
                        m_Person = value;
                        NotifyPropertyChanged("CurrentPerson");
                    }
                }
    
                //set // This works
                //{
                //  if (m_AvailablePersons.Contains(value)) {
                //     m_Person = m_AvailablePersons.Where(p => p.Equals(value)).First();
                //  }
                //  else throw new ArgumentOutOfRangeException("value");
                //  NotifyPropertyChanged("CurrentPerson");
                //}
            }
    
            private void NotifyPropertyChanged(string name)
            {
                if (PropertyChanged != null)
                {
                    PropertyChanged(this, new PropertyChangedEventArgs(name));
                }
            }
    
            public event PropertyChangedEventHandler PropertyChanged;
        }
    
        public class Person
        {
            public string Firstname { get; set; }
            public string Surname { get; set; }
    
            public Person(string firstname, string surname)
            {
                this.Firstname = firstname;
                this.Surname = surname;
            }
    
            public override string ToString()
            {
                return Firstname + "  " + Surname;
            }
    
            public override bool Equals(object obj)
            {
                if (obj is Person)
                {
                    Person other = obj as Person;
                    if (other.Firstname == Firstname && other.Surname == Surname)
                    {
                        Debug.WriteLine(string.Format("{0} == {1}", other.ToString(), this.ToString()));
                        return true;
                    }
                    else
                    {
                        Debug.WriteLine(string.Format("{0} <> {1}", other.ToString(), this.ToString()));
                        return false;
                    }
                }
                return base.Equals(obj);
            }
        }
    }
    

    MainPage.xaml

    <UserControl x:Class="PersonTests.MainPage"
        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:scm="clr-namespace:System.ComponentModel;assembly=System.Windows" mc:Ignorable="d"
        d:DesignHeight="300" d:DesignWidth="400">
        <UserControl.Resources>
            <CollectionViewSource x:Key="PersonsViewSource" Source="{Binding AvailablePersons}">
                <CollectionViewSource.SortDescriptions>
                    <scm:SortDescription PropertyName="Surname" Direction="Ascending" />
                </CollectionViewSource.SortDescriptions>
            </CollectionViewSource>
        </UserControl.Resources>
        <StackPanel x:Name="LayoutRoot" Background="LightBlue" Width="150">
            <!--<ComboBox ItemsSource="{Binding AvailablePersons}"
                  SelectedItem="{Binding Path=CurrentPerson, Mode=TwoWay}" />-->
            <ComboBox ItemsSource="{Binding Source={StaticResource PersonsViewSource}}"
              SelectedItem="{Binding Path=CurrentPerson, Mode=TwoWay}" />
            <Button Content="Select Mike Smith" Height="23" Name="button1" Click="button1_Click" />
            <Button Content="Select Anne Aardvark" Height="23" Name="button2" Click="button2_Click" />
        </StackPanel>
    </UserControl>
    

    MainPage.xaml.cs

    using System.Windows;
    using System.Windows.Controls;
    
    namespace PersonTests
    {
        public partial class MainPage : UserControl
        {
            public MainPage()
            {
                InitializeComponent();
                this.DataContext = new PersonViewModel();
            }
    
            private void button1_Click(object sender, RoutedEventArgs e)
            {
                (this.DataContext as PersonViewModel).CurrentPerson = new Person("Mike", "Smith");
            }
    
            private void button2_Click(object sender, RoutedEventArgs e)
            {
                (this.DataContext as PersonViewModel).CurrentPerson = new Person("Anne", "Aardvark");
    
            }
        }
    }
    
    0 讨论(0)
  • 2021-01-04 03:53

    Have you considered using a CollectionView and set IsSynchronizedWithCurrentItem on the combobox?

    That is what I would do - instead of having your CurrentPerson property you have the selected person on your collectionView.CurrentItem and the combobox following currentitem on the collectionview.

    I Have used collectionview with sorting and grouping without problems - and you get a nice decoupling from ui with it.

    I would move the collectionview to code and bind to it there

    public ICollectionView AvailablePersonsView {get;private set;}

    in ctor:

    AvailablePersonsView = CollectionViewSource.GetDefaultView(AvailablePersons)

    0 讨论(0)
提交回复
热议问题