在WPF中的UserControl中的控件上显示验证错误模板

时间:2013-02-19 18:49:31

标签: c# wpf validation mvvm idataerrorinfo

如何让WPF错误模板出现在WPF中UserControl的控件上?

我有一个包含两个标签,两个TextBox和一个CheckBox的UserControl。其中一个TextBoxes表示实体的名称,它绑定到ViewModel公开的Model属性的Name属性,该属性是我的Window的DataContext。 Model类实现了IDataErrorInfo接口,我通过单元测试确认,当Name为空时,通过属性索引器实现返回错误。我已经绑定了支持UserControl中的Name TextBox的Dependency Property,当遇到验证错误时,WPF错误模板在整个UserControl周围放置一个红色边框,而不仅仅是Name TextBox。

绑定到UserControl的name字段如下。

<vc:MyUserControl ItemName="{Binding Model.Name, ValidatesOnDataErrors=True}" />

我的UserControl的简化版本和支持DependencyProperty如下。

<UserControl>
    <Grid>
       <TextBox Text="{Binding ItemName}" />
    </Grid>
</UserControl>

public partial class MyUserControl: UserControl
{
    public static readonly DependencyProperty ItemNameProperty = 
        DependencyProperty.Register(
            "ItemName", 
            typeof(string), 
            typeof(MyUserControl), 
            new FrameworkPropertyMetadata(string.Empty, FrameworkPropertyMetadataOptions.BindsTwoWayByDefault)
    );

    public string ItemName
    {
        get { return (string)GetValue(ItemNameProperty); }
        set { SetValue(ItemNameProperty, value); }
    }
}

到目前为止,我发现的与此问题相关的信息都是关于Silverlight或使用转换器不显示红色边框(这对我来说没有意义)。这些信息都在stackoverflow上找到。

有没有人能够用WPF解决这个问题?我忽略了一些明显的东西吗?

1 个答案:

答案 0 :(得分:10)

如果绑定到ErrorTemplate使用ValidatesOnDataErrors=True,则会UserControl使用UserControl。但您可以使用Validation.ErrorTemplate Attached Property删除红色边框。

UserControl中的所有控件只有在通过为支持DependencyProperties实现IDataErrorInfo验证其绑定时才会显示红色边框。

public class MyUserControl : UserControl, IDataErrorInfo
{
   public static readonly DependencyProperty ItemNameProperty =
       DependencyProperty.Register(
           "ItemName",
           typeof(string),
           typeof(MyUserControl),
           new FrameworkPropertyMetadata(string.Empty, FrameworkPropertyMetadataOptions.BindsTwoWayByDefault)
   );

   public string ItemName
   {
      get { return (string)GetValue(ItemNameProperty); }
      set { SetValue(ItemNameProperty, value); }
   }

   public string Error
   {
      get { throw new NotImplementedException(); }
   }

   public string this[string columnName]
   {
      get
      {
         // use a specific validation or ask for UserControl Validation Error 
         return Validation.GetHasError(this) ? "UserControl has Error" : null;
      }
   }
}

这里是简化的XAML

<UserControl Validation.ErrorTemplate="{x:Null}">
   <Grid DataContext="{Binding RelativeSource={RelativeSource AncestorType=UserControl}}">
      <TextBox Text="{Binding ItemName, ValidatesOnDataErrors=True}" />
   </Grid>
</UserControl>

<强>加成

如果您想区分错误,可以获取DependencyProperty的BindingExpression并查看HasError Property

BindingExpression be = BindingOperations.GetBindingExpression(this, ItemNameProperty);
return be != null && be.HasError ? "ItemName has Error" : null;