有时候我们为了方便使用TreeView,会改变它的ItemTemplate的模板,但是有时候,我们无法获取TreeView的SelectedItem,如下是在TreeViewItem添加一个多选框的模板,当我们单击CheckBox时,TreeView的SelectedItem就为null,即使界面选中一项,再改变其它项的值时,我们获得的SelectedItem也是界面选中的哪项,而不是我们改变值的那项,这不符合我们的要求,我希望当我改变任何一项的值时,获取的是改变值这项的TreeViewItem或者其相关联的数据项。
<TreeView.ItemTemplate> <HierarchicalDataTemplate ItemsSource="{Binding Children}"> <StackPanel Orientation="Horizontal"> <TextBlock Text="{Binding Id}"></TextBlock> <CheckBox Margin="6,0,0,0" IsChecked="{Binding IsEnabled, UpdateSourceTrigger=PropertyChanged}" Focusable="False"
FlowDirection="RightToLeft" Content="是否启用"> </StackPanel> </HierarchicalDataTemplate> </TreeView.ItemTemplate>
在这里我想的的方法是在鼠标按下之前,递归遍历TreeView的数据项,通过VisualTreeHelper获取数据项的在界面显示的范围,获取鼠标的位置,如果鼠标位置在数据项的显示范围内,则选中当前项,以下是我的代码实现
public TreeViewItem SelectItemByClick(ItemsControl itemsControl) { foreach (object item in itemsControl.Items) { var currentItem = itemsControl.ItemContainerGenerator.ContainerFromItem(item) as TreeViewItem; //如果鼠标处于当前项内容边界框内 if (VisualTreeHelper.GetDescendantBounds(currentItem).Contains(Mouse.GetPosition(currentItem))) { //如果当前项展开则递归设置子项选中 if (currentItem.IsExpanded) { var selectedItem = SelectItemByClick(currentItem); if (selectedItem != null) { selectedItem.IsSelected = true; return selectedItem; } } currentItem.IsSelected = true; return currentItem; } } return null; }
最后,我们只需要在TreeView的PreviewMouseDown调用即可
private void Tree_PreviewMouseDown(object sender, MouseButtonEventArgs e) { SelectItemByClick(orgTree); }
至于为什么要在这个事件中做,大家可以查查按钮事件的触发顺序,这样可以在Click事件中获取当前改变值的项了。
这样,我们的SelectedItem在单击CheckBox的时候也不为null了