10

tl; dr:强制值不会跨数据绑定传播。当代码隐藏不知道绑定的另一端时,如何强制更新数据绑定?


我在CoerceValueCallbackWPF 依赖属性上使用 a ,但我遇到了强制值不会传播到绑定的问题。

Window1.xaml.cs

using System;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Media;

namespace CoerceValueTest
{
    public class SomeControl : UserControl
    {
        public SomeControl()
        {
            StackPanel sp = new StackPanel();

            Button bUp = new Button();
            bUp.Content = "+";
            bUp.Click += delegate(object sender, RoutedEventArgs e) {
                Value += 2;
            };

            Button bDown = new Button();
            bDown.Content = "-";
            bDown.Click += delegate(object sender, RoutedEventArgs e) {
                Value -= 2;
            };

            TextBlock tbValue = new TextBlock();
            tbValue.SetBinding(TextBlock.TextProperty,
                               new Binding("Value") {
                                Source = this
                               });

            sp.Children.Add(bUp);
            sp.Children.Add(tbValue);
            sp.Children.Add(bDown);

            this.Content = sp;
        }

        public static readonly DependencyProperty ValueProperty = DependencyProperty.Register("Value",
                                                                                              typeof(int),
                                                                                              typeof(SomeControl),
                                                                                              new PropertyMetadata(0, ProcessValueChanged, CoerceValue));

        private static object CoerceValue(DependencyObject d, object baseValue)
        {
            if ((int)baseValue % 2 == 0) {
                return baseValue;
            } else {
                return DependencyProperty.UnsetValue;
            }
        }

        private static void ProcessValueChanged(object source, DependencyPropertyChangedEventArgs e)
        {
            ((SomeControl)source).ProcessValueChanged(e);
        }

        private void ProcessValueChanged(DependencyPropertyChangedEventArgs e)
        {
            OnValueChanged(EventArgs.Empty);
        }

        protected virtual void OnValueChanged(EventArgs e)
        {
            if (e == null) {
                throw new ArgumentNullException("e");
            }

            if (ValueChanged != null) {
                ValueChanged(this, e);
            }
        }

        public event EventHandler ValueChanged;

        public int Value {
            get {
                return (int)GetValue(ValueProperty);
            }
            set {
                SetValue(ValueProperty, value);
            }
        }
    }

    public class SomeBiggerControl : UserControl
    {
        public SomeBiggerControl()
        {
            Border parent = new Border();
            parent.BorderThickness = new Thickness(2);
            parent.Margin = new Thickness(2);
            parent.Padding = new Thickness(3);
            parent.BorderBrush = Brushes.DarkRed;

            SomeControl ctl = new SomeControl();
            ctl.SetBinding(SomeControl.ValueProperty,
                           new Binding("Value") {
                            Source = this,
                            Mode = BindingMode.TwoWay
                           });
            parent.Child = ctl;

            this.Content = parent;
        }

        public static readonly DependencyProperty ValueProperty = DependencyProperty.Register("Value",
                                                                                              typeof(int),
                                                                                              typeof(SomeBiggerControl),
                                                                                              new PropertyMetadata(0));

        public int Value {
            get {
                return (int)GetValue(ValueProperty);
            }
            set {
                SetValue(ValueProperty, value);
            }
        }
    }

    public partial class Window1 : Window
    {
        public Window1()
        {
            InitializeComponent();
        }
    }
}

Window1.xaml

<Window x:Class="CoerceValueTest.Window1"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    Title="CoerceValueTest" Height="300" Width="300"
    xmlns:local="clr-namespace:CoerceValueTest"
    >
    <StackPanel>
        <local:SomeBiggerControl x:Name="sc"/>
        <TextBox Text="{Binding Value, ElementName=sc, Mode=TwoWay}" Name="tb"/>
        <Button Content=" "/>
    </StackPanel>
</Window>

即两个用户控件,一个嵌套在另一个内部,一个嵌套在一个窗口中。内部用户控件具有Value绑定到Value外部控件的依赖属性的依赖属性。在窗口中,一个TextBox.Text属性被绑定到Value外部控件的属性上。

内部控件CoerceValueCallback注册了一个Value属性,其效果是该Value属性只能分配偶数。

请注意,出于演示目的,此代码已简化。真实版本不会在构造函数中初始化任何东西;这两个控件实际上具有控件模板,这些模板可以完成此处相应构造函数中所做的所有事情。也就是说,在实际代码中,外部控件不知道内部控件。

当将偶数写入文本框并更改焦点时(例如,通过将虚拟按钮聚焦在文本框下方),两个Value属性都会得到适当更新。但是,当在文本框中写入奇数时,Value内部控件的属性不会改变,而Value外部控件的属性以及TextBox.Text属性都会显示奇数。

我的问题是:如何在文本框中强制更新(理想情况下也可以在外部控件的Value属性中,当我们在它的时候)?

在同一个问题上找到了一个 SO question,但并没有真正提供解决方案。它暗示使用属性更改事件处理程序来重置值,但据我所知,这意味着将评估代码复制到外部控件......这并不可行,因为我的实际评估代码依赖于一些信息基本上只有内部控制才知道(不费力气)。

此外,这篇博文建议调用UpdateTarget.内在控制。我不知道在哪里执行此操作,但在方法中,尚未设置强制值(因此更新绑定还为时过早),而在值由 重置的情况下,属性值将保持原样,因此不会调用属性更改的回调(如本讨论中所暗示的那样)。TextBox.TextCoerceValueCallbackUpdateSourceValueCoerceValueCoerceValue

我想到的一种可能的解决方法是用SomeControl常规属性和实现替换依赖属性(因此即使值已被强制INotifyPropertyChanged,我也可以手动触发事件)。PropertyChanged但是,这意味着我不能再声明对该属性的绑定,因此这不是一个真正有用的解决方案。

4

1 回答 1

4

一段时间以来,我一直在寻找这个相当讨厌的错误的答案。一种不需要在绑定上强制 UpdateTarget 的方法是:

  • 删除您的CoerceValue回调。
  • 将CoerceValue回调的逻辑转移到ProcessValueChanged回调中。
  • 在适用时(当数字为奇数时)将您的强制值分配给您的Value属性
  • 您最终会遇到两次ProcessValueChanged回调,但您的强制值最终会被有效地推送到您的绑定中。

根据您的代码,您的依赖属性声明将变为:

public static readonly DependencyProperty ValueProperty = 
                       DependencyProperty.Register("Value",
                                                   typeof(int),
                                                   typeof(SomeControl),
                                                   new PropertyMetadata(0, ProcessValueChanged, null));

然后,您的 ProcessValueChanged将变为:

private static void ProcessValueChanged(object source, DependencyPropertyChangedEventArgs e)
    {
        int baseValue = (int) e.NewValue;
        SomeControl someControl = source as SomeControl;
        if (baseValue % 2 != 0) 
        {
            someControl.Value = DependencyProperty.UnsetValue;
        }
        else
        {
            someControl.ProcessValueChanged(e);
        }
    }

我稍微修改了您的逻辑,以防止在需要强制值时引发事件。如前所述,将强制值分配给someControl.Value将导致您的ProcessValueChanged被连续调用两次。放置 else 语句只会引发一次具有有效值的事件。

我希望这有帮助!

于 2013-08-28T14:18:10.403 回答