1.如何展开结点时改变图标(注意:不是选中时)
要在目录中使用图标首先要加入一个控件ImageList(命名为imageList1),然后可以按图片的index或名称引用图片.
然后需要在TreeView控件的ImageList属性中指向imageList1.
TreeView有两个属性:
SelectImageIndex:选中该结点时显示的图片的索引
ImageIndex:未选中该结点时显示的图片的索引
可以实现的效果是,选中某个结点时该结点的图片进行改变,如果我们的目标也是如此,万事已经大吉了.
但我希望的效果是:展开某个结点时该结点的图片改变(如显示为打开的盒子),折叠时该结点的图片改变(如包装好的盒子).直接使用属性无法实现该效果.
实现原理是:展开某个结点时将SelectImageIndex和ImageIndex统统指向打开盒子的图片
折叠某个结点时将SelectImageIndex和ImageIndex统统指向包装盒子的图片
自然需要用到两个事件:TreeView的AfterExpand和AfterCollapse事件
private void treeView1_AfterExpand(object sender, TreeViewEventArgs e) { e.Node.ImageIndex = 1; //指向展开的图标 e.Node.SelectedImageIndex = 1;//指向展开的图标 } private void treeView1_AfterCollapse(object sender, TreeViewEventArgs e) { e.Node.ImageIndex = 0; //指向关闭的图标 e.Node.SelectedImageIndex = 0;//指向关闭的图标 }
PS:当对某个结点的属性
ImageIndex和SelectedImageIndex赋值后,向它添加的子结点如未指定该属性,似乎会沿用父结点的属性,所以通常我会对子结点单独设置该值 .
2.重命名结点名称
重命名结点的功能很好实现,只要将TreeView的LabelEdit属性改为True即可.选中某个结点后再单击(提示:间隔短的话算做双击),然后结点名称变成可以编辑状态,修改文本后即可完成.
但我的程序设计是这样的:结点的名称会在后面自动加入子结点的数量(注:树的深度为2).
效果如图:
这样就会带来麻烦(我们总是擅长给自己制造麻烦?),因为我在重命名的时候并不想在编辑状态下名称中包含后面数字(它只在显示的时候出现).
假设我希望将"新建分组"重命名为"我的分组",我希望进入编辑状态时,名称为"新建分组",修改后的名称为"我的分组",退出编辑状态时显示为"我的分组[1]".
那实际的逻辑就是这样:
1.进入编辑状态时将名称的数字去掉
2.退出编辑状态时将名称加入数字.
so easy!
那如何获取进入编辑状态时的事件呢?我没有找到,或者可能就没有,但我们可以在结点的单击事件中进行捕获,再判断结点的文本是否为选中状态
private void treeView1_NodeMouseClick(object sender, TreeNodeMouseClickEventArgs e) { if (e.Node.IsSelected) { e.Node.Text = "替换掉数字,请自己实现"; } }
然后捕捉编辑结束事件,实现加数字的方法
private void treeView1_AfterLabelEdit(object sender, NodeLabelEditEventArgs e) { if (!e.CancelEdit) { e.Node.Text=e.Label+string.Format(" [{0}]", e.Node.Nodes.Count.ToString());
e.CancelEdit = true; //注意这里
}
}
以下是重点:
在上面的代码没加入 e.CancelEdit = true 之前,会有一个奇怪的现象,当我将结点的名称修改后不会出现后面的数字.假设我修改的顺序是这样:将A->B->B
很有趣的现象:当从A变成B时后面的数字不会出现,当从B变成B时(实际没有发行变动),数字又出现了.跟踪了很久的程序加上了很长的时间分析,终于明白原因了.
当我们修改结点的文本时是这样的一个顺序:结点进入编辑状态->触发BeforeLabelEdit事件->触发AterLabelEdit事件->刷新界面用新值代替旧值.
在 treeView1_AfterLabelEdit事件的参数中有两个属性:s.Label 修改之后的值(新值,可能为null,只读) 和s.Node.Text 修改之前的值(旧值,可读写).
仔细分析后可以理解上面的现象:当从A->修改为B时,我们的代码对e.Node.Text(旧值)进行了赋值加入数字,最后程序刷新界面时会用新值代替旧值.这时我们加入的数字被新值Label覆盖了(代码做了无用功)
而我们从B->修改为B时,由于新值(Label)为null,则系统不会执行最后一步,所以我们添加的数字得到了保留.
最终的做法,就是将参数的e.CancelEdit改为true,告诉系统我们就当我们没有编辑过吧.这样系统就不会执行最后一步了,我们代码的修改得到了保留.