diff --git a/src/Avalonia.Base/Data/Core/BindingExpression.cs b/src/Avalonia.Base/Data/Core/BindingExpression.cs index ba0ce7bb9e6..a7d1d456def 100644 --- a/src/Avalonia.Base/Data/Core/BindingExpression.cs +++ b/src/Avalonia.Base/Data/Core/BindingExpression.cs @@ -507,8 +507,10 @@ private void OnTargetPropertyChanged(object? sender, AvaloniaPropertyChangedEven Debug.Assert(_mode is BindingMode.TwoWay or BindingMode.OneWayToSource); Debug.Assert(UpdateSourceTrigger is UpdateSourceTrigger.PropertyChanged); - if (e.Property == TargetProperty) - WriteValueToSource(e.NewValue); + // The value must be read from the target object instead of using the value from the event + // because the value may have changed again between the time the event was raised and now. + if (e.Property == TargetProperty && TryGetTarget(out var target)) + WriteValueToSource(target.GetValue(TargetProperty)); } private object? ConvertFallback(object? fallback, string fallbackName) diff --git a/tests/Avalonia.Markup.UnitTests/Data/BindingTests.cs b/tests/Avalonia.Markup.UnitTests/Data/BindingTests.cs index d4a799da6fc..9fb52fee388 100644 --- a/tests/Avalonia.Markup.UnitTests/Data/BindingTests.cs +++ b/tests/Avalonia.Markup.UnitTests/Data/BindingTests.cs @@ -667,6 +667,31 @@ public void OneWayToSource_Binding_Does_Not_Override_TwoWay_Binding() Assert.Equal("TwoWay", source.Foo); } + [Fact] + public void Target_Undoing_Property_Change_During_TwoWay_Binding_Does_Not_Cause_StackOverflow() + { + var source = new TestStackOverflowViewModel { BoolValue = true }; + var target = new TwoWayBindingTest(); + + source.ResetSetterInvokedCount(); + + // The AlwaysFalse property is set to false in the PropertyChanged callback. Ensure + // that binding it to an initial `true` value with a two-way binding does not cause a + // stack overflow. + target.Bind( + TwoWayBindingTest.AlwaysFalseProperty, + new Binding(nameof(TestStackOverflowViewModel.BoolValue)) + { + Mode = BindingMode.TwoWay, + }); + + target.DataContext = source; + + Assert.Equal(1, source.SetterInvokedCount); + Assert.False(source.BoolValue); + Assert.False(target.AlwaysFalse); + } + private class StyledPropertyClass : AvaloniaObject { public static readonly StyledProperty DoubleValueProperty = @@ -725,10 +750,24 @@ private class TestStackOverflowViewModel : INotifyPropertyChanged public const int MaxInvokedCount = 1000; + private bool _boolValue; private double _value; public event PropertyChangedEventHandler PropertyChanged; + public bool BoolValue + { + get => _boolValue; + set + { + if (_boolValue != value) + { + _boolValue = value; + SetterInvokedCount++; + PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(BoolValue))); + } + } } + public double Value { get => _value; @@ -754,20 +793,38 @@ public double Value } } } + + public void ResetSetterInvokedCount() => SetterInvokedCount = 0; } private class TwoWayBindingTest : Control { + public static readonly StyledProperty AlwaysFalseProperty = + AvaloniaProperty.Register(nameof(AlwaysFalse)); public static readonly StyledProperty TwoWayProperty = AvaloniaProperty.Register( "TwoWay", defaultBindingMode: BindingMode.TwoWay); + public bool AlwaysFalse + { + get => GetValue(AlwaysFalseProperty); + set => SetValue(AlwaysFalseProperty, value); + } + public string TwoWay { get => GetValue(TwoWayProperty); set => SetValue(TwoWayProperty, value); } + + protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) + { + base.OnPropertyChanged(change); + + if (change.Property == AlwaysFalseProperty) + SetCurrentValue(AlwaysFalseProperty, false); + } } public class Source : INotifyPropertyChanged