WPF使用ICollectionView过滤ObservableCollection

时间:2016-12-13 11:45:37

标签: c# wpf mvvm

我有一个ObservableCollection,一个ICollectionView,一个过滤器和两个TextBox。 如果我只使用一个TextBox过滤器,它可以正常工作。如果我添加另一个文本框并将过滤器绑定到第二个TB,则过滤正在挣扎。

我已尝试使用两个过滤器(不同的名称 - 相同的功能),这也没有效果。

我认为它可能与ObservableCollection有关..

这是我的过滤器:

this.AllUsers.Filter = i =>
{
      if (string.IsNullOrEmpty(this.SearchUsername)) return true;

      User u = i as User;
      return u.Name.StartsWith(this.SearchUsername);
};

我的ICollectionView包含来自ObservableCollection的数据:

public ICollectionView AllUsers
{
    get
    {
       return CollectionViewSource.GetDefaultView(UserSource);
    }
}

我的ObservableCollection:

public ObservableCollection<User> UserSource
{
    get
    {
        return _UserSource;
    }
    set
    {
        _UserSource = value; OnPropertyChanged();
    }

}

我在我的字符串属性SearchUsername中使用AllUsers.Refresh();更新了视图。

ObservableCollection绑定到ListBox,字符串Property绑定到TextBox。

第二个TextBox也一样。相同的ObservableCollection绑定到不同的ListBox,字符串属性(UserName)绑定到第二个TextBox。

那么有一种简单的方法可以解决这个问题吗?

2 个答案:

答案 0 :(得分:0)

这是一个使用Collection View Source

的决斗过滤器的示例

查看

<Window x:Class="WpfApplication1.MainWindow"
        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:local="clr-namespace:WpfApplication1"
        mc:Ignorable="d"
        Title="MainWindow" Height="350" Width="525">
    <Window.DataContext>
        <local:ViewModel x:Name="vm" />
    </Window.DataContext>

    <DockPanel >
        <TextBox DockPanel.Dock="Top" Text="{Binding Search1}"/>
        <TextBox DockPanel.Dock="Top" Text="{Binding Search2}"/>
        <ListView ItemsSource="{Binding View}">
            <ListView.View>
                <GridView>
                    <GridViewColumn DisplayMemberBinding="{Binding Property1}" Header="Prop1"/>
                    <GridViewColumn DisplayMemberBinding="{Binding Property2}" Header="Prop1"/>
                </GridView>
            </ListView.View>
        </ListView>
    </DockPanel>
</Window>

型号:

public class DataModel
{
    public string Property1 { get; set; }
    public string Property2 { get; set; }
}

查看模型:(使用Prism和c#6)

public class ViewModel:BindableBase
{
    public ViewModel()
    {
        for (int i = 0; i < 200; i++)
        {
            Items.Add(new DataModel()
            {
                Property1 = $"{200 - i}:Prop1",
                Property2 = $"{i}:Prop2"
            });
        }
        //add filter
        CollectionViewSource.Filter += (s, e) =>
        {
            var d = e.Item as DataModel;
            if (d != null)
            {
                e.Accepted = (string.IsNullOrEmpty(Search1) || d.Property1.StartsWith(Search1))//search property 1
                            && (string.IsNullOrEmpty(Search2) || d.Property2.StartsWith(Search2));//search property 2
            }
            else
                e.Accepted = false;
        };
        CollectionViewSource.Source = Items;
    }

    public CollectionViewSource CollectionViewSource { get; } = new CollectionViewSource();
    public ICollectionView View => CollectionViewSource.View;



    private string _Search1;

    public string Search1
    {
        get { return _Search1; }
        set
        {
            if (SetProperty(ref _Search1, value))
                View.Refresh();
        }
    }

    private string _Search2;

    public string Search2
    {
        get { return _Search2; }
        set
        {
            //SetProperty defined as if value is different update, raise PropertyChanged and return true, else return false;
            if (SetProperty(ref _Search2, value))
                View.Refresh();
        }
    }

    public ObservableCollection<DataModel> Items { get; } = new ObservableCollection<DataModel>();
}

答案 1 :(得分:0)

  

ObservableCollection绑定到ListBox,字符串Property绑定到TextBox ...

您应该将ListBox的ItemsSource绑定到 ICollectionView 属性,而不是绑定到ObservableCollection属性。你实际上是过滤的前者。

此外,您应该只在视图模型中创建一个ObservableCollection和ICollectionView实例,例如在构造函数中,然后在ObservableCollection的这个单个实例中添加/删除项目:

public class ViewModel
{
    public ViewModel()
    {
        AllUsers = new ObservableCollection<User>();
        AllUsers = CollectionViewSource.GetDefaultView(UserSource);
        //...
    }

    public ObservableCollection<User> UserSource
    {
        { get; private set; }
    }

    public ICollectionView AllUsers
    {
        { get; private set; }
    }
}