我有一个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。
那么有一种简单的方法可以解决这个问题吗?
答案 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; }
}
}