两个验证规则中的验证规则未正确更新

3

我已经查看了一些关于验证规则的帖子,但没有遇到我正在经历的问题。

我在WPF窗口的文本框中使用验证规则。我有两个检查,一个是针对空文本框的,一个是使用RegEx匹配无效字符的。

我的问题是这样的:

在我的文本框中:

  1. 键入A-有效,不显示任何内容
  2. 按退格键清空字符串-有效,显示验证错误消息“请在所有字段中输入值”
  3. 键入!-无效-它应该显示“找到无效字符”,但仍显示“请在所有字段中输入值”
  4. 按退格键回到空字符串-技术上有效,因为它仍然显示第一个错误“请在所有字段中输入值”
  5. 键入A-有效,没有错误消息
  6. 键入!-好的,显示消息“找到无效字符”。

情况也是相反的。

打开窗口

  1. 键入!-好的,显示“找到无效字符”。
  2. 按退格键清空字符串-仍然显示“找到无效字符”,而不是“请在所有字段中输入值”。

我的代码如下:

ProviderNew.xaml:

<Label Name="lblProviderName" Content="Provider Name: " 
                   Grid.Row="1" Grid.Column="0"/>

<TextBox Name="txtBoxProviderName" Margin="2" 
    MaxLength="20" CharacterCasing="Upper"
    Grid.Row="1" Grid.Column="1" LostFocus="TxtBoxProviderNameLostFocus"   TextChanged="TxtBoxProviderNameTextChanged">
                <TextBox.Text>
                    <Binding ElementName="This" Path="ProviderName" UpdateSourceTrigger="PropertyChanged">
                        <Binding.ValidationRules>
                            <Domain:ProviderValidation />
                        </Binding.ValidationRules>
                    </Binding>
                </TextBox.Text>
            </TextBox>

            <TextBlock x:Name="tbNameValidation"  Foreground="Red" FontWeight="Bold" Margin="10,0,0,0"
                       Text="{Binding ElementName=txtBoxProviderName, Path=(Validation.Errors), Converter={StaticResource eToMConverter}}"
                       Grid.Row="1" Grid.Column="2" />

提供程序后台代码 - ProviderNew.xaml.cs

public static readonly DependencyProperty NewProviderNameProperty = DependencyProperty.Register("ProviderName",
        typeof (string), typeof(ProviderNew), new UIPropertyMetadata(""));


    public string ProviderName
    {
        get { return (string) GetValue(NewProviderNameProperty); }
        set { SetValue(NewProviderNameProperty, value); }
    }

值转换器类

public class ErrorsToMessageConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        var sb = new StringBuilder();
        var errors = value as ReadOnlyCollection<ValidationError>;

        if (errors != null && errors.Count > 0)
        {

            foreach (var e in errors.Where(e => e.ErrorContent != null))
            {
                sb.AppendLine(e.ErrorContent.ToString());
            }
        }
        return sb.ToString();
    }

    public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        throw new NotImplementedException();
    }
}

提供程序类

private string _providerName;
    public string ProviderName
    {
        get { return _providerName; }
        set
        {
            _providerName = value;
            RaisePropertyChanged("ProviderName");
        }
    }

private void RaisePropertyChanged(string propertyName)
    {
        PropertyChangedEventHandler handler = this.PropertyChanged;

        if (handler != null)
        {
            handler(this, new PropertyChangedEventArgs(propertyName));
        }
    }

验证规则类

public class ProviderValidation : ValidationRule
{
    public override ValidationResult Validate(object value, System.Globalization.CultureInfo cultureInfo)
    {
        var str = value as string;

        if (str != null && !Regex.IsMatch(str, @"^[a-zA-Z0-9]*$"))
        {
           return new ValidationResult(false, "Invalid characters were found.");
        }


        if (String.IsNullOrEmpty(str))
        {
            return new ValidationResult(false, "Please enter a value in all fields.");
        }

        return new ValidationResult(true, null);
    }
}

我尝试过设置失去焦点和文本更改事件来强制更新,使用以下方法:
var expression = txtBoxProviderName.GetBindingExpression(TextBox.TextProperty);
if (expression != null)
    expression.UpdateSource();

在Validate方法上设置断点可以显示正确的匹配并返回正确的ValidationResult,但它不能正确更新文本。

我做了一些非常愚蠢的事吗?

任何建议将不胜感激。

编辑1。

是的,我已经通过MultiDataTrigger和绑定到文本框使其工作。

问题在于当我首次显示窗口时,按钮是启用的,这是我不想要的,因为这可能会允许用户在空文本框中单击保存。

验证规则从窗口打开时就不起作用。

我将焦点设置在文本框上,如果失去焦点或输入不正确的数据,则验证规则会触发并禁用按钮。

默认情况下将按钮设置为禁用会使其在打开时被禁用,但是当没有验证错误时它不会启用。

我可以通过在Load事件上强制检查验证规则来使其工作,使用:

var expression = txtBoxProviderName.GetBindingExpression(TextBox.TextProperty);
if (expression != null)
    expression.UpdateSource();

然而,当窗口第一次打开时,它会立即显示验证错误信息“请在所有字段中输入值”,我不太喜欢这个样子。

有没有办法解决这个问题,或者我不能同时拥有最好的两个世界。

以下是按钮代码:

<Button x:Name="btnSave" Height="25" Width="100" Content="Save" HorizontalAlignment="Right" Margin="0,0,120,0"
            Grid.Row="2" Click="BtnSaveClick" >
        <Button.Style>
            <Style TargetType="{x:Type Button}">
                <Setter Property="IsEnabled" Value="False" />
                <Style.Triggers>
                    <MultiDataTrigger>
                        <MultiDataTrigger.Conditions>
                            <Condition Binding="{Binding ElementName=txtBoxProviderName, Path=(Validation.HasError)}" Value="False" />
                            <Condition Binding="{Binding ElementName=txtBoxHelpDesk, Path=(Validation.HasError)}" Value="False" />
                            <Condition Binding="{Binding ElementName=txtBoxRechargeInstructions, Path=(Validation.HasError)}" Value="False" />
                        </MultiDataTrigger.Conditions>
                        <Setter Property="IsEnabled" Value="True" />
                    </MultiDataTrigger>
                </Style.Triggers>
            </Style>
        </Button.Style>     
    </Button>

谢谢,

尼尔

编辑2

一个快速的问题。我找不到RelayCommand命名空间。在搜索其他代码时,我发现了微软的MVVM示例,该示例实现了一个RelayCommand:ICommand类。

这是正确的吗?

代码如下:

public class RelayCommand : ICommand
{
    #region Fields

    readonly Action<object> _execute;
    readonly Predicate<object> _canExecute;

    #endregion // Fields

    #region Constructors

    /// <summary>
    /// Creates a new command that can always execute.
    /// </summary>
    /// <param name="execute">The execution logic.</param>
    public RelayCommand(Action<object> execute)
        : this(execute, null)
    {
    }

    /// <summary>
    /// Creates a new command.
    /// </summary>
    /// <param name="execute">The execution logic.</param>
    /// <param name="canExecute">The execution status logic.</param>
    public RelayCommand(Action<object> execute, Predicate<object> canExecute)
    {
        if (execute == null)
            throw new ArgumentNullException("execute");

        _execute = execute;
        _canExecute = canExecute;
    }

    #endregion // Constructors

    #region ICommand Members

    [DebuggerStepThrough]
    public bool CanExecute(object parameter)
    {
        return _canExecute == null ? true : _canExecute(parameter);
    }

    public event EventHandler CanExecuteChanged
    {
        add { CommandManager.RequerySuggested += value; }
        remove { CommandManager.RequerySuggested -= value; }
    }

    public void Execute(object parameter)
    {
        _execute(parameter);
    }

    #endregion // ICommand Members
}

我在ProviderNew.xaml.cs中实现了以下内容:
private ICommand saveCommand;
    public ICommand SaveCommand
    {
        get
        {
            if (saveCommand == null)
            {
                saveCommand = new RelayCommand(p => DoSave(), p => CanSave() );
            }
            return saveCommand;
        }
    }

    private bool CanSave()
    {
        if (!string.IsNullOrEmpty(ProviderName))
        {
            ??? What goes here? 
        }


        return true;
    }
private bool DoSave()
    {
        // Save the information to DB
    }

说实话,我不确定在“if (!string.IsNullOrEmpty(ProviderName))”中应该编码什么。

此外,您说要在DataContext中添加ICommand代码,所以不确定是否放置在正确的位置,因为当我打开窗口时,保存是启用的,但单击它不起作用,即使所有字段都有正确的数据。

这是按钮的ProviderNew xaml代码:

<Button x:Name="btnSave" Height="25" Width="100" Content="Save" HorizontalAlignment="Right" Margin="0,0,120,0"
            Grid.Row="2" Command="{Binding SaveCommand}" >
        <Button.Style>
            <Style TargetType="{x:Type Button}">
                <Setter Property="IsEnabled" Value="False" />
                <Style.Triggers>
                    <MultiDataTrigger>
                        <MultiDataTrigger.Conditions>
                            <Condition Binding="{Binding ElementName=txtBoxHelpDesk, Path=(Validation.HasError)}" Value="False" />
                            <Condition Binding="{Binding ElementName=txtBoxProviderName, Path=(Validation.HasError)}" Value="False" />
                            <Condition Binding="{Binding ElementName=txtBoxRechargeInstructions, Path=(Validation.HasError)}" Value="False" />
                        </MultiDataTrigger.Conditions>
                        <Setter Property="IsEnabled" Value="True" />
                    </MultiDataTrigger>
                </Style.Triggers>
            </Style>
        </Button.Style>     
    </Button>

非常感谢您的帮助。
此致,
Neill

如果有其他人遇到了与ValidationRule不同的IDataErrorInfo验证问题,那么同样的问题也存在。 - Andreas Duering
1个回答

2

好的,我成功重现了这个问题,而且让我感到困扰的是它没有正常工作。但我找到了解决方法。
问题在于转换器在错误改变时没有被触发,因为实际属性(ProviderName)由于ValidationRule没有改变。如果您在Converter、ValidationRule和Property-setter中设置断点,就可以看到这种行为。

因此,您应该将tbNameValidation的绑定更改为以下内容,而不是使用IValueConverter

<TextBlock x:Name="tbNameValidation"  
           Text="{Binding ElementName=txtBoxProviderName, 
                  Path=(Validation.Errors).CurrentItem.ErrorContent}">

重要的部分是Path=(Validation.Errors).CurrentItem.ErrorContent。这将确保显示当前错误消息。
此外,如果您希望从一开始就显示"请在所有字段中输入值。"消息,您应该将txtBoxProviderName的绑定添加Mode=TwoWay
 <Binding ElementName="This" 
          Path="ProviderName" 
          UpdateSourceTrigger="PropertyChanged"
          Mode="TwoWay">

第二部分的编辑

要修复按钮的启用属性,您可以使用与问题中相同的代码,但是不要使用Click事件来运行保存代码,而是使用Command属性:

在您的DataContext中,创建一个ICommand类型的属性,并将按钮绑定到此属性:

<Button Command="{Binding SaveCommand}" .... />

private ICommand saveCommand; 
public ICommand SaveCommand { 
    get { 
        if (saveCommand == null) {
            saveCommand = new RelayCommand(p => DoSave(), p=> CanSave() ); 
         } 
         return saveCommand; 
      } 
 }

CanSave实现可以检查是否满足所有期望:

private bool CanSave(){
   if (!string.IsNullOrEmpty(ProviderName)){
       //.... etc
   }
 }
CanSave()方法将决定您的按钮的状态,因为CommandBinding会根据绑定命令的CanExecute方法自动启用该按钮。但是,您的验证器的逻辑会再次返回此处,因此最好找到一种重用该代码的方法。
有关更多信息,请参阅Relaying Commands

非常感谢。现在它运行得很好。您已经回答了我的第二个问题。为使规则验证从一开始就起作用,因为如果规则验证失败,则我的保存按钮将无法启用。然而,Mode="TwoWay"并不起到这个作用。不过,两者互动模式难道不是默认模式吗? - Neill
您可以使用 Validation.HasError 属性通过 BindingTrigger 来控制保存按钮的状态。 - RoelF
请查看帖子的编辑1,这里的信息太少了。谢谢。 - Neill
验证在WPF中一直是一个棘手的问题,我个人认为...请参考编辑,希望这能帮到你。 - RoelF
如果你需要的话,可以在聊天框中联系我,那么我可以提供更多信息。问题已经不是原来的了。http://chat.stackoverflow.com/rooms/7/c - RoelF
谢谢,我同意。正如我所说,我从微软那里找到了一个MVVM示例。我打算通过这个示例来解决我的问题,因为它恰好符合我的要求。我们可以认为问题已经解决了,因为你确实解决了我的原始问题。非常感谢你的帮助。 - Neill

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接