TreeView 三种状态 没多大变化 只是增加了很多函数以方便调用
using System.Drawing;
using System.Windows.Forms;
using System.ComponentModel; namespace SimpleCustomControl
{
[ToolboxBitmap(typeof(TreeView))]
public partial class MyTreeView : TreeView
{
private int imageWidth = 0x12; private bool isFirstNodeDifferent = true;
[Category("行为"), DefaultValue(true), Description("确定根节点第一个节点是否作为取消所择使用。")]
public bool IsFirstNodeDifferent
{
get { return isFirstNodeDifferent; }
set { isFirstNodeDifferent = value; }
} public MyTreeView()
{
InitializeComponent();
// 如果更改 ImageList 中图片大小,此处设置可能有用.未测试.可注释掉.
imageWidth = imageList1.ImageSize.Width + ;
} //规则1:取消选定
//规则1.1:检查是否有子节点,需清除所有子节点的选定状态;
//规则1.2:检查是否有父节点,如有,则根据兄弟节点的选定状态修改父节点的选定状态
//规则2:选定
//规则2.1:检查是否有子节点,设置所有子节点为选定状态
//规则2.2:检查是否有父节点,如有,则根据兄弟节点的选定状态修改父节点的选定状态
/// <summary>
/// 鼠标点击节点触发事件
/// </summary>
private void MyTreeView_NodeMouseClick(object sender, TreeNodeMouseClickEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
Rectangle rc = e.Node.Bounds;
rc = new Rectangle(rc.X - imageWidth, rc.Y, rc.Width + imageWidth, rc.Height);
if (rc.Contains(e.Location))
{
if (isFirstNodeDifferent && e.Node.Level == && e.Node.Index == )
NodeUnSelected();
else
NodeClick(e.Node);
}
}
} /// <summary>
/// 节点点击
/// </summary>
/// <param name="tn">点击的节点</param>
public void NodeClick(TreeNode tn)
{
if (tn.ImageIndex == )
{
NodeUnSelected(tn);
}
else
{
NodeSelected(tn);
}
} /// <summary>
/// 不管现在节点状态
/// 将节点设置为选中
/// </summary>
/// <param name="tn">待更改状态的节点</param>
public void NodeSelected(TreeNode tn)
{
tn.SelectedImageIndex = ;
tn.ImageIndex = ;
SetNodeImg21(tn);
SetNodeImg22(tn);
} /// <summary>
/// 不管现在节点状态
/// 去掉节点选中状态
/// </summary>
/// <param name="tn">待更改状态的节点</param>
public void NodeUnSelected(TreeNode tn)
{
tn.SelectedImageIndex = ;
tn.ImageIndex = ;
SetNodeImg11(tn);
SetNodeImg12(tn);
} /// <summary>
/// 查找节点状态 是否被选中
/// </summary>
/// <param name="tn">查看的节点</param>
/// <returns>节点是否被选中</returns>
public bool NodeIsChecked(TreeNode tn)
{
if (tn.ImageIndex == || tn.ImageIndex == )
return true;
return false;
} private bool nodeIsAllChecked;
/// <summary>
/// 返回整个TreeView 选中状态
/// </summary>
/// <returns>返回选中状态信息</returns>
[Browsable(false)]
public bool NodeIsAllChecked
{
get
{
if (isFirstNodeDifferent)
{
// 两种方式 两种效率 未测试哪个更效率
//foreach (TreeNode tn in Nodes)
//{
// if (tn.Index != 0 && NodeCheckState(tn) != CheckState.Checked)
// return false;
//}
int count = Nodes.Count;
for (int i = ; i < count; i++)
{
if (NodeCheckState(Nodes[i]) != CheckState.Checked)
return false;
}
}
else
{
foreach (TreeNode tn in Nodes)
{
if (NodeCheckState(tn) != CheckState.Checked)
return false;
}
}
return true;
}
} /// <summary>
/// 返回指定节点选中状态
/// 有三种状态 Checked 选中 Indeterminate 部分选中 Unchecked 未选中
/// </summary>
/// <param name="tn">节点是否被选中</param>
/// <returns>返回选中状态信息</returns>
public CheckState NodeCheckState(TreeNode tn)
{
switch (tn.ImageIndex)
{
case :
return CheckState.Indeterminate;
case :
return CheckState.Checked;
case :
default:
return CheckState.Unchecked;
}
} /// <summary>
/// 将所有子节点全不选
/// </summary>
public void NodeUnSelected()
{
foreach (TreeNode tn in this.Nodes)
{
NodeUnSelected(tn);
}
} /// <summary>
/// 将所有子节点全选
/// </summary>
public void NodeSelected()
{
foreach (TreeNode tn in this.Nodes)
{
NodeSelected(tn);
}
if (isFirstNodeDifferent && Nodes.Count > )
{
Nodes[].SelectedImageIndex = ;
Nodes[].ImageIndex = ;
}
} //设置节点选定状态:
//规则.1:检查是否有子节点,需清除所有子节点的选定状态;
void SetNodeImg11(TreeNode tn)
{
foreach (TreeNode t in tn.Nodes)
{
t.SelectedImageIndex = ;
t.ImageIndex = ;
if (t.Nodes.Count != )
SetNodeImg11(t);
}
} //设置节点选定状态:
//规则.2:检查是否有父节点,如有,则根据兄弟节点的选定状态修改父节点的选定状态
void SetNodeImg12(TreeNode tn)
{
if (tn.Parent == null)
return;
int Img0Num = , Img1Num = , Img2Num = ;
//统计同级节点选中情况
foreach (TreeNode t in tn.Parent.Nodes)
{
switch (t.ImageIndex)
{
case :
Img1Num++;
break;
case :
Img2Num++;
break;
case :
default:
Img0Num++;
break;
}
}
//如果同级节点选中和未选中的都有
if (Img1Num != || (Img0Num != && Img2Num != ))
{
tn.Parent.SelectedImageIndex = ;
tn.Parent.ImageIndex = ;
}
else
{
tn.Parent.StateImageIndex = ;
tn.Parent.ImageIndex = ;
}
SetNodeImg12(tn.Parent);
} //设置节点选定状态:
//规则.1:检查是否有子节点,设置所有子节点为选定状态
void SetNodeImg21(TreeNode tn)
{
foreach (TreeNode t in tn.Nodes)
{
t.SelectedImageIndex = ;
t.ImageIndex = ;
if (t.Nodes.Count != )
{
SetNodeImg21(t);
}
}
} //设置节点选定状态:
//规则.2:检查是否有父节点,如有,则根据兄弟节点的选定状态修改父节点的选定状态
void SetNodeImg22(TreeNode tn)
{
if (tn.Parent == null)
return;
int Img0Num = , Img1Num = , Img2Num = ;
foreach (TreeNode t in tn.Parent.Nodes)
{
//if (t.ImageIndex == 0)
// Img0Num++;
//if (t.ImageIndex == 1)
// Img1Num++;
//if (t.ImageIndex == 2)
// Img2Num++;
switch (t.ImageIndex)
{
case :
Img1Num++;
break;
case :
Img2Num++;
break;
case :
default:
Img0Num++;
break;
}
}
if (Img1Num != || (Img0Num != && Img2Num != ))
{
tn.Parent.SelectedImageIndex = ;
tn.Parent.ImageIndex = ;
}
//else if (Img1Num == 0 && Img2Num == 0)
//{
// tn.Parent.SelectedImageIndex = 0;
// tn.Parent.ImageIndex = 0;
//}
else
{
tn.Parent.StateImageIndex = ;
tn.Parent.ImageIndex = ;
}
SetNodeImg22(tn.Parent);
} }
}
TreeView 三种状态 没多大变化 只是增加了很多函数以方便调用的更多相关文章
- WinForm TreeView 三种状态
private void treeView1_NodeMouseClick(object sender, TreeNodeMouseClickEventArgs e) { var node = e.N ...
- TreeView的三种状态,全选,全不选,半选中
我知道的设置treeview节点的三种状态,如果不是买的控件,那么通过代码,只能设置两种状态,我知道的有三种方法, 第一种是重写treeview,第二种是把三种状态做成小图标,让节点复选框随着不同的状 ...
- hibernate 三种状态的转换
一.遇到的神奇的事情 使用jpa操作数据库,当我使用findAll()方法查处一个List的对象后,给对这个list的实体进行了一些操作,并没有调用update 或者 saveOrUpdate方法,更 ...
- SSH框架之-hibernate 三种状态的转换
一.遇到的神奇的事情 使用jpa操作数据库,当我使用findAll()方法查处一个List的对象后,给对这个list的实体进行了一些操作,并没有调用update 或者 saveOrUpdate方法,更 ...
- Hibernate 对象的三种状态
hibernate对象的三种状态: (一) 瞬时(临时)状态: 对象被创建时的状态,数据库里面没有与之对应的记录! (二) 持久状态: 处于session的管理中,并且数据库里面存在与之对应的 ...
- Hibernate中对象的三种状态及相互转化
1. 瞬时状态 (Transient) 当我们通过Java的new关键字来生成一个实体对象时,这时这个实体对象就处于自由状态,如下: Customer customer=new Customer(“ ...
- 菜鸟学SSH(八)——Hibernate对象的三种状态
前面写了几篇关于SSH的博客,但不是Struts就是Spring,Hibernate还从来没写过呢.说好是SSH的,怎么可以光写那两个,而不写Hibernate呢对吧.今天就先说说Hibernate对 ...
- java对象中的三种状态和脏检查及刷新缓存机制
瞬时状态 瞬时状态又称临时状态.如果java对象与数据库中的数据没有任何的关联,即此java对象在数据库中没有相关联的记录,此时java对象的状态为瞬时状态,session对于 瞬时状态的ava对象是 ...
- Git 深度学习填坑之旅二(文件三种状态、打标签)
0x01 三种状态 Git 有三种状态,你的文件可能处于其中之一: 已提交(committed).已修改(modified)和已暂存(staged). 已提交表示数据已经安全的保存在本地数据库中. 已 ...
随机推荐
- 【Mac】 开启原生的 NTFS 硬盘格式支持
一.MacOS 10.13 之前 二.MacOS 10.13 及之后 一.MacOS 10.13 之前 直接跳到引用地址查看,下面的草记只是为了防止链接丢失 引用地址 打开终端 切换至root身份,输 ...
- php的小数位数最长多少位
在php中, echo 0.1234567890123456;exit; // 结果为:0.12345678901235, 整数部分为0时,最多到14位小数,如果后面还有,就自动四舍五入 echo 7 ...
- Jupyter安装和环境配置
配置: 1. 命令行启动 jupyter notebook 2. 也可以Anaconda直接启动 3. 设置token,如下图所示,命令行中输入 jupyter notebook list C:\Us ...
- Django_01_创建图书管理项目
在django中,项目的组织结构为一个项目包含多个应用,一个应用对应一个业务模块 示例:创建项目的名称为test1,完成“图书-英雄”信息的维护,创建应用名称为booktest 创建项目:首先进入到虚 ...
- macos下简单的socket服务器+客户端
TCP客户端服务器编程模型: 服务器: 调用socket函数创建套接字 调用bind绑定本地IP和端口 调用listen启动监听(准备好接收客户端链接的队列) 调用accept从已连接队列中提取第一个 ...
- 对称加密实现重要日志上报Openresty接口服务
记录后端接收日志的流程: 由于记录的是广告数据,单次计费数据都会上报,全国内约10几万终端上报. 终端上报:Android电视端Apk上报 接收终端:Openresty(Nginx+lua)利用ngi ...
- Python3学习笔记36-PEP8代码规范
在使用PyCharm时,最右边会有波浪线警告提示代码不符合PEP8代码规范.记录一下犯的错和解决方式 PEP8是风格错误,而不是编码错误.只是为了让代码更具有阅读性. 1)block comment ...
- vue.js 父子组件间 props 数据同步处理
常见的场景: 在一个vue组件A中,使用另外一个组件B.A将自己的数据通过B组件的Props属性(propX)传递到B组件实例内部,B组件内部会修改该Props属性(propX)的值,此时在A组件内部 ...
- 统计连接到主机前十的ip地址和连接数
常用脚本–tcp #!/bin/bash # #******************************************************************** #encodi ...
- redis四种部署方式
1.单点 2.主从 3.哨兵 4.集群