// Copyright (c) The Avalonia Project. All rights reserved. // Licensed under the MIT license. See licence.md file in the project root for full license information. using System; using System.Collections.Generic; using System.Collections.ObjectModel; using System.Reactive.Linq; using System.Threading.Tasks; using Avalonia.Collections; using Avalonia.Diagnostics; using Avalonia.Data.Core; using Avalonia.UnitTests; using Xunit; namespace Avalonia.Base.UnitTests.Data.Core { public class ExpressionObserverTests_Indexer { [Fact] public async Task Should_Get_Array_Value() { var data = new { Foo = new [] { "foo", "bar" } }; var target = new ExpressionObserver(data, "Foo[1]"); var result = await target.Take(1); Assert.Equal("bar", result); GC.KeepAlive(data); } [Fact] public async Task Should_Get_UnsetValue_For_Invalid_Array_Index() { var data = new { Foo = new[] { "foo", "bar" } }; var target = new ExpressionObserver(data, "Foo[invalid]"); var result = await target.Take(1); Assert.Equal(AvaloniaProperty.UnsetValue, result); GC.KeepAlive(data); } [Fact] public async Task Should_Get_UnsetValue_For_Invalid_Dictionary_Index() { var data = new { Foo = new Dictionary { { 1, "foo" } } }; var target = new ExpressionObserver(data, "Foo[invalid]"); var result = await target.Take(1); Assert.Equal(AvaloniaProperty.UnsetValue, result); GC.KeepAlive(data); } [Fact] public async Task Should_Get_UnsetValue_For_Object_Without_Indexer() { var data = new { Foo = 5 }; var target = new ExpressionObserver(data, "Foo[noindexer]"); var result = await target.Take(1); Assert.Equal(AvaloniaProperty.UnsetValue, result); GC.KeepAlive(data); } [Fact] public async Task Should_Get_MultiDimensional_Array_Value() { var data = new { Foo = new[,] { { "foo", "bar" }, { "baz", "qux" } } }; var target = new ExpressionObserver(data, "Foo[1, 1]"); var result = await target.Take(1); Assert.Equal("qux", result); GC.KeepAlive(data); } [Fact] public async Task Should_Get_Value_For_String_Indexer() { var data = new { Foo = new Dictionary { { "foo", "bar" }, { "baz", "qux" } } }; var target = new ExpressionObserver(data, "Foo[foo]"); var result = await target.Take(1); Assert.Equal("bar", result); GC.KeepAlive(data); } [Fact] public async Task Should_Get_Value_For_Non_String_Indexer() { var data = new { Foo = new Dictionary { { 1.0, "bar" }, { 2.0, "qux" } } }; var target = new ExpressionObserver(data, "Foo[1.0]"); var result = await target.Take(1); Assert.Equal("bar", result); GC.KeepAlive(data); } [Fact] public async Task Array_Out_Of_Bounds_Should_Return_UnsetValue() { var data = new { Foo = new[] { "foo", "bar" } }; var target = new ExpressionObserver(data, "Foo[2]"); var result = await target.Take(1); Assert.Equal(AvaloniaProperty.UnsetValue, result); GC.KeepAlive(data); } [Fact] public async Task Array_With_Wrong_Dimensions_Should_Return_UnsetValue() { var data = new { Foo = new[] { "foo", "bar" } }; var target = new ExpressionObserver(data, "Foo[1,2]"); var result = await target.Take(1); Assert.Equal(AvaloniaProperty.UnsetValue, result); GC.KeepAlive(data); } [Fact] public async Task List_Out_Of_Bounds_Should_Return_UnsetValue() { var data = new { Foo = new List { "foo", "bar" } }; var target = new ExpressionObserver(data, "Foo[2]"); var result = await target.Take(1); Assert.Equal(AvaloniaProperty.UnsetValue, result); GC.KeepAlive(data); } [Fact] public async Task Should_Get_List_Value() { var data = new { Foo = new List { "foo", "bar" } }; var target = new ExpressionObserver(data, "Foo[1]"); var result = await target.Take(1); Assert.Equal("bar", result); GC.KeepAlive(data); } [Fact] public void Should_Track_INCC_Add() { var data = new { Foo = new AvaloniaList { "foo", "bar" } }; var target = new ExpressionObserver(data, "Foo[2]"); var result = new List(); using (var sub = target.Subscribe(x => result.Add(x))) { data.Foo.Add("baz"); } Assert.Equal(new[] { AvaloniaProperty.UnsetValue, "baz" }, result); Assert.Null(((INotifyCollectionChangedDebug)data.Foo).GetCollectionChangedSubscribers()); GC.KeepAlive(data); } [Fact] public void Should_Track_INCC_Remove() { var data = new { Foo = new AvaloniaList { "foo", "bar" } }; var target = new ExpressionObserver(data, "Foo[0]"); var result = new List(); using (var sub = target.Subscribe(x => result.Add(x))) { data.Foo.RemoveAt(0); } Assert.Equal(new[] { "foo", "bar" }, result); Assert.Null(((INotifyCollectionChangedDebug)data.Foo).GetCollectionChangedSubscribers()); GC.KeepAlive(data); } [Fact] public void Should_Track_INCC_Replace() { var data = new { Foo = new AvaloniaList { "foo", "bar" } }; var target = new ExpressionObserver(data, "Foo[1]"); var result = new List(); using (var sub = target.Subscribe(x => result.Add(x))) { data.Foo[1] = "baz"; } Assert.Equal(new[] { "bar", "baz" }, result); Assert.Null(((INotifyCollectionChangedDebug)data.Foo).GetCollectionChangedSubscribers()); GC.KeepAlive(data); } [Fact] public void Should_Track_INCC_Move() { // Using ObservableCollection here because AvaloniaList does not yet have a Move // method, but even if it did we need to test with ObservableCollection as well // as AvaloniaList as it implements PropertyChanged as an explicit interface event. var data = new { Foo = new ObservableCollection { "foo", "bar" } }; var target = new ExpressionObserver(data, "Foo[1]"); var result = new List(); var sub = target.Subscribe(x => result.Add(x)); data.Foo.Move(0, 1); Assert.Equal(new[] { "bar", "foo" }, result); GC.KeepAlive(sub); GC.KeepAlive(data); } [Fact] public void Should_Track_INCC_Reset() { var data = new { Foo = new AvaloniaList { "foo", "bar" } }; var target = new ExpressionObserver(data, "Foo[1]"); var result = new List(); var sub = target.Subscribe(x => result.Add(x)); data.Foo.Clear(); Assert.Equal(new[] { "bar", AvaloniaProperty.UnsetValue }, result); GC.KeepAlive(sub); GC.KeepAlive(data); } [Fact] public void Should_Track_NonIntegerIndexer() { var data = new { Foo = new NonIntegerIndexer() }; data.Foo["foo"] = "bar"; data.Foo["baz"] = "qux"; var target = new ExpressionObserver(data, "Foo[foo]"); var result = new List(); using (var sub = target.Subscribe(x => result.Add(x))) { data.Foo["foo"] = "bar2"; } var expected = new[] { "bar", "bar2" }; Assert.Equal(expected, result); Assert.Equal(0, data.Foo.PropertyChangedSubscriptionCount); GC.KeepAlive(data); } [Fact] public void Should_SetArrayIndex() { var data = new { Foo = new[] { "foo", "bar" } }; var target = new ExpressionObserver(data, "Foo[1]"); using (target.Subscribe(_ => { })) { Assert.True(target.SetValue("baz")); } Assert.Equal("baz", data.Foo[1]); GC.KeepAlive(data); } [Fact] public void Should_Set_ExistingDictionaryEntry() { var data = new { Foo = new Dictionary { {"foo", 1 } } }; var target = new ExpressionObserver(data, "Foo[foo]"); using (target.Subscribe(_ => { })) { Assert.True(target.SetValue(4)); } Assert.Equal(4, data.Foo["foo"]); GC.KeepAlive(data); } [Fact] public void Should_Add_NewDictionaryEntry() { var data = new { Foo = new Dictionary { {"foo", 1 } } }; var target = new ExpressionObserver(data, "Foo[bar]"); using (target.Subscribe(_ => { })) { Assert.True(target.SetValue(4)); } Assert.Equal(4, data.Foo["bar"]); GC.KeepAlive(data); } [Fact] public void Should_Set_NonIntegerIndexer() { var data = new { Foo = new NonIntegerIndexer() }; data.Foo["foo"] = "bar"; data.Foo["baz"] = "qux"; var target = new ExpressionObserver(data, "Foo[foo]"); using (target.Subscribe(_ => { })) { Assert.True(target.SetValue("bar2")); } Assert.Equal("bar2", data.Foo["foo"]); GC.KeepAlive(data); } [Fact] public async Task Indexer_Only_Binding_Works() { var data = new[] { 1, 2, 3 }; var target = new ExpressionObserver(data, "[1]"); var value = await target.Take(1); Assert.Equal(data[1], value); } private class NonIntegerIndexer : NotifyingBase { private readonly Dictionary _storage = new Dictionary(); public string this[string key] { get { return _storage[key]; } set { _storage[key] = value; RaisePropertyChanged(CommonPropertyNames.IndexerName); } } } } }