闲话WPF之十二(Attached属性 )
我们仍然用前面的一个简单XAML代码为例:
<Window>
<StackPanel>
<Label>LabelText</Lable>
</StackPanel>
</Window>
现在,如果需要对StackPanel及其子元素设置字体大小,应该如何做呢?在Window元素中,它有一个属性FontSize,可以直接设置。但是,StackPanel自己本身并没有FontSize这样的属性。这就该Attached属性出场了。这里我们需要用定义在TextElement元素中的Attached属性FontSize来设置StackPanel的字体。
<Window>
<StackPanel TextElement.FontSize=”30”>
<Label>LabelText</Lable>
</StackPanel>
</Window>
这样,StackPanel的子元素就能通过属性值继承得到新的FontSize属性。对于这样的XAML代码,XAML编译器或者解析器看到这种语法时,就要求TextElement(有时也称为Attached属性提供者)有相应的静态方法SetFontSize来设置对应的属性值。因此,上面的Attached属性设置代码,可以如下用C#实现:
StackPanel panel = new StackPanel();
TextElement.SetFontSize(panel, 30);
从这里的代码可以看出,Attached属性并不神秘。只是调用方法把元素和不相关的属性关联起来。而SetFontSize实现也比较简单。它只是调用了Dependency属性访问函数所调用的DependencyObject.SetValue方法。注意调用的对象是传入的DependencyObject,而不是当前的实例:
public static void SetFontSize(DependencyObject element, double value)
{
element.SetValue(TextElement.FontSizeProperty, value);
}
同样地,Attached属性也定义了对应的GetXXX函数。它调用的DependencyObject.GetValue方法:
public static double GetFontSize(DependencyObject element)
{
return (double)element.GetValue(TextElement.FontSizeProperty);
}
与普通的Dependency属性一样,这些GetXXX和SetXXX方法除了实现对GetValue和SetValue的调用,不能做任何其他额外的工作。
其实,在WPF应用中,Attached属性更多的用来控制UI的布局。除了前面的StackPanel,还有Grid等等。
补充说明:上面的代码还有一个问题需要说明。我们设置StackPanel的字体属性时用的是TextElement元素。为什么不用其他的元素Control、Button呢?
这个问题的关键之处在于Dependency属性的注册方法。我曾在Dependency属性[1]做过简单的说明。我们看看Element的FontSizeProperty属性的注册代码:
TextElement.FontSizeProperty = DependencyProperty.RegisterAttached(
“FontSize”, typeof(double), typeof(TextElement), new FrameworkPropertyMetadata(
SystemFonts.MessageFontSize, FrameworkPropertyMetadataOptions.Inherits |
FrameworkPropertyMetadataOptions.AffectsRender |
FrameworkPropertyMetadataOptions.AffectsMeasure),
new ValidateValueCallback(TextElement.IsValidFontSize));
这里与我们前面的IsDefault属性类似,只是RisterAttached方法优化了Attached属性需要的属性元数据的处理过程。
另一方面,Control的FontSize属性是在TextElement元素已经注册的属性之上调用AddOwner方法,获取一个完全相同的实例引用:
Control.FontSizeProperty = TextElement.FontSizeProperty.AddOwner(
typeof(Control), new FrameworkPropertyMetadata(SystemFonts.MessageFontSize,
FrameworkPropertyMetadataOptions.Inherits));
所以,在实现Attached属性时我们使用的是TextElement,而不是Control等等。