gpt4 book ai didi

wpf - 为什么将无操作转换器放在 Binding 上会改变其行为?

转载 作者:行者123 更新时间:2023-12-04 12:50:02 27 4
gpt4 key购买 nike

我正在测试我构建的用户控件,我遇到了一些对我来说莫名其妙的事情。

该控件是处理特定自定义类型值的 ComboBox 的扩展。它具有该自定义类型的依赖属性,该属性是绑定(bind)的目标属性。

我在 setter 中有一个跟踪语句,我可以看到该属性正在设置。但它没有出现在我的用户控件中。

现在,通常我会说,好吧,我的用户控件中有一个错误。我可能会,虽然我对此感到困惑。但这个问题不是关于在我的控制中找到错误。继续阅读;这就是奇怪的地方。

我还使用 Bea Stollnitz 的小值转换器来帮助调试绑定(bind):

public class DebuggingConverter : IValueConverter
{
public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
return value; // Add the breakpoint here!!
}
public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
throw new NotImplementedException("This method should never be called");
}
}

这背后的想法是我将此转换器添加到我的绑定(bind)中,并且可以设置断点以查看将什么值推送到目标。好的,这工作得很好。我可以看到值(value)被推出。

事实上,它工作得有点太好了。如果 DebuggingConverter 附加到 Binding,则用户控件将显示该值。如果不是,它不会。

这怎么可能呢?什么都不做的值转换器如何影响绑定(bind)控件的行为?

编辑:

并不是说它可能会有所帮助,但这是用户控件的 XAML:
<a:CodeLookupBox
Grid.Column="1"
Grid.IsSharedSizeScope="True"
MinWidth="100"
Style="{Binding Style}">
<a:CodeLookupBox.CodeLookupTable>
<Binding Path="Codes" Mode="OneWay"/>
</a:CodeLookupBox.CodeLookupTable>
<a:CodeLookupBox.SelectedCode>
<Binding Path="Value" Mode="TwoWay" ValidatesOnDataErrors="True"/>
</a:CodeLookupBox.SelectedCode>
</a:CodeLookupBox>

在第二个绑定(bind)上没有转换器,控件的行为就像我没有设置 SelectedCode .即使 OnSelectedCodePropertyChanged 中的跟踪语句处理程序显示 e.Value确实包含正确的值。无论转换器是否连接,都会发生这种情况。

我一直在尝试通过思想实验对这个问题进行逆向工程:如果你想创建一个绑定(bind)的用户控件,如果一个无操作转换器附加到它的绑定(bind),它的行为会发生变化,你会怎么做?我对绑定(bind)知之甚少,无法给出答案。

最佳答案

好消息是,我知道为什么 SelectedCode当我不使用值转换器时没有设置。坏消息是,我仍然有一些谜团,但问题已经被推到了食物链上一点,我有一个解决方法。

这个控件本质上是一个强类型的组合框,它具有许多附加功能,因为它知道其中有哪些类型的项目。 SelectedCodeCodeLookupTable属性是强类型的,它们隐藏了底层的 SelectedItemItemsSource属性,不是。 (顺便说一句,这就是为什么这是一个用户控件而不是 ComboBox 的子类;我不希望这些属性可见,因为如果设置不当会发生很多事情,但没有一个是好的.)

这是正在发生的事情。这是附加值转换器时的调试输出(数字是控件的哈希码,因为我有一堆在程序初始化时同时绘制的):

14626603: OnCodeLookupTablePropertyChanged
CodeLookupTable property set to Proceedings.Model.CodeLookupTable
box.MainComboBox.ItemsSource = MS.Internal.Data.EnumerableCollectionView
14626603: OnSelectedCodePropertyChanged:
SelectedCode property set to Unlicensed Driver [VC12500(A)]
box.MainComboBox.ItemsSource = MS.Internal.Data.EnumerableCollectionView

这是预期的行为。 CodeLookupTable属性已设置,因此设置 SelectedCode到该集合中的一项正确设置 SelectedItem在基础 ComboBox .

但是如果没有值转换器,我们会得到:
16143157: OnSelectedCodePropertyChanged:
SelectedCode property set to Unlicensed Driver [VC12500(A)]
box.MainComboBox.ItemsSource =
16143157: OnCodeLookupTablePropertyChanged
CodeLookupTable property set to Proceedings.Model.CodeLookupTable
box.MainComboBox.ItemsSource = MS.Internal.Data.EnumerableCollectionView

在这里, SelectedCodeCodeLookupTable 之前设置属性属性(property)是。所以当方法尝试设置 SelectedItem在基础 ComboBox ,没有任何 react ,因为 ItemsSource一片空白。

这就是问题的根源。我愚蠢地假设绑定(bind)更新其目标的顺序与它们在 XAML 中声明的顺序相同。 (我将绑定(bind)表示为元素而不是属性的原因之一是因为 XML 文档中元素的顺序是确定性的,而属性的顺序不是。这并不是我没有考虑到这一点。)显然不是这样。

我还假设,也许不那么愚蠢,绑定(bind)更新其目标的顺序不取决于它们是否具有附加值转换器。嗯,是的。我想知道它还取决于什么。

幸运的是,我有办法解决这个问题。由于我的 CodeLookup对象包含对 CodeLookupTable 的引用,我可以制作 SelectedCode setter 设置 CodeLookupTable (因此是 ItemsSource )属性,如果尚未设置。这将使这个问题消失,而不必在绑定(bind)上粘贴一个假值转换器,并希望绑定(bind)的行为方式永远不会改变。

编辑

下面是属性声明的样子:
#region SelectedCode

public static readonly DependencyProperty SelectedCodeProperty = DependencyProperty.Register(
"SelectedCode", typeof(CodeLookup), typeof(CodeLookupBox),
new FrameworkPropertyMetadata(OnSelectedCodePropertyChanged));

private static void OnSelectedCodePropertyChanged(DependencyObject source, DependencyPropertyChangedEventArgs e)
{
CodeLookupBox box = (CodeLookupBox)source;
CodeLookup code = e.NewValue as CodeLookup;
// this right here is the fix to the original problem:
if (box.CodeLookupTable == null && code != null)
{
box.CodeLookupTable = code.Table;
}
box.MainComboBox.SelectedItem = e.NewValue;
}

public CodeLookup SelectedCode
{
get { return GetValue(SelectedCodeProperty) as CodeLookup; }
set { SetValue(SelectedCodeProperty, value); }
}

#endregion

#region CodeLookupTable

public static readonly DependencyProperty CodeLookupTableProperty = DependencyProperty.Register(
"CodeLookupTable", typeof(CodeLookupTable), typeof(CodeLookupBox),
new FrameworkPropertyMetadata(OnCodeLookupTablePropertyChanged));

private static void OnCodeLookupTablePropertyChanged(DependencyObject source,
DependencyPropertyChangedEventArgs e)
{
CodeLookupBox box = (CodeLookupBox)source;
CodeLookupTable table = (CodeLookupTable)e.NewValue;

box.ViewSource = new CollectionViewSource { Source = table.Codes };
box.View = box.ViewSource.View;
box.MainComboBox.ItemsSource = box.View;

}

public CodeLookupTable CodeLookupTable
{
get { return GetValue(CodeLookupTableProperty) as CodeLookupTable; }
set { SetValue(CodeLookupTableProperty, value); }
}

#endregion

关于wpf - 为什么将无操作转换器放在 Binding 上会改变其行为?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2285812/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com