5

介绍

我创建了一个DecimalTextBox UserControl,其中包含我需要完成的一些十进制验证,因此我不需要每次都重新创建验证,而可以使用它UserControl根据Josh Smith 的这篇文章,此验证具有需要绑定的属性,因此我创建了DependencyProperties以便可以绑定到它们。


问题

控件的验证行为异常。当我在 中输入错误值时TextBox,它会显示为错误。但是,当我尝试将值更改回代码中时,文本框中显示的值保持不变。

以下是我执行的导致此错误的步骤(在此示例中 1 是无效值):

  1. 加载表单,默认值为0
  2. 在文本框中输入1 (由于验证结果为错误,文本框变为红色)
  3. 在代码中,我将绑定到文本框的属性设置为0
  4. 表单仍然在红色文本框中显示1

代码示例

我准备了一个演示问题的示例,可以在此处下载。

我会在这里发布一些代码,如果您想了解更多,请告诉我。

ValidationTestControl 的 XAML

<UserControl x:Class="WPFTestProject.ValidationTestControl"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:v="clr-namespace:WPFTestProject"
    x:Name="ValidationTest"
    Height="50" Width="525">
<Grid>
    <Grid.RowDefinitions>
        <RowDefinition></RowDefinition>          

    </Grid.RowDefinitions>
    <StackPanel Orientation="Horizontal" VerticalAlignment="Center" HorizontalAlignment="Center">
        <TextBlock Text="Type 'Banana' here: "></TextBlock>
        <TextBox MinWidth="100">
            <TextBox.Text>
                <Binding ElementName="ValidationTest"  Path="Text" UpdateSourceTrigger="PropertyChanged" Mode="TwoWay" ValidatesOnDataErrors="True" ValidatesOnExceptions="True">
                    <Binding.ValidationRules>
                        <v:NotBananaValidationRule>
                            <v:NotBananaValidationRule.NotWhatBinding>
                                <v:NotBananaBinding x:Name="NotBananaValidationBinding"></v:NotBananaBinding>
                            </v:NotBananaValidationRule.NotWhatBinding>
                        </v:NotBananaValidationRule>
                    </Binding.ValidationRules>
                </Binding>
            </TextBox.Text>
        </TextBox>
        <TextBlock Text=" (the text will give error when = 'Banana')"></TextBlock>
    </StackPanel>
</Grid>

ValidationTestControls 代码背后

(是的,我不太了解 MVVM,但我觉得这个独立控件没问题)

 public partial class ValidationTestControl : UserControl
{
    public ValidationTestControl()
    {
        InitializeComponent();
        Banana = "Banana";

        Binding BananaBinding = new Binding("Banana");
        BananaBinding.Source = this;

        NotBananaValidationBinding.SetBinding(NotBananaBinding.NotWhatProperty, BananaBinding);
    }

    public static DependencyProperty TextProperty = DependencyProperty.Register("Text", typeof(string), typeof(ValidationTestControl), new PropertyMetadata());
    public static DependencyProperty BananaProperty = DependencyProperty.Register("Banana", typeof(string), typeof(ValidationTestControl), new PropertyMetadata());

    public string Text
    {
        get
        {
            return (string)GetValue(TextProperty);
        }
        set
        {
            SetValue(TextProperty, value);
        }
    }


    public string Banana
    {
        get
        {
            return (string)GetValue(BananaProperty);
        }
        set
        {
            SetValue(BananaProperty, value);
        }
    }


}

为绑定创建的 ValidationRule 和 FrameWorkElement

 public class NotBananaValidationRule:ValidationRule
{
    private NotBananaBinding _notWhatBinding;
    public NotBananaBinding NotWhatBinding
    {
        get { return _notWhatBinding; }
        set { _notWhatBinding = value; }
    }

    public override ValidationResult Validate(object value, System.Globalization.CultureInfo cultureInfo)
    {
        string what = value.ToString();

        if(what == _notWhatBinding.NotWhat||string.IsNullOrEmpty(what))
            return new ValidationResult(false,
                       "Please enter a string that is not " + _notWhatBinding.NotWhat);
        else
            return new ValidationResult(true, null);

    }

}


public class NotBananaBinding : FrameworkElement
{
    public static readonly DependencyProperty NotWhatProperty = DependencyProperty.Register(
      "NotWhat", typeof(string), typeof(NotBananaBinding), new UIPropertyMetadata());

    public string NotWhat
    {
        get { return (string)GetValue(NotWhatProperty); }
        set { SetValue(NotWhatProperty, value); }
    }

    public NotBananaBinding() { }
}

基本上,这段代码的作用是检查您是否输入了“Banana”,然后返回验证错误。该控件公开了依赖属性,因为我希望在使用该控件时能够绑定到它们。NotBananaBinding让我可以创建依赖属性(因为它是一个所以我可以为验证绑定东西。ValidationRule 有一个FrameworkElement NotBananaBinding属性,它存储依赖属性并在 validate 方法中使用它。DependencyObject

我知道我的财产名称有点蹩脚,对不起。问题是该示例在显示错误方面做得很好。在我匆忙做一个例子时,我没有很好地命名变量。如果您发现代码很糟糕,请在此处下载示例


到目前为止我发现了什么

基本上这个问题似乎是由于我实际上并没有改变 value 引起的

即使我在属性上调用 OnPropertyChanged,因为值没有不同,它也不会尝试重新评估验证。

我显然可以将值更改为任意 Valid 值,然后将其更改为我想要的值,它会起作用,但我希望有某种方法可以手动获取呼叫验证,重新评估值然后更改它等等。把它换回来有点乱。


结论

我做错了吗(也许是关于我从 Josh Smiths 帖子中实现验证和绑定的方式)

这只是 ac# 错误,还是预期的行为?如果是,那为什么?

有什么优雅的方法可以解决吗?

u_u

4

1 回答 1

1

验证阻止设置Text属性。在 setter 上放一个断点,你会发现当你输入最后一个 'a' 时它不会中断。如果您键入 Bananan 并按 Backspace 并出现错误,请按下按钮,它将起作用。验证确保您的属性中没有无效值。因此,如果您在出错时将其保存到数据库,则不会保存无效值。

于 2012-03-10T02:11:24.647 回答