考虑这个非常简单的例子,我有一个这样的用户控件:
用户控件 XAML:
<UserControl x:Class="BindingTest.SomeControl"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
x:Name="SomeControlElement">
<Grid>
<TextBlock Text="{Binding ElementName=SomeControlElement, Path=Counter}" />
</Grid>
</UserControl>
代码隐藏:
using System;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Threading;
namespace BindingTest
{
public partial class SomeControl : UserControl
{
public SomeControl()
{
InitializeComponent();
var timer = new DispatcherTimer();
timer.Interval = new TimeSpan(0, 0, 5);
timer.Tick += (s, e) => Counter = Counter + 1;
timer.Start();
}
public int Counter
{
get { return (int)GetValue(CounterProperty); }
set { SetValue(CounterProperty, value); }
}
public static readonly DependencyProperty CounterProperty = DependencyProperty.Register(nameof(Counter), typeof(int), typeof(SomeControl), new PropertyMetadata(0));
}
}
因此,控件仅显示一个文本块,并且每 5 秒递增计数器。然后我当然有一个消费者:
主窗口 XAML:
<Window x:Class="BindingTest.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:BindingTest"
x:Name="MainWindowName" Width="200" Height="300">
<Grid HorizontalAlignment="Center" VerticalAlignment="Center">
<StackPanel>
<local:SomeControl Counter="{Binding ElementName=MainWindowName, Path=SomeSource, Mode=OneWay}" />
<local:SomeControl Counter="{Binding ElementName=MainWindowName, Path=SomeSource, Mode=TwoWay}" />
</StackPanel>
</Grid>
</Window>
最后是后面的主代码:
using System;
using System.Windows;
using System.ComponentModel;
using System.Windows.Threading;
namespace BindingTest
{
public partial class MainWindow : Window, INotifyPropertyChanged
{
public MainWindow()
{
InitializeComponent();
var timer = new DispatcherTimer();
timer.Interval = new TimeSpan(0, 0, 1);
timer.Tick += (s, e) => SomeSource = SomeSource + 1;
timer.Start();
}
private int someSource;
public int SomeSource
{
get => someSource;
set
{
if (someSource != value)
{
someSource = value;
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(SomeSource)));
}
}
}
public event PropertyChangedEventHandler PropertyChanged;
}
}
对,所以 main 在代码后面有一个计数器,每秒更新一个属性。XAML 有 2 个用户控件实例。一个具有单向绑定,另一个具有双向绑定。
我在这里看到的是,当"SomeControl.cs"中的计数器更新时,第一个UserControl(OneWay(的绑定被破坏了。带有TwoWay的那个不断更新。
这是设计使然(为什么(?更重要的是,如果我需要更新用户控件中的属性,我将如何在我的示例中执行此操作 - 以支持 OneWay 绑定?请注意,我真的对此示例中的 TwoWay 绑定不感兴趣,因为它会更新"MySource",这不是我想要的!
谢谢。
这是设计使然。将所谓的本地值分配给依赖项属性时,将替换以前分配的单向绑定。双向绑定保持活动状态并更新其源属性。
但是,有一个解决方法。不要设置本地值,而是设置"当前值"。取代
timer.Tick += (s, e) => Counter = Counter + 1;
跟
timer.Tick += (s, e) => SetCurrentValue(CounterProperty, Counter + 1);