我刚刚根据Rachel Lim的博客实现了我的业务逻辑验证。一切都很顺利,直到我决定在我的视图中设置一个触发器,绑定到IsValid属性,如下所示:
<ListBox ItemsSource="{Binding EdgedBoards}" SelectedItem="{Binding SelEdgedBoard, Mode=TwoWay}" DisplayMemberPath="Name">
<ListBox.ItemContainerStyle>
<Style TargetType="{x:Type ListBoxItem}" BasedOn="{StaticResource {x:Type ListBoxItem}}">
<Setter Property="Focusable" Value="True" />
<Style.Triggers>
<DataTrigger Binding="{Binding Path=IsValid}" Value="False">
<Setter Property="Focusable" Value="False" />
</DataTrigger>
</Style.Triggers>
</Style>
</ListBox.ItemContainerStyle>
</ListBox>
问题是,当绑定项出现错误(SelEdgedBoard.IsValid==false)时,不会通知触发器重新评估,并且该项将其可聚焦属性保持为true。
我已经尝试在GetValidationError()返回其值之前放入NotifyPropertyChanged("IsValid"),但通过这种方式我得到了一个stackerflow异常:
#region IsValid Property
public bool IsValid
{
get
{
return string.IsNullOrWhiteSpace(GetValidationError());
}
}
public string GetValidationError()
{
string error = null;
if (ValidatedProperties != null)
{
foreach (string s in ValidatedProperties)
{
error = GetValidationError(s);
if (!string.IsNullOrWhiteSpace(error))
{
break;
}
}
}
NotifyPropertyChanged("IsValid");
return error;
}
#endregion
当然,它会导致堆栈溢出。当你这样做:
NotifyPropertyChanged("IsValid")
您可以强制WPF基础结构重新评估IsValid的值。它通过调用IsValid getter来实现这一点,而该getter又再次调用GetValidationError!
有几种方法可以解决这个问题。我可能会创建一个私有成员变量来保存IsValid的最后一个值,然后在调用NotifyPropertyChanged之前将当前值与旧值进行比较。
另一种方法可能是,只有当一个已验证的属性发生更改时(因此在每个属性的setter中),才调用NotifyPropertyChanged("IsValid"),因为可能会导致IsValid发生更改。
private bool _isValid;
public bool IsValid
{
get
{
string.IsNullOrWhiteSpace(GetValidationError())
return _isValid;
}
set
{
_isValid = value;
NotifyPropertyChanged("IsValid");
}
}
public string GetValidationError()
{
string error = null;
if (ValidatedProperties != null)
{
foreach (string s in ValidatedProperties)
{
error = GetValidationError(s);
if (!string.IsNullOrWhiteSpace(error))
{
break;
}
}
}
IsValid=string.IsNullOrWhiteSpace(error);
return error;
}
我希望这会有所帮助。