• 关于Attached Property(学习)


    在前面,我用三篇短小的PostDependency属性进行了说明。现在,我们再继续看一种特殊的Dependency属性:Attached属性。Attached属性可以非常高效地Attach到其他的对象中。

    我们仍然用前面的一个简单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属性一样,这些GetXXXSetXXX方法除了实现对GetValueSetValue的调用,不能做任何其他额外的工作。

    其实,在WPF应用中,Attached属性更多的用来控制UI的布局。除了前面的StackPanel,还有Grid等等。

    补充说明:上面的代码还有一个问题需要说明。我们设置StackPanel的字体属性时用的是TextElement元素。为什么不用其他的元素Control、Button呢?

    这个问题的关键之处在于Dependency属性的注册方法。我曾在Dependency属性[1]做过简单的说明。我们看看ElementFontSizeProperty属性的注册代码:

    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属性需要的属性元数据的处理过程。

    另一方面,ControlFontSize属性是在TextElement元素已经注册的属性之上调用AddOwner方法,获取一个完全相同的实例引用:

    Control.FontSizeProperty = TextElement.FontSizeProperty.AddOwner(
    
         typeof(Control), new FrameworkPropertyMetadata(SystemFonts.MessageFontSize,
    
         FrameworkPropertyMetadataOptions.Inherits));

    所以,在实现Attached属性时我们使用的是TextElement,而不是Control等等。

  • 相关阅读:
    bugfree3.0.1-修改“优先级”为中文引起的PHP Error
    【vue】vue单元测试:karma+mocha
    【node】node的核心模块---http模块,http的服务器和客户端
    【node】fs模块,文件和目录的操作
    【webpack】从零开始学webpack
    【vue】webpack插件svg-sprite-loader---实现自己的icon组件
    【vue】vue的路由权限管理
    【node】用koa搭建一个增删改服务(一)
    【小程序】小程序之发送模板消息
    【小程序】返回顶部wx.pageScrollTo和scroll-view的对比
  • 原文地址:https://www.cnblogs.com/shawnzxx/p/2710076.html
Copyright © 2020-2023  润新知