我目前有几个模型对象如下:
public class ModelBaseClass : INotifyPropertyChanged
{
//Members omitted for brevity, but there is a DateTime property
//that becomes important further down (see dummy method in application controller)
}
另一个:
public class ModelCollectionClass : INotifyPropertyChanged
{
private ObservableCollection<ModelBaseClass> modelBaseClasses;
public ObservableCollection<ModelBaseClass> ModelBaseClasses
{
get { return modelBaseClasses; }
set
{
modelBaseClasses = value;
OnPropertyChanged(new PropertyChangedEventArgs("ModelBaseClasses");
}
}
private void modelBaseClasses_CollectionChanged(object sender, NotifyCollectionChangedEventArgs e)
{
OnPropertyChanged(new PropertyChangedEventArgs("ModelBaseClasses"));
}
}
然后在我的视图的一个视图模型中,我有这样的可观察集合:
private ObservableCollection<ModelBaseClass> modelBaseClasses;
public ObservableCollection<ModelBaseClass> ModelBaseClasses
{
get{ return modelBaseClasses ?? (modelBaseClasses = new ObservableCollection<ModelBaseClass>()); }
set
{
modelBaseClasses = value;
OnPropertyChanged(new PropertyChangedEventArgs("ModelBaseClasses"));
}
}
private ObservableCollection<ModelCollectionClass> modelCollectionClasses;
public ObservableCollection<ModelCollectionClass> ModelCollectionClasses
{
get { return modelCollectionClasses ?? (modelCollectionClasses = new ObservableCollection<ModelCollectionClass>()); }
set
{
modelCollectionClasses = value;
OnPropertyChanged(new PropertyChangedEventArgs("ModelCollectionClasses"));
}
}
然后在我的应用程序控制器(代表工作到视图模型)中,我有一个虚拟方法可以加载 ModelBaseClass 集合,如下所示:
for (int i = 0; i < 8; i++)
{
myViewModel.ModelBaseClasses.Add(new ModelBaseClass()
{
DateTimeProperty = Convert.ToDateTime("06/2" + i + "/2012 09:00:00 AM")
//Other properties get set as well, using the i variable...
}
}
应该注意的是,这种方法中有多个 for 循环,因此我可以处理大量数据。
接下来,我有一个方法可以获取大量对象并按日期对它们进行分组,如下所示:
myViewModel.ModelBaseClasses
.GroupBy(modelBaseClass => ((DateTime)modelBaseClass.DateTimeProperty).Date)
.Select(group => group.ToList())
.ToList()
.ForEach(
list => myiewModel.ModelBaseCollections.Add(new ModelBaseCollection() { ModelBaseClasses = new ObservableCollection<ModelBaseClass>(list) })
);
现在这就是它开始变得更加复杂的地方。在我的视图的 XAML 中,我声明了一个 ItemsControl,它的 ItemsSource 绑定到我的视图模型中的 ModelCollectionClasses 属性。它看起来像这样:
<ItemsControl ItemsSource="{Binding Path=DataContext.ModelCollectionClasses, RelativeSource={RelativeSource AncestorType={x:Type views:MyView}}}"
ItemTemplate="{StaticResource ItemsTemplate}">
<ItemsControl.ItemsPanel>
<ItemsPanelTemplate>
<StackPanel Width="{Binding ElementName=scroller, Path=ViewportWidth}"/>
</ItemsPanelTemplate>
</ItemsControl.ItemsPanel>
</ItemsControl>
然后在这个特定视图的资源字典中,我有一个 DataTemplate 声明如下:
<DataTemplate x:Key="ItemsTemplate">
<Grid DataContext="{Binding ModelBaseClasses}">
<DataGrid AutoGenerateColumns="False"
HeadersVisibility="None"
CanUserAddRows="False"
CanUserDeleteRows="False"
ItemsSource="{Binding Path=DataContext, RelativeSource={RelativeSource Mode=FindAncestor, AncestorType={x:Type Grid}}}"
SelectedItem="{Binding Path=DataContext.SelectedModelBaseClass, RelativeSource={RelativeSource Mode=FindAncestor, AncestorType={x:Type views:MyView}}, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}">
<DataGrid.Columns>
...Omitted for brevity...
</DataGrid.Columns>
</DataGrid>
</Grid>
</DataTemplate>
现在,综上所述,我得到了我最初想要的数据的所需可视化表示。但是,我现在遇到的问题是当我想删除 ModelBaseClass 对象之一时。因为 DataGrid 的 ItemsSource 被绑定到 ModelCollectionClass 的 ModelBaseClasses 属性,而不是我的视图模型中的 ModelBaseClasses,所以当我从视图模型中找到的 ModelCollectionClass 中删除 ModelBaseClass 时,我需要以某种方式通知 ModelCollectionClass此更改也会使 UI 正确更新。
我已经看到了一些其他示例,其中有些人创建了包装类来同步模型集合和视图模型集合,但是我很难用我自己的应用程序来实现这样的事情。任何见解或有用的提示/想法将不胜感激。