TreeView SelectedItem行为 - 双向绑定在单向上不起作用

时间:2015-08-13 12:18:27

标签: c# wpf xaml treeview behavior

我创建了一个非常简单的例子来展示我的问题。也许我只是想错了。

我想选择TreeView的一个项目 - 我希望在视图中看到它(蓝色背景)。

要实现TwoWayBinding,我使用此行为:Data binding to SelectedItem in a WPF Treeview

public class BindableSelectedItemBehavior : Behavior<TreeView>
{
    #region SelectedItem Property

    public object SelectedItem
    {
        get { return (object)GetValue(SelectedItemProperty); }
        set { SetValue(SelectedItemProperty, value); }
    }

    public static readonly DependencyProperty SelectedItemProperty =
        DependencyProperty.Register("SelectedItem", typeof(object), typeof(BindableSelectedItemBehavior), new UIPropertyMetadata(null, OnSelectedItemChanged));

    private static void OnSelectedItemChanged(DependencyObject sender, DependencyPropertyChangedEventArgs e)
    {
        var item = e.NewValue as TreeViewItem;
        if (item != null)
        {
            item.SetValue(TreeViewItem.IsSelectedProperty, true);
        }
    }

    #endregion

    protected override void OnAttached()
    {
        base.OnAttached();

        this.AssociatedObject.SelectedItemChanged += OnTreeViewSelectedItemChanged;
    }

    protected override void OnDetaching()
    {
        base.OnDetaching();

        if (this.AssociatedObject != null)
        {
            this.AssociatedObject.SelectedItemChanged -= OnTreeViewSelectedItemChanged;
        }
    }

    private void OnTreeViewSelectedItemChanged(object sender, RoutedPropertyChangedEventArgs<object> e)
    {
        this.SelectedItem = e.NewValue;
    }
}

但如果我点击某个项目,它就不会进入OnSelectedItemChanged的'if',因为e.newValue as TreeViewItemnull

我的XAML非常简单:

<StackPanel>
    <TreeView xmlns:i="clr-namespace:System.Windows.Interactivity;assembly=System.Windows.Interactivity"
      ItemsSource="{Binding Items}">
        <i:Interaction.Behaviors>
            <local:BindableSelectedItemBehavior
                SelectedItem="{Binding Item}" />
        </i:Interaction.Behaviors>
        <TreeView.ItemTemplate>
            <HierarchicalDataTemplate>
                <TextBlock Text="{Binding Text}"/>
            </HierarchicalDataTemplate>
        </TreeView.ItemTemplate>
    </TreeView>
    <TextBox Text="{Binding Item.Text}"/>
</StackPanel>

谢谢你们!

3 个答案:

答案 0 :(得分:2)

为了方便起见,这里结合了OPghrod's answer的最终解决方案:

namespace MyPoject.Behaviors
{
  using System.Windows;
  using System.Windows.Controls;
  using System.Windows.Interactivity;

  public class BindableSelectedItemBehavior : Behavior<TreeView>
  {
    #region SelectedItem Property

    public object SelectedItem
    {
      get { return (object)GetValue(SelectedItemProperty); }
      set { SetValue(SelectedItemProperty, value); }
    }
    public static readonly DependencyProperty SelectedItemProperty =
        DependencyProperty.Register(
          nameof(SelectedItem),
          typeof(object),
          typeof(BindableSelectedItemBehavior),
          new FrameworkPropertyMetadata(null, 
            FrameworkPropertyMetadataOptions.BindsTwoWayByDefault, 
            OnSelectedItemChanged));

    static void OnSelectedItemChanged(DependencyObject sender, 
      DependencyPropertyChangedEventArgs e)
    {
      var behavior = (BindableSelectedItemBehavior)sender;
      var generator = behavior.AssociatedObject.ItemContainerGenerator;
      if (generator.ContainerFromItem(e.NewValue) is TreeViewItem item)
        item.SetValue(TreeViewItem.IsSelectedProperty, true);
    }
    #endregion

    protected override void OnAttached()
    {
      base.OnAttached();

      AssociatedObject.SelectedItemChanged += OnTreeViewSelectedItemChanged;
    }

    protected override void OnDetaching()
    {
      base.OnDetaching();

      if (this.AssociatedObject != null)
        AssociatedObject.SelectedItemChanged -= OnTreeViewSelectedItemChanged;
    }

    void OnTreeViewSelectedItemChanged(object sender, 
        RoutedPropertyChangedEventArgs<object> e) =>
      SelectedItem = e.NewValue;
  }
}

答案 1 :(得分:1)

SelectedItem的{​​{1}}属性在您的情况下不会返回TreeView。它返回绑定的TreeViewItem集合中当前选定的项目。要从Items获取TreeViewItem,您需要在此使用SelectedItem

ItemContainerGenerator

答案 2 :(得分:0)

如果您的实际模型对象的类型为TreeViewItem,则OnSelectedItemChanged将仅传递TreeViewItem类型的对象,99%的情况下不会出现这种情况。您将不得不从模型对象中检索TreeViewItem,但如果节点当前已折叠,则它将不可用,这使得选择折叠节点非常重要。

我已经努力在我的博客中彻底解释这一点,包括代码示例here