代码之家  ›  专栏  ›  技术社区  ›  PostMan

每列的DataBoud DataGridView验证

c#
  •  0
  • PostMan  · 技术社区  · 14 年前

    所以我有一个DataGridView,用于显示自定义模型的列表。 下面是模型的一些示例代码:

    public class TestModel
    {
        public IEnumerable<string> GetValidationErrors()
        {
            if (Value1 > 100 || Value1 <= 0)
                yield return "Value1 can only be between 1 and 100 (inclusive)";
    
            if (string.IsNullOrEmpty(Value2))
                yield return "Value2 can not be empty";
        }
    
        public bool IsValid
        {
            get { return GetValidationErrors().Count() == 0; }
        }
    
        public int Value1
        {
            get; set;
        }
    
        public string Value2
        {
            get; set;
        }
    }
    

    现在假设我正在绑定一个这样的模型列表(或IEnumerable)

    List<TestModel> list = Helpers.GetListOfTestModels();
    dataGridView1.DataSource = list;
    

    (我们可以假设列表返回是有效的,否则它将不会被存储)。

    现在在 RowValidating 事件,我可以通过访问 IsValid 属性并设置 Rows[].ErrorText ,像这样:

    private void dataGridView1_RowValidating(object sender, DataGridViewCellCancelEventArgs e)
    {
        var item = dataGridView1.Rows[e.RowIndex].DataBoundItem as TestModel;
        if (item == null)
            return;
    
        if(!item.IsValid)
        {
            dataGridView1.Rows[e.RowIndex].ErrorText = "Failed Validation"; // Or use GetValidationErrors and concat them, but to be simple I've left that out
            e.Cancel = true;
        }
        else
        {
            dataGridView1.Rows[e.RowIndex].ErrorText = string.Empty;
        }
    }
    

    现在,我真正想要做的是(不需要在整个地方复制和粘贴代码)验证每个属性并设置 Rows[].Cells[].ErrorText 属性(这将显示每个单元格的错误,而不是整行)。

    我该怎么办?

    也许是一些具有自定义属性的东西,然后反射以获取属性名并以这种方式访问单元?

    希望这一切都有意义!

    2 回复  |  直到 14 年前
        1
  •  1
  •   PostMan    14 年前

    解决了!

    具有这样的抽象属性:

    [AttributeUsage(AttributeTargets.Property, AllowMultiple = true, Inherited = true)]
    public abstract class ValidationAttribute : Attribute
    {
        public abstract void Validate(object value, PropertyInfo propertyInfo, ref IList<string> errors);
    }
    

    以及类似这样的实现:

    public class ValidStringAttribute  : ValidationAttribute
    {
        #region Overrides of ValidationAttribute
    
        public override void Validate(object value, PropertyInfo propertyInfo, ref IList<string> errors)
        {
            var v = propertyInfo.GetValue(value, null);
    
            if(!string.IsNullOrEmpty(v as string))
            {
                errors.Add(string.format("`{0}` cannot be null or empty",propertyInfo.Name);
            }
        }
    
        #endregion
    }
    

    然后修改 Value2 类似财产:

    [ValidString]
    public string Value2
    { 
       get; set;
    }
    

    在任何一个 RowValidating CellValidating 事件的作用如下:

    if(!item.IsValid)
    {
        foreach(var propertyInfo in item.GetType().GetProperties())
        {
            IList<string> list = new List<string>();
            foreach (ValidationAttribute attribute in propertyInfo.GetCustomAttributes(typeof(ValidationAttribute),true))
            {
                attribute.Validate(item,propertyInfo,ref list);
            }
    
            if(list.Count > 0)
            {
                // make sure it's not ignored
                var browsable = propertyInfo.GetCustomAttributes(typeof (BrowsableAttribute), true);
                if(browsable.Count() == 0)
                {
                    dataGridView1.Rows[e.RowIndex].Cells[propertyInfo.Name].ErrorText = list[0];
                }
            }
        }
    
        e.Cancel = true;
    }
    

    每个属性的BAM错误,数据绑定!

        2
  •  0
  •   Adam Houldsworth    14 年前

    你能不能利用 CellValidating 事件?此事件与 RowValidating 但当你退出单元格时会发生这种情况。

    或者,如果您的行表示业务对象,则可以公开 Validate 将对属性执行验证并返回 Errors 未通过验证且提供错误消息的属性数组。它不会自动链接到UI,但会在对象本身而不是UI中保持验证。