using System; using System.Collections; using System.Collections.Generic; using System.Collections.ObjectModel; using System.Linq; using Avalonia.Collections; using Avalonia.Controls.Presenters; using Avalonia.Controls.Primitives; using Avalonia.Controls.Selection; using Avalonia.Controls.Templates; using Avalonia.Data; using Avalonia.Input; using Avalonia.Input.Platform; using Avalonia.Interactivity; using Avalonia.UnitTests; using Moq; using Xunit; namespace Avalonia.Controls.UnitTests.Primitives { public class SelectingItemsControlTests_Multiple { private MouseTestHelper _helper = new MouseTestHelper(); [Fact] public void Setting_SelectedIndex_Should_Add_To_SelectedItems() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar" }, Template = Template(), }; target.ApplyTemplate(); target.SelectedIndex = 1; Assert.Equal(new[] { "bar" }, target.SelectedItems.Cast().ToList()); } [Fact] public void Adding_SelectedItems_Should_Set_SelectedIndex() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar" }, Template = Template(), }; target.ApplyTemplate(); target.SelectedItems.Add("bar"); Assert.Equal(1, target.SelectedIndex); } [Fact] public void Assigning_Single_SelectedItems_Should_Set_SelectedIndex() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar" }, Template = Template(), }; target.ApplyTemplate(); ((Control)target.Presenter).ApplyTemplate(); target.SelectedItems = new AvaloniaList("bar"); Assert.Equal(1, target.SelectedIndex); Assert.Equal(new[] { "bar" }, target.SelectedItems); Assert.Equal(new[] { 1 }, SelectedContainers(target)); } [Fact] public void Assigning_Multiple_SelectedItems_Should_Set_SelectedIndex() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar", "baz" }, Template = Template(), }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); target.SelectedItems = new AvaloniaList("foo", "bar", "baz"); Assert.Equal(0, target.SelectedIndex); Assert.Equal(new[] { "foo", "bar", "baz" }, target.SelectedItems); Assert.Equal(new[] { 0, 1, 2 }, SelectedContainers(target)); } [Fact] public void Selected_Items_Should_Be_Marked_When_Panel_Created_After_SelectedItems_Is_Set() { // Issue #2565. var target = new TestSelector { ItemsSource = new[] { "foo", "bar", "baz" }, Template = Template(), }; target.ApplyTemplate(); target.SelectedItems = new AvaloniaList("foo", "bar", "baz"); target.Presenter.ApplyTemplate(); Assert.Equal(0, target.SelectedIndex); Assert.Equal(new[] { "foo", "bar", "baz" }, target.SelectedItems); Assert.Equal(new[] { 0, 1, 2 }, SelectedContainers(target)); } [Fact] public void Reassigning_SelectedItems_Should_Clear_Selection() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar" }, Template = Template(), }; target.ApplyTemplate(); target.SelectedItems.Add("bar"); target.SelectedItems = new AvaloniaList(); Assert.Equal(-1, target.SelectedIndex); Assert.Null(target.SelectedItem); } [Fact] public void Adding_First_SelectedItem_Should_Raise_SelectedIndex_SelectedItem_Changed() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar" }, Template = Template(), }; bool indexRaised = false; bool itemRaised = false; target.PropertyChanged += (s, e) => { indexRaised |= e.Property.Name == "SelectedIndex" && (int)e.OldValue == -1 && (int)e.NewValue == 1; itemRaised |= e.Property.Name == "SelectedItem" && (string)e.OldValue == null && (string)e.NewValue == "bar"; }; target.ApplyTemplate(); target.SelectedItems.Add("bar"); Assert.True(indexRaised); Assert.True(itemRaised); } [Fact] public void Adding_Subsequent_SelectedItems_Should_Not_Raise_SelectedIndex_SelectedItem_Changed() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar" }, Template = Template(), }; target.ApplyTemplate(); target.SelectedItems.Add("foo"); bool raised = false; target.PropertyChanged += (s, e) => raised |= e.Property.Name == "SelectedIndex" || e.Property.Name == "SelectedItem"; target.SelectedItems.Add("bar"); Assert.False(raised); } [Fact] public void Removing_Last_SelectedItem_Should_Raise_SelectedIndex_Changed() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar" }, Template = Template(), }; target.ApplyTemplate(); target.SelectedItems.Add("foo"); bool raised = false; target.PropertyChanged += (s, e) => raised |= e.Property.Name == "SelectedIndex" && (int)e.OldValue == 0 && (int)e.NewValue == -1; target.SelectedItems.RemoveAt(0); Assert.True(raised); } [Fact] public void Adding_SelectedItems_Should_Set_Item_IsSelected() { var target = new TestSelector { Items = { new ListBoxItem(), new ListBoxItem(), new ListBoxItem(), }, Template = Template(), }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); target.SelectedItems.Add(target.Items[0]); target.SelectedItems.Add(target.Items[1]); var items = target.Items.Cast().ToList(); Assert.True(items[0].IsSelected); Assert.True(items[1].IsSelected); Assert.False(items[2].IsSelected); } [Fact] public void Assigning_SelectedItems_Should_Set_Item_IsSelected() { var target = new TestSelector { Items = { new ListBoxItem(), new ListBoxItem(), new ListBoxItem(), }, Template = Template(), }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); target.SelectedItems = new AvaloniaList { target.Items[0], target.Items[1] }; var items = target.Items.Cast().ToList(); Assert.True(items[0].IsSelected); Assert.True(items[1].IsSelected); Assert.False(items[2].IsSelected); } [Fact] public void Removing_SelectedItems_Should_Clear_Item_IsSelected() { var target = new TestSelector { Items = { new ListBoxItem(), new ListBoxItem(), new ListBoxItem(), }, Template = Template(), }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); target.SelectedItems.Add(target.Items[0]); target.SelectedItems.Add(target.Items[1]); target.SelectedItems.Remove(target.Items[1]); var items = target.Items.Cast().ToList(); Assert.True(items[0].IsSelected); Assert.False(items[1].IsSelected); } [Fact] public void Reassigning_SelectedItems_Should_Clear_Item_IsSelected() { var target = new TestSelector { Items = { new ListBoxItem(), new ListBoxItem(), new ListBoxItem(), }, Template = Template(), }; target.ApplyTemplate(); target.SelectedItems.Add(target.Items[0]); target.SelectedItems.Add(target.Items[1]); target.SelectedItems = new AvaloniaList { target.Items[0], target.Items[1] }; var items = target.Items.Cast().ToList(); Assert.False(items[0].IsSelected); Assert.False(items[1].IsSelected); } [Fact] public void Setting_SelectedIndex_Should_Unmark_Previously_Selected_Containers() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar", "baz" }, Template = Template(), }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); target.SelectedItems.Add("foo"); target.SelectedItems.Add("bar"); Assert.Equal(new[] { 0, 1 }, SelectedContainers(target)); target.SelectedIndex = 2; Assert.Equal(new[] { 2 }, SelectedContainers(target)); } [Fact] public void Range_Select_Should_Select_Range() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar", "baz", "qux", "qiz", "lol", }, Template = Template(), }; target.ApplyTemplate(); target.SelectedIndex = 1; target.SelectRange(3); Assert.Equal(new[] { "bar", "baz", "qux" }, target.SelectedItems.Cast().ToList()); } [Fact] public void Range_Select_Backwards_Should_Select_Range() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar", "baz", "qux", "qiz", "lol", }, SelectionMode = SelectionMode.Multiple, Template = Template(), }; target.ApplyTemplate(); target.SelectedIndex = 3; target.SelectRange(1); Assert.Equal(new[] { "qux", "bar", "baz" }, target.SelectedItems.Cast().ToList()); } [Fact] public void Second_Range_Select_Backwards_Should_Select_From_Original_Selection() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar", "baz", "qux", "qiz", "lol", }, SelectionMode = SelectionMode.Multiple, Template = Template(), }; target.ApplyTemplate(); target.SelectedIndex = 2; target.SelectRange(5); target.SelectRange(4); Assert.Equal(new[] { "baz", "qux", "qiz" }, target.SelectedItems.Cast().ToList()); } [Fact] public void Setting_SelectedIndex_After_Range_Should_Unmark_Previously_Selected_Containers() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar", "baz", "qux" }, Template = Template(), SelectedIndex = 0, SelectionMode = SelectionMode.Multiple, }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); target.SelectRange(2); Assert.Equal(new[] { 0, 1, 2 }, SelectedContainers(target)); target.SelectedIndex = 3; Assert.Equal(new[] { 3 }, SelectedContainers(target)); } [Fact] public void Toggling_Selection_After_Range_Should_Work() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar", "baz", "foo", "bar", "baz" }, Template = Template(), SelectedIndex = 0, SelectionMode = SelectionMode.Multiple, }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); target.SelectRange(3); Assert.Equal(new[] { 0, 1, 2, 3 }, SelectedContainers(target)); target.Toggle(4); Assert.Equal(new[] { 0, 1, 2, 3, 4 }, SelectedContainers(target)); } [Fact] public void Suprious_SelectedIndex_Changes_Should_Not_Be_Triggered() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar", "baz" }, Template = Template(), }; target.ApplyTemplate(); var selectedIndexes = new List(); target.GetObservable(TestSelector.SelectedIndexProperty).Subscribe(x => selectedIndexes.Add(x)); target.SelectedItems = new AvaloniaList { "bar", "baz" }; target.SelectedItem = "foo"; Assert.Equal(0, target.SelectedIndex); Assert.Equal(new[] { -1, 1, 0 }, selectedIndexes); } [Fact] public void Can_Set_SelectedIndex_To_Another_Selected_Item() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar", "baz" }, Template = Template(), }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); target.SelectedItems.Add("foo"); target.SelectedItems.Add("bar"); Assert.Equal(0, target.SelectedIndex); Assert.Equal(new[] { "foo", "bar" }, target.SelectedItems); Assert.Equal(new[] { 0, 1 }, SelectedContainers(target)); var raised = false; target.SelectionChanged += (s, e) => { raised = true; Assert.Empty(e.AddedItems); Assert.Equal(new[] { "foo" }, e.RemovedItems); }; target.SelectedIndex = 1; Assert.True(raised); Assert.Equal(1, target.SelectedIndex); Assert.Equal(new[] { "bar" }, target.SelectedItems); Assert.Equal(new[] { 1 }, SelectedContainers(target)); } /// /// Tests a problem discovered with ListBox with selection. /// /// /// - Items is bound to DataContext first, followed by say SelectedIndex /// - When the ListBox is removed from the visual tree, DataContext becomes null (as it's /// inherited) /// - This changes Items to null, which changes SelectedIndex to null as there are no /// longer any items /// - However, the news that DataContext is now null hasn't yet reached the SelectedItems /// binding and so the unselection is sent back to the ViewModel /// /// This is a similar problem to that tested by XamlBindingTest.Should_Not_Write_To_Old_DataContext. /// However, that tests a general property binding problem: here we are writing directly /// to the SelectedItems collection - not via a binding - so it's something that the /// binding system cannot solve. Instead we solve it by not clearing SelectedItems when /// DataContext is in the process of changing. /// [Fact] public void Should_Not_Write_SelectedItems_To_Old_DataContext() { var vm = new OldDataContextViewModel(); var target = new TestSelector(); var itemsBinding = new Binding { Path = "Items", Mode = BindingMode.OneWay, }; var selectedItemsBinding = new Binding { Path = "SelectedItems", Mode = BindingMode.OneWay, }; // Bind ItemsSource and SelectedItems to the VM. target.Bind(TestSelector.ItemsSourceProperty, itemsBinding); target.Bind(TestSelector.SelectedItemsProperty, selectedItemsBinding); // Set DataContext and SelectedIndex target.DataContext = vm; target.SelectedIndex = 1; // Make sure SelectedItems are written back to VM. Assert.Equal(new[] { "bar" }, vm.SelectedItems); // Clear DataContext and ensure that SelectedItems is still set in the VM. target.DataContext = null; Assert.Equal(new[] { "bar" }, vm.SelectedItems); // Ensure target's SelectedItems is now clear. Assert.Empty(target.SelectedItems); } /// /// See . /// [Fact] public void Should_Not_Write_SelectionModel_To_Old_DataContext() { var vm = new OldDataContextViewModel(); var target = new TestSelector(); var itemsBinding = new Binding { Path = "Items", Mode = BindingMode.OneWay, }; var selectionBinding = new Binding { Path = "Selection", Mode = BindingMode.OneWay, }; // Bind ItemsSource and Selection to the VM. target.Bind(TestSelector.ItemsSourceProperty, itemsBinding); target.Bind(TestSelector.SelectionProperty, selectionBinding); // Set DataContext and SelectedIndex target.DataContext = vm; target.SelectedIndex = 1; // Make sure selection is written to selection model Assert.Equal(1, vm.Selection.SelectedIndex); // Clear DataContext and ensure that selection is still set in model. target.DataContext = null; Assert.Equal(1, vm.Selection.SelectedIndex); // Ensure target's SelectedItems is now clear. Assert.Empty(target.SelectedItems); } [Fact] public void Unbound_SelectedItems_Should_Be_Cleared_When_DataContext_Cleared() { var data = new { Items = new[] { "foo", "bar", "baz" }, }; var target = new TestSelector { DataContext = data, Template = Template(), }; var itemsBinding = new Binding { Path = "Items" }; target.Bind(TestSelector.ItemsSourceProperty, itemsBinding); Assert.Same(data.Items, target.ItemsSource); target.SelectedItems.Add("bar"); target.DataContext = null; Assert.Empty(target.SelectedItems); } [Fact] public void Adding_To_SelectedItems_Should_Raise_SelectionChanged() { var items = new[] { "foo", "bar", "baz" }; var target = new TestSelector { DataContext = items, Template = Template(), ItemsSource = items, }; var called = false; target.SelectionChanged += (s, e) => { Assert.Equal(new[] { "bar" }, e.AddedItems.Cast().ToList()); Assert.Empty(e.RemovedItems); called = true; }; target.SelectedItems.Add("bar"); Assert.True(called); } [Fact] public void Removing_From_SelectedItems_Should_Raise_SelectionChanged() { var items = new[] { "foo", "bar", "baz" }; var target = new TestSelector { ItemsSource = items, Template = Template(), SelectedItem = "bar", }; var called = false; target.SelectionChanged += (s, e) => { Assert.Equal(new[] { "bar" }, e.RemovedItems.Cast().ToList()); Assert.Empty(e.AddedItems); called = true; }; target.SelectedItems.Remove("bar"); Assert.True(called); } [Fact] public void Assigning_SelectedItems_Should_Raise_SelectionChanged() { var items = new[] { "foo", "bar", "baz" }; var target = new TestSelector { ItemsSource = items, Template = Template(), SelectedItem = "bar", }; var called = false; target.SelectionChanged += (s, e) => { Assert.Equal(new[] { "foo", "baz" }, e.AddedItems.Cast()); Assert.Equal(new[] { "bar" }, e.RemovedItems.Cast()); called = true; }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); target.SelectedItems = new AvaloniaList("foo", "baz"); Assert.True(called); } [Fact] public void Shift_Selecting_From_No_Selection_Selects_From_Start() { using (UnitTestApplication.Start()) { var target = new ListBox { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz" }, SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); AvaloniaLocator.CurrentMutable.Bind().ToConstant(new Mock().Object); _helper.Click((Interactive)target.Presenter.Panel.Children[2], modifiers: KeyModifiers.Shift); Assert.Equal(new[] { "Foo", "Bar", "Baz" }, target.SelectedItems); Assert.Equal(new[] { 0, 1, 2 }, SelectedContainers(target)); } } [Fact] public void Ctrl_Selecting_Raises_SelectionChanged_Events() { using (UnitTestApplication.Start()) { var target = new ListBox { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz", "Qux" }, SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); AvaloniaLocator.CurrentMutable.Bind().ToConstant(new Mock().Object); SelectionChangedEventArgs receivedArgs = null; target.SelectionChanged += (_, args) => receivedArgs = args; void VerifyAdded(string selection) { Assert.NotNull(receivedArgs); Assert.Equal(new[] { selection }, receivedArgs.AddedItems); Assert.Empty(receivedArgs.RemovedItems); } void VerifyRemoved(string selection) { Assert.NotNull(receivedArgs); Assert.Equal(new[] { selection }, receivedArgs.RemovedItems); Assert.Empty(receivedArgs.AddedItems); } _helper.Click((Interactive)target.Presenter.Panel.Children[1]); VerifyAdded("Bar"); receivedArgs = null; _helper.Click((Interactive)target.Presenter.Panel.Children[2], modifiers: KeyModifiers.Control); VerifyAdded("Baz"); receivedArgs = null; _helper.Click((Interactive)target.Presenter.Panel.Children[3], modifiers: KeyModifiers.Control); VerifyAdded("Qux"); receivedArgs = null; _helper.Click((Interactive)target.Presenter.Panel.Children[1], modifiers: KeyModifiers.Control); VerifyRemoved("Bar"); } } [Fact] public void Ctrl_Selecting_SelectedItem_With_Multiple_Selection_Active_Sets_SelectedItem_To_Next_Selection() { using (UnitTestApplication.Start()) { var target = new ListBox { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz", "Qux" }, SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); AvaloniaLocator.CurrentMutable.Bind().ToConstant(new Mock().Object); _helper.Click((Interactive)target.Presenter.Panel.Children[1]); _helper.Click((Interactive)target.Presenter.Panel.Children[2], modifiers: KeyModifiers.Control); _helper.Click((Interactive)target.Presenter.Panel.Children[3], modifiers: KeyModifiers.Control); Assert.Equal(1, target.SelectedIndex); Assert.Equal("Bar", target.SelectedItem); Assert.Equal(new[] { "Bar", "Baz", "Qux" }, target.SelectedItems); _helper.Click((Interactive)target.Presenter.Panel.Children[1], modifiers: KeyModifiers.Control); Assert.Equal(2, target.SelectedIndex); Assert.Equal("Baz", target.SelectedItem); Assert.Equal(new[] { "Baz", "Qux" }, target.SelectedItems); } } [Fact] public void Ctrl_Selecting_Non_SelectedItem_With_Multiple_Selection_Active_Leaves_SelectedItem_The_Same() { using (UnitTestApplication.Start()) { var target = new ListBox { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz" }, SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); AvaloniaLocator.CurrentMutable.Bind().ToConstant(new Mock().Object); _helper.Click((Interactive)target.Presenter.Panel.Children[1]); _helper.Click((Interactive)target.Presenter.Panel.Children[2], modifiers: KeyModifiers.Control); Assert.Equal(1, target.SelectedIndex); Assert.Equal("Bar", target.SelectedItem); _helper.Click((Interactive)target.Presenter.Panel.Children[2], modifiers: KeyModifiers.Control); Assert.Equal(1, target.SelectedIndex); Assert.Equal("Bar", target.SelectedItem); } } [Fact] public void Should_Ctrl_Select_Correct_Item_When_Duplicate_Items_Are_Present() { using (UnitTestApplication.Start()) { var target = new ListBox { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz", "Foo", "Bar", "Baz" }, SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); AvaloniaLocator.CurrentMutable.Bind().ToConstant(new Mock().Object); _helper.Click((Interactive)target.Presenter.Panel.Children[3]); _helper.Click((Interactive)target.Presenter.Panel.Children[4], modifiers: KeyModifiers.Control); var panel = target.Presenter.Panel; Assert.Equal(new[] { "Foo", "Bar" }, target.SelectedItems); Assert.Equal(new[] { 3, 4 }, SelectedContainers(target)); } } [Fact] public void Should_Shift_Select_Correct_Item_When_Duplicates_Are_Present() { using (UnitTestApplication.Start()) { var target = new ListBox { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz", "Foo", "Bar", "Baz" }, SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); AvaloniaLocator.CurrentMutable.Bind().ToConstant(new Mock().Object); _helper.Click((Interactive)target.Presenter.Panel.Children[3]); _helper.Click((Interactive)target.Presenter.Panel.Children[5], modifiers: KeyModifiers.Shift); var panel = target.Presenter.Panel; Assert.Equal(new[] { "Foo", "Bar", "Baz" }, target.SelectedItems); Assert.Equal(new[] { 3, 4, 5 }, SelectedContainers(target)); } } [Fact] public void Can_Shift_Select_All_Items_When_Duplicates_Are_Present() { using (UnitTestApplication.Start()) { var target = new ListBox { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz", "Foo", "Bar", "Baz" }, SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); AvaloniaLocator.CurrentMutable.Bind().ToConstant(new Mock().Object); _helper.Click((Interactive)target.Presenter.Panel.Children[0]); _helper.Click((Interactive)target.Presenter.Panel.Children[5], modifiers: KeyModifiers.Shift); var panel = target.Presenter.Panel; Assert.Equal(new[] { "Foo", "Bar", "Baz", "Foo", "Bar", "Baz" }, target.SelectedItems); Assert.Equal(new[] { 0, 1, 2, 3, 4, 5 }, SelectedContainers(target)); } } [Fact] public void Shift_Selecting_Raises_SelectionChanged_Events() { using (UnitTestApplication.Start()) { var target = new ListBox { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz", "Qux" }, SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); AvaloniaLocator.CurrentMutable.Bind().ToConstant(new Mock().Object); SelectionChangedEventArgs receivedArgs = null; target.SelectionChanged += (_, args) => receivedArgs = args; void VerifyAdded(params string[] selection) { Assert.NotNull(receivedArgs); Assert.Equal(selection, receivedArgs.AddedItems); Assert.Empty(receivedArgs.RemovedItems); } void VerifyRemoved(string selection) { Assert.NotNull(receivedArgs); Assert.Equal(new[] { selection }, receivedArgs.RemovedItems); Assert.Empty(receivedArgs.AddedItems); } _helper.Click((Interactive)target.Presenter.Panel.Children[1]); VerifyAdded("Bar"); receivedArgs = null; _helper.Click((Interactive)target.Presenter.Panel.Children[3], modifiers: KeyModifiers.Shift); VerifyAdded("Baz", "Qux"); receivedArgs = null; _helper.Click((Interactive)target.Presenter.Panel.Children[2], modifiers: KeyModifiers.Shift); VerifyRemoved("Qux"); } } [Fact] public void Duplicate_Items_Are_Added_To_SelectedItems_In_Order() { using (UnitTestApplication.Start()) { var target = new ListBox { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz", "Foo", "Bar", "Baz" }, SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); AvaloniaLocator.CurrentMutable.Bind().ToConstant(new Mock().Object); _helper.Click((Interactive)target.Presenter.Panel.Children[0]); Assert.Equal(new[] { "Foo" }, target.SelectedItems); _helper.Click((Interactive)target.Presenter.Panel.Children[4], modifiers: KeyModifiers.Control); Assert.Equal(new[] { "Foo", "Bar" }, target.SelectedItems); _helper.Click((Interactive)target.Presenter.Panel.Children[3], modifiers: KeyModifiers.Control); Assert.Equal(new[] { "Foo", "Bar", "Foo" }, target.SelectedItems); _helper.Click((Interactive)target.Presenter.Panel.Children[1], modifiers: KeyModifiers.Control); Assert.Equal(new[] { "Foo", "Bar", "Foo", "Bar" }, target.SelectedItems); } } [Fact] public void SelectAll_Sets_SelectedIndex_And_SelectedItem() { var target = new TestSelector { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz" }, SelectionMode = SelectionMode.Multiple, }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); target.SelectAll(); Assert.Equal(0, target.SelectedIndex); Assert.Equal("Foo", target.SelectedItem); } [Fact] public void SelectAll_Raises_SelectionChanged_Event() { var target = new TestSelector { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz" }, SelectionMode = SelectionMode.Multiple, }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); SelectionChangedEventArgs receivedArgs = null; target.SelectionChanged += (_, args) => receivedArgs = args; target.SelectAll(); Assert.NotNull(receivedArgs); Assert.Equal(target.ItemsSource, receivedArgs.AddedItems); Assert.Empty(receivedArgs.RemovedItems); } [Fact] public void UnselectAll_Clears_SelectedIndex_And_SelectedItem() { var target = new TestSelector { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz" }, SelectionMode = SelectionMode.Multiple, SelectedIndex = 0, }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); target.UnselectAll(); Assert.Equal(-1, target.SelectedIndex); Assert.Equal(null, target.SelectedItem); } [Fact] public void SelectAll_Handles_Duplicate_Items() { var target = new TestSelector { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz", "Foo", "Bar", "Baz" }, SelectionMode = SelectionMode.Multiple, }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); target.SelectAll(); Assert.Equal(new[] { "Foo", "Bar", "Baz", "Foo", "Bar", "Baz" }, target.SelectedItems); } [Fact] public void Adding_Item_Before_SelectedItems_Should_Update_Selection() { var items = new ObservableCollection { "Foo", "Bar", "Baz" }; var target = new ListBox { Template = Template(), ItemsSource = items, SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); target.SelectAll(); items.Insert(0, "Qux"); root.LayoutManager.ExecuteLayoutPass(); Assert.Equal(1, target.SelectedIndex); Assert.Equal("Foo", target.SelectedItem); Assert.Equal(new[] { "Foo", "Bar", "Baz" }, target.SelectedItems); Assert.Equal(new[] { 1, 2, 3 }, SelectedContainers(target)); } [Fact] public void Removing_Item_Before_SelectedItem_Should_Update_Selection() { var items = new ObservableCollection { "Foo", "Bar", "Baz" }; var target = new TestSelector { Template = Template(), ItemsSource = items, SelectionMode = SelectionMode.Multiple, }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); target.SelectedIndex = 1; target.SelectRange(2); Assert.Equal(new[] { "Bar", "Baz" }, target.SelectedItems); items.RemoveAt(0); Assert.Equal(0, target.SelectedIndex); Assert.Equal("Bar", target.SelectedItem); Assert.Equal(new[] { "Bar", "Baz" }, target.SelectedItems); Assert.Equal(new[] { 0, 1 }, SelectedContainers(target)); } [Fact] public void Removing_SelectedItem_With_Multiple_Selection_Active_Should_Update_Selection() { var items = new ObservableCollection { "Foo", "Bar", "Baz" }; var target = new ListBox { Template = Template(), ItemsSource = items, SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); target.SelectAll(); items.RemoveAt(0); Assert.Equal(0, target.SelectedIndex); Assert.Equal("Bar", target.SelectedItem); Assert.Equal(new[] { "Bar", "Baz" }, target.SelectedItems); Assert.Equal(new[] { 0, 1 }, SelectedContainers(target)); } [Fact] public void Replacing_Selected_Item_Should_Update_SelectedItems() { var items = new ObservableCollection { "Foo", "Bar", "Baz" }; var target = new ListBox { Template = Template(), ItemsSource = items, SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); target.SelectAll(); items[1] = "Qux"; Assert.Equal(new[] { "Foo", "Baz" }, target.SelectedItems); } [Fact] public void Left_Click_On_SelectedItem_Should_Clear_Existing_Selection() { using (UnitTestApplication.Start()) { var target = new ListBox { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz" }, ItemTemplate = new FuncDataTemplate((x, _) => new TextBlock { Width = 20, Height = 10 }), SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); target.SelectAll(); Assert.Equal(3, target.SelectedItems.Count); _helper.Click((Interactive)target.Presenter.Panel.Children[0]); Assert.Equal(1, target.SelectedItems.Count); Assert.Equal(new[] { "Foo", }, target.SelectedItems); Assert.Equal(new[] { 0 }, SelectedContainers(target)); } } [Fact] public void Right_Click_On_SelectedItem_Should_Not_Clear_Existing_Selection() { using (UnitTestApplication.Start()) { var target = new ListBox { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz" }, ItemTemplate = new FuncDataTemplate((x, _) => new TextBlock { Width = 20, Height = 10 }), SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); target.SelectAll(); Assert.Equal(3, target.SelectedItems.Count); _helper.Click((Interactive)target.Presenter.Panel.Children[0], MouseButton.Right); Assert.Equal(3, target.SelectedItems.Count); } } [Fact] public void Right_Click_On_UnselectedItem_Should_Clear_Existing_Selection() { using (UnitTestApplication.Start()) { var target = new ListBox { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz" }, ItemTemplate = new FuncDataTemplate((x, _) => new TextBlock { Width = 20, Height = 10 }), SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); AvaloniaLocator.CurrentMutable.Bind().ToConstant(new Mock().Object); _helper.Click((Interactive)target.Presenter.Panel.Children[0]); _helper.Click((Interactive)target.Presenter.Panel.Children[1], modifiers: KeyModifiers.Shift); Assert.Equal(2, target.SelectedItems.Count); _helper.Click((Interactive)target.Presenter.Panel.Children[2], MouseButton.Right); Assert.Equal(1, target.SelectedItems.Count); } } [Fact] public void Adding_Selected_ItemContainers_Should_Update_Selection() { var target = new TestSelector { Items = { new ItemContainer(), new ItemContainer(), }, SelectionMode = SelectionMode.Multiple, Template = Template(), }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); target.Items.Add(new ItemContainer { IsSelected = true }); target.Items.Add(new ItemContainer { IsSelected = true }); Assert.Equal(2, target.SelectedIndex); Assert.Equal(target.Items[2], target.SelectedItem); Assert.Equal(new[] { target.Items[2], target.Items[3] }, target.SelectedItems); } [Fact] public void Shift_Right_Click_Should_Not_Select_Multiple() { using (UnitTestApplication.Start()) { var target = new ListBox { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz" }, ItemTemplate = new FuncDataTemplate((x, _) => new TextBlock { Width = 20, Height = 10 }), SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); AvaloniaLocator.CurrentMutable.Bind().ToConstant(new Mock().Object); _helper.Click((Interactive)target.Presenter.Panel.Children[0]); _helper.Click((Interactive)target.Presenter.Panel.Children[2], MouseButton.Right, modifiers: KeyModifiers.Shift); Assert.Equal(1, target.SelectedItems.Count); } } [Fact] public void Ctrl_Right_Click_Should_Not_Select_Multiple() { using (UnitTestApplication.Start()) { var target = new ListBox { Template = Template(), ItemsSource = new[] { "Foo", "Bar", "Baz" }, ItemTemplate = new FuncDataTemplate((x, _) => new TextBlock { Width = 20, Height = 10 }), SelectionMode = SelectionMode.Multiple, Width = 100, Height = 100, }; var root = new TestRoot(target); root.LayoutManager.ExecuteInitialLayoutPass(); AvaloniaLocator.CurrentMutable.Bind().ToConstant(new Mock().Object); _helper.Click((Interactive)target.Presenter.Panel.Children[0]); _helper.Click((Interactive)target.Presenter.Panel.Children[2], MouseButton.Right, modifiers: KeyModifiers.Control); Assert.Equal(1, target.SelectedItems.Count); } } [Fact] public void Adding_To_Selection_Should_Set_SelectedIndex() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar" }, Template = Template(), }; target.ApplyTemplate(); target.SelectedItems.Add("bar"); Assert.Equal(1, target.SelectedIndex); } [Fact] public void Assigning_Null_To_Selection_Should_Create_New_SelectionModel() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar" }, Template = Template(), }; var oldSelection = target.Selection; target.Selection = null; Assert.NotNull(target.Selection); Assert.NotSame(oldSelection, target.Selection); } [Fact] public void Assigning_SelectionModel_With_Different_Source_To_Selection_Should_Fail() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar" }, Template = Template(), }; var selection = new SelectionModel { Source = new[] { "baz" } }; Assert.Throws(() => target.Selection = selection); } [Fact] public void Assigning_SelectionModel_With_Null_Source_To_Selection_Should_Set_Source() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar" }, Template = Template(), }; var selection = new SelectionModel(); target.Selection = selection; Assert.Same(target.ItemsSource, selection.Source); } [Fact] public void Assigning_Single_Selected_Item_To_Selection_Should_Set_SelectedIndex() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar" }, Template = Template(), }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); var selection = new SelectionModel { SingleSelect = false }; selection.Select(1); target.Selection = selection; Assert.Equal(1, target.SelectedIndex); Assert.Equal(new[] { "bar" }, target.Selection.SelectedItems); Assert.Equal(new[] { 1 }, SelectedContainers(target)); } [Fact] public void Assigning_Multiple_Selected_Items_To_Selection_Should_Set_SelectedIndex() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar", "baz" }, Template = Template(), }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); var selection = new SelectionModel { SingleSelect = false }; selection.SelectRange(0, 2); target.Selection = selection; Assert.Equal(0, target.SelectedIndex); Assert.Equal(new[] { "foo", "bar", "baz" }, target.Selection.SelectedItems); Assert.Equal(new[] { 0, 1, 2 }, SelectedContainers(target)); } [Fact] public void Reassigning_Selection_Should_Clear_Selection() { var target = new TestSelector { ItemsSource = new[] { "foo", "bar" }, Template = Template(), }; target.ApplyTemplate(); target.Selection.Select(1); target.Selection = new SelectionModel(); Assert.Equal(-1, target.SelectedIndex); Assert.Null(target.SelectedItem); } [Fact] public void Assigning_Selection_Should_Set_Item_IsSelected() { var target = new TestSelector { Items = { new ListBoxItem(), new ListBoxItem(), new ListBoxItem(), }, Template = Template(), }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); var selection = new SelectionModel { SingleSelect = false }; selection.SelectRange(0, 1); target.Selection = selection; var items = target.Items.Cast().ToList(); Assert.True(items[0].IsSelected); Assert.True(items[1].IsSelected); Assert.False(items[2].IsSelected); } [Fact] public void Assigning_Selection_Should_Raise_SelectionChanged() { var items = new[] { "foo", "bar", "baz" }; var target = new TestSelector { ItemsSource = items, Template = Template(), SelectedItem = "bar", }; var raised = 0; target.SelectionChanged += (s, e) => { if (raised == 0) { Assert.Empty(e.AddedItems.Cast()); Assert.Equal(new[] { "bar" }, e.RemovedItems.Cast()); } else { Assert.Equal(new[] { "foo", "baz" }, e.AddedItems.Cast()); Assert.Empty(e.RemovedItems.Cast()); } ++raised; }; target.ApplyTemplate(); target.Presenter.ApplyTemplate(); var selection = new SelectionModel { Source = items, SingleSelect = false }; selection.Select(0); selection.Select(2); target.Selection = selection; Assert.Equal(2, raised); } private static IEnumerable SelectedContainers(SelectingItemsControl target) { return target.Presenter.Panel.Children .Select(x => x.Classes.Contains(":selected") ? target.IndexFromContainer(x) : -1) .Where(x => x != -1); } private static FuncControlTemplate Template() { return new FuncControlTemplate((control, scope) => new ItemsPresenter { Name = "PART_ItemsPresenter", [~ItemsPresenter.ItemsPanelProperty] = control[~ItemsControl.ItemsPanelProperty], }.RegisterInNameScope(scope)); } private class TestSelector : SelectingItemsControl { public static readonly new AvaloniaProperty SelectedItemsProperty = SelectingItemsControl.SelectedItemsProperty; public static readonly new DirectProperty SelectionProperty = SelectingItemsControl.SelectionProperty; public TestSelector() { SelectionMode = SelectionMode.Multiple; } public new IList SelectedItems { get { return base.SelectedItems; } set { base.SelectedItems = value; } } public new ISelectionModel Selection { get => base.Selection; set => base.Selection = value; } public new SelectionMode SelectionMode { get { return base.SelectionMode; } set { base.SelectionMode = value; } } public void SelectAll() => Selection.SelectAll(); public void UnselectAll() => Selection.Clear(); public void SelectRange(int index) => UpdateSelection(index, true, true); public void Toggle(int index) => UpdateSelection(index, true, false, true); } private class OldDataContextViewModel { public OldDataContextViewModel() { Items = new List { "foo", "bar" }; SelectedItems = new List(); Selection = new SelectionModel(); } public List Items { get; } public List SelectedItems { get; } public SelectionModel Selection { get; } } private class ItemContainer : Control, ISelectable { public string Value { get; set; } public bool IsSelected { get; set; } } } }