简单的通用TreeView(WPF)
工作中要为很多类创建TreeView, 很多时候仅仅是因为要显示字段不同, 就得Ctrl+C、Ctrl+V复制一份几乎相同的代码, 这难免让人生厌, 于是希望像泛型集合的扩展方法那样, 可以在使用的时候灵活指定要显示哪个字段.
下面的TreeView要实现这样的逻辑: 父项目 被勾选 或者 取消勾选, 那么它的所有子项目全部改成 被勾选 或者 取消勾选; 只有所有的子项目都被勾选时, 父项目才自发的改成被勾选
- 创建基本的CommonTreeViewItemModel
/// <summary>
/// 通用的TreeViewItem模型, 仅包含最基础CheckBox(如果你觉得不好看, 在CommonTreeView.xaml中修改样式), 还包含一个 Tag(包含的对象)
/// 业务逻辑: 父项目 被勾选 或者 取消勾选, 那么它的所有子项目全部改成 被勾选 或者 取消勾选; 只有所有的子项目都被勾选时, 父项目才自发的改成被勾选
/// 所有的字段都改为protected, 方便继承修改
/// </summary>
public class CommonTreeViewItemModel : INotifyPropertyChanged
{ #region 属性 public event PropertyChangedEventHandler PropertyChanged; protected object id;
/// <summary>
/// 唯一性Id
/// </summary>
public object Id
{
get { return id; }
set { id = value; }
} protected string caption;
/// <summary>
/// 标题
/// </summary>
public string Caption
{
get { return caption; }
set { caption = value; if (PropertyChanged != null) PropertyChanged.Invoke(this, new PropertyChangedEventArgs("Caption")); }
} protected bool isChecked;
/// <summary>
/// 是否被勾选
/// </summary>
public bool IsChecked
{
get { return isChecked; }
set
{
isChecked = value;
if (PropertyChanged != null) PropertyChanged.Invoke(this, new PropertyChangedEventArgs("IsChecked"));
SetIsCheckedByParent(value);
if (Parent != null) Parent.SetIsCheckedByChild(value);
}
} protected bool isExpanded;
/// <summary>
/// 是否被展开
/// </summary>
public bool IsExpanded
{
get { return isExpanded; }
set { isExpanded = value; if (PropertyChanged != null) PropertyChanged.Invoke(this, new PropertyChangedEventArgs("IsExpanded")); }
} protected CommonTreeViewItemModel parent;
/// <summary>
/// 父项目
/// </summary>
public CommonTreeViewItemModel Parent
{
get { return parent; }
set { parent = value; }
} protected List<CommonTreeViewItemModel> children = new List<CommonTreeViewItemModel>();
/// <summary>
/// 含有的子项目
/// </summary>
public List<CommonTreeViewItemModel> Children
{
get { return children; }
set { children = value; }
} /// <summary>
/// 包含的对象
/// </summary>
public object Tag { get; set; } /// <summary>
/// 包含对象的类型
/// </summary>
public Type TagType { get; set; }
#endregion #region 业务逻辑, 如果你需要改成其他逻辑, 要修改的也就是这两行 /// <summary>
/// 子项目的isChecked改变了, 通知 是否要跟着改变 isChecked
/// </summary>
/// <param name="value"></param>
public virtual void SetIsCheckedByChild(bool value)
{
if (this.isChecked == value)
{
return;
} bool isAllChildrenChecked = this.Children.All(c => c.IsChecked == true);
this.isChecked = isAllChildrenChecked;
if (PropertyChanged != null) PropertyChanged.Invoke(this, new PropertyChangedEventArgs("IsChecked"));
if (Parent != null) Parent.SetIsCheckedByChild(value);
} /// <summary>
/// 自己的isChecked改变了, 所有子项目都要跟着改变
/// </summary>
/// <param name="value"></param>
public virtual void SetIsCheckedByParent(bool value)
{
this.isChecked = value;
if (PropertyChanged != null) PropertyChanged.Invoke(this, new PropertyChangedEventArgs("IsChecked"));
foreach (var child in Children)
{
child.SetIsCheckedByParent(value);
}
}
#endregion } - 创建通用的 CommonTreeView, 这其中最关键的就是泛型方法SetItemsSourceData<TSource, TId>
public partial class CommonTreeView : UserControl
{
public IList<CommonTreeViewItemModel> ItemsSourceData
{
get { return (IList<CommonTreeViewItemModel>)innerTree.ItemsSource; }
} public CommonTreeView()
{
InitializeComponent();
} /// <summary>
/// 设置数据源, 以及各个字段
/// </summary>
/// <typeparam name="TSource">数据源类型</typeparam>
/// <typeparam name="TId">主键类型</typeparam>
/// <param name="itemsArray">数据源列表</param>
/// <param name="captionSelector">指定显示为Caption的属性</param>
/// <param name="idSelector">指定主键属性</param>
/// <param name="parentIdSelector">指定父项目主键属性</param>
public void SetItemsSourceData<TSource, TId>(IEnumerable<TSource> itemsArray, Func<TSource, string> captionSelector, Func<TSource, TId> idSelector, Func<TSource, TId> parentIdSelector)
where TId : IEquatable<TId>
{
var list = new List<CommonTreeViewItemModel>(); foreach (var item in itemsArray.Where(a => object.Equals(default(TId), parentIdSelector(a))))
{
var tvi = new CommonTreeViewItemModel();
tvi.Caption = captionSelector(item).ToString();
tvi.Id = idSelector(item);
tvi.Tag = item;
tvi.TagType = item.GetType();
list.Add(tvi);
RecursiveAddChildren(tvi, itemsArray, captionSelector, idSelector, parentIdSelector);
} innerTree.ItemsSource = list;
return;
} /// <summary>
/// 递归加载children
/// </summary>
/// <typeparam name="TSource"></typeparam>
/// <typeparam name="TId"></typeparam>
/// <param name="parent"></param>
/// <param name="itemsArray"></param>
/// <param name="captionSelector"></param>
/// <param name="idSelector"></param>
/// <param name="parentIdSelector"></param>
/// <returns></returns>
private CommonTreeViewItemModel RecursiveAddChildren<TSource, TId>(CommonTreeViewItemModel parent, IEnumerable<TSource> itemsArray, Func<TSource, string> captionSelector, Func<TSource, TId> idSelector, Func<TSource, TId> parentIdSelector)
{ foreach (var item in itemsArray.Where(a => parent.Id.Equals(parentIdSelector(a))))
{
var tvi = new CommonTreeViewItemModel();
tvi.Caption = captionSelector(item);
tvi.Id = idSelector(item);
tvi.Tag = item;
tvi.TagType = item.GetType();
tvi.Parent = parent;
parent.Children.Add(tvi);
RecursiveAddChildren(tvi, itemsArray, captionSelector, idSelector, parentIdSelector);
}
return parent;
} - CommonTreeView 对应的xaml
<UserControl x:Class="WPFCommonTreeView.CommonTreeView"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local="clr-namespace:WPFCommonTreeView"
mc:Ignorable="d"
d:DesignHeight="300" d:DesignWidth="300">
<Grid>
<TreeView Name="innerTree">
<TreeView.ItemContainerStyle>
<Style TargetType="TreeViewItem">
<Setter Property="IsExpanded" Value="{Binding IsExpanded, Mode=TwoWay}"></Setter>
</Style>
</TreeView.ItemContainerStyle>
<TreeView.ItemTemplate>
<HierarchicalDataTemplate DataType="{x:Type local:CommonTreeViewItemModel}" ItemsSource="{Binding Children}">
<CheckBox FontSize="14" FontFamily="微软雅黑" Tag="{Binding Children}" IsChecked="{Binding IsChecked, Mode=TwoWay}" Content="{Binding Caption}" />
</HierarchicalDataTemplate>
</TreeView.ItemTemplate>
</TreeView>
</Grid>
</UserControl> - 演示效果

后记: 说实在的, 这个通用TreeView其实一点也不通用, 简单倒是真的, 主要原因是实际业务中TreeView的层次和逻辑千差万别, 就当自己的一点尝试吧.
转载请注明出处: http://www.cnblogs.com/zhouandke/p/6201064.html
简单的通用TreeView(WPF)的更多相关文章
- [译]聊聊C#中的泛型的使用(新手勿入) Seaching TreeVIew WPF 可编辑树Ztree的使用(包括对后台数据库的增删改查) 字段和属性的区别 C# 遍历Dictionary并修改其中的Value 学习笔记——异步 程序员常说的「哈希表」是个什么鬼?
[译]聊聊C#中的泛型的使用(新手勿入) 写在前面 今天忙里偷闲在浏览外文的时候看到一篇讲C#中泛型的使用的文章,因此加上本人的理解以及四级没过的英语水平斗胆给大伙进行了翻译,当然在翻译的过程中发 ...
- 一个简单的通用Makefile实现
一个简单的通用Makefile实现 Makefile是Linux下程序开发的自动化编译工具,一个好的Makefile应该准确的识别编译目标与源文件的依赖关系,并且有着高效的编译效率,即每次重新ma ...
- wpf中,一个简单的自定义treeview
首先创建一个自定义控件,在里面定义好treeview的样式,将本来的三角形的图标变为加号的图标,并且添加节点之间的连线. <UserControl x:Class="TreeViewE ...
- 微信小程序实现简单的树形图treeview
H5有很多树形图(树状图)的组件,echarts也有.比如像bootstrap的treeview,定制性很强.不过这些都无法方便地为小程序所用,除非整个页面用H5搭建再用webview框进去,有点粗暴 ...
- Seaching TreeVIew WPF
项目中有一个树形结构的资源,需要支持搜索功能,搜索出来的结果还是需要按照树形结构展示,下面是简单实现的demo. 1.首先创建TreeViewItem的ViewModel,一般情况下,树形结构都包含D ...
- C#简单的通用分页
通用分页技术分析 需要返回不同的类型的数据--采用泛型实现该操作 需要提供不同的方法 上一页 上一页 第一页 最后一页 跳转到指定页 Demo 代码 using System; using Syste ...
- Task(TPL)简单的实现Winform(WPF)异步
很多时候,我们要实现Winform异步操作,你可以用传统的方法,但个人感觉代码不好理解,而且使用真有点不舒服.也可以用Task来实现,Task(.net4.0新添加的对象)其实就是对线程池线程的一个封 ...
- flowable笔记 - 简单的通用流程
简介 通用流程可以用于一些基本的申请,例如请假.加班. 大致过程是: 1. 创建申请 2. 分配给审批人(需要审批人列表,当前审批人) -> 有下一个审批人 -> 3 -> 无 -& ...
- 简单的介绍下WPF中的MVVM框架
最近在研究学习Swift,苹果希望它迅速取代复杂的Objective-C开发,引发了一大堆热潮去学它,放眼望去各个培训机构都已打着Swift开发0基础快速上手的招牌了.不过我觉得,等同于无C++基础上 ...
随机推荐
- Entity Framework学习笔记——记一个错误解决方式及思路
继续之前设定的学习目标前,先来一篇小小的外篇.按照第一篇里的配置方式配置好的工程前两天还能正常工作,昨天却突然无法通过Add-Migration命令进行数据库的升级.错误信息如下: System.Da ...
- C#有关数组内存的释放及动态数组问题
一.数组内存释放问题 数组内存的释放可以按照如下语句实现: string [] aa=new string[2]; aa[0]="A"; aa[1]="B"; ...
- jQuery validate兼容IE8写法
最近做项目的时候遇到一个validate插件在IE8下面点击submit按钮没有执行检查的BUG 在chrome和FF,还有IE9以上都可以.百度了好多文章都没有找到解决方法,后面自己测试找到了问题. ...
- swift 多线程及GCD
1.基本概念 1)进程: 进程是指在系统中正在运行的一个应用程序.每个进程之间是独立的,每个进程运行在其专用且受保护的内存空间里.某进程内的线程在其它进程不可见 2)线程: 1个进程要执行任务,必须有 ...
- 外网无法访问本地IIS站点
自己申请了个花生壳动态域名,本来是打算在IIs上建个站点测试一下的,路由器上的端口也配置好了,把80端口指向我本机.域名也通过 ping 测试确实指向了我这里的路由器.但是用这个域名老是访问不了我这个 ...
- asp.net mvc4 Html.BeginForm表单提交
默认是get提交,如果是post提交需要在控制器ActionResult上加:[AcceptVerbs(HttpVerbs.Post)] 举例: 在HelpController中,会定义如下的Acti ...
- Java开发的命名规范
Java的命名规范 定义规范的目的是为了使项目的代码样式统一,使程序有良好的可读性,便于日后维护. 1.工程的命名(全用小写字母) 工程的命名一般全用小写字母,单词之间用下划线“_”隔开. 2.包的命 ...
- Vagrant的一个BUG - 不支持'change_host_name'
==> master: Setting hostname... Vagrant attempted to execute the capability 'change_host_name' on ...
- NodeJs解析web一例
var http = require('http'); var fs = require('fs'); var url = require('url'); http.createServer(func ...
- CSS中如何让元素隐藏
在CSS中,让元素隐藏(指屏幕范围内肉眼不可见)的方法很多,有的占据空间,有的不占据空间:有的可以响应点击,有的不能响应点击.下面一个个列出,选一个适合你的 { display: none; /* 不 ...