ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

WPF 数据验证

2022-08-23 08:31:35  阅读:149  来源: 互联网

标签:return string 验证 value WPF 数据 public 属性


 

Validation ValidationRule iDataErrorInfo+DataAnimation 的用法

//Validation:依赖属性用(在验证回调中返回false)

//ValidationRule:依赖属性和普通类的属性都能用
//IdataErrorInfo:依赖属性和普通类的属性都能用,谁想用的话 就继承iDataErrorInfo

区别?

ValidationRule可以自定义错误信息,Validation和ValidationRule在xaml中的用法一样,比较繁琐

iDataErrorErrorInfo可以使用属性索引器,在xaml中只需咋binding表达式中打开ValidatesOnDataErrors=True即可,xaml代码比较少

ValidationRule和iDataErrorInfo的错误信息都是被全局静态对象Validation接收的

1.Validation 依赖属性的验证

界面上有2个控件,一个textbox绑定一个自定义的依赖属性,一个textblock绑定前面依赖属性的验证结果

依赖属性都有一个private static bool ValidateValueCallback(object value)验证回调函数,返回值是验证的结果bool类型的,value是依赖属性的数据,在验证回调中做一下判断,当不满足条件时,返回一个false,

这时候全局静态对象Validation就能捕获到这个验证结果

<TextBox Name="tb">
  <TextBox.Text>
     <Binding Path="MyProperty" UpdateSourceTrigger="PropertyChanged">
      <Binding.ValidationRules>
        <ExceptionValidationRule/>
      </Binding.ValidationRules>
    </Binding>
  </TextBox.Text>
</TextBox>
<TextBox Text="{Binding Path=(Validation.Errors)[0].ErrorContent,ElementName=tb}"/>-->

在依赖属性的验证回调中判断,

运行结果

 2.ValidationRule 依赖属性的验证

上面的验证信息(“123456”不是属性“Myproperty”的有效值)是由ExceptionValidationRule默认提供的错误提示消息

 错误信息能不能自定义呢?这时候要自己写一个ValidationRule

可以看到ExceptionValidationRule也是继承自ValidationRule的

新建一个类继承自ValidationRule,重写ValidationResult

 那怎么用自己定义的去代替ExceptionValidationRule?

运行结果:

 

 3.iDataErrorInfo和DataAnnotations特性

普通属性所在的类继承iDataErrorInfo,并实现iDataErrorInfo提供的接口和索引器

 public class Person : INotifyPropertyChanged, IDataErrorInfo
    {
        private int _id;
        private string _name;

        public string Name
        {
            get { return _name; }
            set { _name = value;
                PropertyChanged?.Invoke(this, new PropertyChangedEventArgs("Name"));
            }
        }

        public int ID
        {
            get { return _id; }
            set { _id = value;
                PropertyChanged?.Invoke(this, new PropertyChangedEventArgs("ID"));
            }
        }

        //INotifyPropertyChanged的接口
        public event PropertyChangedEventHandler? PropertyChanged;
        //IDataErrorInfo的接口
        public string Error { get { return ""; } }

        public string this[string columnName]
        {
            get
            {
                 if (columnName == "ID")
                {
                       if (_id >18)
                    {
                        return ">18";
                    }
                }

                 if (columnName == "Name")
                {
                    if (Name.Length > 3)
                        return "leng>3";
                }

                 return string.Empty;
            }
        }
    }

Person类提供了2个属性,继承了iDataErrorInfo,在索引器中判断每个属性的值是否符合要求,在索引器中的return 返回的值会被全局静态对象Validation接受,这是binding表达式提供的功能,会在绑定的目标和数据源之间检测是否存在检验信息,在控件上如何获取呢?

只需要在需要接受验证的对象上打开

ValidatesOnDataErrors=True,ValidatesOnExceptions=True

xaml代码:

<Grid>
        <StackPanel>
            <Label Content="ID"/>
            <TextBox x:Name="tb" Text="{Binding ID, UpdateSourceTrigger=PropertyChanged,Mode=TwoWay,ValidatesOnDataErrors=True,ValidatesOnExceptions=True}"/>
            <TextBox Text="{Binding Path=(Validation.Errors)[0].ErrorContent,ElementName=tb}"/>
            <Label Content="Name"/>
            <TextBox x:Name="tbname" Text="{Binding Name,UpdateSourceTrigger=PropertyChanged,Mode=TwoWay,ValidatesOnDataErrors=True,ValidatesOnExceptions=True}"/>
            <TextBox Text="{Binding Path=(Validation.Errors)[0].ErrorContent,ElementName=tbname}"/>
        </StackPanel>
    </Grid>

运行结果:

 

 

 当使用IdataErrorInfo时,需要在索引器中以此判断每个属性和属性的值,为了简化索引器中的代码,通过反射去获取每个属性,并通过添加特性的方法去完成验证

 public class Person : INotifyPropertyChanged, IDataErrorInfo
    {
        private int _id;
        private string _name;
        [Required]
        [MyArrtibute ]
        public string Name
        {
            get { return _name; }
            set
            {
                _name = value;
                PropertyChanged?.Invoke(this, new PropertyChangedEventArgs("Name"));
            }
        }
        [Range(0,18,ErrorMessage ="超出范围了")]
        [Required]
        public int ID
        {
            get { return _id; }
            set
            {
                _id = value;
                PropertyChanged?.Invoke(this, new PropertyChangedEventArgs("ID"));
            }
        }

        //INotifyPropertyChanged的接口
        public event PropertyChangedEventHandler? PropertyChanged;
        //IDataErrorInfo的接口
        public string Error { get { return ""; } }
        public string this[string columnName]
        {
            get
            {
                //引入命名空间using System.ComponentModel.DataAnnotations;
                var vc = new ValidationContext(this, null, null);
                vc.MemberName = columnName;
                var res = new List<ValidationResult>();
                var result = Validator.TryValidateProperty(this.GetType().GetProperty(columnName).GetValue(this, null), vc, res);
                if (res.Count > 0)
                {
                    return string.Join(Environment.NewLine, res.Select(r => r.ErrorMessage).ToArray());
                }
                return string.Empty;
            }
        }
    }

一个类继承了IdataErrorInfi后,又使用DataAnnotations去改善索引器后,属性的验证逻辑需要通过特性的方式去完成,特性分为系统自带的和自定义的,新建特性的方法,新建一个类并继承ValidationAttribute,在该类中返回一个ValidationResult即可。

新建特性代码:

public class MyArrtibute : ValidationAttribute
    {
        protected override ValidationResult? IsValid(object? value, ValidationContext validationContext)
        {
            if (value.ToString().Length > 10)
                return new ValidationResult("名字长度大于10了");

            return base.IsValid(value, validationContext);
        }
    }

第一个参数value就是在binding表达式,从数据源到目标的值,这里的return返回的值依旧是被全局静态对象Validation接受,这是由binding功能提供的

特性的使用方法:

    [Required]
        [MyArrtibute ]
        public string Name
        {
            get { return _name; }
            set
            {
                _name = value;
                PropertyChanged?.Invoke(this, new PropertyChangedEventArgs("Name"));
            }
        }

在需要验证的属性前加上特性即可

运行结果:

 

标签:return,string,验证,value,WPF,数据,public,属性
来源: https://www.cnblogs.com/1024E/p/15744305.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有