当存在多个验证错误时,WPF DataGrid TextBox保留以前的值

时间:2017-04-13 13:52:54

标签: c# wpf xaml datagrid idataerrorinfo

我创建了一个DataGrid,其中有两列都使用TextBox来编辑ViewModel的属性。当两列都有验证错误,并且属性值从ViewModel更改时,在其中一个单元格中输入编辑模式会保留以前编辑的值。

这是一个简短的例子:

视图

<Window ...>
    <Window.DataContext>
        <ViewModels:MainPresenter />
    </Window.DataContext>

    <DockPanel>
        <Button Command="{Binding ResetValuesCommand}"
                Margin="5" DockPanel.Dock="Top">Reset Values</Button>

        <DataGrid ItemsSource="{Binding Items}" AutoGenerateColumns="False" Margin="5">
            <DataGrid.Columns>
                <DataGridTextColumn Header="Value 1"
                    Binding="{Binding Value1, ValidatesOnDataErrors=True}" />
                <DataGridTextColumn Header="Value 2"
                    Binding="{Binding Value2, ValidatesOnDataErrors=True}" />
            </DataGrid.Columns>
        </DataGrid>
    </DockPanel>
</Window>

视图模型

public class MainPresenter : INotifyPropertyChanged
{
    public event PropertyChangedEventHandler PropertyChanged;

    public IEnumerable<ItemPresenter> Items { get; }
        = new ObservableCollection<ItemPresenter> {new ItemPresenter()};

    public ICommand ResetValuesCommand => new ResetCommand(Items);

    private class ResetCommand : ICommand
    {
        private readonly IEnumerable<ItemPresenter> _items;

        public ResetCommand(IEnumerable<ItemPresenter> items) { _items = items; }

        public void Execute(object parameter) => _items.ToList().ForEach(i => i.Reset());

        public bool CanExecute(object parameter) => true;

        public event EventHandler CanExecuteChanged { add { } remove { } }
    }
}

public class ItemPresenter : INotifyPropertyChanged, IDataErrorInfo
{
    public event PropertyChangedEventHandler PropertyChanged;

    public string Value1 { get; set; } = "A";

    public string Value2 { get; set; } = "B";

    public string this[string columnName] => "ERROR";

    public string Error => "ERROR";

    public void Reset()
    {
        Value1 = "A";
        PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Value1)));
        Value2 = "B";
        PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Value2)));
    }
}

重现步骤

运行应用时,两列都会突出显示为无效。

  • 双击“值1”列中的单元格(当前值为“A”)并更改它,例如到“Z”;
  • 按Enter(或等效)提交更改;
  • 按“重置值”按钮(使我们刚刚编辑过的单元格变为“A”);
  • 再次双击“值1”列中的单元格。

“值1”列中的单元格更改为编辑模式,该值再次显示为“Z”。

有一点需要注意:只有当其他列出现验证错误时才会发生这种情况。如果这是唯一具有验证错误的列,则进入编辑模式时编辑模式下的TextBox将正确显示“A”。

部分修复

奇怪的是,明确地将绑定中的模式设置为TwoWay(可能是默认值,因为这是明显的行为)可以解决问题。

但是,如果我想要一些自定义单元格模板(并将DataGridTextColumn替换为DataGridTemplateColumns s),但仍然使用TextBox进行编辑:

<DataGrid ItemsSource="{Binding Items}" AutoGenerateColumns="False" Margin="5">
    <DataGrid.Columns>
        <DataGridTemplateColumn Header="Value 1">
            <DataGridTemplateColumn.CellTemplate>
                <DataTemplate DataType="{x:Type ViewModels:ItemPresenter}">
                    <TextBlock Text="{Binding Value1, ValidatesOnDataErrors=True}" />
                </DataTemplate>
            </DataGridTemplateColumn.CellTemplate>
            <DataGridTemplateColumn.CellEditingTemplate>
                <DataTemplate DataType="{x:Type ViewModels:ItemPresenter}">
                    <TextBox Text="{Binding Value1, ValidatesOnDataErrors=True}" />
                </DataTemplate>
            </DataGridTemplateColumn.CellEditingTemplate>
        </DataGridTemplateColumn>

        <DataGridTemplateColumn Header="Value 2">
            <DataGridTemplateColumn.CellTemplate>
                <DataTemplate DataType="{x:Type ViewModels:ItemPresenter}">
                    <TextBlock Text="{Binding Value2, ValidatesOnDataErrors=True}" />
                </DataTemplate>
            </DataGridTemplateColumn.CellTemplate>
            <DataGridTemplateColumn.CellEditingTemplate>
                <DataTemplate DataType="{x:Type ViewModels:ItemPresenter}">
                    <TextBox Text="{Binding Value2, ValidatesOnDataErrors=True}" />
                </DataTemplate>
            </DataGridTemplateColumn.CellEditingTemplate>
        </DataGridTemplateColumn>
    </DataGrid.Columns>
</DataGrid>

我遇到了同样的问题,但明确地将绑定模式设置为TwoWay并没有解决它。

我在某处做错了什么,我忽略了什么?或者,有人有解决方法吗?

1 个答案:

答案 0 :(得分:0)

我找到了解决方法。

如果我更改为使用INotifyDataErrorInfo仅在.NET 4.5及以上版本中使用),则它会按预期工作。

视图模型

public class ItemPresenter : INotifyPropertyChanged, INotifyDataErrorInfo
{
    public event PropertyChangedEventHandler PropertyChanged;
    public event EventHandler<DataErrorsChangedEventArgs> ErrorsChanged;

    public string Value1 { get; set; } = "A";

    public string Value2 { get; set; } = "B";

    public IEnumerable GetErrors(string propertyName) => new[] { "ERROR" };

    public bool HasErrors => true;

    public void Reset()
    {
        Value1 = "A";
        PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Value1)));
        ErrorsChanged?.Invoke(this, new DataErrorsChangedEventArgs(nameof(Value1)));
        Value2 = "B";
        PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Value2)));
        ErrorsChanged?.Invoke(this, new DataErrorsChangedEventArgs(nameof(Value2)));
    }
}

视图

<DataGrid ItemsSource="{Binding Items}" AutoGenerateColumns="False" Margin="5">
    <DataGrid.Columns>
        <DataGridTemplateColumn Header="Value 1">
            <DataGridTemplateColumn.CellTemplate>
                <DataTemplate DataType="{x:Type ViewModels:ItemPresenter}">
                    <TextBlock Text="{Binding Value1}" />
                </DataTemplate>
            </DataGridTemplateColumn.CellTemplate>
            <DataGridTemplateColumn.CellEditingTemplate>
                <DataTemplate DataType="{x:Type ViewModels:ItemPresenter}">
                    <TextBox Text="{Binding Value1}" />
                </DataTemplate>
            </DataGridTemplateColumn.CellEditingTemplate>
        </DataGridTemplateColumn>
        <DataGridTemplateColumn Header="Value 2">
            <DataGridTemplateColumn.CellTemplate>
                <DataTemplate DataType="{x:Type ViewModels:ItemPresenter}">
                    <TextBlock Text="{Binding Value2}" />
                </DataTemplate>
            </DataGridTemplateColumn.CellTemplate>
            <DataGridTemplateColumn.CellEditingTemplate>
                <DataTemplate DataType="{x:Type ViewModels:ItemPresenter}">
                    <TextBox Text="{Binding Value2}" />
                </DataTemplate>
            </DataGridTemplateColumn.CellEditingTemplate>
        </DataGridTemplateColumn>
    </DataGrid.Columns>
</DataGrid>