前一个博客,介绍了依赖属性的基本定义,在定义的过程中register中的的两个参数,并没有传入参数,不知道其是用来干什么的,以下,我们将介绍这两个参数的真正用途FrameworkPropertyMetadata和ValidateValueCallback。

1、FrameworkPropertyMetadata 参数

FrameworkPropertyMetadata是一个重要的参数,其用于配置当前的依赖属性的特性,这个对象的有很多构造函数,我们使用一个比较常用的构造函数

public FrameworkPropertyMetadata( Object defaultValue, FrameworkPropertyMetadataOptions flags, PropertyChangedCallback propertyChangedCallback )

那么FrameworkPropertyMetadata 的作用是什么呢?我们看看这个FrameworkPropertyMetadata对象的几个属性:

编号 名称 说明
1 AffectsArrange 获取或设置一个值,该值指示依赖项属性是否会影响布局引擎运行期间的排列处理过程。
2 AffectsMeasure 获取或设置一个值,该值指示依赖项属性是否会影响布局引擎运行期间的测量处理过程。
3 AffectsParentArrange 获取或设置一个值,该值指示依赖项属性是否会影响布局引擎运行期间其父元素布局的排列处理过程。
4 AffectsParentMeasure 获取或设置一个值,该值指示依赖项属性是否会影响布局引擎运行期间其父元素布局的测量处理过程。
5 AffectsRender 获取或设置一个值,该值指示依赖项属性是否会以某种方式(这种方式不会具体影响排列或测量,但将请求重绘)对常规布局造成潜在影响。
6 BindsTwoWayByDefault 获取或设置一个值,该值指示默认情况下属性是否双向绑定。
7 CoerceValueCallback 获取或设置对此元数据中所指定 CoerceValueCallback 实现的引用
8 DefaultUpdateSourceTrigger 获取或设置在应用具有该元数据的属性的绑定时要使用的 UpdateSourceTrigger 的默认值,这些绑定的 UpdateSourceTrigger 设置为 Default。
9 DefaultValue 获取或设置依赖项属性的默认值.(继承自 PropertyMetadata。)
10 IsAnimationProhibited 获取或设置一个值,声明是否应在应用了包含元数据实例的依赖项对象上禁用动画。(继承自 UIPropertyMetadata。)
11 IsDataBindingAllowed 获取一个值,该值指示依赖项属性是否支持数据绑定。
12 IsNotDataBindable 获取或设置一个值,该值指示依赖项属性是否支持数据绑定。
13 IsSealed 获取一个值,该值确定是否已通过某种方式将元数据应用于属性,从而导致该元数据实例变为不可变状态。(继承自 PropertyMetadata。)
14 Journal 获取或设置一个值,该值指示该属性是否包含应用程序可以或应该存储为日记实现一部分的日记信息。
15 OverridesInheritanceBehavior 获取或设置一个值,该值指示属性值继承计算是否应跨越元素的逻辑树中的某些内容边界。
16 PropertyChangedCallback 获取或设置对此元数据中所指定 PropertyChangedCallback 实现的引用。 (继承自 PropertyMetadata。)
17 SubPropertiesDoNotAffectRender 获取或设置一个值,该值指示依赖项属性的子属性是否影响包含对象的呈现。

我们对几个关键属性进行分析:

AffectsArrange、AffectsMeasure、AffectsParentArrange、AffectsParentMeasure、AffectsRender。

五个属性,主要是用于定义当前依赖属性的改变,是否影响控件调整。如果设置AffectsArrange=true,属性的变化会造成控件重新排列。

BindsTwoWayByDefault: 指定属性是否是双向绑定,这个会在绑定的地方详细讲解

CoerceValueCallback:一个回调函数,用于在验证依赖属性之前尝试“纠正”属性值。我们可以简单的案例说明:

class Example:DependencyObject
     {
         static Example()
         {
             FrameworkPropertyMetadata metadata = new FrameworkPropertyMetadata();
             metadata.DefaultValue = "example";
             metadata.CoerceValueCallback = new CoerceValueCallback(CoerceMaxValue);
             NameProperty = DependencyProperty.Register("Name", typeof(string), typeof(Example), metadata);
         }
        public static DependencyProperty NameProperty;
         public String Name {
            set { SetValue(NameProperty, value); }
             get { return (string)GetValue(NameProperty); }
         }
        private static  object CoerceMaxValue(DependencyObject d, object baseValue)
         {
             string base1 = baseValue.ToString();
             if (base1.Length>)
             {
                 return "err";
             }
             return baseValue;
         }
     }

我们使用这个依赖属性

Example e = new Example();
e.Name = "";
MessageBox.Show(e.Name);
e.Name = "";
MessageBox.Show(e.Name);

我们可以到,第一个弹出框显示333,第二个弹出框,显示err,所以CoerceValueCallback的作用就是验证传入的依赖属性,并且可以修改属性,并且可以用来修改类型,可以把字符串red修改为color对象。

IsAnimationProhibited、IsDataBindingAllowed、IsSealed、IsSealed、SubPropertiesDoNotAffectRender:

标记依赖属性的一些特性,比如是否可以应用与动画或者数据绑定等。

PropertyChangedCallback:

属性,这个属性比较有用,代表属性发生变化的时候,调用的函数,这个函数在CoerceValueCallback函数调用之后调用,代表属性已经确定完成的最后一个步骤,我们简单案例如下:

static Example()
        {
            FrameworkPropertyMetadata metadata = new FrameworkPropertyMetadata();
            metadata.DefaultValue = "example";
            metadata.CoerceValueCallback = new CoerceValueCallback(CoerceMaxValue);
            metadata.PropertyChangedCallback = new PropertyChangedCallback(OnItemsSourceChanged);
            NameProperty = DependencyProperty.Register("Name", typeof(string), typeof(Example), metadata);
        }
       public static DependencyProperty NameProperty;
        public String Name {
           set { SetValue(NameProperty, value); }
            get { return (string)GetValue(NameProperty); }
        }
       private static  object CoerceMaxValue(DependencyObject d, object baseValue)
        {
            string base1 = baseValue.ToString();
            if (base1.Length>)
            {
                return "err";
            }
            return baseValue;
        }
       private static void OnItemsSourceChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            MessageBox.Show("finish");
        }

调用代码:

Example e = new Example();
e.Name = "";
e.Name = "";

我们可以看到,finish被弹出了两次,每次属性改变都会调用此函数。

2、ValidateValueCallback参数

  在定义任何类型属性的时候,都需要面对错误设置属性的情况,对于传统属性来说,可以通过get/set方法中,对属性进行验证,但是wpf虽然也有get/set,但是其设置方式和传统属性完全不同,无法通过get/set中进行验证,那么wpf提供FrameworkPropertyMetadata中的回调函数进行判断,在CoerceValueCallback 中对值进行判断合法性,但在设置给对象之前,还需要调用ValidateValueCallback函数,最终确定是否修改值,ValidateValueCallback回调会返回true和false,赋值和不赋值,假如赋值发生修改,则会调用PropertyChangedCallback回调,基本案例如下:

class Example:DependencyObject
   {
       static Example()
       {
           FrameworkPropertyMetadata metadata = new FrameworkPropertyMetadata();
           metadata.DefaultValue = "example";
           metadata.CoerceValueCallback = new CoerceValueCallback(CoerceMaxValue);
           metadata.PropertyChangedCallback = new PropertyChangedCallback(OnItemsSourceChanged);
           NameProperty = DependencyProperty.Register("Name", typeof(string), typeof(Example), metadata,new ValidateValueCallback(ValidateValue));
       }
      public static DependencyProperty NameProperty;
       public String Name {
          set { SetValue(NameProperty, value); }
           get { return (string)GetValue(NameProperty); }
       }
      private static  object CoerceMaxValue(DependencyObject d, object baseValue)
       {
           string base1 = baseValue.ToString();
           if (base1.Length>)
           {
               return "err";
           }
           return baseValue;
       }
       static bool ValidateValue(object obj)
       {
           if (obj.ToString() == "err") { return false; }
           return true;
       }
       private static void OnItemsSourceChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
       {
           MessageBox.Show("finish");
       }
   }

代码经过以上的修改,我们发现finish只会弹出一次,返回false后,则程序会抛出异常

WPF依赖属性2的更多相关文章

  1. WPF依赖属性详解

    WPF依赖属性详解 WPF 依赖属性 英文译为 Dependency Properties,是WPF引入的一种新类型的属性,在WPF中有着极为广泛的应用,在WPF中对于WPF Dependency P ...

  2. WPF自学入门(五)WPF依赖属性

    在.NET中有事件也有属性,WPF中加入了路由事件,也加入了依赖属性.最近在写项目时还不知道WPF依赖属性是干什么用的,在使用依赖项属性的时候我都以为是在用.NET中的属性,但是确实上不是的,通过阅读 ...

  3. WPF依赖属性值源(BaseValueSource)

    原文:WPF依赖属性值源(BaseValueSource)   WPF依赖属性提供一个机制,可以获取依赖属性提供值的来源 其以BaseValueSource枚举表示 1.Default public ...

  4. WPF依赖属性(续)(3)依赖属性存储

    原文:WPF依赖属性(续)(3)依赖属性存储          在之前的两篇,很多朋友参与了讨论,也说明各位对WPF/SL计数的热情,对DP系统各抒已见,当然也出现了一些分歧. 以下简称DP为依赖属性 ...

  5. WPF依赖属性(续)(1)

    原文:WPF依赖属性(续)(1)                 之前有写过几篇文章,详细地介绍了依赖属性的基本使用方法,如果你不想了解其内部实现机制的话,那么通过那两篇文章的介绍,足以应付平时的应用 ...

  6. WPF依赖属性(续)(2)依赖属性与附加属性的区别

    原文:WPF依赖属性(续)(2)依赖属性与附加属性的区别        接上篇,感谢各位的评论,都是认为依赖属性的设计并不是为了节省内存,从大的方面而讲是如此.样式,数据绑定,动画样样都离不开它.这篇 ...

  7. 监听WPF依赖属性

    原文:监听WPF依赖属性 当我们使用依赖属性的时候,有时需要监听它的变化,这在写自定义控件的时候十分有用, 下面介绍一种简单的方法.   如下使用DependencyPropertyDescripto ...

  8. WPF依赖属性的正确学习方法

    前言 我在学习WPF的早期,对依赖属性理解一直都非常的不到位,其恶果就是,我每次在写依赖属性的时候,需要翻过去的代码来复制黏贴. 相信很多朋友有着和我相同的经历,所以这篇文章希望能帮助到那些刚刚开始学 ...

  9. WPF 依赖属性前言

    WPF 依赖属性前言 ​ 在.net中,我们可以属性来获取或设置字段的值,不需要在编写额外的get和set方法,但这有一个前提,那就是需要在对象中拥有一个字段,才能在此字段的基础上获取或设置字段的值, ...

  10. WPF 依赖属性

    依赖属性,简单的说,在WPF控件应用过程中,界面上直接可以引用的属性 如:<Button Content="aaa"></Button> Content称为 ...

随机推荐

  1. java泛型(整理)

    1 泛型基础知识 泛型需要理解两个关键点:1)类型擦除 2)类型转换 1)类型擦除 泛型有个很重要的概念,是类型擦除.正确理解泛型概念的首要前提是理解类型擦除(type erasure). Java中 ...

  2. Java之IO流补充

    IO流例子 package com.hanqi.maya.util; import java.io.BufferedReader; import java.io.BufferedWriter; imp ...

  3. Unity3D 中材质球(Material)预制体打包成AB(AssetBundle)出现材质丢失问题的解决方案

    关于Unity3D中打AB(AssetBundle)资源包,默认是不连Shader一同打包进去的,所以得自行添加,添加方式比较简单,不需要往项目中添加Shader,只是点选一下就可以了,具体实现方式如 ...

  4. 使用ide编程时候 不知为何突然光标变宽,如何恢复成原有的细竖光标

    各位朋友们, 你们在编程时候有没有这样的情况: 码着码着,突然不知什么原因,光标变成这样了: 这种宽的光标,不知道怎么调都调不回去,而且网上也没有类似的问题描述 就对我们编程极其不便(因为这种光标是操 ...

  5. PL/SQL 编程(二)游标、存储过程、函数

    游标--数据的缓存区 游标:类似集合,可以让用户像操作数组一样操作查询出来的数据集,实质上,它提供了一种从集合性质的结果中提取单条记录的手段. 可以将游标形象的看成一个变动的光标,他实质上是一个指针, ...

  6. EF对于已有数据库的Code First支持

    EF对于已有数据库的Code First支持 原文链接 本文将逐步介绍怎样用Code First的方式基于已有数据库进行开发.Code First支持你使用C#或者VB.Net定义类.并使用数据模型标 ...

  7. Python2和Python3的一些语法区别

    Python2和Python3的一些语法区别 python 1.print 在版本2的使用方法是: print 'this is version 2 也可以是 print('this is versi ...

  8. iOS之Cocoapods安装

    网上关于cocoapods的教程很多,关于它的优点我不赘述:但是我根据多次安装的经验,把我遇到的问题写一下,希望对新手有所帮助. 1. 设置输入源(由于默认的gem资源是国外的,由于历史原因,访问比较 ...

  9. html4与html5的区别及html5的一些新特性

    区别 1.html5语法的改变 HTML5简化了很多细微的语法,例如: 1.1doctype的声明; html4: <!DOCTYPE HTML PUBLIC "-//W3C//DTD ...

  10. Clang之词法分析Lex

    Clang是LLVM编译器框架的前端(Frontend)编译器,可编译链接C.C++.Objective-C和Objective-C++四种语言的项目代码.Clang 的开发目标是提供一个可以替代 G ...