Проверка данных в MVVM

У меня есть приложение с множеством ViewModels. У некоторых свойств есть DataAnnotations.

 [Required(ErrorMessage = "Field 'Range' is required.")]
    [Range(1, 10, ErrorMessage = "Field 'Range' is out of range.")]
    public int Password
    {
        get
        {
            return _password;
        }
        set
        {
            if (_password != value)
            {
                _password = value;
                RaisePropertyChanged("Password");
            }
        }
    }

Как можно выполнить проверку, реализовав интерфейс IDataErrorInfo или INotifyDataErrorInfo для всех моделей просмотра?

Я использую эту статью, но проверяю, когда проверить обязательное поле.


person ar.gorgin    schedule 04.08.2013    source источник


Ответы (2)


Вот простой пример, в котором используется IDataErrorInfo. Это должно помочь вам начать работу.

XAML:

<Grid>
    <Grid.Resources>
        <ControlTemplate x:Key="LeftErrorTemplate">
            <StackPanel Orientation="Horizontal">
                <TextBlock Text="{Binding AdornedElement.(Validation.Errors).[0].ErrorContent, ElementName=ErrorAdorner}" Background="Red" Foreground="White" FontWeight="Bold" VerticalAlignment="Center"/>
                <AdornedElementPlaceholder x:Name="ErrorAdorner">
                    <Border BorderBrush="Red" BorderThickness="1" />
                </AdornedElementPlaceholder>
            </StackPanel>
        </ControlTemplate>
    </Grid.Resources>
    <TextBlock Height="23" HorizontalAlignment="Left" Margin="158,66,0,0" Name="textBlock1" Text="Name" VerticalAlignment="Top" Width="44" />
    <TextBox Height="23" HorizontalAlignment="Left" Margin="217,65,0,0" Name="textBox1" VerticalAlignment="Top" Width="120" 
             Text="{Binding Name, UpdateSourceTrigger=PropertyChanged, ValidatesOnDataErrors=True}" 
             Validation.ErrorTemplate="{StaticResource LeftErrorTemplate}"/>
</Grid>

Код позади:

using System;
using System.Windows;
using System.ComponentModel;

namespace WpfApplication1
{
    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
            var vm = new ViewModel();

            this.DataContext = vm;
        }
    }

    public class ViewModel : ObservableBase, IDataErrorInfo
    {
        private string _Name;

        public string Name
        {
            get { return _Name; }
            set
            {
                _Name = value;
                OnPropertyChanged("Name");
            }
        }

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

        public string this[string columnName]
        {
            get
            {
                string errorMessage = string.Empty;

                switch (columnName)
                {
                    case "Name":
                        if (string.IsNullOrEmpty(Name))
                            errorMessage = "Enter name";
                        else if (Name.Trim() == string.Empty)
                            errorMessage = "Enter valid name";
                        break;
                }
                return errorMessage;
            }
        }

    }

    public class ObservableBase : INotifyPropertyChanged
    {

        public event PropertyChangedEventHandler PropertyChanged;

        public void OnPropertyChanged(string propertyName)
        {
            if (PropertyChanged != null)
                PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
        }
    }
}
person Anand Murali    schedule 04.08.2013
comment
Спасибо, но я хочу создать класс проверки для всего приложения и всех ViewModel, унаследованных от класса проверки. - person ar.gorgin; 05.08.2013

Привет, вам нужно будет создать какой-то метод, который проверяет аннотации, такие как ValidateMethod в моем коде

public class ViewModel:INotifyPropertyChanged,IDataErrorInfo
    {
        int _password;

        [Required(ErrorMessage = "Field 'Range' is required.")]
        [Range(1, 10, ErrorMessage = "Field 'Range' is out of range.")]
        public int Password
        {
            get
            {
                return _password;
            }
            set
            {
                if (_password != value)
                {
                    _password = value;
                    Validate("Password", value);
                    Notify("Password");
                }
            }
        }

        private void Validate(string propertyName, object value)
        {
            if (string.IsNullOrEmpty(propertyName))
                throw new ArgumentNullException("propertyName");

            string error = string.Empty;

            var results = new List<System.ComponentModel.DataAnnotations.ValidationResult>(2);

            bool result = Validator.TryValidateProperty(
                value,
                new ValidationContext(this, null, null)
                {
                    MemberName = propertyName
                },
                results);

            if (!result && (value == null || ((value is int || value is long) && (int)value == 0) || (value is decimal && (decimal)value == 0)))
                return;

            if (!result)
            {
                System.ComponentModel.DataAnnotations.ValidationResult validationResult = results.First();
                if (!errorMessages.ContainsKey(propertyName))
                    errorMessages.Add(propertyName, validationResult.ErrorMessage);
            }

            else if (errorMessages.ContainsKey(propertyName))
                errorMessages.Remove(propertyName);
        }

        #region INotifyPropertyChanged 

        public void Notify(string propName)
        {
            if (PropertyChanged != null)
                PropertyChanged(this, new PropertyChangedEventArgs(propName));

        }
        public event PropertyChangedEventHandler PropertyChanged;

        #endregion

        #region IDataErrorInfo

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

        private Dictionary<string, string> errorMessages = new Dictionary<string, string>();

        public string this[string columnName]
        {
            get 
            { 
                if(errorMessages.ContainsKey(columnName))
                    return errorMessages[columnName];
                return null;

            }
        }

        #endregion
    }

> xaml

<TextBox Text="{Binding Password, ValidatesOnDataErrors=True, UpdateSourceTrigger=PropertyChanged}" Height="70" Width="200" />

xaml.cs

public MainWindow()
    {
        InitializeComponent();
        DataContext = new ViewModel();
    }

Вам необходимо вызвать метод Validate из установщика свойства, которое применяет DataAnnotations, и обязательно вызвать его перед уведомлением PropertyChanged.

person yo chauhan    schedule 04.08.2013
comment
Спасибо, но PropertyChanged не уведомляет об обязательной проверке, и я должен вызвать метод проверки для всех необходимых свойств в кнопке отправки. - person ar.gorgin; 05.08.2013
comment
Да, вам нужно будет вызвать это для всех свойств, которые вы хотите подтвердить, и то, что вы имеете в виду под PropertyChanged, не уведомляя при требуемой проверке - person yo chauhan; 05.08.2013
comment
проверки запускаются только тогда, когда происходит привязка. Что происходит, когда пользователь запускает экран и нажимает кнопку отправки напрямую ... Никакой привязки не происходит и, следовательно, не запускается проверка .. - person ar.gorgin; 05.08.2013
comment
вы можете видеть, что есть словарь (errorMessages), в котором есть сообщения об ошибках, которые вы можете использовать. Если количество словаря больше 1, вы оставляете кнопку отключенной, а если вы хотите уведомить, просто RaisePropertyChanged для каждого свойства в ключе словаря. - person yo chauhan; 05.08.2013
comment
Смотрите, это зависит от вашей логики. Как вы это делаете. Для этого вам нужно привязать команду к кнопке. И у меня сейчас нет кода для этого, потому что его нет на моем ноутбуке. - person yo chauhan; 05.08.2013