I’m trying to bind the SelectedItem of a WPF TreeView to a property in my MainViewModel in a pure MVVM setup (no code-behind).
Since TreeView.SelectedItem is not a dependency property, I created an attached property to make it bindable:
public static class TreeViewSelectedItemBehavior
{
public static readonly DependencyProperty SelectedItemProperty =
DependencyProperty.RegisterAttached(
"SelectedItem",
typeof(object),
typeof(TreeViewSelectedItemBehavior),
new FrameworkPropertyMetadata(null, FrameworkPropertyMetadataOptions.BindsTwoWayByDefault, OnSelectedItemChanged));
public static object GetSelectedItem(DependencyObject obj) =>
obj.GetValue(SelectedItemProperty);
public static void SetSelectedItem(DependencyObject obj, object value) =>
obj.SetValue(SelectedItemProperty, value);
private static void OnSelectedItemChanged(DependencyObject sender, DependencyPropertyChangedEventArgs e)
{
if (sender is TreeView treeView)
{
treeView.SelectedItemChanged -= TreeView_SelectedItemChanged;
treeView.SelectedItemChanged += TreeView_SelectedItemChanged;
}
}
private static void TreeView_SelectedItemChanged(object sender, RoutedPropertyChangedEventArgs<object> e)
{
if (sender is TreeView treeView)
{
SetSelectedItem(treeView, e.NewValue);
}
}
}
<TreeView ItemsSource="{Binding tvTodoItems}"
helpers:TreeViewSelectedItemBehavior.SelectedItem="{Binding SelectedTodoItem, Mode=TwoWay}">
<TreeView.ItemTemplate>
<DataTemplate>
<TextBlock Text="{Binding Name}" />
</DataTemplate>
</TreeView.ItemTemplate>
</TreeView>
public class MainViewModel : BaseViewModel
{
public List<TodoItemViewModel> tvTodoItems { get; set; }
private TodoItemViewModel _selectedTodoItem;
public TodoItemViewModel SelectedTodoItem
{
get => _selectedTodoItem;
set
{
if (_selectedTodoItem != value)
{
_selectedTodoItem = value;
OnPropertyChanged();
}
}
}
}
public abstract class BaseViewModel : INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
public void OnPropertyChanged(string propertyName)
{
if (PropertyChanged != null)
{
PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
}
}
}
The problem: When I select a different item in the TreeView, the set accessor of SelectedTodoItem in my ViewModel is never called. The binding does not seem to update from the TreeView to the ViewModel.
Question: How can I bind TreeView.SelectedItem to my ViewModel so that the ViewModel property is updated when the user changes the selection — purely in MVVM, without any code-behind?
Note: This is just the minimal test version. The real TreeView is hierarchical and looks like this:
<TreeView x:Name="NavTree"
Grid.Column="0"
Grid.Row="1"
ItemsSource="{Binding Spaces}"
Margin="8"
helpers:TreeViewSelectedItemBehavior.SelectedItem="{Binding SelectedItem, Mode=TwoWay}">
<TreeView.ItemTemplate>
<HierarchicalDataTemplate ItemsSource="{Binding Folders}">
<TextBlock Text="{Binding Name}" />
<HierarchicalDataTemplate.ItemTemplate>
<HierarchicalDataTemplate ItemsSource="{Binding Lists}">
<TextBlock Text="{Binding Name}" />
</HierarchicalDataTemplate>
</HierarchicalDataTemplate.ItemTemplate>
</HierarchicalDataTemplate>
</TreeView.ItemTemplate>
</TreeView>