写在前面:在(一)中,介绍了TreeView控件MVVM模式下数据绑定的方法。在这篇文章中,将总结给节点添加事件的方法,这样说有些不对,总之实现的效果就是点击某个节点,将出现对应于该节点的页面或者数据。(我这里用的方法肯定不是最好的,但是是我能想到的最佳方法了,WPF初学者,希望大家多多指教。)


Example#1: 实现下图功能,点击左侧treeview姓名节点,在右侧会出现响应的detailed information. 可以将ID的textbox中的text属性绑定到treeview中SelectedItem

先构造两个类,一个是User,一个是TreeNode。User是TreeNode的一个属性。

public class User
{
public string Key { get; set; }
public string Name { get; set; }
public int? Age { get; set; } public User()
{
Key = null;
Name = null;
Age = null;
}
}

User

public class TreeNode
{
public int NodeID { get; set; }
public int ParentID { get; set; }
public string NodeName { get; set; }
public List<TreeNode> ChildNodes { get; set; }
public User user { get; set; } public TreeNode()
{
ChildNodes = new List<TreeNode>();
user = new User();
}
}

TreeNode

绑定:

<TreeView Grid.Column="0" FontSize="15" ItemsSource="{Binding Path=Nodes}" x:Name="treeview">
<TreeView.ItemTemplate>
<HierarchicalDataTemplate DataType="{x:Type local:TreeNode}" ItemsSource="{Binding Path=ChildNodes}">
<Label Content="{Binding Path=NodeName}"/>
</HierarchicalDataTemplate>
</TreeView.ItemTemplate>
</TreeView> <TextBox Text="{Binding ElementName=treeview, Path=SelectedItem.user.Key}" Canvas.Left="70" Width="200" Canvas.Top="8"
FontSize="15"/> <TextBox Text="{Binding ElementName=treeview, Path=SelectedItem.user.Age}" Canvas.Left="70" Width="200" Canvas.Top="8"
FontSize="15"/>

上面两句TextBox控件,就是将项目中名为“treeview"的控件的SelectedItem.user.Key和SeletedItem.user.Age的值绑定到Text属性中。这样点”Lily"节点,右侧就会出现相应的信息。

Example#2:

上面的例子比较简单,第二个例子将button控件作为treeviewitem,并给button控件绑定一个Command。

场景描述:左侧是treeview,其中每个treeviewitem的元素都是button控件,点击每个节点,中间的listview中会出现符合条件的学生的姓名,比如,是Grade1的学生有Lucy, Tom和Lily三人。是Grade2Class1的学生有Sam和Jack两人。点击listview中的学生姓名,右侧会显示学生的ID和Age信息。

TreeView部分的XAML代码:

<TreeView Grid.Column="0" FontSize="15" ItemsSource="{Binding Path=Nodes}">
<TreeView.ItemTemplate>
<HierarchicalDataTemplate DataType="{x:Type local:TreeNode}" ItemsSource="{Binding Path=ChildNodes}">
<Button Content="{Binding NodeName}" Command="{Binding DataContext.TreeViewCommand,
RelativeSource={RelativeSource AncestorType=local:MainWindow}}" CommandParameter="{Binding Path=NodeID}"
Background="White" BorderThickness="0"/>
</HierarchicalDataTemplate>
</TreeView.ItemTemplate>
</TreeView>

可以看见,TreeViewItem元素从Label变成了Button。其中Button控件绑定了一个TreeViewCommand。这里需要指明绑定的是DataContext下的TreeViewCommand,否则默认的是TreeNode类型中的TreeViewCommand属性。因此下面这句是非常关键的。

Command="{Binding DataContext.TreeViewCommand,
RelativeSource={RelativeSource AncestorType=local:MainWindow}}" CommandParameter="{Binding Path=NodeID}"

ListView部分的XAML代码:

<ListView Name="listview" Grid.Column="1" ItemsSource="{Binding Users}" IsSynchronizedWithCurrentItem="True"
BorderBrush="DarkGray" BorderThickness="5">
<ListView.View>
<GridView>
<GridViewColumn Header="Name" DisplayMemberBinding="{Binding Path=Name}" Width="auto"/>
</GridView>
</ListView.View>
</ListView>

ViewModel代码:

public class ViewModel :INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
private List<User> userlist = new List<User>();
private List<TreeNode> nodes;
public List<TreeNode> Nodes
{
get { return nodes; }
set { nodes = value;
if (PropertyChanged != null)
PropertyChanged(this, new PropertyChangedEventArgs("Nodes"));
}
}
private List<User> users = new List<User>();
public List<User> Users
{
get { return users; }
set { users = value;
if (PropertyChanged != null)
PropertyChanged(this, new PropertyChangedEventArgs("Users"));
} } public DelegateCommand TreeViewCommand { get; } public ViewModel()
{
// 初始化Nodes和Users,一般Users是要访问数据库得到的,这里进行了简化。
InitiateNodes();
InitiateUsers();
TreeViewCommand = new DelegateCommand(TreeViewCommandHandler);
} private void TreeViewCommandHandler(object sender, DelegateCommandEventArgs e)
{
int id = Convert.ToInt32(e.Parameter);
switch (id)
{
case 1:
Users = userlist.Where(x => x.GradeNum == 1).ToList();
break;
case 2:
Users = userlist.Where(x => x.GradeNum == 2).ToList();
break;
case 3:
Users = userlist.Where(x => x.GradeNum == 3).ToList();
break;
case 4:
Users = userlist.Where(x => x.GradeNum == 1 && x.ClassNum == 1).ToList();
break;
case 5:
Users = userlist.Where(x => x.GradeNum == 1 && x.ClassNum == 2).ToList();
break;
case 6:
Users = userlist.Where(x => x.GradeNum == 2 && x.ClassNum == 1).ToList();
break;
case 7:
Users = userlist.Where(x => x.GradeNum == 2 && x.ClassNum == 2).ToList();
break;
case 8:
Users = userlist.Where(x => x.GradeNum == 3 && x.ClassNum == 1).ToList();
break;
case 9:
Users = userlist.Where(x => x.GradeNum == 3 && x.ClassNum == 2).ToList();
break;
}
} #region Initial methods
private void InitiateNodes()
{
List<TreeNode> _nodes = new List<TreeNode>()
{
new TreeNode()
{
ParentID=0,NodeID=1,NodeName="Grade1"
},
new TreeNode()
{
ParentID=0,NodeID=2,NodeName="Grade2"
},
new TreeNode()
{
ParentID=0, NodeID=3, NodeName="Grade3"
},
new TreeNode(){ParentID=1,NodeID=4,NodeName="Class1"},
new TreeNode(){ParentID=1,NodeID=5,NodeName="Class2"},
new TreeNode(){ParentID=2,NodeID=6,NodeName="Class1"},
new TreeNode(){ParentID=2, NodeID=7, NodeName="Class2"},
new TreeNode(){ParentID=3, NodeID=8, NodeName="Class1"},
new TreeNode(){ParentID=3, NodeID=9, NodeName="Class2"}
};
Nodes = getChildNodes(0, _nodes); } private List<TreeNode> getChildNodes(int parentID, List<TreeNode> nodes)
{
List<TreeNode> mainNodes = nodes.Where(x => x.ParentID == parentID).ToList();
List<TreeNode> otherNodes = nodes.Where(x => x.ParentID != parentID).ToList();
foreach (TreeNode node in mainNodes)
node.ChildNodes = getChildNodes(node.NodeID, otherNodes);
return mainNodes;
} private void InitiateUsers()
{
User Lily = new User()
{
Name = "Lily",
Age = 12,
GradeNum = 1,
ClassNum = 1
};
User Tom = new User()
{
Name = "Tom",
Age = 11,
GradeNum = 1,
ClassNum = 1
};
User Lucy = new User()
{
Name = "Lucy",
Age = 12,
GradeNum = 1,
ClassNum = 2
};
User Sam = new User()
{
Name = "Sam",
Age = 13,
GradeNum = 2,
ClassNum = 1
};
User Jack = new User() { Name = "Jack", Age = 13, GradeNum = 2, ClassNum = 1 };
User Ray = new User() { Name = "Ray", Age = 13, GradeNum = 2, ClassNum = 2 };
User Lisa = new User() { Name = "Lisa", Age = 14, GradeNum = 3, ClassNum = 1 };
User Liz = new User() { Name = "Liz", Age = 14, GradeNum = 3, ClassNum = 2 };
userlist.Add(Liz);
userlist.Add(Lisa);
userlist.Add(Sam);
userlist.Add(Lucy);
userlist.Add(Tom);
userlist.Add(Lily);
userlist.Add(Jack);
userlist.Add(Ray);
} #endregion
}

ViewModel

其中有两个类,DelegateCommand和DelegateCommandEventArgs,是继承自ICommand,然后委托方法的。

public class DelegateCommand : ICommand
{
// 定义一个名为SimpleEventHandler的委托,两个参数,一个object类,一个是自定义的DelegateCommandEventArgs类
public delegate void SimpleEventHandler(object sender, DelegateCommandEventArgs e);
// handler是方法,别忘了,委托是用于定义方法的类
private SimpleEventHandler handler;
private bool isEnabled = true; public DelegateCommand(SimpleEventHandler handler)
{
this.handler = handler;
}
public void Execute(object parameter)
{
this.handler(this, new DelegateCommandEventArgs(parameter));
}
public bool CanExecute(object parameter)
{
return this.isEnabled;
}
public event EventHandler CanExecuteChanged;
public bool IsEnabled
{
get { return this.isEnabled; }
set
{
this.isEnabled = value;
this.OnCanExecuteChanged();
}
}
private void OnCanExecuteChanged()
{
if (this.CanExecuteChanged != null)
this.CanExecuteChanged(this, EventArgs.Empty);
}
}

DelegateCommand

public class DelegateCommandEventArgs : EventArgs
{
private object parameter;
public DelegateCommandEventArgs(object parameter)
{
this.parameter = parameter;
}
public object Parameter
{
get { return this.parameter; }
}
}

DelegateCommandEventArgs


2020.06.25更新内容

之前使用Button控件作为TreeViewItem,然后为Button的Command属性绑定事件。更新为直接将事件绑定为TreeView的SelectedItemChanged属性,同样可以实现一样的效果。详情可见:

https://www.cnblogs.com/larissa-0464/p/13186486.html

2021.12.14 关于TreeView控件的demo:

WPF中常用控件(TreeView, ComboBox, DataGrid, ListView)使用MVVM模式绑定的demo - 南风小斯 - 博客园 (cnblogs.com)

WPF中TreeView控件数据绑定和后台动态添加数据(二)的更多相关文章

  1. WPF中TreeView控件数据绑定和后台动态添加数据(一)

    数据绑定: 更新内容:补充在MVVM模式上的TreeView控件数据绑定的代码. xaml代码: <TreeView Name="syntaxTree" ItemsSourc ...

  2. WPF中TreeView控件SelectedItemChanged方法的MVVM绑定

    问题描述:左侧treeview控件中点击不同类别的节点时,右侧的页面会显示不同的权限.比如对于My Publications,拥有Modify和Delete两种权限,对于My Subscription ...

  3. WPF中TreeView控件的使用案例

    WPF总体来说还是比较方便的,其中变化最大的主要是Listview和Treeview控件,而且TreeView似乎在WPF是一个备受指责的控件,很多人说他不好用.我这个demo主要是在wpf中使用Tr ...

  4. WPF中PasswordBox控件的Password属性的数据绑定

    原文:WPF中PasswordBox控件的Password属性的数据绑定 英文原文:http://www.wpftutorial.net/PasswordBox.html 中文原文:http://bl ...

  5. WPF中Image控件的Source属性

    原文:WPF中Image控件的Source属性 imgBook 是一个Image控件,在后台代码中我想给它指定Source的属性.我先如下方式进行: Uri uri = new Uri(strImag ...

  6. 示例:WPF中Slider控件封装的缓冲播放进度条控件

    原文:示例:WPF中Slider控件封装的缓冲播放进度条控件 一.目的:模仿播放器播放进度条,支持缓冲任务功能 二.进度: 实现类似播放器中带缓存的播放样式(播放区域.缓冲区域.全部区域等样式) 实现 ...

  7. WPF中Ribbon控件的使用

    这篇博客将分享如何在WPF程序中使用Ribbon控件.Ribbon可以很大的提高软件的便捷性. 上面截图使Outlook 2010的界面,在Home标签页中,将所属的Menu都平铺的布局,非常容易的可 ...

  8. WPF中查找控件的扩展类

    在wpf中查找控件要用到VisualTreeHelper类,但这个类并没有按照名字查找控件的方法,于是搜索网络,整理出下面这个类,感觉用起来很是方便. 贴出来,供大家参考. /// <summa ...

  9. WPF中Popup控件在Win7以及Win10等中的对齐点方式不一样的解决方案 - 简书

    原文:WPF中Popup控件在Win7以及Win10等中的对齐点方式不一样的解决方案 - 简书 最近项目中使用弹出控件Popup,发现弹出框的对齐方式在不同的系统中存在不同(Popup在win10上是 ...

随机推荐

  1. http状态码 200 304 404 503等

    浏览器采用http请求时,会封装http get等信息见下图请求头,然后服务器响应后回发一些信息,包括状态码,响应头,响应信息等等,如下图. 右上图可见两种状态码,一种是200 一种是304.其中20 ...

  2. 使用TortoiseGit和Git Bash不需要输入RSA密码(passphrase)的方法

    1. 安装和配置Putty 安装 官网下载Putty并安装. 生成ppk密钥 打开puttygen.exe(C:\Program Files\PuTTY\puttygen.exe), 点Convers ...

  3. 6. java IO 流

    一.流的分类: * 1.操作数据单位:字节流.字符流 * 2.数据的流向:输入流.输出流 * 3.流的角色:节点流.处理流 *二.流的体系结构 * 抽象基类               节点流(或文件 ...

  4. C++ STL 容器概述

    在STL编程中,容器是经常用到的一种数据结构,在C++标准库中,容器分为: 序列式容器 关联式容器 二者本质区别在于,序列式容器是通过元素在容器中的位置进行顺序存储和元素访问.关联容器则是通过键[ke ...

  5. Matplotlib 3.0 秘籍·翻译完成

    原文:Matplotlib 3.0 Cookbook 协议:CC BY-NC-SA 4.0 欢迎任何人参与和完善:一个人可以走的很快,但是一群人却可以走的更远. 在线阅读 ApacheCN 面试求职交 ...

  6. 男孩和女孩(二)-->相识

    转载请注明来源:https://www.cnblogs.com/hookjc/ 那天是男孩的十九岁生日:男孩还是像平常一样,一大早就起来了(快七点了).一切都是那么的平常,直到第一节课下课,男孩的同窗 ...

  7. 通过ANT生成MANIFEST.MF中的Class-Path属性

    原文地址:http://reason2003.iteye.com/blog/1627353 之前做一个项目,主程序打包成一个jar文件,因为用到了很多第三方的lib包,所以直接通过java命令运行ja ...

  8. iOS应用性能调优的建议和技巧--中高级--王朋

    中级(这些是你可能在一些相对复杂情况下可能用到的) 9. 重用和延迟加载Views 10. Cache, Cache, 还是Cache! 11. 权衡渲染方法 12. 处理内存警告 13. 重用大开销 ...

  9. Ext原码学习之Ext-more.js

    // JavaScript Document Ext.apply(Ext,{ userAgent:navigator.userAgent.toLowerCase(), cache:{}, isSeed ...

  10. 如何将VSCode配置上传到gitee账户,简单几步教你实现

    众所周知,VSCode是一款功能非常强大的代码编写软件,不仅开源免费,其插件商店也是非常广泛.非常之强大.借助这些插件我们可以配置各种语言环境,也可以运行各种代码. 但随之就有问题出现了,我们在更换设 ...