画刷填充区域,不管是元素的背景色、前景色以及边框,还是形状的内部填充和笔画(Stroke)。最简单的画刷类型是SolidColorBrush,这种画刷填充一种固定、连续的颜色。在XAML中设置形状的Stroke或Fill属性时,使用的是SolidColorBrush画刷,他们在后台完成绘制。

  下面是几个与画刷相关的更基本的方面:

  •   画刷支持更改通知,因为他们继承自Freezable类。因此,如果改变了画刷,任何使用画刷的元素都会自动重新绘制自身。
  •   画刷支持部分透明。为此,只需要修改Opacity属性,使背景能够透过前面的内容进行显示。
  •   通过SystemBrushes类可以访问这样的画刷:此类画刷使用Windows系统设置为当前计算机定义的首选颜色。

  SolidColorBrush画刷无疑非常有用,但还有其他几个继承自System.Windows.Media.Brush的类,通过这些类可得到更新颖的效果。下表列出了所有这些类。

表 画刷类

一、SolidColorBrush画刷

  在大多数控件中,通过设置Foreground属性绘制文本颜色,并设置Background属性绘制文本背后的空间。形状使用类似但不同的属性:Stroke属性用于绘制形状的边框,而Fill属性用于绘制形状的内部。

  可在XAML中使用颜色名设置Stroke和Fill属性,对于这种情况,WPF解析器自动创建匹配的SolidColorBrush对象。也可以使用代码设置Stroke和Fill属性,但需要显示地创建SolidColorBrush对象。

//Create a brush from a named color:
cmd.Background=new SolidColorBrush(Colors.AliceBlue); //Create a brush from a system color:
cmd.Background=SystemColors.ControlBrush; //Create a brush from color values:
int red=;int green=; int blue=;
cmd.Foreground=new SolidColorBrush(Color.FromRgb(red,green,blue));

二、LinearGradientBrush画刷

  可通过LinearGradientBrush画刷创建从一种颜色变化到另一种颜色的混合填充。

  下面可能是最简单的渐变。该渐变从蓝色(左上角)到白色(右下角)在对角线上对矩形进行着色。

<Rectangle Width="150" Height="100" Margin="5">
<Rectangle.Fill>
<LinearGradientBrush >
<GradientStop Color="Blue" Offset="0"/>
<GradientStop Color="White" Offset="1" />
</LinearGradientBrush>
</Rectangle.Fill>
</Rectangle>

  为创建这种渐变效果,需要为每种颜色添加一个GradientStop对象,还需要在渐变中使用0~1的偏移值放置每种颜色。在该例中,用于蓝色的GradientStop对象的偏移值为0,这意味着它被放在渐变的开头。用于白色的GradientStop对象的偏移值为1,这意味着将它放在末尾。通过改变这些值,可调整渐变从一种颜色变化到另一种颜色的速度。例如,如果将白色的GradientStop设置为0.5,渐变就会在中间(两个拐角的中点)从蓝色(左上角)混合到白色。矩形的右边将会是纯白色的。

  上面的标记创建了从一个拐角拉伸到另一个拐角的对象填充渐变。然而,可能希望创建自上而下或从一边向另一边混合的渐变,或是使用不同的对角线角度。可使用LinearGradientBrush的StartPoint和EndPoint属性控制这些细节。可以通过这些属性选择第一种颜色开始变化的点,以及最后一种颜色结束变化的点(中间的区域被渐变混合)。但这里存在一个古怪的问题:用于开始点和结束点的坐标不是真实坐标。相反,LinearGradientBrush画刷将点(0,0)指定为希望希望填充的区域的左上角,将点(1,1)指定为希望填充的区域的右下角,而不管该区域实际上有多高和多宽。

  为创建自上而下的横向填充,可将用于左上角的(0,0)点作为开始点,并将(0,1)点作为结束点,该点表示左下角。为了创建从一边到另一边的垂直填充(不倾斜),可以使用点(0,0)作为开始点,并使用右上角的点(1,0)作为结束点。

  通过为渐变提供不是填充区域拐角点的开始点和结束点。可得到更灵活的渐变。例如,渐变可从点(0,0)拉伸到点(0,0.5),该点是左侧边缘上的中点。这会创建压缩的线性渐变——一种颜色从顶部开始,在中间混合到第二种颜色。形状的后半部分使用第二种颜色。但可用LinearGradientBrush.SpreadMethod属性改变这种行为。默认情况下,该属性使用Pad(这意味着渐变之外的区域使用恰当的纯色填充),但也可使用Reflect(翻转渐变,从第二种颜色反向渐变大偶第一种颜色)或Repeat(复制相同的颜色变化过程)。

  LinearGradientBrush画刷还可通过添加两个以上的GradientStop对象,创建具有两种以上的颜色渐变。例如,下面的渐变实现了彩虹效果:

<Rectangle Width="150" Height="100" Grid.Row="4" Margin="5">
<Rectangle.Fill>
<LinearGradientBrush StartPoint="0,0" EndPoint="1,1">
<GradientStop Color="Yellow" Offset="0.0" />
<GradientStop Color="Red" Offset="0.25" />
<GradientStop Color="Blue" Offset="0.75" />
<GradientStop Color="LimeGreen" Offset="1.0" />
</LinearGradientBrush>
</Rectangle.Fill>
</Rectangle>

  唯一的技巧是为每个GradientStop对象何止合适的偏移值。例如,如果希望变换经过5中颜色,可将第1中颜色的偏移值设置为0,第2中颜色的偏移值设置为0.25,将第3中颜色的偏移值设置为0.5,第4种颜色的偏移值设置为0.75,将第5中颜色的偏移值设置为1。或者如果希望开始时渐变速度较快,而在结束速度较慢,可将便宜值设置为0、0.1、0.2、0.4、0.6、1。

  下面是LinearGradientBrush画刷的完整示例以及效果图:

<Window x:Class="Drawing.Gradients"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="Gradients" Height="587" Width="347">
<Grid Margin="5">
<Grid.RowDefinitions>
<RowDefinition Height="Auto"></RowDefinition>
<RowDefinition Height="Auto"></RowDefinition>
<RowDefinition Height="Auto"></RowDefinition>
<RowDefinition Height="Auto"></RowDefinition>
<RowDefinition Height="Auto"></RowDefinition>
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition></ColumnDefinition>
<ColumnDefinition Width="Auto"></ColumnDefinition>
</Grid.ColumnDefinitions> <Rectangle Width="150" Height="100" Margin="5">
<Rectangle.Fill>
<LinearGradientBrush >
<GradientStop Color="Blue" Offset="0"/>
<GradientStop Color="White" Offset="1" />
</LinearGradientBrush>
</Rectangle.Fill>
</Rectangle>
<TextBlock Grid.Column="1" VerticalAlignment="Center" Margin="5">Diagonal Linear Gradient</TextBlock> <Rectangle Width="150" Height="100" Margin="5" Grid.Row="1">
<Rectangle.Fill>
<LinearGradientBrush>
<GradientStop Color="Blue" Offset="0"/>
<GradientStop Color="White" Offset="0.5" />
</LinearGradientBrush>
</Rectangle.Fill>
</Rectangle>
<TextBlock Grid.Row="1" Grid.Column="1" VerticalAlignment="Center" Margin="5">With 0.5 Offset for White</TextBlock> <Rectangle Width="150" Height="100" Grid.Row="2" Margin="5">
<Rectangle.Fill>
<LinearGradientBrush StartPoint="0,0" EndPoint="0,1" >
<GradientStop Color="Blue" Offset="0"/>
<GradientStop Color="White" Offset="1" />
</LinearGradientBrush>
</Rectangle.Fill>
</Rectangle>
<TextBlock Grid.Row="2" Grid.Column="1" VerticalAlignment="Center" Margin="5">Horizontal Linear Gradient</TextBlock> <Rectangle Width="150" Height="100" Grid.Row="3" Margin="5">
<Rectangle.Fill>
<LinearGradientBrush StartPoint="0,0" EndPoint="0,0.5" SpreadMethod="Reflect">
<GradientStop Color="Blue" Offset="0"/>
<GradientStop Color="White" Offset="1" />
</LinearGradientBrush>
</Rectangle.Fill>
</Rectangle>
<TextBlock Grid.Row="3" Grid.Column="1" VerticalAlignment="Center" Margin="5">Reflected Gradient</TextBlock> <Rectangle Width="150" Height="100" Grid.Row="4" Margin="5">
<Rectangle.Fill>
<LinearGradientBrush StartPoint="0,0" EndPoint="1,1">
<GradientStop Color="Yellow" Offset="0.0" />
<GradientStop Color="Red" Offset="0.25" />
<GradientStop Color="Blue" Offset="0.75" />
<GradientStop Color="LimeGreen" Offset="1.0" />
</LinearGradientBrush>
</Rectangle.Fill>
</Rectangle>
<TextBlock Grid.Row="4" Grid.Column="1" VerticalAlignment="Center" Margin="5">Multicolored Gradient</TextBlock>
</Grid>
</Window>

Gradients

  渐变画刷并不限于绘制形状。可在使用SolidColorBrush画刷的任何时候替代LinearGradientBrush——例如,填充元素的背景表面(使用Background属性)、填充元素文本的前景色(使用Foreground属性)或者填充边框(使用BorderBrush属性)。如下示例所示:

<Window x:Class="Drawing.GradientText"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="GradientText" Height="300" Width="300">
<Grid>
<TextBlock Margin="5" FontWeight="Bold" FontSize="65" TextWrapping="Wrap" TextAlignment="Center">
<TextBlock.Text>This text uses a gradient.</TextBlock.Text>
<TextBlock.Foreground>
<LinearGradientBrush StartPoint="0,0" EndPoint="1,1">
<GradientStop Color="Yellow" Offset="0.0" />
<GradientStop Color="Red" Offset="0.25" />
<GradientStop Color="Blue" Offset="0.75" />
<GradientStop Color="LimeGreen" Offset="1.0" />
</LinearGradientBrush>
</TextBlock.Foreground>
</TextBlock>
</Grid>
</Window>

GradientText

三、RadialGradientBrush画刷

  RadialGradientBrush画刷和LinearGradientBrush画刷的工作方式类似,也使用一系列具有不同偏移值的颜色。与LinearGradientBrush画刷一样,可使用希望的任意多种颜色。区域是放置渐变的方式。

  为指定第一种颜色在渐变中的开始点,需要使用GradientOrigin属性。默认情况下,渐变的开始点是(0.5,0.5),该点表示填充区域的中心。

  渐变从开始点以环形的方式向外辐射。渐变最终到达内部渐变圆的边缘,这里是渐变的终点。根据所期望的效果,渐变圆的中心可能和渐变开始点对齐,也可能和渐变开始点不对齐。超出内部渐变圆的区域以及填充区域的最外侧边缘。使用在RadialGradientBrush.GradientStops集合中定义的最后一种颜色进行纯色填充。

  可使用三个属性设置内部渐变圆的边界:Center、RadiusX和RadiusY。默认情况下,Center属性被设置为(0.5,0.5),该设置将限定圆的中心放在填充区域的中央,并且该点同时也是渐变开始点。

  RadiusX和RadiusY属性 决定了限定圆的尺寸,默认情况下着两个属性都被设置为0.5.这些值可能不够直观,因为他们根据填充区域的对角范围(一条从填充区域的左上角延伸到右下角的假想线的长度)进行度量。这意味着半径0.5定义了一个圆,该圆的半径是对角线长度的一半。如果填充区域为正方形,使用勾股定理可计算出,该长度大约是填充区域宽度(或宽度)的0.7倍。因此,如果用默认设置填充正方形区域,渐变就从中心点开始,并拉伸大约正方形宽度0.7倍的距离到达最外侧边界。

  对于填充圆形形状并创建发光效果,径向渐变是非常好的选择(水平高超的美工人员通过组合使用渐变创建具有光晕效果的按钮)。一种常见技巧是稍微偏移GradientOrigin点,为形状创建深度感。

<Window x:Class="Drawing.RadialGradient"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="RadialGradient" Height="534" Width="480">
<Grid>
<Grid.RowDefinitions>
<RowDefinition></RowDefinition>
<RowDefinition></RowDefinition>
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition></ColumnDefinition>
<ColumnDefinition Width="Auto"></ColumnDefinition>
</Grid.ColumnDefinitions>
<Ellipse Margin="5" Stroke="Black" StrokeThickness="1">
<Ellipse.Fill>
<RadialGradientBrush RadiusX="1" RadiusY="1" >
<GradientStop Color="White" Offset="0"/>
<GradientStop Color="Blue" Offset="1"/>
</RadialGradientBrush>
</Ellipse.Fill>
</Ellipse>
<TextBlock Grid.Column="1" VerticalAlignment="Center" Margin="5">A Radial Gradient</TextBlock> <Ellipse Margin="5" Grid.Row="1" Stroke="Black" StrokeThickness="1">
<Ellipse.Fill>
<RadialGradientBrush
RadiusX="1" RadiusY="1" GradientOrigin="0.7,0.3"
>
<GradientStop Color="White" Offset="0" />
<GradientStop Color="Blue" Offset="1" />
</RadialGradientBrush> </Ellipse.Fill>
</Ellipse>
<TextBlock Grid.Row="1" Grid.Column="1" VerticalAlignment="Center" Margin="5" TextWrapping="Wrap">A Radial Gradient with an Offset Center</TextBlock> </Grid>
</Window>

RadialGradient

四、ImageBrush画刷

  可通过ImageBrush画刷使用位图填充区域。可使用最常见的文件类型,包括BMP、PNG、GIF以及JPEG文件。可通过设置ImageSource属性来制定希望使用的图像。例如,下面的画刷使用一幅名为logo.jpg的图像绘制Grid面板的背景,在程序集中作为资源包含了该图像:

<Grid>
<Grid.Background>
<ImageBrush ImageSource="logo.jpg" />
</Grid.Background>
</Grid>

  ImageBrush.ImageSource属性和Image元素的Source属性的工作方式相同,这意味着也可以使用指向资源、外部文件或Web站点的URI设置ImageSource属性。也可通过为ImageSource属性提供DrawingImage对象,创建使用由XAML定义的矢量内容的ImageBrush画刷。可通过这种方法降低开销(通过避免使用更耗资源的Shape类的派生类),或使用矢量图像创建平铺模式。

  在该例中,ImageBrush画刷用于绘制单元格的背景。因此,为了适应填充区域,图像会被拉伸。如果Grid面板比图像的原始尺寸大,就会看到改变图像尺寸造成的显示问题(如常见的模糊效果)。如果Grid面板的形状和图像的宽高比不匹配,为了适应Grid面板,图像会变形。

  为控制该行为,可修改ImageBrush.Stretch属性。例如,可将该属性设置为Uniform,从而为了适应缩放图像时保持图像的高宽比,或将该属性设置为None,使用图像的自然尺寸绘制图像(对于这种情况,为适应容器,部分图像可能被裁减掉)。

  如果绘制的图像比填充区域小,图像会根据AlignmentX和AlignmentY属性进行对齐。未填充的区域保持透明。当使用Uniform设置进行缩放,并且填充区域的形状不同时,就会出现这种情况。如果将Stretch属性设置为None,并且填充区域比图像大,也会出现这种情况。

  还可使用Viewbox属性从图像上裁剪有兴趣使用的一小部分。为此,需要指定4个数值以描述希望从源图上裁剪并使用的矩形部分。前连个数值指定矩形开始的左上角,而后两个数值指定矩形的宽度和高度。唯一的问题是Viewbox属性使用的是相对坐标系统,就像渐变画刷使用的坐标系统那样。这一坐标系统将图像上的左上角指定为(0,0),将右下角指定为(1,1)。

  为理解Viewbox属性的工作原理,分析下面的标记:

<ImageBrush ImageSource="logo.jpg" Stretch="Uniform"
Viewbox="0.4 0.5 0.2 0.2"></ImageBrush>

  现在,Viewbox属性从(0.4,0.5)开始,这差不多是用图像的一半出开始(从技术角度看,X坐标是宽度的0.4倍,Y坐标是高度的0.5倍)。然后伸展矩形以填充一个20%宽度和20%高度的小方块作为整幅图像(从技术角度看,矩形的长度为图像宽度的0.2倍,矩形的高度为图像高度的0.2倍)。根据Stretch、AlignmentX以及AlignmentY属性的设置,被裁剪下来的部分图像会被拉伸或剧中显示。下图显示两个使用不同ImageBrush对象填充自身的矩形。最上面的矩形显示了整幅图像,下面的矩形使用Viewbox放大了图像中的一小部分。这两个矩形都使用了纯黑色的边框。

<Window x:Class="Drawing.ImageBrushes"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="ImageBrushes" Height="389.6" Width="419.6">
<Canvas>
<Rectangle Canvas.Left="10" Canvas.Top="10" Width="271" Height="100" Stroke="Black">
<Rectangle.Fill>
<ImageBrush ImageSource="logo.jpg" Stretch="Fill"></ImageBrush>
</Rectangle.Fill>
</Rectangle>
<Rectangle Canvas.Left="10" Canvas.Top="120" Width="200" Height="200" Stroke="Black">
<Rectangle.Fill>
<ImageBrush ImageSource="logo.jpg" Stretch="Uniform"
Viewbox="0.4 0.5 0.2 0.2"></ImageBrush>
</Rectangle.Fill>
</Rectangle>
</Canvas>
</Window>

ImageBrushes

五、平铺的ImageBrush画刷

  除普遍的ImageBrush画刷外,还有其他令人更加激动的内容。可通过在画刷的表面平铺图像来得到一些有趣的效果。

  当平铺图像时,有两种选择:

  •   按比例平铺。填充区域始终具有相同数量的平铺图像。为适应填充区域,平铺的图像会扩展或收缩。
  •   按固定尺寸平铺。平铺图像始终具有相同的尺寸。填充区域的尺寸决定了显示的平铺图像的数量。

  为了平铺一幅图像,需要设置ImageSource属性(指定希望平铺的图像)以及ViewPort、ViewportUnits与TitleMode属性。后三个属性决定了平铺图像的尺寸和排列方式。

  可使用Viewport属性设置每幅平铺图像的尺寸。为使用按比例平铺模式,必须将ViewportUnits属性这是为RelativeToBoundingBox(默认值)。然后使用在两个方向上的坐标范围都是从0到1的按比例坐标定义平铺图像的尺寸。换句话说,如果一幅平铺图像的左上角位于(0,0),右下角位于(1,1),就会占据整个填充区域。为得到平铺模式,为Viewport属性设置的值应当比整个填充区域的尺寸小。如下所示:

<ImageBrush ImageSource="tile.jpg" TileMode="Tile"
Viewport="0 0 0.5 0.5"></ImageBrush>

  上面的标记创建了一个从填充区域的左上角(0,0)开始,并拉伸到中间点(0.5,0.5)的Viewport方框。因此,不管填充区域的大小如何,填充区域始终包含4幅平铺图像。这样行为非常好,因为可确保平铺图像不会在形状的边缘被裁减(当然,如果使用ImageBrush画刷填充非矩形区域,图像仍会被裁剪)。

  因为这个示例中的平铺图像采用相对于填充区域的尺寸,所以更大的填充区域会使用更大的平铺图像,并且因为改变了图像的尺寸。所以会造成一定的模糊效果。此外,如果填充区域不是完美的正方形,相对坐标系统会相应地进行行挤压,从而每个平铺的正方形都会变成矩形。

  可通过修改Stretch属性(默认设置为Fill)改变这种行为。如果将该属性设置为None,可保证平铺图像永不变形,并且保持正确的形状,然而,如果填充区域不是正方形,将在平铺图像之间显示空白空间。

  第三种选择是将Stretch属性设置为UniformToFill这种设置会根据需要裁减平铺的图像。使用这种方式,平铺图像会保持正确的纵横比,而且平铺的图像之间没有空白空间。然而,如果填充的区域不是正方形,就不会看到完整的平铺图像。

  自动改变平铺图像的尺寸是一项非常有用的功能,但也是需要付出代价。有些位图可能不能正确地改变其尺寸。在某种程度上,可通过提供比所需位图更大的位图,为应对这种情况做好准备。当当缩小图像时,这种技术就会导致更模糊的位图。

  另一种定义平铺图像的尺寸的方法是根据原始图像的尺寸使用绝对坐标。为此,将ViewportUnits属性设置为Absolute。下面举一个示例,该例将每幅平铺图像定义为32X32单位大小,并从左上角开始平铺:

<ImageBrush ImageSource="tile.jpg" TileMode="Tile"
ViewboxUnits="Absolute" Viewbox="0 0 32 32"/>

  这种模式的缺点就是填充区域的高度和宽度必须能被32整除。否则,在填充区域便于就会显示部分平铺图像。如果使用ImageBrush画刷填充可改变尺寸的元素,就无法避免该问题,所以必须接受平铺图像未必能与填充区域的边缘对齐这种情况。

  下面是上面示例的完整XAML:

<Window x:Class="Drawing.TileTypes"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="TileTypes" Height="500" Width="296.8">
<Grid Margin="5">
<Grid.RowDefinitions>
<RowDefinition></RowDefinition>
<RowDefinition></RowDefinition>
<RowDefinition></RowDefinition>
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="Auto"/>
<ColumnDefinition />
</Grid.ColumnDefinitions> <TextBlock Margin="3">固定尺寸<LineBreak></LineBreak>平铺</TextBlock>
<Rectangle Grid.Column="1" Stroke="Black">
<Rectangle.Fill>
<ImageBrush ImageSource="tile.jpg" TileMode="Tile"
ViewboxUnits="Absolute" Viewbox="0 0 32 32"/>
</Rectangle.Fill>
</Rectangle> <TextBlock Grid.Row="1" Margin="3">按比例<LineBreak></LineBreak>平铺</TextBlock>
<Rectangle Grid.Row="1" Grid.Column="1" Stroke="Black">
<Rectangle.Fill>
<ImageBrush ImageSource="tile.jpg" TileMode="Tile"
Viewport="0 0 0.5 0.5"></ImageBrush>
</Rectangle.Fill>
</Rectangle> <TextBlock Grid.Row="2" Margin="3">
按比例<LineBreak></LineBreak>平铺(无拉伸)
</TextBlock>
<Rectangle Grid.Row="2" Grid.Column="1" Stroke="Black">
<Rectangle.Fill>
<ImageBrush ImageSource="tile.jpg" TileMode="Tile" Stretch="None"
Viewport="0 0 0.5 0.5"></ImageBrush>
</Rectangle.Fill>
</Rectangle>
</Grid>
</Window>

TileTypes

  下表列出了TileMode枚举值得所有选项.

表 TileMode枚举值

  如果需要 使平铺图像更无缝地混合,翻转行为通常是有用的。例如,如果使用FlipX,相邻的平铺图像总可以无缝地排列。下面举例说明一下。

<Window x:Class="Drawing.TileFlip"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="TileFlip" Height="300" Width="300">
<!-- Overlay labels by putting one UniformGrid on top of another. -->
<Grid>
<UniformGrid>
<Rectangle Stroke="Black">
<Rectangle.Fill>
<ImageBrush ImageSource="tile.jpg" TileMode="Tile"
ViewportUnits="Absolute" Viewport="0 0 37 37"></ImageBrush>
</Rectangle.Fill>
</Rectangle>
<Rectangle Stroke="Black">
<Rectangle.Fill>
<ImageBrush ImageSource="tile.jpg" TileMode="FlipX"
ViewportUnits="Absolute" Viewport="0 0 37 37"></ImageBrush>
</Rectangle.Fill>
</Rectangle>
<Rectangle Stroke="Black">
<Rectangle.Fill>
<ImageBrush ImageSource="tile.jpg" TileMode="FlipY"
ViewportUnits="Absolute" Viewport="0 0 37 37"></ImageBrush>
</Rectangle.Fill>
</Rectangle>
<Rectangle Stroke="Black">
<Rectangle.Fill>
<ImageBrush ImageSource="tile.jpg" TileMode="FlipXY"
ViewportUnits="Absolute" Viewport="0 0 37 37"></ImageBrush>
</Rectangle.Fill>
</Rectangle>
</UniformGrid>
<UniformGrid>
<UniformGrid.Resources>
<Style TargetType="{x:Type TextBlock}">
<Setter Property="HorizontalAlignment" Value="Center"></Setter>
<Setter Property="VerticalAlignment" Value="Bottom"></Setter>
<Setter Property="FontSize" Value="25"></Setter>
<Setter Property="FontWeight" Value="Bold"></Setter>
<Setter Property="Margin" Value="3"></Setter>
</Style>
</UniformGrid.Resources>
<TextBlock>Tile</TextBlock>
<TextBlock>FlipX</TextBlock>
<TextBlock>FlipY</TextBlock>
<TextBlock>FlipXY</TextBlock>
</UniformGrid>
</Grid>
</Window>

TileFlip

 六、VisualBrush画刷

  VisualBrush画刷不常用,使用这种画刷获取元素的可视化内容,并使用该内容填充任意表面。例如,可使用VisualBrush画刷将窗口中某个按钮的外观复制到同一个窗口中的其他位置。然而,复制的按钮不能被单击,也不能通过任何方式与其进行交互。在此就复制了元素的外观。例如,下面的标记片段定义了一个按钮和用于复制该按钮的VisualBrush画刷:

 <Button Name="cmd" Margin="3" Padding="5">Is this a real button?</Button>
<Rectangle Margin="3" Height="{Binding ElementName=cmd,Path=ActualHeight}">
<Rectangle.Fill>
<VisualBrush Visual="{Binding ElementName=cmd}"></VisualBrush>
</Rectangle.Fill>
</Rectangle>

  尽管可在VisualBrush本身定义希望使用的元素,但通常使用绑定表达式引用当前窗口中的额元素。如本例所示,下图显示了原始按钮(在窗口顶部)和几个形状不同的区域,这些区域是用基于按钮的VisualBrush画刷绘制的。完整XAML如下所示:

<Window x:Class="Drawing.VisualBrush"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="VisualBrush" Height="300" Width="300">
<StackPanel Margin="3">
<Button Name="cmd" Margin="3" Padding="5">Is this a real button?</Button>
<Rectangle Margin="3" Height="{Binding ElementName=cmd,Path=ActualHeight}">
<Rectangle.Fill>
<VisualBrush Visual="{Binding ElementName=cmd}"></VisualBrush>
</Rectangle.Fill>
</Rectangle>
<Rectangle Margin="3" Height="50">
<Rectangle.Fill>
<VisualBrush Visual="{Binding ElementName=cmd}"></VisualBrush>
</Rectangle.Fill>
</Rectangle>
<Rectangle Margin="3" Height="150">
<Rectangle.Fill>
<VisualBrush Visual="{Binding ElementName=cmd}"></VisualBrush>
</Rectangle.Fill>
</Rectangle>
</StackPanel>
</Window>

VisualBrush

  VisualBrush监视元素外观的变化。例如,如果复制某个按钮的可视化外观,而且此外按钮收到焦点,VisualBrush画刷会使用新的可视化内容重新绘制填充区域——一个具有焦点的按钮。VisualBrush类继承自TileBrush类,因此,VisualBrush类也支持所有的裁剪、拉伸以及翻转等特性。

七、BitmapCacheBrush画刷

  BitmapCacheBrush画刷在许多方面和VisualBrush画刷类似。尽管VisualBrush类提供了用于引用其他元素的Visual属性,但BitmapCacheBrush类提供了与此作用相同的Target属性。

  两者之间的关键区别是,BitmapCacheBrush画刷采用可视化内容(这些内容以及通过变换、裁剪、效果以及透明设置进行了改变)并要求显卡在显存中存储该内容。这样一来,当需要时可快速地重新绘制内容,而不必要求WPF执行任何额外的工作。

  为配置位图缓存,设置BitmapCacheBrush.BitmapCache属性(使用可预先确定的BitmapCache对象)。下面是最简单的用法:

<Button Name="cmd" Margin="3" Padding="5">Is this a real button?</Button>
<Rectangle Margin="3" Height="{Binding ElementName=cmd,Path=ActualHeight}">
<Rectangle.Fill>
<BitmapCacheBrush Target="{Binding ElementName=cmd}">
</BitmapCacheBrush>
</Rectangle.Fill>
</Rectangle>

  BitmapCacheBrush画刷存在严重缺点:渲染位图以及将其复制到显存的初始化步骤需要比较短但可察觉到得额外时间。如果在窗口中使用BitmapCacheBrush画刷,在窗口第一次绘制自身之前,当渲染BitmapCacheBrush并复制其位图时,将会注意到延迟。因此,在传统窗口中,BitmapCacheBrush起不到多大的帮助作用。

  然而,如果在用户界面中大量使用动画,值得考虑使用位图缓存。这是因为动画会强制窗口在没一秒内重新绘制多次。如果具有复杂的矢量内容,从缓存位图中绘制窗口内容比从头重新绘制窗口要快。但即使是这种情况,也不应当立即使用BitmapCacheBrush画刷。可能更愿意通过为每个希望缓存的元素设置更高级的UIElement.CacheMode属性来应用缓存。对于这种情况,WPF在后台使用BitmapCacheBrush画刷获取相同的效果,但需要做的工作更少。

  根据这些细节,BitmapCacheBrush画刷本身好像不是很有用。然而,如果需要在几个地方绘制单块复杂的可视化内容,使用BitmapCacheBrush画刷是合理的。对于这种情况,通过使用BitmapCacheBrush画刷缓存整个可视化内容比单独缓存每个元素更节省内存。在此输出,这种节省可能得不偿失,除非用户界面还使用了动画。

【WPF学习】第四十章 画刷的更多相关文章

  1. WPF 动画(形状、画刷)

    一:形状 在WPF用户界面中,可以通过形状(Shape)来绘制直线.椭圆.矩形及一些多边形的类.通过这些基本的图像,组合成为复杂的图形. Shape类中,主要的形状有Rectangle(),Ellip ...

  2. 【WPF学习】第二十章 内容控件

    内容控件(content control)是更特殊的控件类型,它们可包含并显示一块内容.从技术角度看,内容控件时可以包含单个嵌套元素的控件.与布局容器不同的是,内容控件只能包含一个子元素,而布局容器主 ...

  3. WPF学习(四) - 附加属性

    冷静了一晚,我就当这次学习的过程是在看狗血剧情的武打小说吧:没有垃圾的武术,只有垃圾的武者…… 还有个话儿怎么说来着:你们是用户,不是客户,也就有个使用的权力.搞清楚身份,别叽叽歪歪的! 没办法,全世 ...

  4. 【WPF学习】第十章 WPF布局示例

    前几章用了相当大的篇幅研究有关WPF布局容器的复杂内容.在掌握了这些基础知识后,就可以研究几个完整的布局示例.通过研究完整的布局示例,可更好的理解各种WPF布局概念在实际窗口中的工作方式. 一.列设置 ...

  5. WPF学习笔记四之命令

    1.概念 对于程序来说,命令就是一个个任务,例如保存,复制,剪切这些操作都可以理解为一个个命令.即当我们点击一个复杂按钮时,此时就相当于发出了一个复制的命令,即告诉文本框执行一个复杂选中内容的操作,然 ...

  6. 简述WPF中的画刷(Brush)

    原文:简述WPF中的画刷(Brush) -------------------------------------------------------------------------------- ...

  7. 【WPF学习】第四十二章 透明

    WPF支持真正的透明效果.这意味着,如果在一个性质或元素上层叠另外几个形状或元素,并让所有这些形状和元素具有不同的透明度,就会看到所期望的效果.通过该特性能够创建透过上面的元素可以看到的的图像背景,这 ...

  8. WPF学习系列之八(形状,画刷和变换)

    形状,画刷和变换   概述: 在许多用户界面技术中,普通控件和自定义绘图之间具有清晰的区别.通常来说,绘图特性只用于特定的应用程序--如游戏,数据可视化和物理仿真等.而WPF具有一个非常不同的原则.它 ...

  9. 【WPF学习】第四十九章 基本动画

    在前一章已经学习过WPF动画的第一条规则——每个动画依赖于一个依赖项属性.然而,还有另一个限制.为了实现属性的动态化(换句话说,使用基于时间的方式改变属性的值),需要有支持相应数据类型的动画类.例如, ...

随机推荐

  1. 「洛谷P1080」「NOIP2012提高组」国王游戏 解题报告

    P1080 国王游戏 题目描述 恰逢 \(H\)国国庆,国王邀请\(n\)位大臣来玩一个有奖游戏.首先,他让每个大臣在左.右手上面分别写下一个整数,国王自己也在左.右手上各写一个整数.然后,让这 \( ...

  2. IOS系统唤醒微信内置地图

    针对前一篇文章 唤醒微信内置地图 后来发现在IOS系统中运行 唤醒地图会无效的问题.因为在IOS上无法解析这俩个字符串的问题! 需要对经纬度 使用 “parseFloat()”进行转换 返回一个浮点数 ...

  3. Arrays.asList 返回值类型

    public static void main(String[] args) { Integer[] datas = {1,2,3,4,5}; List<Integer> list = A ...

  4. 【Spark 内核】 Spark 内核解析-上

    Spark内核泛指Spark的核心运行机制,包括Spark核心组件的运行机制.Spark任务调度机制.Spark内存管理机制.Spark核心功能的运行原理等,熟练掌握Spark内核原理,能够帮助我们更 ...

  5. KMO检验和Bartlett球形检验

    KMO检验和Bartlett球形检验因子分析前,首先进行KMO检验和巴特利球体检验,KMO检验系数>0.5,(巴特利特球体检验的x2统计值的显著性概率)P值<0.05时,问卷才有结构效度, ...

  6. 深入理解Java虚拟机-类加载连接和初始化解析

    不管学习什么,我一直追求的是知其然,还要知其所以然,对真理的追求可以体现在方方面面.人生短短数十载,匆匆一世似烟云,我认为,既然来了,就应该留下一些有意义的东西.本系列文章是结合张龙老师的<深入 ...

  7. IDEA新建maven项目没有webapp目录解决方法

    转载地址:https://www.cnblogs.com/oldzhang1222/p/10429827.html 先创建的页面修改路径 修改路径如下 添加并完善路径\src\main\webapp ...

  8. springboot的yml不显示的原因

    首先排除插件原因 1 安装好插件Ctrl+Alt+S 2 查看修改的application.yml是什么格 在yaml格式中添加*.yaml和*.yml 3 查看maven是否配置完善

  9. C++中的四个智能指针

    只能指针的行为类似常规指针,重要的区别是它负责自动释放所指向的对象.智能指针定义在memory头文件中. 1. auto_ptr(C++11已经舍弃) 由new expression获得的对象,在au ...

  10. 生产者消费者代码学习,Producer_Consuner

    使用BlockingQuery实现生产者者消费者:考虑并发,解耦. 生产者消费者模式是面向过程的设计模式. 生产者制造数据   ------> 生产者把数据放入缓冲区  -------> ...