DataGrid Looses Focus When Delete Key is Pressed

2019-06-24 13:34发布

I'm doing MVVM where a DataGrid is bound to an ObservableCollection with a DeleteItemCommand hooked up to the DataGrid.InputBindings as follows:

  <DataGrid.InputBindings>
      <KeyBinding Key="Delete" Command="{Binding DeleteItemCommand}" />
  </DataGrid.InputBindings>

The item and row are removed when the user hits the delete key but the grid looses focus. You have to click or tab to the grid so it regains focus before hitting Delete to remove another row (pretty freaking annoying). I tried setting the DataGrid.CanUserDeleteRows="False" but it doesn't make any difference.

I replaced the DataGrid with a ListView and the ListView retains focus.

Is this a bug with the DataGrid or am I doing something wrong? Peace and love, peace and love!

3条回答
再贱就再见
2楼-- · 2019-06-24 14:11

I bumped on that some time ago. Somehow this event is never raised. Try this approach.

Long story short, event PreviewKeyDown will get you where you want.

And in MVVM-friendly manner:

<i:Interaction.Triggers>
    <i:EventTrigger EventName="PreviewKeyDown">
      <i:InvokeCommandAction Command="{Binding DeleteItemCommand}"/>
    </i:EventTrigger>
</i:Interaction.Triggers>
查看更多
我欲成王,谁敢阻挡
3楼-- · 2019-06-24 14:12

I solved this by using the built in functionality of WPF DataGrid. The grid handles removing items by default if the underlying collection is editable (if the collection is dedicated to this purpose that's no problem, otherwise an intermediate collection can be added...). I avoided any key bindings and just set up the grid like this:
<DataGrid ItemsSource="{Binding InvoiceItems}" IsReadOnly="False" CanUserDeleteRows="True" CanUserAddRows="False">
The ItemsSource collection is of type BidningCollection<>

In my ViewModel (my DataContext) I add a handler for CollectionChanged event:
InvoiceItems.CollectionChanged += InvoiceItemsCollectionChanged;

And implement it like this:

private void InvoiceItemsCollectionChanged(object sender, NotifyCollectionChangedEventArgs e)
    {
        if (e.Action != NotifyCollectionChangedAction.Remove)
            return;
        foreach (var oldItem in e.OldItems)
        {
            //do any other processing necessary
        }
    }

That's because you will probably be having at least two ways of removing an item from you underlying collection (keyboard with Del key, some button) and maybe some other things to take care of when an item is deleted.

查看更多
Anthone
4楼-- · 2019-06-24 14:27

Did you check this answer yet?

How to bind delete action (in WPF Datagrid) to a command or property in view model

You probably need to:

  1. make sure CanUserDeleteRows="False"
  2. make sure the key is actually bound to the specified command in your datacontext, like this:

    <DataGrid.InputBindings>
        <KeyBinding Key="Delete" Command="{Binding DataContext.DeleteEntry, RelativeSource={RelativeSource Mode=FindAncestor, AncestorType=DataGrid}}"/>
    </DataGrid.InputBindings>
    

Previously I fail at item #2 and wrote Command="{Binding DeleteEntry}", while in fact I should bind to DataContext.DeleteEntry by RelativeSource.

查看更多
登录 后发表回答