zoukankan      html  css  js  c++  java
  • 8天入门wpf—— 第四天 模板

          今天说下wpf中的模板,前面一篇中我们讲到了style,但是style所能做的仅仅是在现有控件的基础上进行修修补补,但是如果我们想

    彻底颠覆控件样式,那么我们就必须使用这一篇所说的模板。

       老外写书都喜欢在篇头搞一个类图,方便我们宏观认识,这里我也上一个。

    一:控件模板

    1:ControlTemplate

       我们知道wpf的控件都是继承自Control,在Control类中有一个Template属性,类型就是ControlTemplate。

    那么利用这个ControlTemplate就可以彻底的颠覆控件的默认外观,这里我把一个checkbox变成一个小矩形,蛮有意思的。

     1 <Window x:Class="WpfApplication1.MainWindow"
     2          xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
     3          xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
     4          xmlns:sys="clr-namespace:System;assembly=mscorlib"
     5          Title="MainWindow" Height="350" Width="525">
     6     <Window.Resources>
     7         <ControlTemplate x:Key="rect" TargetType="{x:Type CheckBox}">
     8             <StackPanel>
     9                 <Rectangle Name="breakRectangle" Stroke="Red" StrokeThickness="2" Width="20" Height="20">
    10                     <Rectangle.Fill>
    11                         <SolidColorBrush Color="White"/>
    12                     </Rectangle.Fill>
    13                 </Rectangle>
    14             </StackPanel>
    15         </ControlTemplate>
    16     </Window.Resources>
    17     <Canvas>
    18         <CheckBox Template="{StaticResource ResourceKey=rect}" Content="我是CheckBox"/>
    19     </Canvas>
    20 </Window>

    确实,我们干了一件漂亮的事情,把checkbox变成了“小矩形”,但是我们发现了一个小问题,为什么我的Content=“xxx”没有显示到模板上?

    很简单,我们已经重定义了控件模板,默认模板将会被覆盖...

    2:ContentPresenter

         幸好,wpf给我们提供了一个ContentPresenter,它的作用就是把原有模板的属性原封不动的投放到自定义模板中。

     1 <Window x:Class="WpfApplication1.MainWindow"
     2          xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
     3          xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
     4          xmlns:sys="clr-namespace:System;assembly=mscorlib"
     5          Title="MainWindow" Height="350" Width="525">
     6     <Window.Resources>
     7         <ControlTemplate x:Key="rect" TargetType="{x:Type CheckBox}">
     8             <StackPanel>
     9                 <Rectangle Name="breakRectangle" Stroke="Red" StrokeThickness="2" Width="20" Height="20">
    10                     <Rectangle.Fill>
    11                         <SolidColorBrush Color="White"/>
    12                     </Rectangle.Fill>
    13                 </Rectangle>
    14                 <ContentPresenter/>
    15             </StackPanel>
    16         </ControlTemplate>
    17     </Window.Resources>
    18     <Canvas>
    19         <CheckBox Template="{StaticResource ResourceKey=rect}" Content="我是CheckBox"/>
    20     </Canvas>
    21 </Window>

    当然你也可以玩一些小技巧,比如我想在"矩形“和”文字”中间设置边距,那么我们可以设置ContentPresenter的margin。

     1 <Window x:Class="WpfApplication1.MainWindow"
     2          xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
     3          xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
     4          xmlns:sys="clr-namespace:System;assembly=mscorlib"
     5          Title="MainWindow" Height="350" Width="525">
     6     <Window.Resources>
     7         <ControlTemplate x:Key="rect" TargetType="{x:Type CheckBox}">
     8             <StackPanel>
     9                 <Rectangle Name="breakRectangle" Stroke="Red" StrokeThickness="2" Width="20" Height="20">
    10                     <Rectangle.Fill>
    11                         <SolidColorBrush Color="White"/>
    12                     </Rectangle.Fill>
    13                 </Rectangle>
    14                 <ContentPresenter Margin="10" />
    15             </StackPanel>
    16         </ControlTemplate>
    17     </Window.Resources>
    18     <Canvas>
    19         <CheckBox Template="{StaticResource ResourceKey=rect}"  Content="我是CheckBox"/>
    20     </Canvas>
    21 </Window>

    如果你够聪明,你会发现我设置的margin是一个非常呆板的事情,意思就是说能不能根据具体控件灵活控制margin呢?答案肯定是没问题的,

    因为我们记得一个控件可以绑定到另一个控件上,比如这里我将模板中的Margin绑定到原控件中的Padding上去。

     1 <Window x:Class="WpfApplication1.MainWindow"
     2          xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
     3          xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
     4          xmlns:sys="clr-namespace:System;assembly=mscorlib"
     5          Title="MainWindow" Height="350" Width="525">
     6     <Window.Resources>
     7         <ControlTemplate x:Key="rect" TargetType="{x:Type CheckBox}">
     8             <StackPanel>
     9                 <Rectangle Name="breakRectangle" Stroke="Red" StrokeThickness="2" Width="20" Height="20">
    10                     <Rectangle.Fill>
    11                         <SolidColorBrush Color="White"/>
    12                     </Rectangle.Fill>
    13                 </Rectangle>
    14                 <ContentPresenter Margin="{TemplateBinding Padding}" />
    15             </StackPanel>
    16         </ControlTemplate>
    17     </Window.Resources>
    18     <Canvas>
    19         <CheckBox Template="{StaticResource ResourceKey=rect}"  Content="我是CheckBox" Padding="10"/>
    20     </Canvas>
    21 </Window>

    3:Trigger

         我们知道style里面也是有trigger的,废话不多说,上代码说话。

     1 <Window x:Class="WpfApplication1.MainWindow"
     2          xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
     3          xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
     4          xmlns:sys="clr-namespace:System;assembly=mscorlib"
     5          Title="MainWindow" Height="350" Width="525">
     6     <Window.Resources>
     7         <ControlTemplate x:Key="rect" TargetType="{x:Type CheckBox}">
     8             <ControlTemplate.Resources>
     9                 <SolidColorBrush x:Key="redBrush" Color="Red"/>
    10             </ControlTemplate.Resources>
    11             <StackPanel>
    12                 <Rectangle Name="breakRectangle" Stroke="Red" StrokeThickness="2" Width="20" Height="20">
    13                     <Rectangle.Fill>
    14                         <SolidColorBrush Color="White"/>
    15                     </Rectangle.Fill>
    16                 </Rectangle>
    17                 <ContentPresenter/>
    18             </StackPanel>
    19             <ControlTemplate.Triggers>
    20                 <Trigger Property="IsChecked" Value="True">
    21                     <Setter TargetName="breakRectangle" Property="Fill" Value="{StaticResource ResourceKey=redBrush}">
    22                     </Setter>
    23                 </Trigger>
    24             </ControlTemplate.Triggers>
    25         </ControlTemplate>
    26     </Window.Resources>
    27     <Canvas>
    28         <CheckBox Template="{StaticResource ResourceKey=rect}"  Content="我是CheckBox"/>
    29     </Canvas>
    30 </Window>

    最后形成的效果就是当checkbox选中时为实心框,不选中为空心框。

    4:与Style混搭

       可能刚才我也说了,style只能在原有的控件基础上修修补补,如果我们让Style修补Control控件的Template属性时,此时我们是不是

    就可以实现ControlTemplate和Style的混搭呢?

     1 <Window x:Class="WpfApplication1.MainWindow"
     2          xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
     3          xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
     4          xmlns:sys="clr-namespace:System;assembly=mscorlib"
     5          Title="MainWindow" Height="350" Width="525">
     6     <Window.Resources>
     7         <Style x:Key="cbx" TargetType="{x:Type CheckBox}">
     8             <Setter Property="Template">
     9                 <Setter.Value>
    10                     <ControlTemplate TargetType="{x:Type CheckBox}">
    11                         <ControlTemplate.Resources>
    12                             <SolidColorBrush x:Key="redBrush" Color="Red"/>
    13                         </ControlTemplate.Resources>
    14                         <StackPanel>
    15                             <Rectangle Name="breakRectangle" Stroke="Red" StrokeThickness="2" Width="20" Height="20">
    16                                 <Rectangle.Fill>
    17                                     <SolidColorBrush Color="White"/>
    18                                 </Rectangle.Fill>
    19                             </Rectangle>
    20                             <ContentPresenter/>
    21                         </StackPanel>
    22                         <ControlTemplate.Triggers>
    23                             <Trigger Property="IsChecked" Value="True">
    24                                 <Setter TargetName="breakRectangle" Property="Fill" Value="{StaticResource ResourceKey=redBrush}">
    25                                 </Setter>
    26                             </Trigger>
    27                         </ControlTemplate.Triggers>
    28                     </ControlTemplate>
    29                 </Setter.Value>
    30             </Setter>
    31         </Style>
    32 
    33     </Window.Resources>
    34     <Canvas>
    35         <CheckBox Style="{StaticResource ResourceKey=cbx}" Content="我是CheckBox"/>
    36     </Canvas>
    37 </Window>

    二:数据模板

       现在我们已经知道“控件模板”是用于改变控件外观,那么“数据模板”顾名思义就是控制数据的显示方式,下面做个demo让person绑定到listbox上。

     1 namespace WpfApplication1
     2 {
     3     /// <summary>
     4     /// MainWindow.xaml 的交互逻辑
     5     /// </summary>
     6     public partial class MainWindow : Window
     7     {
     8         public static string name = "一线码农";
     9 
    10         public MainWindow()
    11         {
    12             InitializeComponent();
    13         }
    14     }
    15 
    16     public class PersonList : ObservableCollection<Person>
    17     {
    18         public PersonList()
    19         {
    20             this.Add(new Person() { Name = "一线码农", Age = 24, Address = "上海" });
    21             this.Add(new Person() { Name = "小师妹", Age = 20, Address = "上海" });
    22         }
    23     }
    24 
    25     public class Person
    26     {
    27         public string Name { get; set; }
    28 
    29         public int Age { get; set; }
    30 
    31         public string Address { get; set; }
    32     }
    33 }

    xaml:

     1 <Window x:Class="WpfApplication1.MainWindow"
     2          xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
     3          xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
     4          xmlns:sys="clr-namespace:System;assembly=mscorlib"
     5         xmlns:src="clr-namespace:WpfApplication1"
     6          Title="MainWindow" Height="350" Width="525">
     7     <Window.Resources>
     8         <ObjectDataProvider x:Key="personList" ObjectType="{x:Type src:PersonList}"/>
     9     </Window.Resources>
    10     <Grid>
    11         <ListBox ItemsSource="{Binding Source={StaticResource ResourceKey=personList}}"></ListBox>
    12     </Grid>
    13 </Window>

    最后我们发现,listbox中并没有呈现我们需要的数据,只是呈现了当前类的ToString()方法,很简单,因为我们绑定的不是简单的数据类型集合,

    而是多字段的复杂类型,更重要的是我们并没有告诉wpf该如何呈现person数据。

    <1>重写Tostring()

        既然wpf在Render数据的时候呈现的是当前的ToString()形式,那下面我们来重写ToString()试试看。

     1     public class Person
     2     {
     3         public string Name { get; set; }
     4 
     5         public int Age { get; set; }
     6 
     7         public string Address { get; set; }
     8 
     9         public override string ToString()
    10         {
    11             return string.Format("姓名:{0}, 年龄:{1}, 地址:{2}", Name, Age, Address);
    12         }
    13     }

    最后看看效果,如我们所愿,person信息已经呈现。

    <2>DataTemplate重写

       或许有的人比较苛刻,他需要person是作为矩形一块一块的呈现,而不是这些简单的形式,那么此时我们就可以用DataTemplate来颠覆。

     1 <Window x:Class="WpfApplication1.MainWindow"
     2          xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
     3          xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
     4          xmlns:sys="clr-namespace:System;assembly=mscorlib"
     5         xmlns:src="clr-namespace:WpfApplication1"
     6          Title="MainWindow" Height="350" Width="525">
     7     <Window.Resources>
     8         <ObjectDataProvider x:Key="personList" ObjectType="{x:Type src:PersonList}"/>
     9         <DataTemplate x:Key="rect">
    10             <Border Name="border" BorderBrush="Aqua" BorderThickness="1" Padding="5" Margin="5">
    11                 <StackPanel>
    12                     <StackPanel Orientation="Horizontal">
    13                         <TextBlock Text="{Binding Name}" Margin="5,0,0,0"/>
    14                         <TextBlock Text="{Binding Age}" Margin="5,0,0,0"/>
    15                     </StackPanel>
    16                     <StackPanel Orientation="Horizontal">
    17                         <TextBlock Text="{Binding Address}" Margin="5,0,0,0"/>
    18                     </StackPanel>
    19                 </StackPanel>
    20             </Border>
    21         </DataTemplate>
    22     </Window.Resources>
    23     <Grid>
    24         <ListBox ItemsSource="{Binding Source={StaticResource ResourceKey=personList}}"
    25                   ItemTemplate="{StaticResource ResourceKey=rect}"></ListBox>
    26     </Grid>
    27 </Window>

    哈哈,果然是以一块一块的形式展现,大功告成,当然这里的”触发器“和”style混搭“跟ConrolTemplate非常相似,我想应该不需要累赘了。

    三: ItemsPanelTemplate

          在条目控件(ItemControl)里面,有一个属性叫ItemPanel,类型是ItemPanelTemplate。

    那么ItemsPanelTemplate主要用来干什么的呢?首先我们要知道常见的条目控件有:ListBox,Menu,StatusBar,比如拿ListBox来说,

    我们经过仔细研究,发现ItemBox的ItemPanel其实是一个VisualizingStackPanel,就是说ListBox的每一项的排列方式是遵循StackPanel的

    原则,也就是从上到下的排列方式,比如”一线码农“和”小师妹“是按照竖行排列方式,好,我现在的要求就是能够”横排“,该如何做到呢?

     1 <Window x:Class="WpfApplication1.MainWindow"
     2          xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
     3          xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
     4          xmlns:sys="clr-namespace:System;assembly=mscorlib"
     5         xmlns:src="clr-namespace:WpfApplication1"
     6          Title="MainWindow" Height="350" Width="525">
     7     <Window.Resources>
     8         <ObjectDataProvider x:Key="personList" ObjectType="{x:Type src:PersonList}"/>
     9         <DataTemplate x:Key="rect">
    10             <Border Name="border" BorderBrush="Aqua" BorderThickness="1" Padding="5" Margin="5">
    11                 <StackPanel>
    12                     <StackPanel Orientation="Horizontal">
    13                         <TextBlock Text="{Binding Name}" Margin="5,0,0,0"/>
    14                         <TextBlock Text="{Binding Age}" Margin="5,0,0,0"/>
    15                     </StackPanel>
    16                     <StackPanel Orientation="Horizontal">
    17                         <TextBlock Text="{Binding Address}" Margin="5,0,0,0"/>
    18                     </StackPanel>
    19                 </StackPanel>
    20             </Border>
    21         </DataTemplate>
    22         <ItemsPanelTemplate x:Key="items">
    23             <StackPanel Orientation="Horizontal" VerticalAlignment="Center" HorizontalAlignment="Center"/>
    24         </ItemsPanelTemplate>
    25     </Window.Resources>
    26     <Grid>
    27         <ListBox ItemsSource="{Binding Source={StaticResource ResourceKey=personList}}"
    28                   ItemTemplate="{StaticResource ResourceKey=rect}" ItemsPanel="{StaticResource ResourceKey=items}"></ListBox>
    29     </Grid>
    30 </Window>

    哈哈,确实有意思,我们改变了ListBox中Item的默认排序方向,当然在menu,statusBar中我们也可以用同样的方式来更改。

    四: HierarchicalDataTemplate

           它是针对具有分层数据结构的控件设计的,比如说TreeView,相当于可以每一个层级上做DataTemplate,很好很强大。

     1 <Window x:Class="WpfApplication1.MainWindow"
     2          xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
     3          xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
     4          xmlns:sys="clr-namespace:System;assembly=mscorlib"
     5         xmlns:src="clr-namespace:WpfApplication1"
     6          Title="MainWindow" Height="350" Width="525">
     7     <Window.Resources>
     8         <XmlDataProvider x:Key="Info" XPath="Nations">
     9             <x:XData>
    10                 <Nations xmlns="">
    11                     <Nation Name="中国">
    12                         <Provinces>
    13                             <Province Name="安徽">
    14                                 <Citys>
    15                                     <City Name="安庆">
    16                                         <Countrys>
    17                                             <Country Name="潜山"/>
    18                                             <Country Name="桐城"/>
    19                                         </Countrys>
    20                                     </City>
    21                                     <City Name="合肥">
    22                                         <Countrys>
    23                                             <Country Name="长丰"/>
    24                                             <Country Name="肥东"/>
    25                                         </Countrys>
    26                                     </City>
    27                                 </Citys>
    28                             </Province>
    29                             <Province Name="江苏">
    30                                 <Citys>
    31                                     <City Name="南京">
    32                                         <Countys>
    33                                             <Country Name="溧水"/>
    34                                             <Country Name="高淳"/>
    35                                         </Countys>
    36                                     </City>
    37                                     <City Name="苏州">
    38                                         <Countys>
    39                                             <Country Name="常熟"/>
    40                                         </Countys>
    41                                     </City>
    42                                 </Citys>
    43                             </Province>
    44                         </Provinces>
    45                     </Nation>
    46                 </Nations>
    47             </x:XData>
    48         </XmlDataProvider>
    49         <HierarchicalDataTemplate DataType="Nation" ItemsSource="{Binding XPath=Provinces/Province}">
    50             <StackPanel Background="AliceBlue">
    51                 <TextBlock FontSize="20" Text="{Binding XPath=@Name}"/>
    52             </StackPanel>
    53         </HierarchicalDataTemplate>
    54         <HierarchicalDataTemplate DataType="Province" ItemsSource="{Binding XPath=Citys/City}">
    55             <StackPanel Background="LightBlue">
    56                 <TextBlock FontSize="18" Text="{Binding XPath=@Name}"/>
    57             </StackPanel>
    58         </HierarchicalDataTemplate>
    59         <HierarchicalDataTemplate DataType="City" ItemsSource="{Binding XPath=Countrys/Country}">
    60             <StackPanel Background="LightBlue">
    61                 <TextBlock FontSize="18" Text="{Binding XPath=@Name}"/>
    62             </StackPanel>
    63         </HierarchicalDataTemplate>
    64         <HierarchicalDataTemplate DataType="Country">
    65             <StackPanel Background="LightSalmon">
    66                 <TextBlock FontSize="18" Text="{Binding XPath=@Name}"/>
    67             </StackPanel>
    68         </HierarchicalDataTemplate>
    69     </Window.Resources>
    70     <TreeView ItemsSource="{Binding Source={StaticResource ResourceKey=Info},XPath=Nation}"></TreeView>
    71 </Window>

  • 相关阅读:
    树莓派上跑.NET的segment fault错误
    WiFi、ZigBee、BLE用哪个?
    SQL Server 的字段不为NULL时唯一
    Asp.net 子web application的Session共享
    Gnucash数据库结构
    sql server中的merge
    禁止使用的极限用语
    Git 操作指南
    Excel VBA保护工作表
    WPF多语言化的实现
  • 原文地址:https://www.cnblogs.com/huangxincheng/p/2566595.html
Copyright © 2011-2022 走看看