我简化了一个DataGrid,看起来像这样:
<DataGrid AutoGenerateColumns="False"
ItemsSource="{Binding Parts}"
SelectedItem="{Binding SelectedPart}" >
<DataGrid.Columns>
<DataGridTextColumn Header="Name" Binding="{Binding Path=Name, Mode=TwoWay}" />
<DataGridTemplateColumn Header="PartType" >
<DataGridTemplateColumn.CellEditingTemplate>
<DataTemplate>
<ComboBox ItemsSource="{Binding RelativeSource={RelativeSource AncestorType={x:Type DataGrid}}, Path=DataContext.PartTypes}"
SelectedItem="{Binding PartType, Mode=TwoWay, UpdateSourceTrigger=LostFocus}" />
</DataTemplate>
</DataGridTemplateColumn.CellEditingTemplate>
<DataGridTemplateColumn.CellTemplate>
<DataTemplate>
<TextBlock Text="{Binding PartType}" />
</DataTemplate>
</DataGridTemplateColumn.CellTemplate>
</DataGridTemplateColumn>
</DataGrid.Columns>
</DataGrid>
我的ViewModel看起来像这样:
public class PartListViewModel
{
private ObservableCollection<Part> _parts;
public ObservableCollection<Part> Parts
{
get { return _parts; }
set
{
_parts = value;
OnPropertyChanged("Parts");
}
}
private Part _selectedPart;
public Part SelectedPart
{
get { return _selectedPart; }
set
{
_selectedPart = value;
OnPropertyChanged("SelectedPart");
}
}
}
现在,我想更改数据网格单元,将其立即存储到数据库中。如何在MVVM中做到这一点?
当前,我监听DataGrid的OnCellEditEnding
事件,并将记录保存在后面的代码中。但这很丑:
private void DataGrid_OnCellEditEnding(object sender, DataGridCellEditEndingEventArgs e)
{
var viewModel = (PartListViewModel) DataContext;
viewModel.SavePart((Part) e.Row.Item);
}
答案 0 :(得分:2)
您可以执行以下操作。
(您需要添加对System.Windows.Interactivity
,you can download the assemblies from here的引用)
然后在XAML中添加对名称空间的引用。
xmlns:i="http://schemas.microsoft.com/expression/2010/interactivity"
<i:Interaction.Triggers>
<i:EventTrigger EventName="CellEditEnding">
<i:InvokeCommandAction Command="{Binding SomeCommand}"/>
</i:EventTrigger>
</i:Interaction.Triggers>
答案 1 :(得分:1)
其中一种方法是使用Prism库随附的事件代理(例如EventAggregator)。
将每个Part实体(“模型”)添加到Parts集合中时,可以使用其自己的视图模型(PartViewModel)对其进行包装,然后可以在其组成的Part中侦听属性更新,然后通过该事件发布更新消息经纪人。然后,您可以使用一个服务来侦听(订阅)该消息并保存已发布的有效负载(有效负载可以是已修改的Part)。
答案 2 :(得分:1)
您可以为视图模型中的所有PropertyChanged
对象处理Part
事件:
public class PartListViewModel
{
private ObservableCollection<Part> _parts;
public ObservableCollection<Part> Parts
{
get { return _parts; }
set
{
if(_parts != null) _parts.CollectionChanged -= OnCollectionChanged;
_parts = value;
if (_parts != null) _parts.CollectionChanged += OnCollectionChanged;
//OnPropertyChanged("Parts");
}
}
private void OnCollectionChanged(object sender, NotifyCollectionChangedEventArgs e)
{
if (e.NewItems != null)
{
foreach (object part in e.NewItems)
{
(part as INotifyPropertyChanged).PropertyChanged
+= new PropertyChangedEventHandler(PartPropertyChanged);
}
}
if (e.OldItems != null)
{
foreach (object part in e.OldItems)
{
(part as INotifyPropertyChanged).PropertyChanged
-= new PropertyChangedEventHandler(PartPropertyChanged);
}
}
}
private void PartPropertyChanged(object sender, PropertyChangedEventArgs e)
{
if(e.PropertyName == "PartType")
{
//save to database...
}
}
}
这确实通过Part
类来实现INotifyPropertyChanged
。