除非用户编辑单元格,否则在以编程方式更改时不会更新WPF EF6 Datagrid单元格

时间:2013-09-24 06:08:33

标签: c# wpf entity-framework datagrid

我有一个绑定到EF6 Dbcontext的WPF数据网格。我有许多功能,如手动编辑等。

问题 编辑绑定到网格的EF对象不会在网格上更新。但是,如果我在其中一次背景编辑后手动编辑单元格,它会立即显示正确的值。

情景/设置

ViewSources Involved

<Window.Resources>
 <CollectionViewSource x:Key="equipmentViewSource" d:DesignSource="{d:DesignInstance {x:Type HAI_Job_EF_Model:Equipment}, CreateList=True}"/>
 <CollectionViewSource x:Key="equipmentAssociatedDevicesViewSource" 
                              Source="{Binding AssociatedDevices, Source={StaticResource equipmentViewSource}}"/>    
 </Window.Resources>

请注意,关联设备是Equipment对象内的ObservableCollection。

WPF数据网格(仅限示例部分)

<DataGrid x:Name="associatedDevicesDataGrid" Grid.Row="1" AutoGenerateColumns="False" 
                              MaxWidth="1200" EnableRowVirtualization="True" 
                              RowDetailsVisibilityMode="VisibleWhenSelected" CanUserDeleteRows="False"
                              DataContext="{StaticResource equipmentAssociatedDevicesViewSource}" ItemsSource="{Binding}" 
                              CellEditEnding="associatedDevicesDataGrid_CellEditEnding" 
                              SelectionChanged="associatedDevicesDataGrid_SelectionChanged" 
                              PreviewKeyDown="associatedDevicesDataGrid_PreviewKeyDown"
                              LostFocus="associatedDevicesDataGrid_LostFocus">
                <DataGrid.Resources>
                    <!-- DATGRID STYLE CELL: Gives padding space inside cells -->
                    <Style TargetType="DataGridCell">
                        <Setter Property="Padding" Value="5,5"/>
                        <Setter Property="Template">
                            <Setter.Value>
                                <ControlTemplate TargetType="{x:Type DataGridCell}">
                                    <Border Padding="{TemplateBinding Padding}" BorderBrush="{TemplateBinding BorderBrush}" BorderThickness="{TemplateBinding BorderThickness}" Background="{TemplateBinding Background}" SnapsToDevicePixels="True">
                                        <ContentPresenter SnapsToDevicePixels="{TemplateBinding SnapsToDevicePixels}"/>
                                    </Border>
                                </ControlTemplate>
                            </Setter.Value>
                        </Setter>
                    </Style>
                </DataGrid.Resources>
                <DataGrid.Columns>
                    <DataGridTextColumn x:Name="quantityColumn" Header="Qty" MaxWidth="50" 
                                        Binding="{Binding Quantity, StringFormat={}\{0:N0\}, Mode=TwoWay, UpdateSourceTrigger=LostFocus}"/>
                    <DataGridComboBoxColumn x:Name="typeColumn" Header="Type" MaxWidth="150" 
                                            ItemsSource="{Binding Source={StaticResource assDevTypeFilteredViewSource}}"
                                            SelectedItemBinding="{Binding Path=Type, UpdateSourceTrigger=PropertyChanged}" 
                                            TextBinding="{Binding Path=Type, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}">
                        <DataGridComboBoxColumn.EditingElementStyle>
                            <Style TargetType="ComboBox">
                                <Setter Property="IsEditable" Value="True"/>
                                <Setter Property="Text" Value="{Binding Path=Type}"/>
                                <Setter Property="IsSynchronizedWithCurrentItem" Value="True" />
                                <Setter Property="IsTextSearchEnabled" Value="True" />
                                <Setter Property="IsTextSearchCaseSensitive" Value="False" />
                            </Style>
                        </DataGridComboBoxColumn.EditingElementStyle>
                    </DataGridComboBoxColumn>

                    <DataGridTemplateColumn x:Name="certificateNumberColumn" Header="Certificate Number" Width="Auto" MaxWidth="200">
                        <DataGridTemplateColumn.CellTemplate>
                            <DataTemplate>
                                <ComboBox Name="cbxAssDevCertComboBox" 
                                          IsEditable="True" IsTextSearchEnabled="True" IsTextSearchCaseSensitive="False"
                                          ItemsSource="{Binding GenericFilterResults}"
                                          Text="{Binding Path=CertificateNumber, UpdateSourceTrigger=PropertyChanged}"
                                          SelectionChanged="cbxAssDevCertComboBox_SelectionChanged"/>
                            </DataTemplate>
                        </DataGridTemplateColumn.CellTemplate>
                    </DataGridTemplateColumn>
                    <DataGridTextColumn x:Name="manufacturerColumn" Header="Manufacturer" Width="Auto" MaxWidth="150" 
                                        Binding="{Binding Manufacturer, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}"/>

后台更改 组合框“cbxAssDevCertComboBox”selectionchanged事件然后更新其他单元源对象的所有值。这是不刷新的更新。

已审查/考虑的主要选项

  1. 我没有完全使用MVVM而且我没有关注它。我现在太深了,无法改变这个项目。我只需要更新......!

  2. 我无法将每个EF对象包装在一个Observable集合中......没有其他需要它,为什么呢?这不是CollectionViewSource基本上做的事情吗?!

  3. 更新Datagrids ItemSource bindingexpression没有帮助。

  4. 使用更新

  5. 如何在程序编辑后手动编辑时如何更新电池?

    我现在已经失去了试图解决这个问题的日子;非常感谢anyhelp。

2 个答案:

答案 0 :(得分:2)

ObservableCollection非常棒,但只会在添加或删除项目时通知用户界面。在修改项目时(即,项目属性发生更改时),它不会通知UI。

您需要在模型上实现INotifyPropertyChanged界面并启动它。

这是一个实现接口的示例模型类:

public class ModelClass : INotifyPropertyChanged
{
    public event PropertyChangedEventHandler PropertyChanged;

    protected virtual void OnPropertyChanged(string property)
    {
        if (PropertyChanged != null)
            PropertyChanged(this, new PropertyChangedEventArgs(property));
    }

    protected virtual void OnPropertyChanged<T>(Expression<Func<T>> selectorExpression)
    {
        if (selectorExpression == null)
            throw new ArgumentNullException("selectorExpression");
        MemberExpression body = selectorExpression.Body as MemberExpression;
        if (body == null)
            throw new ArgumentException("The body must be a member expression");
        OnPropertyChanged(body.Member.Name);
    }

    string _myValue;
    public string MyValue
    {
        get
        {
            return _myValue;
        }
        set
        {
            _myValue = value;
            OnPropertyChanged(() => MyValue);
        }
    }
}

漂亮的lambda技巧取自this SO question ......

当然,您可以从代码的其他部分手动调用OnPropertyChanged(根据需要调整访问者或编写额外的公共方法)。调用OnPropertyChanged应强制UI为显示的属性调用getter。

答案 1 :(得分:2)

已为其他人提供信息

我找到了答案。我想知道为什么我的其他网格有些更新而且这个网格没有更新。

如果我在OP中创建了一个视图源;调用ViewSource.View.Refresh将刷新网格。 我最初没有这个,因为刷新时会导致ComboboxSelectionChanged事件;然后在无限循环中创建。您也不能只是添加一个切换来阻止每次事件刷新,因为此事件可以在设置期间多次设置。

解决方案 我使用bool作为锁来防止处理选择更改事件中的逻辑。每次处理事件逻辑时都会设置此锁定。该锁只能通过Combobox drop open事件删除;这可确保刷新仅在用户进行选择更改时发生。