4

我有一个绑定到 Integer 属性的文本框。当用户在文本框中输入无法转换为整数的内容(例如名称)时,将引发异常并且原始属性值不会更改。我想捕获异常,以便我可以禁用连接到该属性的命令?一般来说,如果可能的话,我怎么能从定义属性的视图模型中做到这一点?

4

3 回答 3

5

我最近遇到了同样的问题,我使用行为来解决它(但如果你不想要它们就不需要它们,它只是为了在不同的视图中重用我需要的一些代码)。主要思想是在 ViewModel 中定义一些方法,允许视图通知 ViewModel 无法检测到的输入中的错误。

因此,首先在您的 ViewModel 中定义这些方法。为简单起见,我将只跟踪错误的数量,但您可以存储有关它们的更多信息(如实际错误):

private int _errorCount = 0;
void AddUIValidationError()
{
   _errorCount++;
}

void RemoveUIValidationError()
{
   _errorCount--;
}

然后,在您的视图中注册System.Windows.Controls.Validation.ErrorEvent,这是一个路由事件,可让您知道组件(之前配置为通知数据错误)何时检测到错误(如异常验证错误):

public partial class MyView : UserControl // or whatever it is
{
    public MyView(MyViewModel viewModel)
    {
        // Possibly ensure that viewModel is not null
        InitializeComponent();
        _myViewModel = viewModel;

        this.AddHandler(System.Windows.Controls.Validation.ErrorEvent, new RoutedEventHandler(OnValidationRaised));
    }

    private MyViewModel _myViewModel;

    private void OnValidationRaised(object sender, RoutedEventArgs e)
    {
        var args = (System.Windows.Controls.ValidationErrorEventArgs)e;

        if (_myViewModel != null)
        {

            // Check if the error was caused by an exception
            if (args.Error.RuleInError is ExceptionValidationRule)
            {
                // Add or remove the error from the ViewModel
                if (args.Action == ValidationErrorEventAction.Added)
                    _myViewModel.AddUIValidationError();
                else if (args.Action == ValidationErrorEventAction.Removed)
                    _myViewModel.RemoveUIValidationError();
            }
        }
    }
}

在 Command 的 CanExecute 方法中,您将检查 ViewModel 的 _errorCount 字段是否大于 0,在这种情况下,应禁用该命令。

请注意,您必须添加ValidatesOnExceptions=True, NotifyOnValidationError=True到您的绑定中才能正常工作。前任:

<TextBox Text="{Binding Path=MyProperty, ValidatesOnExceptions=True, NotifyOnValidationError=True}" />

编辑:

除了 Riley 提到的(这也很好,但需要将模型中的每个整数属性映射到 ViewModel 中的新字符串属性)之外,另一种方法是使用 ValidationRules。您可以添加ValidationRule在解析和调用属性设置器之前检查的 s。因此,例如,您可以从 ValidationRule 继承并实现 Validate 方法以确保可以将字符串解析为整数。例子:

public class IntegerValidationRule : ValidationRule
{
    public override ValidationResult Validate(object value, CultureInfo cultureInfo)
    {
        int number;
        if(Int32.TryParse((string)value, out number) == false)
            return new ValidationResult(false, "It is not a valid number");
        return new ValidationResult(true, null);
    }
}

然后,在您的视图中定义定义 IntegerValidationRule 的命名空间:

<UserControl 
...
    xmlns:rules="clr-namespace:MyApplication.ValidationRules"
...>

并在绑定中使用规则:

<TextBox>
    <TextBox.Text>
       <Binding Path="MyProperty">
           <Binding.ValidationRules>

              <rules:IntegerValidationRule/>
           </Binding.ValidationRules>
       </Binding>
    </TextBox.Text>
</TextBox>

但无论如何,您需要为每个要验证的非字符串类型创建类,而且我认为 Binding 语法现在看起来有点长。

问候

于 2012-11-11T22:09:12.687 回答
1

我在网上找到的“最佳”解决方案在http://www.wpfsharp.com/2012/02/03/how-to-disable-a-button-on-textbox-validationerrors-in-wpf进行了解释/

简而言之,Validation 错误是在 View 中处理的,而不是在 ViewModel 中。您不需要自己的 ValidationRule。只需在 XAML 中为您的按钮添加样式:

<Button.Style>
            <Style TargetType="{x:Type Button}">
                <Setter Property="IsEnabled" Value="false" />
                <Style.Triggers>
                    <MultiDataTrigger>
                        <MultiDataTrigger.Conditions>
                            <Condition Binding="{Binding ElementName=TextBox1, Path=(Validation.HasError)}" Value="false" />
                            <Condition Binding="{Binding ElementName=TextBox2, Path=(Validation.HasError)}" Value="false" />
                        <Setter Property="IsEnabled" Value="true" />
                    </MultiDataTrigger>
                </Style.Triggers>
            </Style>
        </Button.Style>

希望这对偶然发现这个问题的人有所帮助(我知道多年后它不太可能对 OP 有所帮助)。

于 2015-05-09T15:51:41.213 回答
0

考虑从视图模型而不是视图调用命令。

private int _myProperty;
public int MyProperty
{
    get
    {
        return _myProperty;
    }
    set
    {
        _myProperty = value;
        // See if the value can be parsed to an int.
        int potentialInt;
        if(int.TryParse(_myProperty, out potentialInt))
        {
            // If it can, execute your command with any needed parameters.
            yourCommand.Execute(_possibleParameter)
        }
    }
}

这将允许您处理用户键入的无法解析为整数的内容,并且您只会在用户键入的内容是整数时触发命令。

(我没有测试这段代码,但我认为它可能会有所帮助。)

于 2012-11-12T01:36:27.547 回答