WPF按钮绑定到属性Convertback IMultiValueConverter不会更新值

时间:2019-03-14 14:10:10

标签: c# wpf binding converters multibinding

我有一个绑定到后面代码中的切换按钮的绑定。 我想将isChecked状态从一个按钮绑定到4个视频控件,以切换静音功能。我正在使用多重绑定将切换按钮绑定到4个不同的控件。我的问题是使用断点,我可以看到一切都被触发,直到每个对象的属性Mutate属性都被触发,但是属性“ value”参数从未更新。实例化控件时,它保持默认设置。

所以我首先在后面的代码中创建绑定

        IMultiValueConverter converter = new EmptyMultiValueConverter();
        MultiBinding myMultiBinding = new MultiBinding();
        myMultiBinding.Converter = converter;
        myMultiBinding.UpdateSourceTrigger = UpdateSourceTrigger.Default;
        myMultiBinding.Mode = BindingMode.OneWayToSource;      
        myMultiBinding.NotifyOnSourceUpdated = true;  



        for (int i = 1; i < _maxNumberofPlayers; i++)
        {
            VideoPlayer player = new VideoPlayer()
            {
                Mute = false
            };

          myMultiBinding.Bindings.Add(new Binding("Mute") { Source = player 
         });

        }


     btnMuteToggle.SetBinding(SimpleButton.IsCheckedProperty, myMultiBinding);

这一切似乎都起作用,因为当我单击按钮时,我可以在多值转换器中看到正确的isChecked按钮状态到达断点,在下面的ConvertBack处,我可以确认该值是反映切换按钮状态的正确布尔值

public class EmptyMultiValueConverter : IMultiValueConverter
{
    #region IMultiValueConverter Members

    public object Convert(object[] values, Type targetType, object 
    parameter, System.Globalization.CultureInfo culture)
    {
        // gets from the object source
        return (bool)values[0];
    }

    public object[] ConvertBack(object value, Type[] targetTypes, object 
   parameter, System.Globalization.CultureInfo culture)
    {

        return new Object[] {value,value,value,value};
    }

    #endregion
}

在这一点上,我可以确认它是否达到了Mute属性,并触发了SET 4次,但是当我跟踪它时,value参数保持在先前设置的值,并且不会更新以反映通过它传递给它的值ConvertBack

   // mute property in the media player user control
    public bool Mute
    {
        get { return  _media.IsMuted; }
        set
        {       
            if (_media.IsMuted == value)
                return;
            else
            {
                _media.IsMuted = value;
                NotifyPropertyChanged("Mute");
            }

        }
    }

任何人都可以帮忙。 已经把我的头发扯了三天了。

在我看来,使用多重绑定是连接4个单独的控件播放器并将其绑定到一个按钮上的有效方法。

2 个答案:

答案 0 :(得分:0)

好吧,我再次尝试了几个选项,除了单击按钮事件中的代码外,它们都无法工作

多重绑定根本不起作用。 我可以在断点处看到4个控件的Mute属性被调用了4次,但是value参数从未更新为新的属性值。

我还尝试将一个按钮绑定到4个不同的控件

foreach(Player player in lsPlayers)
{
    btnMuteToggle.SetBinding(SimpleButton.IsCheckedProperty, new Binding("Mute")
       {
         Source = player,
         UpdateSourceTrigger = UpdateSourceTrigger.PropertyChanged,
         Mode = BindingMode.OneWayToSource
        });
}

这只会导致调用一个玩家对象的静音属性。 您不能将多个控件绑定到同一按钮。 只是不起作用。

所以我只是在一次点击事件中调用了4个对象。

答案 1 :(得分:0)

我相信在这种情况下,您可能需要将这4个属性分别绑定到Mutate属性,并确保将每个属性都设置为ModeWay。但是,如果MultiValueConverter支持它,我不能特别反对您的方法。

看来您没有为Mute设置Dependency属性。这对于WPF绑定工作流至关重要,并且比NotifyPropertyChanged易于扩展,因为NotifyPropertyChanged倾向于要求您进行更多管理。 DependencyProperty将提取和发送对属性的更新,而Xaml编辑器对此属性的支持会更好。

https://docs.microsoft.com/en-us/dotnet/framework/wpf/advanced/how-to-implement-a-dependency-property

我发现WPF绑定在理论上是不错的,但仍然需要更多的腿部工作。有功能。我将其添加到所有WPF视图模型和控件中。

/// <summary>
/// simplify wpf binding
/// </summary>
/// <param name="name"></param>
/// <param name="type"></param>
/// <param name="preferNull">Fair Warning. if you create an instance, you will have also created a singleton of this property for all instances of this class. </param>
/// <returns></returns>
private static DependencyProperty AddDp(string name, Type type, bool preferNull = true)
{
    return DependencyProperty.Register(name, type, typeof(Setting),
        new PropertyMetadata((!preferNull || type.IsValueType) ? Activator.CreateInstance(type) : null));
}

仅从DependencyObject继承,您可以使静音看起来像这样:

public static readonly DependencyProperty MuteDp = AddDp(nameof(Mute), typeof(bool));
public bool Mute
{
    get => (bool)GetValue(MuteDp);
    set { if(value != Mute) SetValue(MuteDp, value); }
}

但要警告! 有时WPF绑定到依赖项属性会更新依赖项属性的内部值,而无需输入其相应属性的设置器!我使用的MultiValueBinding就是这种情况。 表示它永远都不会触摸Mute :: Set访问器。这可能很成问题,要解决此问题,您可以使用DependencyObject可用的各种回调!

private static DependencyProperty AddDp(string name, Type type, bool preferNull = true)
{
    var dp = DependencyProperty.Register(name, type, typeof(Setting),
        new PropertyMetadata(
            (!preferNull || type.IsValueType) ? Activator.CreateInstance(type) : null
            ,new PropertyChangedCallback((dobj, dpe)=>
            {
                //after property changed.
                //forcing usage of set accessor
                ((MuteContainerObj)dobj).Value = ((strong text)dobj).Value;
                //or forcibly use an update function
                ((MuteContainerObj)dobj).MuteUpdated();
            }),new CoerceValueCallback((dobj, o)=> {
                //before property changed
                return o;
            })), new ValidateValueCallback((o)=> {
                //before property change events
                return true;
            }));
    return dp;
        
}