123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738 |
- using System;
- using System.Collections.Generic;
- using System.Reactive;
- using System.Reactive.Linq;
- using System.Reactive.Subjects;
- using System.Threading.Tasks;
- using Avalonia.Data;
- using Avalonia.Data.Core;
- using Avalonia.Threading;
- using Avalonia.UnitTests;
- using Microsoft.Reactive.Testing;
- using Xunit;
- namespace Avalonia.Base.UnitTests.Data.Core
- {
- public class ExpressionObserverTests_Property
- {
- [Fact]
- public async Task Should_Get_Simple_Property_Value()
- {
- var data = new { Foo = "foo" };
- var target = ExpressionObserver.Create(data, o => o.Foo);
- var result = await target.Take(1);
- Assert.Equal("foo", result);
- GC.KeepAlive(data);
- }
- [Fact]
- public void Should_Get_Simple_Property_Value_Type()
- {
- var data = new { Foo = "foo" };
- var target = ExpressionObserver.Create(data, o => o.Foo);
- target.Subscribe(_ => { });
- Assert.Equal(typeof(string), target.ResultType);
- GC.KeepAlive(data);
- }
- [Fact]
- public async Task Should_Get_Simple_Property_Value_Null()
- {
- var data = new { Foo = (string)null };
- var target = ExpressionObserver.Create(data, o => o.Foo);
- var result = await target.Take(1);
- Assert.Null(result);
- GC.KeepAlive(data);
- }
- [Fact]
- public async Task Should_Get_Simple_Property_From_Base_Class()
- {
- var data = new Class3 { Foo = "foo" };
- var target = ExpressionObserver.Create(data, o => o.Foo);
- var result = await target.Take(1);
- Assert.Equal("foo", result);
- GC.KeepAlive(data);
- }
- [Fact]
- public async Task Should_Return_BindingNotification_Error_For_Root_Null()
- {
- var target = ExpressionObserver.Create(default(Class3), o => o.Foo);
- var result = await target.Take(1);
- Assert.Equal(
- new BindingNotification(
- new MarkupBindingChainException("Null value", "o => o.Foo", string.Empty),
- BindingErrorType.Error,
- AvaloniaProperty.UnsetValue),
- result);
- }
- [Fact]
- public async Task Should_Return_BindingNotification_Error_For_Root_UnsetValue()
- {
- var target = ExpressionObserver.Create(AvaloniaProperty.UnsetValue, o => (o as Class3).Foo);
- var result = await target.Take(1);
- Assert.Equal(
- new BindingNotification(
- new MarkupBindingChainException("Null value", "o => (o As Class3).Foo", string.Empty),
- BindingErrorType.Error,
- AvaloniaProperty.UnsetValue),
- result);
- }
- [Fact]
- public async Task Should_Return_BindingNotification_Error_For_Observable_Root_Null()
- {
- var target = ExpressionObserver.Create(Observable.Return(default(Class3)), o => o.Foo);
- var result = await target.Take(1);
- Assert.Equal(
- new BindingNotification(
- new MarkupBindingChainException("Null value", "o => o.Foo", string.Empty),
- BindingErrorType.Error,
- AvaloniaProperty.UnsetValue),
- result);
- }
- [Fact]
- public async void Should_Return_BindingNotification_Error_For_Observable_Root_UnsetValue()
- {
- var target = ExpressionObserver.Create<object, string>(Observable.Return(AvaloniaProperty.UnsetValue), o => (o as Class3).Foo);
- var result = await target.Take(1);
- Assert.Equal(
- new BindingNotification(
- new MarkupBindingChainException("Null value", "o => (o As Class3).Foo", string.Empty),
- BindingErrorType.Error,
- AvaloniaProperty.UnsetValue),
- result);
-
- }
- [Fact]
- public async Task Should_Get_Simple_Property_Chain()
- {
- var data = new { Foo = new { Bar = new { Baz = "baz" } } };
- var target = ExpressionObserver.Create(data, o => o.Foo.Bar.Baz);
- var result = await target.Take(1);
- Assert.Equal("baz", result);
- GC.KeepAlive(data);
- }
- [Fact]
- public void Should_Get_Simple_Property_Chain_Type()
- {
- var data = new { Foo = new { Bar = new { Baz = "baz" } } };
- var target = ExpressionObserver.Create(data, o => o.Foo.Bar.Baz);
- target.Subscribe(_ => { });
- Assert.Equal(typeof(string), target.ResultType);
- GC.KeepAlive(data);
- }
- [Fact]
- public void Should_Return_BindingNotification_Error_For_Chain_With_Null_Value()
- {
- var data = new { Foo = default(Class1) };
- var target = ExpressionObserver.Create(data, o => o.Foo.Foo.Length);
- var result = new List<object>();
- target.Subscribe(x => result.Add(x));
- Assert.Equal(
- new[]
- {
- new BindingNotification(
- new MarkupBindingChainException("Null value", "o => o.Foo.Foo.Length", "Foo"),
- BindingErrorType.Error,
- AvaloniaProperty.UnsetValue),
- },
- result);
- GC.KeepAlive(data);
- }
- [Fact]
- public void Should_Track_Simple_Property_Value()
- {
- var data = new Class1 { Foo = "foo" };
- var target = ExpressionObserver.Create(data, o => o.Foo);
- var result = new List<object>();
- var sub = target.Subscribe(x => result.Add(x));
- data.Foo = "bar";
- Assert.Equal(new[] { "foo", "bar" }, result);
- sub.Dispose();
- // Forces WeakEvent compact
- Dispatcher.UIThread.RunJobs();
-
- Assert.Equal(0, data.PropertyChangedSubscriptionCount);
- GC.KeepAlive(data);
- }
- [Fact]
- public void Should_Trigger_PropertyChanged_On_Null_Or_Empty_String()
- {
- var data = new Class1 { Bar = "foo" };
- var target = ExpressionObserver.Create(data, o => o.Bar);
- var result = new List<object>();
- var sub = target.Subscribe(x => result.Add(x));
- Assert.Equal(new[] { "foo" }, result);
- data.Bar = "bar";
- Assert.Equal(new[] { "foo" }, result);
- data.RaisePropertyChanged(string.Empty);
- Assert.Equal(new[] { "foo", "bar" }, result);
- data.RaisePropertyChanged(null);
- Assert.Equal(new[] { "foo", "bar", "bar" }, result);
-
- sub.Dispose();
-
- // Forces WeakEvent compact
- Dispatcher.UIThread.RunJobs();
- Assert.Equal(0, data.PropertyChangedSubscriptionCount);
- GC.KeepAlive(data);
- }
- [Fact]
- public void Should_Track_End_Of_Property_Chain_Changing()
- {
- var data = new Class1 { Next = new Class2 { Bar = "bar" } };
- var target = ExpressionObserver.Create(data, o => (o.Next as Class2).Bar);
- var result = new List<object>();
- var sub = target.Subscribe(x => result.Add(x));
- ((Class2)data.Next).Bar = "baz";
- ((Class2)data.Next).Bar = null;
- Assert.Equal(new[] { "bar", "baz", null }, result);
- sub.Dispose();
- // Forces WeakEvent compact
- Dispatcher.UIThread.RunJobs();
-
- Assert.Equal(0, data.PropertyChangedSubscriptionCount);
- Assert.Equal(0, data.Next.PropertyChangedSubscriptionCount);
- GC.KeepAlive(data);
- }
- [Fact]
- public void Should_Track_Property_Chain_Changing()
- {
- var data = new Class1 { Next = new Class2 { Bar = "bar" } };
- var target = ExpressionObserver.Create(data, o => (o.Next as Class2).Bar);
- var result = new List<object>();
- var sub = target.Subscribe(x => result.Add(x));
- var old = data.Next;
- data.Next = new Class2 { Bar = "baz" };
- data.Next = new Class2 { Bar = null };
- Assert.Equal(new[] { "bar", "baz", null }, result);
- sub.Dispose();
-
- // Forces WeakEvent compact
- Dispatcher.UIThread.RunJobs();
- Assert.Equal(0, data.PropertyChangedSubscriptionCount);
- Assert.Equal(0, data.Next.PropertyChangedSubscriptionCount);
- Assert.Equal(0, old.PropertyChangedSubscriptionCount);
- GC.KeepAlive(data);
- }
- [Fact]
- public void Should_Track_Property_Chain_Breaking_With_Null_Then_Mending()
- {
- var data = new Class1
- {
- Next = new Class2
- {
- Next = new Class2
- {
- Bar = "bar"
- }
- }
- };
- var target = ExpressionObserver.Create(data, o => ((o.Next as Class2).Next as Class2).Bar);
- var result = new List<object>();
- var sub = target.Subscribe(x => result.Add(x));
- var old = data.Next;
- data.Next = new Class2 { Bar = "baz" };
- data.Next = old;
- Assert.Equal(
- new object[]
- {
- "bar",
- new BindingNotification(
- new MarkupBindingChainException("Null value", "o => ((o.Next As Class2).Next As Class2).Bar", "Next.Next"),
- BindingErrorType.Error,
- AvaloniaProperty.UnsetValue),
- "bar"
- },
- result);
- sub.Dispose();
- // Forces WeakEvent compact
- Dispatcher.UIThread.RunJobs();
-
- Assert.Equal(0, data.PropertyChangedSubscriptionCount);
- Assert.Equal(0, data.Next.PropertyChangedSubscriptionCount);
- Assert.Equal(0, old.PropertyChangedSubscriptionCount);
- GC.KeepAlive(data);
- }
- [Fact]
- public void Should_Track_Property_Chain_Breaking_With_Missing_Member_Then_Mending()
- {
- var data = new Class1 { Next = new Class2 { Bar = "bar" } };
- var target = ExpressionObserver.Create(data, o => (o.Next as Class2).Bar);
- var result = new List<object>();
- var sub = target.Subscribe(x => result.Add(x));
- var old = data.Next;
- var breaking = new WithoutBar();
- data.Next = breaking;
- data.Next = new Class2 { Bar = "baz" };
- Assert.Equal(
- new object[]
- {
- "bar",
- new BindingNotification(
- new MissingMemberException("Could not find a matching property accessor for 'Bar' on 'Avalonia.Base.UnitTests.Data.Core.ExpressionObserverTests_Property+WithoutBar'"),
- BindingErrorType.Error),
- "baz",
- },
- result);
- sub.Dispose();
-
- // Forces WeakEvent compact
- Dispatcher.UIThread.RunJobs();
- Assert.Equal(0, data.PropertyChangedSubscriptionCount);
- Assert.Equal(0, data.Next.PropertyChangedSubscriptionCount);
- Assert.Equal(0, breaking.PropertyChangedSubscriptionCount);
- Assert.Equal(0, old.PropertyChangedSubscriptionCount);
- GC.KeepAlive(data);
- }
- [Fact]
- public void Empty_Expression_Should_Track_Root()
- {
- var data = new Class1 { Foo = "foo" };
- var update = new Subject<ValueTuple>();
- var target = ExpressionObserver.Create(() => data.Foo, o => o, update);
- var result = new List<object>();
- target.Subscribe(x => result.Add(x));
- data.Foo = "bar";
- update.OnNext(default);
- Assert.Equal(new[] { "foo", "bar" }, result);
- GC.KeepAlive(data);
- }
- [Fact]
- public void Should_Track_Property_Value_From_Observable_Root()
- {
- var scheduler = new TestScheduler();
- var source = scheduler.CreateColdObservable(
- OnNext(1, new Class1 { Foo = "foo" }),
- OnNext(2, new Class1 { Foo = "bar" }));
- var target = ExpressionObserver.Create(source, o => o.Foo);
- var result = new List<object>();
- using (target.Subscribe(x => result.Add(x)))
- {
- scheduler.Start();
- }
- Assert.Equal(new[] { "foo", "bar" }, result);
- Assert.All(source.Subscriptions, x => Assert.NotEqual(Subscription.Infinite, x.Unsubscribe));
- }
- [Fact]
- public void Subscribing_Multiple_Times_Should_Return_Values_To_All()
- {
- var data = new Class1 { Foo = "foo" };
- var target = ExpressionObserver.Create(data, o => o.Foo);
- var result1 = new List<object>();
- var result2 = new List<object>();
- var result3 = new List<object>();
- target.Subscribe(x => result1.Add(x));
- target.Subscribe(x => result2.Add(x));
- data.Foo = "bar";
- target.Subscribe(x => result3.Add(x));
- Assert.Equal(new[] { "foo", "bar" }, result1);
- Assert.Equal(new[] { "foo", "bar" }, result2);
- Assert.Equal(new[] { "bar" }, result3);
- GC.KeepAlive(data);
- }
- [Fact]
- public void Subscribing_Multiple_Times_Should_Only_Add_PropertyChanged_Handlers_Once()
- {
- var data = new Class1 { Foo = "foo" };
- var target = ExpressionObserver.Create(data, o => o.Foo);
- var sub1 = target.Subscribe(x => { });
- var sub2 = target.Subscribe(x => { });
- Assert.Equal(1, data.PropertyChangedSubscriptionCount);
- sub1.Dispose();
- sub2.Dispose();
- // Forces WeakEvent compact
- Dispatcher.UIThread.RunJobs();
-
- Assert.Equal(0, data.PropertyChangedSubscriptionCount);
- GC.KeepAlive(data);
- }
- [Fact]
- public void SetValue_Should_Set_Simple_Property_Value()
- {
- var data = new Class1 { Foo = "foo" };
- var target = ExpressionObserver.Create(data, o => o.Foo);
- using (target.Subscribe(_ => { }))
- {
- Assert.True(target.SetValue("bar"));
- }
- Assert.Equal("bar", data.Foo);
- GC.KeepAlive(data);
- }
- [Fact]
- public void SetValue_Should_Set_Property_At_The_End_Of_Chain()
- {
- var data = new Class1 { Next = new Class2 { Bar = "bar" } };
- var target = ExpressionObserver.Create(data, o => (o.Next as Class2).Bar);
- using (target.Subscribe(_ => { }))
- {
- Assert.True(target.SetValue("baz"));
- }
- Assert.Equal("baz", ((Class2)data.Next).Bar);
- GC.KeepAlive(data);
- }
- [Fact]
- public void SetValue_Should_Return_False_For_Missing_Property()
- {
- var data = new Class1 { Next = new WithoutBar() };
- var target = ExpressionObserver.Create(data, o => (o.Next as Class2).Bar);
- using (target.Subscribe(_ => { }))
- {
- Assert.False(target.SetValue("baz"));
- }
- GC.KeepAlive(data);
- }
- [Fact]
- public void SetValue_Should_Notify_New_Value_With_Inpc()
- {
- var data = new Class1();
- var target = ExpressionObserver.Create(data, o => o.Foo);
- var result = new List<object>();
- target.Subscribe(x => result.Add(x));
- target.SetValue("bar");
- Assert.Equal(new[] { null, "bar" }, result);
- GC.KeepAlive(data);
- }
- [Fact]
- public void SetValue_Should_Notify_New_Value_Without_Inpc()
- {
- var data = new Class1();
- var target = ExpressionObserver.Create(data, o => o.Bar);
- var result = new List<object>();
- target.Subscribe(x => result.Add(x));
- target.SetValue("bar");
- Assert.Equal(new[] { null, "bar" }, result);
- GC.KeepAlive(data);
- }
- [Fact]
- public void SetValue_Should_Return_False_For_Missing_Object()
- {
- var data = new Class1();
- var target = ExpressionObserver.Create(data, o => (o.Next as Class2).Bar);
- using (target.Subscribe(_ => { }))
- {
- Assert.False(target.SetValue("baz"));
- }
- GC.KeepAlive(data);
- }
- [Fact]
- public void Can_Replace_Root()
- {
- var first = new Class1 { Foo = "foo" };
- var second = new Class1 { Foo = "bar" };
- var root = first;
- var update = new Subject<ValueTuple>();
- var target = ExpressionObserver.Create(() => root, o => o.Foo, update);
- var result = new List<object>();
- var sub = target.Subscribe(x => result.Add(x));
- root = second;
- update.OnNext(default);
- root = null;
- update.OnNext(default);
- Assert.Equal(
- new object[]
- {
- "foo",
- "bar",
- new BindingNotification(
- new MarkupBindingChainException("Null value", "o => o.Foo", string.Empty),
- BindingErrorType.Error,
- AvaloniaProperty.UnsetValue)
- },
- result);
- // Forces WeakEvent compact
- Dispatcher.UIThread.RunJobs();
- Assert.Equal(0, first.PropertyChangedSubscriptionCount);
- Assert.Equal(0, second.PropertyChangedSubscriptionCount);
- GC.KeepAlive(first);
- GC.KeepAlive(second);
- }
- [Fact]
- public void Should_Not_Keep_Source_Alive()
- {
- Func<Tuple<ExpressionObserver, WeakReference>> run = () =>
- {
- var source = new Class1 { Foo = "foo" };
- var target = ExpressionObserver.Create(source, o => o.Foo);
- return Tuple.Create(target, new WeakReference(source));
- };
- var result = run();
- result.Item1.Subscribe(x => { });
- // Mono trickery
- GC.Collect(2);
- GC.WaitForPendingFinalizers();
- GC.WaitForPendingFinalizers();
- GC.Collect(2);
-
- Assert.Null(result.Item2.Target);
- }
- [Fact]
- public void Should_Not_Throw_Exception_On_Unsubscribe_When_Already_Unsubscribed()
- {
- var source = new Class1 { Foo = "foo" };
- var target = new PropertyAccessorNode("Foo", false);
- Assert.NotNull(target);
- target.Target = new WeakReference<object>(source);
- target.Subscribe(_ => { });
- target.Unsubscribe();
- target.Unsubscribe();
- Assert.True(true);
- }
- [Fact]
- public void Should_Not_Throw_Exception_When_Enabling_Data_Validation_On_Missing_Member()
- {
- var source = new Class1();
- var target = new PropertyAccessorNode("NotFound", true);
- target.Target = new WeakReference<object>(source);
- var result = new List<object>();
- target.Subscribe(x => result.Add(x));
- Assert.Equal(
- new object[]
- {
- new BindingNotification(
- new MissingMemberException("Could not find a matching property accessor for 'NotFound' on 'Avalonia.Base.UnitTests.Data.Core.ExpressionObserverTests_Property+Class1'"),
- BindingErrorType.Error),
- },
- result);
- }
- [Fact]
- public void Should_Not_Throw_Exception_On_Duplicate_Properties()
- {
- // Repro of https://github.com/AvaloniaUI/Avalonia/issues/4733.
- var source = new MyViewModel();
- var target = new PropertyAccessorNode("Name", false);
-
- target.Target = new WeakReference<object>(source);
-
- var result = new List<object>();
-
- target.Subscribe(x => result.Add(x));
- }
- [Fact]
- public void RootGetter_Is_Reevaluated_On_Subscribe()
- {
- var data = "foo";
- var target = new ExpressionObserver(() => data, new EmptyExpressionNode(), new Subject<ValueTuple>(), null);
- var result = new List<object>();
- var sub = target.Subscribe(x => result.Add(x));
- Assert.Equal(new object[] { "foo" }, result);
- sub.Dispose();
- data = "bar";
- target.Subscribe(x => result.Add(x));
- Assert.Equal(new object[] { "foo", "bar" }, result);
- }
- public class MyViewModelBase { public object Name => "Name"; }
-
- public class MyViewModel : MyViewModelBase { public new string Name => "NewName"; }
- private interface INext
- {
- int PropertyChangedSubscriptionCount { get; }
- }
- private class Class1 : NotifyingBase
- {
- private string _foo;
- private INext _next;
- public string Foo
- {
- get { return _foo; }
- set
- {
- _foo = value;
- RaisePropertyChanged(nameof(Foo));
- }
- }
- private string _bar;
- public string Bar
- {
- get { return _bar; }
- set { _bar = value; }
- }
- public INext Next
- {
- get { return _next; }
- set
- {
- _next = value;
- RaisePropertyChanged(nameof(Next));
- }
- }
- }
- private class Class2 : NotifyingBase, INext
- {
- private string _bar;
- private INext _next;
- public string Bar
- {
- get { return _bar; }
- set
- {
- _bar = value;
- RaisePropertyChanged(nameof(Bar));
- }
- }
- public INext Next
- {
- get { return _next; }
- set
- {
- _next = value;
- RaisePropertyChanged(nameof(Next));
- }
- }
- }
- private class Class3 : Class1
- {
- }
- private class WithoutBar : NotifyingBase, INext
- {
- }
- private static Recorded<Notification<T>> OnNext<T>(long time, T value)
- {
- return new Recorded<Notification<T>>(time, Notification.CreateOnNext<T>(value));
- }
- }
- }
|