WPF ListBox moves focus to ListBox after item deleted
Opened this issue · 7 comments
Have a ListBox that can delete selected item.
Verify an item is both selected and focused (not the listbox itself), delete it by showing a confirmation dialog.
When the dialog is dismissed and the item is deleted, the selected item is correctly updated to the next item, but the focus returns to the ListBox itself (apparently because the item is gone).
So the focus and selection are dissynchronized, selection remains in the right place, but the focus switches to the parent listbox, so pressing Up or Down will just switch both the focus and the selection to the first element instead of navigating to the previous/next element.
PresentationCore UIElement.Focus Line 2630
PresentationFramework ListBoxItem.OnVisualParentChanged Line 359
PresentationCore Visual.FireOnVisualParentChanged Line 4000
PresentationCore Visual.RemoveVisualChild Line 2735
PresentationCore Visual.InternalRemoveVisualChild Line 2598
PresentationCore VisualCollection.DisconnectChild Line 468
PresentationCore VisualCollection.RemoveRange Line 830
PresentationFramework UIElementCollection.RemoveRangeInternal Line 369
PresentationFramework VirtualizingPanel.RemoveInternalChildRange Line 512
PresentationFramework VirtualizingStackPanel.RemoveChildRange Line 8987
PresentationFramework VirtualizingStackPanel.OnItemsRemove Line 8936
PresentationFramework VirtualizingStackPanel.OnItemsChanged Line 3596
PresentationFramework VirtualizingPanel.OnItemsChangedInternal Line 581
PresentationFramework Panel.OnItemsChanged Line 686
PresentationFramework ItemContainerGenerator.OnItemRemoved Line 2583
PresentationFramework ItemContainerGenerator.OnCollectionChanged Line 2420
Selection works correctly because it operates on the view, not on the ListBox itself:
See ListCollectionView.MoveCurrencyOffDeletedElement
PresentationFramework Selector.OnSelectionChanged Line 1323
PresentationFramework ListBox.OnSelectionChanged Line 287
PresentationFramework Selector.InvokeSelectionChanged Line 1804
PresentationFramework Selector.SelectionChanger.End Line 2369
PresentationFramework Selector.SelectionChanger.SelectJustThisItem Line 2708
PresentationFramework Selector.SetSelectedToCurrent Line 1568
PresentationFramework Selector.OnCurrentChanged Line 1489
PresentationFramework CollectionView.OnCurrentChanged Line 1066
PresentationFramework ItemCollection.OnCurrentChanged Line 1932
WindowsBase WeakEventManager.ListenerList`1.DeliverEvent
WindowsBase WeakEventManager.DeliverEventToList
WindowsBase WeakEventManager.DeliverEvent
WindowsBase CurrentChangedEventManager.OnCurrentChanged
PresentationFramework CollectionView.OnCurrentChanged Line 1066
PresentationFramework ListCollectionView.MoveCurrencyOffDeletedElement Line 2669
PresentationFramework ListCollectionView.ProcessCollectionChangedWithAdjustedIndex Line 2154
PresentationFramework ListCollectionView.ProcessCollectionChanged Line 1837
PresentationFramework CollectionView.OnCollectionChanged Line 1186
Workaround:
// Workaround for https://github.com/KirillOsenkov/PublicBugs/issues/20
private void CurrentFoldersAndFilesView_CurrentChanged(object sender, EventArgs e)
{
if (Keyboard.FocusedElement == fileListBox && fileListBox.SelectedItem is { } selectedItem)
{
if (fileListBox.ItemContainerGenerator.ContainerFromItem(selectedItem) is { } itemContainer && itemContainer is FrameworkElement frameworkElement)
{
Keyboard.Focus(frameworkElement);
}
}
}
Workaround for TreeView:
public void MoveSelectionOut()
{
var parent = Parent;
if (parent == null)
{
return;
}
var next = parent.FindNext<SelectableViewModel>(this);
if (next != null)
{
IsSelected = false;
next.IsSelected = true;
return;
}
var previous = parent.FindPrevious<SelectableViewModel>(this);
if (previous != null)
{
IsSelected = false;
previous.IsSelected = true;
}
else
{
IsSelected = false;
parent.IsSelected = true;
}
}