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.
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.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With