原文:WPF的逻辑树与视觉树(2)Visual容器

 

一.摘要

虽然我们平时几乎不会从该类派生,但要想了解视觉树就必须要了解Visual,Visual是一个基本抽象类,继承自DependencyObject.其是所有控件的基类.并提供了视觉树操作的基本方法.

二.提纲

  1. 视觉树是一棵树
  2. 遍历视觉树
  3. 内置Visual集合容器ContainerVisual
  4. 小结

视觉树是一棵树

这好像是一句废话,但也没有错.我们来看下Visual提供的一些基本的成员。

首先我们创立一个测试的对象

public class DefaultVisual : Visual
{
public string Key { get; set; } protected override void OnVisualChildrenChanged(DependencyObject visualAdded, DependencyObject visualRemoved)
{
Console.WriteLine(this.Key + " ChildrenChanged");
if (visualAdded != null) Console.WriteLine((visualAdded as DefaultVisual).Key+" Added");
if (visualRemoved != null) Console.WriteLine((visualRemoved as DefaultVisual).Key+"Removed");
base.OnVisualChildrenChanged(visualAdded, visualRemoved);
} protected override void OnVisualParentChanged(DependencyObject oldParent)
{
Console.WriteLine(this.Key + " ParentChanged");
if (oldParent != null) Console.WriteLine((oldParent as DefaultVisual).Key);
base.OnVisualParentChanged(oldParent);
}
}

测试代码

public static void Test()
{
var test1 = new DefaultVisual();
test1.Key = "test1";
var test2 = new DefaultVisual();
test2.Key = "test2";
test1.AddVisualChild(test2);
var test3 = new DefaultVisual();
test3.Key = "test3";
test2.AddVisualChild(test3);
var test4 = new DefaultVisual();
test4.Key = "test4";
test1.AddVisualChild(test4);
test1.RemoveVisualChild(test4);
}

结果

2.遍历视觉树

在调用AddVisualChild的时候,将会为两个Visual之间建立父子关系,子级知道父级,但父级却不知道有几个子级.所以很难遍历全部节点.需要把子节点给保存下来.Visual提供了两个成员用于视觉树的遍历,只要实现这两个成员就可以使用VisualTreeHelper进行遍历了.

下面我们就来实现这两个成员

3.Visual容器

Visual本身具备一些功能,同时也可以充当容器。

在实际情况下,容器分为两种,单容器和集合容器.比如Border就是一个单容器,其内部只可以放一个元素.Panel是一个集合容器.可以放多个元素.

单容器实现

public class SigletonVisual : DefaultVisual
{
public Visual _child;
public Visual Child
{
get
{
return _child;
}
set
{
this.RemoveVisualChild(_child);
this.AddVisualChild(value);
_child = value;
}
} protected override Visual GetVisualChild(int index)
{
return _child;
} protected override int VisualChildrenCount
{
get
{
if (this._child != null)
{
return 1;
}
return 0;
}
}
}

集合容器实现

public class PanelVisual : DefaultVisual
{
public List<Visual> Visuals { get; set; } public PanelVisual()
{
Visuals = new List<Visual>(5);
} public void Add(Visual visual)
{
Visuals.Add(visual);
this.AddVisualChild(visual);
} protected override Visual GetVisualChild(int index)
{
return Visuals[index];
} protected override int VisualChildrenCount
{
get
{
return Visuals.Count;
}
}
}

遍历测试

void Test()
{
var test1 = new PanelVisual();
test1.Key = "test1";
var test2 = new PanelVisual();
test2.Key = "test2";
test1.Add(test2);
var test3 = new PanelVisual();
test3.Key = "test3";
test2.Add(test3);
var test4 = new PanelVisual();
test4.Key = "test4";
test1.Add(test4);
PrintVisualTree(0, test1);
} public void PrintVisualTree(int depth, PanelVisual obj)
{
Console.WriteLine(new string(' ', depth) + obj.Key);
for (int i = 0; i < VisualTreeHelper.GetChildrenCount(obj); i++)
{
PrintVisualTree(depth + 1, VisualTreeHelper.GetChild(obj, i) as PanelVisual);
}
}

测试结果

3.内置Visual集合容器ContainerVisual

其实我们不用这么复杂,WPF内置类ContainerVisual已经默认实现了Visual集合容器,

ContainerVisual内部采用VisualCollection集合来维护视觉树,所以当我们添加Visual的时候,不需要调用AddVisualChild方法,而是应该调用VisualCollection的Add和Remove等方法

如下测试

public class TestVisual : ContainerVisual
{
public static void Test2()
{
var test1 = new TestVisual();
test1.Key = "test1";
var test2 = new TestVisual();
test2.Key = "test2";
test1.Children.Add(test2);
var test3 = new TestVisual();
test3.Key = "test3";
test2.Children.Add(test3);
var test4 = new TestVisual();
test4.Key = "test4";
test1.Children.Add(test4);
PrintVisualTree(0, test1);
} public static void PrintVisualTree(int depth, TestVisual obj)
{
Console.WriteLine(new string(' ', depth) + obj.Key);
for (int i = 0; i < VisualTreeHelper.GetChildrenCount(obj); i++)
{
PrintVisualTree(depth + 1, VisualTreeHelper.GetChild(obj, i) as TestVisual);
}
} public string Key { get; set; }
}

测试结果是一样的,但我们就可以省却手动实现VisualChildrenCount和GetVisualChild这两个成员了.

如果不从ContainerVisual 继承又想简单的维护Visual的话,可以使用VisualCollection来维护.

4.小结

这篇讲到了Visual的基本功能,Visual本身具备父子级关系的功能,但默认没有容器,需要我们自己实现.内置的ContainerVisual 使用VisualCollection实现了一个Visual容器功能.有了容器才能遍历整个视觉树,对Visual进行一些交互.

可能到这里却还没有看到具体UI的呈现.那就下篇了.

WPF的逻辑树与视觉树(2)Visual容器的更多相关文章

  1. 【WPF】逻辑树和视觉树

    WPF中提供了遍历逻辑树和视觉树的辅助类:System.Windows.LogicalTreeHelper和 System.Windows.Media.VisualTreeHelper. 注意遍历的位 ...

  2. WPF的逻辑树与视觉树(1)基本概念

    原文:WPF的逻辑树与视觉树(1)基本概念     一.摘要 逻辑树与视觉树属于WPF的基本概念,学过WPF或者Silverlight的朋友一定会对其有所耳闻,这篇文章将来探讨逻辑树与视觉树的特质以及 ...

  3. WPF的逻辑树和视觉树

    原文:WPF的逻辑树和视觉树 这部分的内容来自于即将出版的新书<WPF Unleashed>的第三章样章.关于什么是逻辑树,我们先看下面的一个伪XAML代码的例子: <Window ...

  4. 理解WPF中的视觉树和逻辑树

    轉載地址:http://blog.csdn.net/changtianshuiyue/article/details/26981797 理解WPF中的视觉树和逻辑树  Understanding th ...

  5. WPF的逻辑树与视觉树(3)Visual呈现

    原文:WPF的逻辑树与视觉树(3)Visual呈现 这篇就点到为止,挑重点讲 绘图方式有两种 1.继承UIElement,重写OnRender方法 public partial class Windo ...

  6. wpf 逻辑树与可视化树

    XAML天生就是用来呈现用户界面的,这是由于它具有层次化的特性.在WPF中,用户界面由一个对象树构建而成,这棵树叫作逻辑树.逻辑树的概念很直观,但是为什么要关注它呢?因为几乎WPF的每一方面(属性.事 ...

  7. WPF学习(4)逻辑树和可视树

    前面几节说了一些WPF的基础,包括XAML和布局等.在接下来的几节,我们来说说WPF的核心概念,包括逻辑树和可视树.依赖对象和依赖属性.路由事件.命令这几个部分.本节介绍下逻辑树(Logical Tr ...

  8. WPF中的逻辑树和可视化树

    WPF中的逻辑树是指XAML元素级别的嵌套关系,逻辑树中的节点对应着XAML中的元素. 为了方便地自定义控件模板,WPF在逻辑树的基础上进一步细化,形成了一个“可视化树(Visual Tree)”,树 ...

  9. WPF路由事件一:逻辑树和可视树

    一.什么是逻辑树 逻辑树就是描述WPF界面元素的实际构成,它是由程序在XAML中所有的UI元素组成.最显著的特点就是由布局控件.或者其他常用的控件组成. <Window x:Class=&quo ...

随机推荐

  1. x264代码剖析(十五):核心算法之宏块编码中的变换编码

    x264代码剖析(十五):核心算法之宏块编码中的变换编码 为了进一步节省图像的传输码率.须要对图像进行压缩,通常採用变换编码及量化来消除图像中的相关性以降低图像编码的动态范围.本文主要介绍变换编码的相 ...

  2. js课程 5-13 js事件绑定和鼠标事件注意事项有哪些

    js课程 5-13  js事件绑定和鼠标事件注意事项有哪些 一.总结 一句话总结:js代码的灵魂就是改变标签的属性和样式,就这两种. 1.js触发改的东西是哪两样? 属性和样式 2.js如何让页面用标 ...

  3. (转)Nginx在RedHat中系统服务配置脚本

    转自:http://binyan17.iteye.com/blog/1688308 以下代码是在前人的基础上,结合自己服务器实际情况修改的,本人服务器环境是:CentOS 6.31.创建启动脚本,  ...

  4. 小型机与pc服务器区别

    按CPU的类型来区分,小型机是基于RISC(精简指令集)架构的专用服务器,而服务器是基于CISC(复杂指令集)架构的PC服务器.小型机相对于普通服务器来说,一直有不可比拟的优势.(1)如必能稳定,它具 ...

  5. 用FATFS在SD卡里写一串数字

    用FATFS写SD卡,如写入数组 s[] ={1,2,3,4,5,6} 想要在txt中显示“123456” 就要把 s[0]=1+'0'    或 s[0]=1+48   或 s[0]=1+0x30  ...

  6. [Angular Unit Testing] Debug unit testing -- component rendering

    If sometime you want to log out the comonent html to see whether the html render correctly, you can ...

  7. Hadoop基本原理之一:MapReduce 分类: A1_HADOOP 2014-08-17 19:26 1113人阅读 评论(0) 收藏

    1.为什么需要Hadoop 目前,一块硬盘容量约为1TB,读取速度约为100M/S,因此完成一块硬盘的读取需时约2.5小时(写入时间更长).若把数据放在同一硬盘上,且全部数据均需要同一个程序进行处理, ...

  8. rabbitMQ服务安装

    1.rabbitMQ简介 1.1.rabbitMQ的优点(适用范围)1. 基于erlang语言开发具有高可用高并发的优点,适合集群服务器.2. 健壮.稳定.易用.跨平台.支持多种语言.文档齐全.3. ...

  9. 【30.49%】【codeforces 569A】Music

    time limit per test2 seconds memory limit per test256 megabytes inputstandard input outputstandard o ...

  10. FullPage.js全屏滚动插件解说

    1.主要功能 1).支持鼠标滚动 2).多个回调函数 3).支持手机.平板触屏事件 4).支持css3动画 5).支持窗口缩放 6).窗口缩放时自动调整 7).可设置滚动宽度.背景颜色.滚动速度.循环 ...