属性和事件是.NET抽象模型的核心部分,WPF在这之上增加了Dependency Property(依赖属性)。依赖属性使用更高效的保存机制,并且支持附加的功能,更改通知、属性值继承(在元素树中向下传播默认属性值)以及减少属性存储空间。依赖属性是WPF动画、数据绑定和样式的基础。通过封装,依赖属性和.NET属性的访问方式一致,但是其背后的实现方式是不一样的。
使用依赖属性的原因很简单,出于性能考虑,如果WPF设计者只是简单的在.NET属性系统之上添加额外的功能,那么就需要为编写代码创建一个复杂庞大的层次,如果不承受这一额外的负担,普通属性就不能支持依赖属性的所有功能。
什么时候使用依赖属性
当该属性可能会被绑定表达式、用于动画计算、更改通知、共享以及属性值继承时,应选择使用依赖属性。
限制
依赖属性所属类必须继承或间接继承自:System.Windows.DependencyObject类。
依赖属性语法
定义
public static readonly DependencyProperty XXXXProperty;
根据约定,依赖属性的字段的名称为普通属性名称的末尾加上单词"Property"。
初始化
XXXXProperty = DependencyProperty.Register(…);
属性包装器
public object XXXX
{
set{ this.SetValue(XXXXProperty,value);}
get{ return this.GetValue(XXXXProperty) as object;}
}
属性包装器不应当添加额外的验证属性值或引发事件的等等代码,这是因为WPF中的其它功能可能会忽略属性包装器,并直接调用SetValue()和GetValue()方法(例如:在运行时解析编译过的XAML文件)。
属性值验证:
DependencyProperty.ValidateValueCallback
可以接受或拒绝新值。
DependencyProperty.CoerceValueCallback
可以将新值修改为更能被接受的值。
事件触发:FrameworkPropertyMetadata.PropertyChangedCallback
执行顺序:
首先,CoerceValueCallback被调用。
接下来调用ValidateValueCallback。
如果上述两个方法都成功,调用PropertyChangedCallback。
依赖属性的值
依赖属性遵循严格的优先规则以确定当前值。即使没有直接设置依赖属性,它可能已经有了数值(可能由数据绑定、样式或者动画提供的,也可能是通过元素树继承来的)。但是,只要直接设置了属性值,设置的属性值就会覆盖所有其它的影响。
如果希望删除本地值设置,并让属性好像从来没有设置过那样确定其取值,可以使用DependencyObject.ClearValue()方法。
例如:element.ClearValue(FrameworkElement.MarginProperty);
共享依赖属性
一些类会共享同一个依赖属性,尽管这些类具有不同的继承层次。例如:TextBlock.FontFamily属性和Control.FontFamily属性指向同一个静态的依赖属性,该属性实际上是在TextElement类中定义的TextElement.FontFamilyProperty依赖属性。
例如:TextBlock.FontFamilyProperty = TextElement.FontFamilyProperty.AddOwner(typeof(TextBlock));
附加的依赖属性
附加属性是一种依赖属性,并且它由WPF属性系统管理。和依赖属性不同的是,附加属性被应用到一个非定义该属性的类。
例如:Grid类的Row和Column就是最常见的例子。
为了定义附加属性,需要使用DependencyProperty.RegisterAttached(…);
当创建附加属性时,不需要定义属性包装器,这是因为附加属性可以被用于任何依赖对象。附加属性需要调用两个静态方法来设置和获取属性的值,这两个方法应当命名为SetPropertyName()和GetPropertyName().
例如:
public static int GetRow(DependencyObject element)
{
return element.GetValue(Grid.RowProperty) as int;
}
public static void SetRow(DependencyObject element , int value)
{
element.SetValue(Grid.RowProperty , value);
}
分类: WPF