| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382 | 
							- using System;
 
- using System.Collections.Generic;
 
- using System.Globalization;
 
- using System.Reactive.Linq;
 
- using System.Threading;
 
- using System.Threading.Tasks;
 
- using Avalonia.Data;
 
- using Avalonia.Data.Converters;
 
- using Avalonia.Data.Core;
 
- using Avalonia.Markup.Parsers;
 
- using Avalonia.UnitTests;
 
- using Moq;
 
- using Xunit;
 
- namespace Avalonia.Base.UnitTests.Data.Core
 
- {
 
-     public class BindingExpressionTests : IClassFixture<InvariantCultureFixture>
 
-     {
 
-         [Fact]
 
-         public async Task Should_Get_Simple_Property_Value()
 
-         {
 
-             var data = new Class1 { StringValue = "foo" };
 
-             var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(string));
 
-             var result = await target.Take(1);
 
-             Assert.Equal("foo", result);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public void Should_Set_Simple_Property_Value()
 
-         {
 
-             var data = new Class1 { StringValue = "foo" };
 
-             var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(string));
 
-             target.OnNext("bar");
 
-             Assert.Equal("bar", data.StringValue);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public void Should_Set_Indexed_Value()
 
-         {
 
-             var data = new { Foo = new[] { "foo" } };
 
-             var target = new BindingExpression(ExpressionObserver.Create(data, o => o.Foo[0]), typeof(string));
 
-             target.OnNext("bar");
 
-             Assert.Equal("bar", data.Foo[0]);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public async Task Should_Convert_Get_String_To_Double()
 
-         {
 
-             var data = new Class1 { StringValue = $"{5.6}" };
 
-             var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(double));
 
-             var result = await target.Take(1);
 
-             Assert.Equal(5.6, result);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public async Task Getting_Invalid_Double_String_Should_Return_BindingError()
 
-         {
 
-             var data = new Class1 { StringValue = "foo" };
 
-             var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(double));
 
-             var result = await target.Take(1);
 
-             Assert.IsType<BindingNotification>(result);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public void Should_Convert_Set_String_To_Double()
 
-         {
 
-             var data = new Class1 { StringValue = $"{5.6}" };
 
-             var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(double));
 
-             target.OnNext(6.7);
 
-             Assert.Equal($"{6.7}", data.StringValue);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public async Task Should_Convert_Get_Double_To_String()
 
-         {
 
-             var data = new Class1 { DoubleValue = 5.6 };
 
-             var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue), typeof(string));
 
-             var result = await target.Take(1);
 
-             Assert.Equal($"{5.6}", result);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public void Should_Convert_Set_Double_To_String()
 
-         {
 
-             var data = new Class1 { DoubleValue = 5.6 };
 
-             var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue), typeof(string));
 
-             target.OnNext($"{6.7}");
 
-             Assert.Equal(6.7, data.DoubleValue);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public async Task Should_Return_BindingNotification_With_FallbackValue_For_NonConvertibe_Target_Value()
 
-         {
 
-             var data = new Class1 { StringValue = "foo" };
 
-             var target = new BindingExpression(
 
-                 ExpressionObserver.Create(data, o => o.StringValue),
 
-                 typeof(int),
 
-                 42,
 
-                 AvaloniaProperty.UnsetValue,
 
-                 DefaultValueConverter.Instance,
 
-                 CultureInfo.InvariantCulture);
 
-             var result = await target.Take(1);
 
-             Assert.Equal(
 
-                 new BindingNotification(
 
-                     new InvalidCastException("'foo' is not a valid number."),
 
-                     BindingErrorType.Error,
 
-                     42),
 
-                 result);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public async Task Should_Return_BindingNotification_With_FallbackValue_For_NonConvertibe_Target_Value_With_Data_Validation()
 
-         {
 
-             var data = new Class1 { StringValue = "foo" };
 
-             var target = new BindingExpression(
 
-                 ExpressionObserver.Create(data, o => o.StringValue, true),
 
-                 typeof(int),
 
-                 42,
 
-                 AvaloniaProperty.UnsetValue,
 
-                 DefaultValueConverter.Instance,
 
-                 CultureInfo.InvariantCulture);
 
-             var result = await target.Take(1);
 
-             Assert.Equal(
 
-                 new BindingNotification(
 
-                     new InvalidCastException("'foo' is not a valid number."),
 
-                     BindingErrorType.Error,
 
-                     42),
 
-                 result);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public async Task Should_Return_BindingNotification_For_Invalid_FallbackValue()
 
-         {
 
-             var data = new Class1 { StringValue = "foo" };
 
-             var target = new BindingExpression(
 
-                 ExpressionObserver.Create(data, o => o.StringValue),
 
-                 typeof(int),
 
-                 "bar",
 
-                 AvaloniaProperty.UnsetValue,
 
-                 DefaultValueConverter.Instance,
 
-                 CultureInfo.InvariantCulture);
 
-             var result = await target.Take(1);
 
-             Assert.Equal(
 
-                 new BindingNotification(
 
-                     new AggregateException(
 
-                         new InvalidCastException("'foo' is not a valid number."),
 
-                         new InvalidCastException("Could not convert FallbackValue 'bar' to 'System.Int32'")),
 
-                     BindingErrorType.Error),
 
-                 result);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public async Task Should_Return_BindingNotification_For_Invalid_FallbackValue_With_Data_Validation()
 
-         {
 
-             var data = new Class1 { StringValue = "foo" };
 
-             var target = new BindingExpression(
 
-                 ExpressionObserver.Create(data, o => o.StringValue, true),
 
-                 typeof(int),
 
-                 "bar",
 
-                 AvaloniaProperty.UnsetValue,
 
-                 DefaultValueConverter.Instance,
 
-                 CultureInfo.InvariantCulture);
 
-             var result = await target.Take(1);
 
-             Assert.Equal(
 
-                 new BindingNotification(
 
-                     new AggregateException(
 
-                         new InvalidCastException("'foo' is not a valid number."),
 
-                         new InvalidCastException("Could not convert FallbackValue 'bar' to 'System.Int32'")),
 
-                     BindingErrorType.Error),
 
-                 result);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public void Setting_Invalid_Double_String_Should_Not_Change_Target()
 
-         {
 
-             var data = new Class1 { DoubleValue = 5.6 };
 
-             var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue), typeof(string));
 
-             target.OnNext("foo");
 
-             Assert.Equal(5.6, data.DoubleValue);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public void Setting_Invalid_Double_String_Should_Use_FallbackValue()
 
-         {
 
-             var data = new Class1 { DoubleValue = 5.6 };
 
-             var target = new BindingExpression(
 
-                 ExpressionObserver.Create(data, o => o.DoubleValue),
 
-                 typeof(string),
 
-                 "9.8",
 
-                 AvaloniaProperty.UnsetValue,
 
-                 DefaultValueConverter.Instance,
 
-                 CultureInfo.InvariantCulture);
 
-             target.OnNext("foo");
 
-             Assert.Equal(9.8, data.DoubleValue);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public void Should_Coerce_Setting_UnsetValue_Double_To_Default_Value()
 
-         {
 
-             var data = new Class1 { DoubleValue = 5.6 };
 
-             var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue), typeof(string));
 
-             target.OnNext(AvaloniaProperty.UnsetValue);
 
-             Assert.Equal(0, data.DoubleValue);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public void Should_Pass_ConverterParameter_To_Convert()
 
-         {
 
-             var data = new Class1 { DoubleValue = 5.6 };
 
-             var converter = new Mock<IValueConverter>();
 
-             var target = new BindingExpression(
 
-                 ExpressionObserver.Create(data, o => o.DoubleValue),
 
-                 typeof(string),
 
-                 converter.Object,
 
-                 CultureInfo.CurrentCulture,
 
-                 converterParameter: "foo");
 
-             target.Subscribe(_ => { });
 
-             converter.Verify(x => x.Convert(5.6, typeof(string), "foo", CultureInfo.CurrentCulture));
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public void Should_Pass_ConverterParameter_To_ConvertBack()
 
-         {
 
-             var data = new Class1 { DoubleValue = 5.6 };
 
-             var converter = new Mock<IValueConverter>();
 
-             var target = new BindingExpression(
 
-                 ExpressionObserver.Create(data, o => o.DoubleValue),
 
-                 typeof(string),
 
-                 converter.Object,
 
-                 CultureInfo.CurrentCulture,
 
-                 converterParameter: "foo");
 
-             target.OnNext("bar");
 
-             converter.Verify(x => x.ConvertBack("bar", typeof(double), "foo", CultureInfo.CurrentCulture));
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public void Should_Handle_DataValidation()
 
-         {
 
-             var data = new Class1 { DoubleValue = 5.6 };
 
-             var converter = new Mock<IValueConverter>();
 
-             var target = new BindingExpression(ExpressionObserver.Create(data, o => o.DoubleValue, true), typeof(string));
 
-             var result = new List<object>();
 
-             target.Subscribe(x => result.Add(x));
 
-             target.OnNext(1.2);
 
-             target.OnNext($"{3.4}");
 
-             target.OnNext("bar");
 
-             Assert.Equal(
 
-                 new[]
 
-                 {
 
-                     new BindingNotification($"{5.6}"),
 
-                     new BindingNotification($"{1.2}"),
 
-                     new BindingNotification($"{3.4}"),
 
-                     new BindingNotification(
 
-                         new InvalidCastException("'bar' is not a valid number."),
 
-                         BindingErrorType.Error)
 
-                 },
 
-                 result);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public void Second_Subscription_Should_Fire_Immediately()
 
-         {
 
-             var data = new Class1 { StringValue = "foo" };
 
-             var target = new BindingExpression(ExpressionObserver.Create(data, o => o.StringValue), typeof(string));
 
-             object result = null;
 
-             target.Subscribe();
 
-             target.Subscribe(x => result = x);
 
-             Assert.Equal("foo", result);
 
-             GC.KeepAlive(data);
 
-         }
 
-         [Fact]
 
-         public void Null_Value_Should_Use_TargetNullValue()
 
-         {
 
-             var data = new Class1 { StringValue = "foo" };
 
-             var target = new BindingExpression(
 
-                 ExpressionObserver.Create(data, o => o.StringValue),
 
-                 typeof(string),
 
-                 AvaloniaProperty.UnsetValue,
 
-                 "bar",
 
-                 DefaultValueConverter.Instance,
 
-                 CultureInfo.InvariantCulture);
 
-             object result = null;
 
-             target.Subscribe(x => result = x);
 
-             Assert.Equal("foo", result);
 
-             
 
-             data.StringValue = null;
 
-             Assert.Equal("bar", result);
 
-             GC.KeepAlive(data);
 
-         }
 
-         private class Class1 : NotifyingBase
 
-         {
 
-             private string _stringValue;
 
-             private double _doubleValue;
 
-             public string StringValue
 
-             {
 
-                 get { return _stringValue; }
 
-                 set { _stringValue = value; RaisePropertyChanged(); }
 
-             }
 
-             public double DoubleValue
 
-             {
 
-                 get { return _doubleValue; }
 
-                 set { _doubleValue = value; RaisePropertyChanged(); }
 
-             }
 
-         }
 
-     }
 
- }
 
 
  |