Data Annotations, IDataErrorInfo and MVVM

2020-05-04 05:24发布

I'm trying to find the best way to validate data in MVVM. Currently, I'm trying to use IDataErrorInfo with Data Annotations using the MVVM pattern.

However, nothing seems to work and I'm not sure what I could be doing wrong. I have something like this.

Model

public class Person : IDataErrorInfo
{
    [Required(ErrorMessage="Please enter your name")]
    public string Name { get; set; }

    public string Error
    {
        get { throw new NotImplementedException(); }
    }

    string IDataErrorInfo.this[string propertyName]
    {
        get
        {
            return OnValidate(propertyName);
        }
    }

    protected virtual string OnValidate(string propertyName)
    {
        if (string.IsNullOrEmpty(propertyName))
            throw new ArgumentException("Property may not be null or empty", propertyName);

        string error = string.Empty;

        var value = this.GetType().GetProperty(propertyName).GetValue(this, null);
        var results = new List<ValidationResult>();

        var context = new ValidationContext(this, null, null) { MemberName = propertyName };

        var result = Validator.TryValidateProperty(value, context, results);

        if(!result)
        {
            var validationResult = results.First();
            error = validationResult.ErrorMessage;
        }
        return error;
    }
}

Model code courtesy of the solution at How to catch DataAnnotations Validation in MVVM (This answer does not meet my criteria unfortunately.)

ViewModel

public class PersonViewModel
{
    private Person _person;

    public string Name
    {
        get
        {
            return _person.Name
        }
        set
        {
            _person.Name = value;
        }
    }
}


View

<Label Content="Name:" />

<TextBox Text="{Binding UpdateSourceTrigger=LostFocus,
                        Path=Name,
                        ValidatesOnDataErrors=True,
                        NotifyOnValidationError=true}" />


Is there any way to keep the seperation between model, view, and viewmodel while still utilizing data annotations for validation with IDataErrorInfo?

2条回答
唯我独甜
2楼-- · 2020-05-04 05:46

To keep validation running, IDataErrorInfo must be implemented by the data context, which property is bound to the control. So, it should be something like:

public class PersonViewModel : IDataErrorInfo 
{
    [Required(AllowEmptyStrings = false)]
    public string Name
    {
        get
        {
             return _person.Name
        }
        set
        {
             _person.Name = value;
        }
    }    

    public string Error
    {
        get { throw new NotImplementedException(); }
    }

    string IDataErrorInfo.this[string propertyName]
    {
        get
        {
            return OnValidate(propertyName);
        }
    }

    protected virtual string OnValidate(string propertyName)
    {
        /* ... */
    }
}

There's no need to implement IDataErrorInfo in model, this is view model's responsibility. Usually, IDataErrorInfo is implemented by the base class for your view models.

By the way, why OnValidate is protected? How do you imagine overriding of this method?

查看更多
唯我独甜
3楼-- · 2020-05-04 05:55

Keep your definition in XAML notifyonvalidation error and validatesondata errors set to true. And in VM use simple Viewmodel which has validation on setter function for desired properties, if validation fails it should throw validation error and xaml will know that it is invalid.

public class PersonViewModel
{
    private Person _person;

    public string Name
    {
        get
        {
            return _person.Name
        }
        set
        {
            if(value == string.empty)
               throw new ValidationException("Name cannot be empty");
            _person.Name = value;
        }
    }
}

Now you will need to handle this in xaml and display error text from exception

查看更多
登录 后发表回答