我想绑定到XAML中的ObservableCollection
并在那里应用分组。原则上,这很好。
<UserControl.Resources>
<CollectionViewSource x:Key="cvs" Source="{Binding Path=TestTemplates}">
<CollectionViewSource.SortDescriptions>
<scm:SortDescription PropertyName="Title"/>
</CollectionViewSource.SortDescriptions>
<CollectionViewSource.GroupDescriptions>
<PropertyGroupDescription PropertyName="TestCategory"/>
</CollectionViewSource.GroupDescriptions>
</CollectionViewSource>
</UserControl.Resources>
然后数据绑定表达式变为ItemsSource="{Binding Source={StaticResource ResourceKey=cvs}}"
而不是ItemsSource="{Binding Path=TestTemplates}"
。
起初,一切看起来都很酷,直到我想从视图模型中刷新UI。问题是,CollectionViewSource.GetDefaultView(TestTemplates)
返回的视图与应用分组的XAML视图不同。因此,我无法设置选择或做任何有用的事情。
我可以通过将列表再次直接绑定到视图模型的属性并在代码隐藏中设置分组来修复它。但我对这个解决方案并不满意。
private void UserControlLoaded(object sender, RoutedEventArgs e)
{
IEnumerable source = TemplateList.ItemsSource;
var cvs = (CollectionView)CollectionViewSource.GetDefaultView(source);
if (cvs != null)
{
cvs.SortDescriptions.Add(new SortDescription("Title", ListSortDirection.Ascending));
cvs.GroupDescriptions.Add(new PropertyGroupDescription("TestCategory"));
}
}
我认为,其原因已经是given by John Skeet here。
尽管如此,我希望应该有办法获得正确的观点。我错了吗?
答案 0 :(得分:6)
你不能这样做吗?
var _viewSource = this.FindResource("cvs") as CollectionViewSource;
如果数据已连接,我认为会有更新的视图。
答案 1 :(得分:5)
我倾向于只从VM公开集合视图,而不是让视图定义它:
public ICollection<Employee> Employees
{
get { ... }
}
public ICollectionView EmployeesView
{
get { ... }
}
这样,您的VM可以完全控制对视图的显示内容。例如,它可以更改排序顺序以响应某些用户操作。
答案 2 :(得分:1)
基于J. Lennon's回答找到了一种方法。如果我使用my命令传递了可以访问资源的内容,那么我可以在那里查找CollectionViewSource
。
在XAML中(如上所述CollectionViewResource
):
<Button Command="{Binding Command}" CommandParameter="{Binding RelativeSource={RelativeSource Self}}">Do it!</Button>
在VM代码中:
private void Execute(object parm)
{
var fe = (FrameworkElement)parm;
var cvs = (CollectionViewSource)fe.FindResource("cvs");
cvs.View.Refresh();
}
Execute
是赋予RelayCommand的那个。
这会回答这个问题,但我不太喜欢。意见?