我有一个绑定到后面代码中的切换按钮的绑定。 我想将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个单独的控件播放器并将其绑定到一个按钮上的有效方法。
答案 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编辑器对此属性的支持会更好。
我发现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;
}