导航

TreeView.AfterCheck和TreeNode.Checked赋值的问题

Posted on 2005-03-22 22:05  bullfinch  阅读(1587)  评论(0编辑  收藏  举报
下午在按照Programming C#的例程写一个TreeView的Event,其中一个是在Check一个Node之后,把它的所有的子结点以及子结点的子结点都Check上,Uncheck时也一样。按照书上的实现方法,我添加了一个TreeView.AfterCheck event handler,并在里面设置了TreeNode.Checked = check。但是运行的时候出错,debug过程中发现,总是在TreeNode.Checked = check之后,又回到TreeView.AfterCheck的event handler。原来是在设置TreeNode.Checked的值之后,又引发了AfterCheck event,如此往复,进入死循环。

google求解,在msdn上发现解答:
http://msdn.microsoft.com/library/default.asp?url=/library/en-s/cpref/html/frlrfsystemwindowsformstreeviewclassafterchecktopic.asp

摘录一段:
Note   Setting the TreeNode.Checked property from within the BeforeCheck or AfterCheck event causes the event to be raised multiple times and can result in unexpected behavior. For example, you might set the Checked property in the event handler when you are recursively updating the child nodes, so the user does not have to expand and check each one individually. To prevent the event from being raised multiple times, add logic to your event handler that only executes your recursive code if the Action property of the TreeViewEventArgs is not set to TreeViewAction.Unknown.

原来关键就是靠TreeViewEventArgs.Action是否是TreeViewAction.Unknown来判断是否是用户Check了Node。

msdn上还有一段例程代码:

Example

The following example updates all the child tree nodes of a TreeNode when the user changes its checked state. This code assumes you have a Form with a TreeView that has TreeNode objects in its TreeNodeCollection. The TreeNodeCollection should have tree nodes with child nodes.

[C#] 
// Updates all child tree nodes recursively.
private void CheckAllChildNodes(TreeNode treeNode, bool nodeChecked)
{
   
foreach(TreeNode node in treeNode.Nodes)
   
{
      node.Checked 
= nodeChecked;
      
if(node.Nodes.Count > 0)
      
{
         
// If the current node has child nodes, call the CheckAllChildsNodes method recursively.
         this.CheckAllChildNodes(node, nodeChecked);
      }

   }

}


// NOTE   This code can be added to the BeforeCheck event handler instead of the AfterCheck event.
// After a tree node's Checked property is changed, all its child nodes are updated to the same value.
private void node_AfterCheck(object sender, TreeViewEventArgs e)
{
   
// The code only executes if the user caused the checked state to change.
   if(e.Action != TreeViewAction.Unknown)
   
{
      
if(e.Node.Nodes.Count > 0)
      
{
         
/* Calls the CheckAllChildNodes method, passing in the current 
         Checked value of the TreeNode whose checked state changed. 
*/

         
this.CheckAllChildNodes(e.Node, e.Node.Checked);
      }

   }

}