WPF 自定义模板

回顾

上一篇,我们简单介绍了几个基本的控件,本节我们将讲解每个控件的样式的自定义和数据模板的自定义,我们会结合项目中的具体的要求和场景来分析,给出我们实现的方案和最终的运行效果。

本文大纲

1、控件模板及数据模板

2、ListBox深度定制模板。

3、TreeView高级模板使用实例。

控件模板及数据模板

控件模板

什么是控件模板,指定可以在控件的多个实例之间共享 Control 的可视结构和性能方面的方面。控件模板其实就是我们在可视方面的自定义模板,ControlTemplate 允许您指定控件的可视结构。 重写 ControlTemplate 重新生成该控件的可视结构。

模板化控件是 WPF 提供的许多功能之一样式设置和模板化模型。 该样式和模板化模型提供了许多情况下您不需要编写拥有控件这样的大的灵活性。

控件模板包含二方面的内容:VisualTree和Tigger。本篇介绍的内容,完全都是基于这二块的内容进行讨论和说明。

什么是ViewTree

VisualTree就是对应WPF控件的可视元素的定义,下面来举例说明:

上面,我们通过了lable重写了button按钮的控件模板,我们还可以采用更复杂的控件来重写它:

运行后的效果效果就是上面的预览图,我们当然还可以构建更复杂的情况,WPF中基本上所有的控件,都可以定义控件模板。

上面的情况是我们针对一个按钮重写这样的控件模板,如果我们一个页面中有多个控件,并且这些控件的样式都是一样的,唯一的区别是控件的内容或文本不同而已,我们应该如何做呢?这个时候我们就需要把控件模板定义为资源,如下所示:

我们下面添加多个按钮,来看看应用的具体效果:

够简单吧,其实很简单,我们就可以重写控件的模板了,好了,下面来看看更复杂一些的,我们可能想当鼠标滑过,或者按下后,按钮有一个不同的样式,这时候我们就会涉及到前面介绍的Tigger了,下面我们就来看看

具体的代码如下:

在WPF中,为了提高用户体验的效果,实现界面特殊的效果,我们会大量的使用动画来完成。

前面用了大量的篇幅来说明,控件模板和触发器,实现界面的特殊的效果,使用WPF来做事非常的简单。

在第二小节中,我们将会举几个例子来说明项目中的具体用法。

数据模板

数据模板与控件模板不同,主要是针对某种类型的数据而定制的模板,该模板会自动根据绑定的数据类型,在构造界面显示时,根据预先设定的数据模板来组织页面显示的内容。数据模板和控件模板的定义差不多。我们先来定义一个数据模板,然后看看如何使用。

我们来看看代码是如何设定的,才能实现,这样的效果:

上面用到了,绑定,关于更多的绑定,我们在后面的MVVM的实例中大量的使用了绑定。

基于人员信息,构造人员信息绑定集合

将ViewModel与界面建立关联关系

最后,采用ListBox来显示数据项,通过数据绑定来实现

这里我们发现数据模板的效果,并不是非常的好看,这时候,我们可以采用样式模板来完成样式设定。

然后我们重新设置Listview的样式后,运行:

F5运行后效果如下:

ListBox深度模板定制

定制Demo1

上面我们虽然应用了样式,但是还是感觉不好看,而且鼠标点击后没有效果,是因为在触发器中没有做任何的效果设定。

我们如果在触发器中修改为Button或者Border的背景色,再试试呢?

果然是我们想要的效果,那么我们来看看我们只需要在模板中书写如下的简单几行代码即可:

下面我们先来看看一个效果。

ListView具有黑色的背景。可以采用图片或者颜色值

我们将上面的例子,一步步的实现这个效果。

哈哈,就是这个效果,其实实现起来非常的简单,就是重写控件模板即可:

接着:重写触发器,当鼠标滑过时的样式

当鼠标获得焦点也就是按下时的样式。

设置Grid的样式

只需要简单的几步,就完全可以实现一组特别不错的效果。

上面我们把样式都写到页面当中了,对于具体的情况,可能我们希望能够将样式通用,所以,我们会定义一个单独的样式文件,关于样式文件,我们前面的文章中也有提到过,这里就不做特别的说明了。

下面我们来看看另外的一个效果:

对于这样的效果,我们也可以通过ListBox来实现,无非就是重写ListBox的控件模板

下面我们也来一步步的分析下效果的实现

选中后的高亮效果,无非就是设置边框。

然后就是横向显示,需要重写ListBox的ItemsPanel修改为WrapPanel。

这样,我们的列表项就可以横向显示,然后设置每个列表项的样式:

为了能够看到Border的边框,请设置borderThickness和颜色。

同时保持border内部的子控件与border的边距值,否则出现不了,刚才展示的效果。

代码本身就是这些东西,都是比较简单的。

TreeView高级模板使用实例

Treeview的效果

首先给大家看看实现的具体效果。

上面的效果,还是不错的,不过实现起来的难度就会大很多。

下面给出设计思路和具体的实现代码。

我们实现的效果如上图,屏蔽了原始的+-符号,太难看了,所以,我们重写了相关的样式和模板来达到上述的效果,下面给出具体的实现

A、先定义基本的ViewModel

具体的代码,会提供下载

B、定义TreeView的数据库结构

上面定义的ViewModel只是为了DataTemplate使用的,在DataTemplate那里一看就明白了。

修改原来的数据结构,如上。

添加一个负责界面绑定的ViewModel

public class PersonViewModelCollection : INotifyPropertyChanged
   {
       System.Collections.ObjectModel.ObservableCollection<ResourceSturctViewModel> persons =
           new System.Collections.ObjectModel.ObservableCollection<ResourceSturctViewModel>();

public PersonViewModelCollection()
       {
           Person tempA = new Person()
           {
               Address = "北京",
               Name = "A",
               Photo = "/Samples.04.Template;Component/Images/logo_small.gif",
               Sex = "男"
           };

Person tempB = new Person()
           {
               Address = "北京",
               Name = "B",
               Photo = "/Samples.04.Template;Component/Images/logo_small.gif",
               Sex = "未知"
           };

List<Person> tempABPersons = new List<Person>();
           tempABPersons.Add(tempA);
           tempABPersons.Add(tempB);
           Person[] tempPersons = tempABPersons.ToArray();

persons.Add(new ResourceSturctViewModel(new Person()
           {
               Address = "北京",
               Name = "A",
               Photo = "/Samples.04.Template;Component/Images/logo_small.gif",
               Sex = "男",
               Childerns = tempPersons

}));

persons.Add(new ResourceSturctViewModel(new Person()
           {
               Address = "河北",
               Name = "B",
               Photo = "/Samples.04.Template;Component/Images/logo_small.gif",
               Sex = "女",
               Childerns = tempPersons
           }));

persons.Add(new ResourceSturctViewModel(new Person()
           {
               Address = "山西",
               Name = "C",
               Photo = "/Samples.04.Template;Component/Images/logo_small.gif",
               Sex = "男",
               Childerns = tempPersons
           }));
       }

public System.Collections.ObjectModel.ObservableCollection<ResourceSturctViewModel> PersonList
       {
           get
           {
               return this.persons;
           }
       }

#region INotifyPropertyChanged 成员

public event PropertyChangedEventHandler PropertyChanged;

#endregion
   }

后台的相关代码,都构建完毕了,下面我们就来看看界面的设计和组织了:

TreeView节点前的展开折叠样式

TrewViewItem的每个节点项的样式:

上面没有设置具体的控件和绑定,而是通过ContentPresenter和ItemsHost来处理的,这样我们就可以结合数据模板来做统一处理,最终将DataTemplate设置的控件自动显示到当前的ContentPresenter和ItemsHost中。

如果不按照上述要求,那么当我们重写TreeView时就会遇到很多莫名其妙的问题,我也是遇到了,才总结出来。

在前面的样式中,我们加入了如下事件,务必写上,这是为了触发Lazyload的操作的。

等你自己试一遍,就会发现其实也不难,只要掌握了对自定义模板的规则,就完全可以自定义更复杂的样式和效果。

出处:https://www.cnblogs.com/hegezhou_hot/archive/2012/10/29/2744197.html

原文地址:https://www.cnblogs.com/mq0036/p/12344806.html

时间: 2024-10-09 21:15:26

WPF 自定义模板的相关文章

WPF 自定义模板 Button闪亮效果

Button的选中Effect,我们看下下面的效果: 让我们再放大一点: 怎么设置上面样式呢?直接设置Button的Effect,有点问题,因为Effect不是四周环绕的.那我们也只能重新定义Template了.下面看控件模板: <ControlTemplate x:Key="LightedBtnTemplate" TargetType="RadioButton"> <Grid> <Grid.RowDefinitions> <

WPF Step By Step 自定义模板

WPF Step By Step 自定义模板 回顾 上一篇,我们简单介绍了几个基本的控件,本节我们将讲解每个控件的样式的自定义和数据模板的自定义,我们会结合项目中的具体的要求和场景来分析,给出我们实现的方案和最终的运行效果. 本文大纲 1.控件模板及数据模板 2.ListBox深度定制模板. 3.TreeView高级模板使用实例. 控件模板及数据模板 控件模板 什么是控件模板,指定可以在控件的多个实例之间共享 Control 的可视结构和性能方面的方面.控件模板其实就是我们在可视方面的自定义模板

WPF三大模板简介

WPF支持以下类型的模板: (1) 控件模板.控件模板可以将自定义模板应用到某一特定类型的所有控件,或是控件的某一实例.决定控件外观的是ControlTemplate,它决定了控件“长成什么样子”,因此控件模板由ControlTemplate类表示.控件模板实际在资源集合或资源字典中定义.例子详见:通过设计ControlTemplate,制作圆角文本框与圆角按钮(http://www.cnblogs.com/zhouhb/p/3284780.html). (2) 数据模板.在WPF中,决定数据外

WPF自定义样式篇-DataGrid

WPF自定义样式篇-DataGrid 先上效果图: 样式: <!--DataGrid样式-->    <Style TargetType="DataGrid">        <Setter Property="RowHeaderWidth" Value="0"></Setter>        <Setter Property="AutoGenerateColumns"

WPF自定义路由事件

一 概要 本文通过实例演示WPF自定义路由事件的使用,进而探讨了路由事件与普通的CLR事件的区别(注:"普通的CLR事件"这个说法可能不太专业,但是,我暂时也找不到什么更好的称呼,就这么着吧,呵呵.)(扩展阅读:例说.NET事件的使用). 二 实例演示与说明 1 新建DetailReportEventArgs类,该类派生自RoutedEventArgs类,RoutedEventArgs类包含与路由事件相关的状态信息和事件数据.DetailReportEventArgs类中定义了属性Ev

WPF 自定义柱状图 BarChart

WPF 自定义柱状图 1. <UserControl x:Class="WpfApplication11.BarChartControl" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:mc="http://schemas.o

RazorEngine在非MVC下的使用,以及使用自定义模板

---恢复内容开始--- RazorEngine模板引擎大大的帮助了我们简化字符串的拼接与方法的调用,开源之后,现在在简单的web程序,winform程序,甚至控制台程序都可以利用它来完成. 但如何在使用中调用方法和使用自定义模板呢?来看这样一个例子 1 string str="hello @Model.Name"; 2 string parse=Razor.Prase(str,new {Name="Tom"}); 3 Console.WriteLine(parse

WPF自定义窗口基类

WPF自定义窗口基类时,窗口基类只定义.cs文件,xaml文件不定义.继承自定义窗口的类xaml文件的根节点就不再是<Window>,而是自定义窗口类名(若自定义窗口与继承者不在同一个命名空间,还得加上命名空间),继承自定义窗口类后台代码也得修改为继承自自定义窗口exp: //继承Window类的自定义窗口类 namespace WPF_Study.Entity { using System.Windows; public class WindowBase:Window { private c

Django 自定义模板标签和过滤器

1.创建一个模板库 使用模板过滤器的时候,直接把过滤器写在app里,例如:在app里新建一个templatetags的文件夹,这个目录应当和 models.py . views.py 等处于同一层次.例如: books/     __init__.py     models.py     templatetags/     views.py 在 templatetags 中创建两个空文件:一个 __init__.py (告诉Python这是一个包含了Python代码的包)和一个用来存放你自定义的