TreeView無法綁定SelectedItem,而又想知道treeview的selecteditem的變化,當然目前有很多方法,我這裡簡單的提供一個。 目前主要思路就是通過處理xaml的TreeViewItem的IsSelected屬性來進行綁定。 <TreeView BorderThicknes ...
TreeView無法綁定SelectedItem,而又想知道treeview的selecteditem的變化,當然目前有很多方法,我這裡簡單的提供一個。
目前主要思路就是通過處理xaml的TreeViewItem的IsSelected屬性來進行綁定。
<TreeView BorderThickness="0" Width="220" ItemsSource="{Binding Items, IsAsync=True}" HorizontalAlignment="Left"> <TreeView.ItemContainerStyle> <Style TargetType="{x:Type TreeViewItem}"> <Setter Property="IsSelected" Value="{Binding IsSelected, Mode=TwoWay}" /> <Setter Property="FontWeight" Value="Normal" /> <Setter Property="HorizontalAlignment" Value="Left" /> <Style.Triggers> <Trigger Property="IsSelected" Value="True"> <Setter Property="FontWeight" Value="Bold" /> </Trigger> </Style.Triggers> </Style> </TreeView.ItemContainerStyle> <TreeView.ItemTemplate> <HierarchicalDataTemplate ItemsSource="{Binding Items, IsAsync=True}"> <Label Content="{Binding Name}" /> </HierarchicalDataTemplate> </TreeView.ItemTemplate> </TreeView>
主要部分是
<Setter Property="IsSelected" Value="{Binding IsSelected, Mode=TwoWay}" />
這裡的意思就是綁定到集合泛型的IsSelected屬性。
這裡呢主要是通過抽象類去實現的也就是:
public abstract class TreeViewItemModel<T> : Notify { private bool _IsSelected; public bool IsSelected { get { return _IsSelected; } set { _IsSelected = value; OnChanged(); if (value) { IChanged<T> changed = ModelChangedManger.Sub.Get<T>(); if (changed?.Accepted == true) { changed.OnModelChanged((T)Convert.ChangeType(this, typeof(T))); } } } } }
具體的使用類再去繼承即可,也就是
public class GroupModel : TreeViewItemModel<GroupModel> { public GroupModel() { Id = Guid.NewGuid(); } private Guid _id; public Guid Id { get { return _id; } set { _id = value; OnChanged(); } } private string _name; public string Name { get { return _name; } set { _name = value; OnChanged(); } } private ObservableCollection<GroupModel> _Item = new ObservableCollection<GroupModel>(); public ObservableCollection<GroupModel> Items { get { return _Item; } set { _Item = value; OnChanged(); } } }
處理完這樣,我們只是完成了一部分。這裡這來說下
if (value) { IChanged<T> changed = ModelChangedManger.Sub.Get<T>(); if (changed?.Accepted == true) { changed.OnModelChanged((T)Convert.ChangeType(this, typeof(T))); } }
這個是乾什麼的。
很明顯,Isslected屬性觸發,我們想通知viewmodel,treeview的selecteditem屬性變了,或者叫做selectindex也變了的意思。所以簡單的實現了一個事件聚合器,這個部分代碼比較簡單,整體如下
public interface IChanged<T> { void OnModelChanged(T Model); bool Accepted { get; set; } } public sealed class ModelChangedManger { private ModelChangedManger() { } static ModelChangedManger() { } private class Inner { private Inner() { } internal readonly static ModelChangedManger manger = new ModelChangedManger(); } private ConcurrentDictionary<Type, object> Keys { get; set; } = new ConcurrentDictionary<Type, object>(); public IChanged<T> Get<T>() { if (Keys.TryGetValue(typeof(T), out object val)) { return val as IChanged<T>; } return null; } public void Set<T>(IChanged<T> model) { if (!Keys.ContainsKey(typeof(T))) { Keys.TryAdd(typeof(T), model); } } public void Remove<T>() { Keys.TryRemove(typeof(T), out _); } public static ModelChangedManger Sub { get => Inner.manger; } }
主要是Ichanged介面和manger的一起搭配使用。使用字典將要引發和被引發的類型,類存儲起來,這樣就可以全局或者大範圍的使用了。
同理 我們需要viewmodel上繼承一個引發事件的ichanged類
public abstract class TreeViewModel<T> : Notify, IChanged<T> { public virtual void OnModelChanged(T Model) { SelectedItem = Model; } private bool _Accepted; public bool Accepted { get { return _Accepted; } set { _Accepted = value; OnChanged(); } } private T _SelectedItem; public T SelectedItem { get { return _SelectedItem; } set { _SelectedItem = value; OnChanged(); } } } public class ViewModel : TreeViewModel<GroupModel> { private System.Collections.ObjectModel.ObservableCollection<GroupModel> _list = new System.Collections.ObjectModel.ObservableCollection<GroupModel>(); public System.Collections.ObjectModel.ObservableCollection<GroupModel> Items { get { return _list; } set { _list = value; OnChanged(); } } public ViewModel() { GroupModel groupModel1 = new GroupModel(); groupModel1.Name = "Ken"; GroupModel groupModel_1 = new GroupModel(); groupModel_1.Name = "A"; GroupModel groupModel_2 = new GroupModel(); groupModel_2.Name = "B"; GroupModel groupModel_3 = new GroupModel(); groupModel_3.Name = "C"; GroupModel groupModel_4 = new GroupModel(); groupModel_4.Name = "D"; groupModel1.Items.Add(groupModel_1); groupModel1.Items.Add(groupModel_2); groupModel1.Items.Add(groupModel_3); groupModel1.Items.Add(groupModel_4); GroupModel groupModel2 = new GroupModel(); groupModel2.Name = "TOM"; GroupModel groupModel_5 = new GroupModel(); groupModel_5.Name = "a"; GroupModel groupModel_6 = new GroupModel(); groupModel_6.Name = "b"; GroupModel groupModel_7 = new GroupModel(); groupModel_7.Name = "c"; GroupModel groupModel_8 = new GroupModel(); groupModel_8.Name = "d"; groupModel2.Items.Add(groupModel_5); groupModel2.Items.Add(groupModel_6); groupModel2.Items.Add(groupModel_7); groupModel2.Items.Add(groupModel_8); Items.Add(groupModel1); Items.Add(groupModel2); ModelChangedManger.Sub.Set(this); this.Accepted = true; } }
這樣,就可以簡單的實現了對selectedItem的變化的監控了