感谢迈克尔的帮助。不幸的是,我需要绑定保持不断同步,只有延迟验证。否则,迈克尔的解决方案就可以完成这项工作。所以,我走向了一个稍微不同的方向。这是我最终实施的解决方案。
简单演示:让我们从最简单的案例开始:我的业务对象实现IDataErrorInfo
;假设我可以修改它的实现方式。我给每个业务对象一个额外的布尔属性 ,ValidationEnabled
并且我修改了IDataErrorInfo
实现以在该属性为假时始终返回空结果。
我在原始帖子中提到的演示使用具有两个属性的 Contact 对象;名字和姓氏。我添加了一个 ValidationEnabled 属性并修改了IDataErrorInfo
实现,如下所示:
#region IDataErrorInfo Members
public string Error
{
get { throw new System.NotImplementedException(); }
}
public string this[string columnName]
{
get
{
// Initialize
string result = null;
// Perform validation only if enabled
if (ValidationEnabled)
{
switch (columnName)
{
// Validate 'First Name'
case "FirstName":
if (string.IsNullOrEmpty(FirstName))
{
result = "First name has to be set";
}
else if(FirstName.Length < 5)
{
result = "First name must be at least five characters";
}
break;
// Validate "Last Name"
case "LastName":
if (string.IsNullOrEmpty(LastName))
{
result = "Last name has to be set";
}
else if (LastName.Length < 5)
{
result = "Last name must be at least five characters";
}
break;
}
}
// Set return value
return result;
}
}
#endregion
在我的简单演示版本中,我将提交按钮连接到代码隐藏中的事件处理程序,该处理程序启用验证并刷新两个文本框的绑定:
private void OnButtonClick(object sender, RoutedEventArgs e)
{
var contact = (Contact) DataContext;
contact.ValidationEnabled = true;
var binding = FirstNameBox.GetBindingExpression(TextBox.TextProperty);
binding.UpdateSource();
binding = LastNameBox.GetBindingExpression(TextBox.TextProperty);
binding.UpdateSource();
}
现在,在我尝试提交表单之前,不会出现红色轮廓。
真实世界:在我的生产应用程序中,我无法控制业务对象,也无法像在简单演示中那样修改它们。因此,我为业务对象创建了一个简单的传递包装器,该包装器将要数据绑定到视图的属性公开,并将这些属性链接到被包装的业务对象的相应属性。包装器实现 IDataErrorInfo 并包含该ValidationEnabled
属性。
如果未启用验证,包装器的 IDataErrorInfo 实现始终返回 null。如果启用了验证,则包装器调用被包装对象上的 IDataErrorInfo 实现并返回它从那里得到的内容。
任何使用 Model-View-ViewModel 模式的人都会熟悉这种方法。我们正在做的是为业务对象创建一个视图模型包装器,这被许多开发人员认为是 MVVM 最佳实践。它将 UI 关注点(在页面提交之前抑制红色边框)与业务模型关注点(简单对象验证)分开。
此外,我的生产应用程序不会在代码隐藏中使用事件处理程序。根据 MVVM,按钮将连接到 ICommand,该命令将包含来自简单演示中 OnButtonClick() 处理程序的逻辑。
我希望这对其他研究此问题的人有所帮助。