我正在研究.NET 4.0项目,该项目处理特定数据类型范围的验证。例如,32位int应该只在Int32.MinValue
和Int32.MaxValue
之间,或者应用程序定义的任何其他值。我希望能够在自定义验证器上指定数据类型和范围,以便可以通过绑定直接从xaml调用它们:<CheckIfValueRangeValidator>
这就是我想到的,我不确定它是否会起作用,或者它是否可以通过xaml完成。
class CheckIfValueInRangeValidator<T> : ValidationRule
{
public T Max { get; set; }
public T Min { get; set; }
public override ValidationResult Validate(object value, System.Globalization.CultureInfo cultureInfo)
{
// Implementation...
}
}
答案 0 :(得分:3)
我的不好,实际上你不能使用x:TypeArguments,因为它会引发 x:XAML版本低于2009的对象元素中不允许TypeArguments ,它只对松散的XAML文件或root有效element(我的窗口)......
http://msdn.microsoft.com/en-us/library/ms750476.aspx
但作为一种解决方法,您可以使用以下模式:
<TextBox x:Name="textBox1">
<Binding Path="MyValue"
Source="{StaticResource MyObject}"
UpdateSourceTrigger="PropertyChanged">
<Binding.ValidationRules>
<wpfApplication6:MyValidationRule ObjectType="{x:Type system:Int32}" />
</Binding.ValidationRules>
</Binding>
</TextBox>
代码背后:
public class MyObject
{
public object MyValue { get; set; }
}
public class MyValidationRule : ValidationRule
{
public Type ObjectType { get; set; }
public override ValidationResult Validate(object value, CultureInfo cultureInfo)
{
throw new NotImplementedException();
}
}
看一下这个: Using a Generic IValueConverter from XAML
@Blam评论值得考虑,要检查的范围通常适用于整数或双精度,对于其他类型我会说你可以添加一个布尔值来返回该对象的有效性并在其中执行此类验证对象本身。
对于您拥有RangeAttribute的号码,但它并不是WPF验证基础架构的一部分。
您还有另一个验证选项:在这种情况下,INotifyDataErrorInfo验证发生在对象内部。
我在这里写了一个冗长的答案:https://softwareengineering.stackexchange.com/questions/203590/is-there-an-effective-way-for-creating-complex-forms你可能会发现一些有用的东西。
根据我的经验,我会说一般的验证规则可能不明智。
您应该将您的问题编辑为 generic ;-),但更具体一点,您可以从此处获得更多帮助。给出您要验证的对象的一个或两个具体案例。
修改强>
您还可以使用BindingGroup验证对象:
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
private void TextBox1_OnTextChanged(object sender, TextChangedEventArgs e)
{
grid.BindingGroup.CommitEdit();
}
}
public class Indices
{
public int ColorIndex { get; set; }
public string ColorPrefix { get; set; }
public int GradientIndex { get; set; }
public string GradientPrefix { get; set; }
}
public class ValidateMe : ValidationRule
{
public override ValidationResult Validate(object value, CultureInfo cultureInfo)
{
var bindingGroup = value as BindingGroup;
var o = bindingGroup.Items[0] as Indices;
return new ValidationResult(true, null);
}
}
<Window x:Class="WpfApplication6.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:wpfApplication6="clr-namespace:WpfApplication6"
Title="MainWindow"
Width="525"
Height="350">
<Window.Resources>
<wpfApplication6:Indices x:Key="Indices"
ColorIndex="1"
ColorPrefix="MyColor"
GradientIndex="1"
GradientPrefix="MyGradient" />
</Window.Resources>
<Grid x:Name="grid" DataContext="{StaticResource Indices}">
<Grid.BindingGroup>
<BindingGroup>
<BindingGroup.ValidationRules>
<wpfApplication6:ValidateMe />
</BindingGroup.ValidationRules>
</BindingGroup>
</Grid.BindingGroup>
<TextBox TextChanged="TextBox1_OnTextChanged">
<Binding Path="ColorIndex" UpdateSourceTrigger="PropertyChanged" />
</TextBox>
</Grid>
</Window>
使用RangeAtribute:
private void test()
{
Indices indices = new Indices();
indices.ColorIndex = 20;
var validationContext = new ValidationContext(indices);
var validationResults = new List<System.ComponentModel.DataAnnotations.ValidationResult>();
var tryValidateObject = Validator.TryValidateObject(indices, validationContext, validationResults,true);
}
public class Indices
{
[Range(1, 10)]
public int ColorIndex { get; set; }
public string ColorPrefix { get; set; }
public int GradientIndex { get; set; }
public string GradientPrefix { get; set; }
}