Search code examples
c#wpftelerik-gridvalidationrule

WPF Grid With Validation Rule And Dependency Property


At the moment I have a grid and I'm trying to have a cell with validation rules. To validate it, I require the row's min and max value.

Validation Class:

public decimal Max { get; set; }

public decimal Min { get; set; }

public override ValidationResult Validate(object value, System.Globalization.CultureInfo cultureInfo)
{
    var test = i < Min;
    var test2 = i > Max;

    if (test || test2)
        return new ValidationResult(false, String.Format("Fee out of range Min: ${0} Max: ${1}", Min, Max));
    else
        return new ValidationResult(true, null);
}

User Control:

<telerik:RadGridView SelectedItem ="{Binding SelectedScript}"
                     ItemsSource="{Binding ScheduleScripts}">
    <telerik:RadGridView.Columns>
        <telerik:GridViewDataColumn
            DataMemberBinding="{Binding Amount}" Header="Amount" 
            CellTemplate="{StaticResource AmountDataTemplate}" 
            CellEditTemplate="{StaticResource AmountDataTemplate}"/>   
        <telerik:GridViewComboBoxColumn
            Header="Fee Type" 
            Style="{StaticResource FeeTypeScriptStyle}" 
            CellTemplate="{StaticResource FeeTypeTemplate}"/>           
    </telerik:RadGridView.Columns>
</telerik:RadGridView>

FeeType Class:

public class FeeType
{
    public decimal Min { get; set; }
    public decimal Max { get; set; }
    public string Name { get; set; }
}

I've tried this solution here WPF ValidationRule with dependency property and it works great. But now I come across the issue that the proxy can't be instantiated through the viewmodel. It's based on the row's selected ComboBox Value's Min and Max property.

For example, that combo box sample values are below

Admin Min: $75 Max $500
Late  Min: $0  Max $50

Since a grid can have virtually as many rows as it wants, I can't see how creating proxies would work in my situation. If I can get some tips of guidance, would be greatly appreciated.


Solution

  • Alert: this is not a definitive solution, but shows you a correct way to implement the validation logic putting it totally on ViewModels.

    For semplicity purpose, I create the list of FeeTypes as static property of the FeeType class:

    public class FeeType
    {
        public decimal Min { get; set; }
        public decimal Max { get; set; }
        public string Name { get; set; }
    
        public static readonly FeeType[] List = new[]
        {
            new FeeType { Min = 0, Max = 10, Name = "Type1", },
            new FeeType { Min = 2, Max = 20, Name = "Type2", },
        };
    }
    

    This is the ViewModel for a single Grid row. I put only Amount and Fee properties.

    public class RowViewModel : INotifyPropertyChanged, INotifyDataErrorInfo
    {
        public RowViewModel()
        {
            _errorFromProperty = new Dictionary<string, string>
            {
                { nameof(Fee), null },
                { nameof(Amount), null },
            };
    
            PropertyChanged += OnPropertyChanged;
        }
    
        private void OnPropertyChanged(object sender, PropertyChangedEventArgs e)
        {
            switch(e.PropertyName)
            {
                case nameof(Fee):
                    OnFeePropertyChanged();
                    break;
                case nameof(Amount):
                    OnAmountPropertyChanged();
                    break;
                default:
                    break;
            }
        }
    
        private void OnFeePropertyChanged()
        {
            if (Fee == null)
                _errorFromProperty[nameof(Fee)] = "You must select a Fee!";
            else
                _errorFromProperty[nameof(Fee)] = null;
    
            NotifyPropertyChanged(nameof(Amount));
        }
    
        private void OnAmountPropertyChanged()
        {
            if (Fee == null)
                return;
    
            if (Amount < Fee.Min || Amount > Fee.Max)
                _errorFromProperty[nameof(Amount)] = $"Amount must be between {Fee.Min} and {Fee.Max}!";
            else
                _errorFromProperty[nameof(Amount)] = null;
        }
    
        public decimal Amount
        {
            get { return _Amount; }
            set
            {
                if (_Amount != value)
                {
                    _Amount = value;
                    NotifyPropertyChanged();
                }
            }
        }
        private decimal _Amount;
    
        public FeeType Fee
        {
            get { return _Fee; }
            set
            {
                if (_Fee != value)
                {
                    _Fee = value;
                    NotifyPropertyChanged();
                }
            }
        }
        private FeeType _Fee;
    
        #region INotifyPropertyChanged
        public event PropertyChangedEventHandler PropertyChanged;
        public void NotifyPropertyChanged([CallerMemberName] string propertyName = null)
        {
            PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
        }
        #endregion
    
        #region INotifyDataErrorInfo
        public event EventHandler<DataErrorsChangedEventArgs> ErrorsChanged;
    
        public bool HasErrors
        {
            get
            {
                return _errorFromProperty.Values.Any(x => x != null);
            }
        }
    
        public IEnumerable GetErrors(string propertyName)
        {
            if (string.IsNullOrEmpty(propertyName))
                return _errorFromProperty.Values;
    
            else if (_errorFromProperty.ContainsKey(propertyName))
            {
                if (_errorFromProperty[propertyName] == null)
                    return null;
                else
                    return new[] { _errorFromProperty[propertyName] };
            }
    
            else
                return null;
        }
    
        private Dictionary<string, string> _errorFromProperty;
        #endregion
    }
    

    Now, I tested it with a native DataGrid, but the result should be the same in Telerik:

    <DataGrid AutoGenerateColumns="False" ItemsSource="{Binding Rows}">
      <DataGrid.Columns>
        <DataGridTextColumn Binding="{Binding Amount}"/>
        <DataGridComboBoxColumn SelectedItemBinding="{Binding Fee, UpdateSourceTrigger=PropertyChanged}"
                                ItemsSource="{x:Static local:FeeType.List}"
                                DisplayMemberPath="Name"
                                Width="200"/>
      </DataGrid.Columns>
    </DataGrid>
    
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
    
            Rows = new List<RowViewModel>
            {
                new RowViewModel(),
                new RowViewModel(),
            };
    
            DataContext = this;
        }
    
        public List<RowViewModel> Rows { get; } 
    }
    

    If a FeeType instance can modify Min and Max at runtime, you need to implement INotifyPropertyChanged also on that class, handling the value changes appropriately.

    If you're new to things "MVVM", "ViewModels", "Notification changes" etc, give a look to this article. If you usually work on middle-big project on WPF, it is worth learning how to decouple View and Logic through the MVVM pattern. This allows you to test the logic in a faster and more automatic way, and to keep things organized and focused.