技术总结:自动扩张WPF树型表格列宽

简介:

问题描述

    今天测试人员提了一个易用性的BUG,主要是说系统目前使用的树型控件不支持自动扩张列的宽度。其实客户那边已经对这个问题提了多次,不过由于对WPF只是入门级,所以一直都没改。这两天项目比较闲,就花了些时间把这个问题改了。原问题如下:

image 

图1 问题描述

 

背景

    树型控件在GIX4系统中已经被大量使用。这个控件是一年前其它同事在网上搜索到,再引入的。

    一开始的时候,要解决这个问题,想到的最直接的方案是这样的:找到第一列中的Expander控件(加号:image),然后监听它的“Expanded”事件;在事件处理程序中,计算所需要的宽度,然后设置为控件的宽度。

    按照这个方案去实际写代码时,发现并没有想象中那么简单,发现了很多问题。例如,Expander并不是Expander控件,而是一个ToggleButton,而且是写在模板中的,TreeGridRowPresenter中的Expander的类型也只是UIElement,也就是说,不能把Expander从UIElement转换为ToggleButton,这样程序会写得很死。又如,如何计算第一列的所需要宽度。

    虽然我们项目中是有整个控件的源码,但是整合进来后别的同事已经对它进行了很多修改,所以只有在网上找到最原始的源码来研究。发现,原来这个树型控件的方案是Avalon Team自己给出的:《TreeListView: Show Hierarchy Data with Details in Columns》。然后Ricciolo对它进行了一些研究:《Fun With GridView*RowPresenter》,最后他给出了一个较完整的版本:《A complete WPF TreeListView control》。

    学习并研究了它的源码,最后总结出以下几个子问题,这些问题是要上面提及的BUG所需要解决的:

 

四个待解决的问题

    1. 何时触发是最合适的?在何处触发调整宽度的代码?

    2. 如何找到树型控件的所有GridViewRowPresenter。

    3. GridViewRowPresenter中,如何把第一列的控件找到。

    4. 第一列控件的组成结构是怎么样的,它所需要的大小如何求出,是否可以直接使用Measure和DesiredSize。

 

一步一步解决

    第一个问题,何时触发这个功能?其实我是要在点击后,当子节点都加载好后,然后计算出合适的大小,再设置给列对象。我先在TreeListView的OnExpanded事件处理程序中尝试编写代码获取每一个TreeListView,但是发现这个事件在发生时,所有的子节点并没有生成,所以不能通过ItemContainerGenerator.GetContainerForItem方法获取到窗口,此方案失败。接着,我查看了ItemsControl的接口声明,发现ItemContainerGenerator属性有事件StatusChanged。所以我就改为监听这个事件,并判断如果当它的Status变为ContainersGenerated时,就表示所有子节点已经生成了。代码如下:

1
2
3
4
5
6
7
this .ItemContainerGenerator.StatusChanged += (o, e) =>
{
     if  ( this .ItemContainerGenerator.Status == GeneratorStatus.ContainersGenerated)
     {
         this .AdjustFirstColumnWidth();
     }
};

但是同样发现新的问题,这时候虽然窗口对象TreeListView已经生成,但是它下面的所有Visual Child都没有生成,这样同样无法获取到它里面用来显示每一行的GridRowPresenter。所以只有改成了这样:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
public  TreeListViewItem()
{
     this .PrepareToAdjustFirstColumnWidth();
}
 
private  void  PrepareToAdjustFirstColumnWidth()
{
     this .ItemContainerGenerator.StatusChanged += (o, e) =>
     {
         if  ( this .ItemContainerGenerator.Status == GeneratorStatus.ContainersGenerated)
         {
             if  ( this .Items.Count > 0)
             {
                 var  item = this .Items[ this .Items.Count - 1];
                 var  treeItem = this .ItemContainerGenerator.ContainerFromItem(item) as  TreeListViewItem;
                 treeItem.Loaded += (oo, ee) =>
                 {
                     this .AdjustFirstColumnWidth();
                 };
             }
         }
     };
}

这样,最后一个孩子的可视内容都加载好后,才会触发调整宽度的代码。

    第二个问题比较简单,看了TreeListView的源码后,发现它在TreeListViewItem类的模板中使用了GridViewRowPresenter类,然后为它定义了名字:“PART_Header”。在模板中以PART_起头的控件是控件的约定,具体内容见:《WPF Parts Control Model》。所以我可以使用以下方法找到它,而不用考虑新的模板是否有它:

1
2
3
4
5
private  TreeGridViewRowPresenter FindGridRow()
{
     var  rowPresenter = this .Template.FindName( "PART_Header" , this ) as  TreeGridViewRowPresenter;
     return  rowPresenter;
}

    要解决第三个问题,我们需要知道GridViewRowPresenter中如何生成一行,并知道最后生成的控件结构。先看看GridViewRowPresenter最后生成的控件结构,这里我使用的是Snoop:

image

图2 用Snoop查看TreeGridViewRowPresenter的可视化结构

我们发现,GridViewRowPresenter下只是简单的包含了几个可视元素,它们刚好是每一列所显示的内容。再查看GridViewRowPresenter的源代码,发现它拥有以下属性:public GridViewColumnCollection Columns{get;set;}、internal UIElementCollection InternalCollection{get;set;},进一步分析后,我猜测性地得出以下结论:GridViewRowPresenter.InternalCollection简单地包含了所有列的显示元素,它会根据Columns属性中各行对这些可视元素进行维护,让它们显示得跟表格一样。

至此,第三个问题解决了:

1
var  firstColumn = VisualTreeHelper.GetChild(rowPresenter, 0) as  UIElement;

    最后一个问题,是过程中最麻烦的一个问题。我们看到,图2中该行下的第一个元素是第一列的显示元素,显示了“2.1”。但是文本左边的Expander控件却是TreeGridViewRowPresenter的最后一个可视化孩子。而且缩进并不是一个控件。那么这是怎么一回事呢?看了TreeGridViewRowPresenter的源码后,发现原来是它主动把Expander放在了最后:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
public  class  TreeGridViewRowPresenter : GridViewRowPresenter
{
     protected  override  System.Windows.Media.Visual GetVisualChild( int  index)
     {
         // Last element is always the expander
         // called by render engine
         if  (index < base .VisualChildrenCount) return  base .GetVisualChild(index);
         if  (index == base .VisualChildrenCount) return  this .lbRowNo;
         return  this .Expander;
     }
 
     protected  override  int  VisualChildrenCount
     {
         get
         {
             // Last element is always the expander
             if  ( this .Expander != null )
                 return  base .VisualChildrenCount + 2;
             else
                 return  base .VisualChildrenCount + 1;
         }
     }
}
1
而文本前面先显示缩进,然后再显示Expander的原因是由于TreeGridViewRowPresenter类重写了FrameworkElement.ArrangeOverride方法。在该方法中,它把第一列的元素显示的长度变短,在之前显示一段缩进的空白和Expander控件:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
protected  override  Size ArrangeOverride(Size arrangeSize)
{
     Size s = base .ArrangeOverride(arrangeSize);
 
     if  ( this .Columns == null  || this .Columns.Count == 0) return  s;
     UIElement expander = this .Expander;
 
     double  current = 0;
     double  max = arrangeSize.Width;
     for  ( int  x = 0; x < this .Columns.Count; x++)
     {
         GridViewColumn column = this .Columns[x];
         // Actual index needed for column reorder
         UIElement uiColumn = (UIElement) base .GetVisualChild(( int )ActualIndexProperty.GetValue(column, null ));
 
         // Compute column width
         double  w = Math.Min(max, (Double.IsNaN(column.Width)) ? ( double )DesiredWidthProperty.GetValue(column, null ) : column.Width);
 
         // First column indent
         if  (x == 0 && expander != null )
         {
             double  indent = FirstColumnIndent + expander.DesiredSize.Width;
             uiColumn.Arrange( new  Rect(current + indent, 0, w - indent, arrangeSize.Height));
         }
         else
         {
             uiColumn.Arrange( new  Rect(current, 0, w, arrangeSize.Height));
         }
         max -= w;
         current += w;
     }
 
     // Show expander
     if  (expander != null )
     {
         expander.Arrange( new  Rect( this .FirstColumnIndent, 0, expander.DesiredSize.Width, expander.DesiredSize.Height));
     }
 
     return  s;
}

分析到这里,就知道如何计算出第一列的最终宽度了:

1
2
3
4
5
6
7
8
9
10
11
12
13
private  double  GetFirstColumnDesiredWidth()
{
     var  rowPresenter = this .FindGridRow();
     if  (VisualTreeHelper.GetChildrenCount(rowPresenter) <= 0) return  0;
 
     //GridViewRowPresenter中的每一个元素表示一列。
     var  firstColumn = VisualTreeHelper.GetChild(rowPresenter, 0) as  UIElement;
     var  desiredWidth = firstColumn.DesiredSize.Width;
 
     //需要的宽度前,需要加上列的缩进和Expander的宽度。
     var  indent = rowPresenter.FirstColumnIndent + rowPresenter.Expander.DesiredSize.Width;
     return  indent + desiredWidth + ENSURE_SIZE;
}

加上以下这段代码后,程序终于可以正确运行了。

 

总结

    解决这个问题,花了一天多的时间,主要原因还是因为对WPF还是处在入门的级别。其中学到了以下内容:

熟悉了TreeView、TreeViewItem、ItemsControl的使用及树型控件的原理。

树型表格控件TreeListView的设计过程(见之前的文章)。

熟悉了Measure的使用。


本文转自BloodyAngel博客园博客,原文链接:http://www.cnblogs.com/zgynhqf/archive/2010/08/05/1793405.html,如需转载请自行联系原作者

相关文章
|
缓存 C# 虚拟化
WPF列表性能提高技术
WPF数据绑定系统不仅需要绑定功能,还需要能够处理大量数据而不会降低显示速度和消耗大量内存,WPF提供了相关的控件以提高性能,所有继承自`ItemsControl`的控件都支持该技术。
|
4月前
|
开发框架 前端开发 JavaScript
循序渐进介绍基于CommunityToolkit.Mvvm 和HandyControl的WPF应用端开发(5) -- 树列表TreeView的使用
循序渐进介绍基于CommunityToolkit.Mvvm 和HandyControl的WPF应用端开发(5) -- 树列表TreeView的使用
|
4月前
|
vr&ar C# 图形学
WPF与AR/VR的激情碰撞:解锁Windows Presentation Foundation应用新维度,探索增强现实与虚拟现实技术在现代UI设计中的无限可能与实战应用详解
【8月更文挑战第31天】增强现实(AR)与虚拟现实(VR)技术正迅速改变生活和工作方式,在游戏、教育及工业等领域展现出广泛应用前景。本文探讨如何在Windows Presentation Foundation(WPF)环境中实现AR/VR功能,通过具体示例代码展示整合过程。尽管WPF本身不直接支持AR/VR,但借助第三方库如Unity、Vuforia或OpenVR,可实现沉浸式体验。例如,通过Unity和Vuforia在WPF中创建AR应用,或利用OpenVR在WPF中集成VR功能,从而提升用户体验并拓展应用功能边界。
79 0
|
4月前
|
C# Windows 开发者
当WPF遇见OpenGL:一场关于如何在Windows Presentation Foundation中融入高性能跨平台图形处理技术的精彩碰撞——详解集成步骤与实战代码示例
【8月更文挑战第31天】本文详细介绍了如何在Windows Presentation Foundation (WPF) 中集成OpenGL,以实现高性能的跨平台图形处理。通过具体示例代码,展示了使用SharpGL库在WPF应用中创建并渲染OpenGL图形的过程,包括开发环境搭建、OpenGL渲染窗口创建及控件集成等关键步骤,帮助开发者更好地理解和应用OpenGL技术。
296 0
|
4月前
|
开发者 C# 容器
【独家揭秘】当WPF邂逅DirectX:看这两个技术如何联手打造令人惊艳的高性能图形渲染体验,从环境搭建到代码实践,一步步教你成为图形编程高手
【8月更文挑战第31天】本文通过代码示例详细介绍了如何在WPF应用中集成DirectX以实现高性能图形渲染。首先创建WPF项目并使用SharpDX作为桥梁,然后在XAML中定义承载DirectX内容的容器。接着,通过C#代码初始化DirectX环境,设置渲染逻辑,并在WPF窗口中绘制图形。此方法适用于从简单2D到复杂3D场景的各种图形处理需求,为WPF开发者提供了高性能图形渲染的技术支持和实践指导。
269 0
|
4月前
|
C# 开发者 Windows
WPF遇上Office:一场关于Word与Excel自动化操作的技术盛宴,从环境搭建到代码实战,看WPF如何玩转文档处理的那些事儿
【8月更文挑战第31天】Windows Presentation Foundation (WPF) 是 .NET Framework 的重要组件,以其强大的图形界面和灵活的数据绑定功能著称。本文通过具体示例代码,介绍如何在 WPF 应用中实现 Word 和 Excel 文档的自动化操作,包括文档的读取、编辑和保存等。首先创建 WPF 项目并设计用户界面,然后在 `MainWindow.xaml.cs` 中编写逻辑代码,利用 `Microsoft.Office.Interop` 命名空间实现 Office 文档的自动化处理。文章还提供了注意事项,帮助开发者避免常见问题。
281 0
|
4月前
|
C# UED 开发者
WPF打印功能实现秘籍:从页面到纸张,带你玩转WPF打印技术大揭秘!
【8月更文挑战第31天】在WPF应用开发中,打印功能至关重要,不仅能提升用户体验,还增强了应用的实用性。本文介绍WPF打印的基础概念与实现方法,涵盖页面元素打印、打印机设置及打印预览。通过具体案例,展示了如何利用`PrintDialog`和`PrintDocument`控件添加打印支持,并使用`PrinterSettings`类进行配置,最后通过`PrintPreviewWindow`实现打印预览功能。
432 0
|
存储 自然语言处理 C#
WPF技术之Binding
WPF(Windows Presentation Foundation)是微软推出的一种用于创建应用程序用户界面的框架。Binding(绑定)是WPF中的一个重要概念,它用于在界面元素和数据源之间建立关联。通过Binding,可以将界面元素(如文本框、标签、列表等)与数据源(如对象、集合、属性等)进行绑定,从而实现数据的双向传递和同步更新。
292 2
WPF技术之Binding
|
前端开发 C# 容器
WPF技术之Command
WPF Command是一种在MVVM(Model-View-ViewModel)模式中用于处理用户界面交互的机制。它可以将用户界面事件(如按钮点击、菜单项选择等)与应用程序逻辑(命令处理)解耦,同时提供了一种便捷的方式来管理和执行命令
317 2
|
C#
WPF技术之动画系列-上下运动
本例子展现动画小球上下循环运动
226 0