| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516 |
- using System.Linq;
- using Avalonia.Collections;
- using Avalonia.Controls.Presenters;
- using Avalonia.Controls.Primitives;
- using Avalonia.Controls.Templates;
- using Avalonia.Input;
- using Avalonia.LogicalTree;
- using Avalonia.Styling;
- using Avalonia.UnitTests;
- using Avalonia.VisualTree;
- using Xunit;
- namespace Avalonia.Controls.UnitTests
- {
- public class ListBoxTests
- {
- private MouseTestHelper _mouse = new MouseTestHelper();
-
- [Fact]
- public void Should_Use_ItemTemplate_To_Create_Item_Content()
- {
- var target = new ListBox
- {
- Template = ListBoxTemplate(),
- Items = new[] { "Foo" },
- ItemTemplate = new FuncDataTemplate<string>((_, __) => new Canvas()),
- };
- Prepare(target);
- var container = (ListBoxItem)target.Presenter.Panel.Children[0];
- Assert.IsType<Canvas>(container.Presenter.Child);
- }
- [Fact]
- public void ListBox_Should_Find_ItemsPresenter_In_ScrollViewer()
- {
- var target = new ListBox
- {
- Template = ListBoxTemplate(),
- };
- Prepare(target);
- Assert.IsType<ItemsPresenter>(target.Presenter);
- }
- [Fact]
- public void ListBox_Should_Find_Scrollviewer_In_Template()
- {
- var target = new ListBox
- {
- Template = ListBoxTemplate(),
- };
- ScrollViewer viewer = null;
- target.TemplateApplied += (sender, e) =>
- {
- viewer = target.Scroll as ScrollViewer;
- };
- Prepare(target);
- Assert.NotNull(viewer);
- }
- [Fact]
- public void ListBoxItem_Containers_Should_Be_Generated()
- {
- using (UnitTestApplication.Start(TestServices.MockPlatformRenderInterface))
- {
- var items = new[] { "Foo", "Bar", "Baz " };
- var target = new ListBox
- {
- Template = ListBoxTemplate(),
- Items = items,
- };
- Prepare(target);
- var text = target.Presenter.Panel.Children
- .OfType<ListBoxItem>()
- .Select(x => x.Presenter.Child)
- .OfType<TextBlock>()
- .Select(x => x.Text)
- .ToList();
- Assert.Equal(items, text);
- }
- }
- [Fact]
- public void LogicalChildren_Should_Be_Set_For_DataTemplate_Generated_Items()
- {
- using (UnitTestApplication.Start(TestServices.MockPlatformRenderInterface))
- {
- var target = new ListBox
- {
- Template = ListBoxTemplate(),
- Items = new[] { "Foo", "Bar", "Baz " },
- };
- Prepare(target);
- Assert.Equal(3, target.GetLogicalChildren().Count());
- foreach (var child in target.GetLogicalChildren())
- {
- Assert.IsType<ListBoxItem>(child);
- }
- }
- }
- [Fact]
- public void DataContexts_Should_Be_Correctly_Set()
- {
- using (UnitTestApplication.Start(TestServices.MockPlatformRenderInterface))
- {
- var items = new object[]
- {
- "Foo",
- new Item("Bar"),
- new TextBlock { Text = "Baz" },
- new ListBoxItem { Content = "Qux" },
- };
- var target = new ListBox
- {
- Template = ListBoxTemplate(),
- DataContext = "Base",
- DataTemplates =
- {
- new FuncDataTemplate<Item>((x, _) => new Button { Content = x })
- },
- Items = items,
- };
- Prepare(target);
- var dataContexts = target.Presenter.Panel.Children
- .Cast<Control>()
- .Select(x => x.DataContext)
- .ToList();
- Assert.Equal(
- new object[] { items[0], items[1], "Base", "Base" },
- dataContexts);
- }
- }
- [Fact]
- public void Selection_Should_Be_Cleared_On_Recycled_Items()
- {
- var target = new ListBox
- {
- Template = ListBoxTemplate(),
- Items = Enumerable.Range(0, 20).Select(x => $"Item {x}").ToList(),
- ItemTemplate = new FuncDataTemplate<string>((x, _) => new TextBlock { Height = 10 }),
- SelectedIndex = 0,
- };
- Prepare(target);
- // Make sure we're virtualized and first item is selected.
- Assert.Equal(10, target.Presenter.Panel.Children.Count);
- Assert.True(((ListBoxItem)target.Presenter.Panel.Children[0]).IsSelected);
- // Scroll down a page.
- target.Scroll.Offset = new Vector(0, 10);
- // Make sure recycled item isn't now selected.
- Assert.False(((ListBoxItem)target.Presenter.Panel.Children[0]).IsSelected);
- }
- [Fact]
- public void ScrollViewer_Should_Have_Correct_Extent_And_Viewport()
- {
- var target = new ListBox
- {
- Template = ListBoxTemplate(),
- Items = Enumerable.Range(0, 20).Select(x => $"Item {x}").ToList(),
- ItemTemplate = new FuncDataTemplate<string>((x, _) => new TextBlock { Width = 20, Height = 10 }),
- SelectedIndex = 0,
- };
- Prepare(target);
- Assert.Equal(new Size(20, 20), target.Scroll.Extent);
- Assert.Equal(new Size(100, 10), target.Scroll.Viewport);
- }
- [Fact]
- public void Containers_Correct_After_Clear_Add_Remove()
- {
- // Issue #1936
- var items = new AvaloniaList<string>(Enumerable.Range(0, 11).Select(x => $"Item {x}"));
- var target = new ListBox
- {
- Template = ListBoxTemplate(),
- Items = items,
- ItemTemplate = new FuncDataTemplate<string>((x, _) => new TextBlock { Width = 20, Height = 10 }),
- SelectedIndex = 0,
- };
- Prepare(target);
- items.Clear();
- items.AddRange(Enumerable.Range(0, 11).Select(x => $"Item {x}"));
- items.Remove("Item 2");
- Assert.Equal(
- items,
- target.Presenter.Panel.Children.Cast<ListBoxItem>().Select(x => (string)x.Content));
- }
- [Fact]
- public void Toggle_Selection_Should_Update_Containers()
- {
- var items = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToArray();
- var target = new ListBox
- {
- Template = ListBoxTemplate(),
- Items = items,
- SelectionMode = SelectionMode.Toggle,
- ItemTemplate = new FuncDataTemplate<string>((x, _) => new TextBlock { Height = 10 })
- };
- Prepare(target);
- var lbItems = target.GetLogicalChildren().OfType<ListBoxItem>().ToArray();
- var item = lbItems[0];
- Assert.Equal(false, item.IsSelected);
- RaisePressedEvent(target, item, MouseButton.Left);
- Assert.Equal(true, item.IsSelected);
- RaisePressedEvent(target, item, MouseButton.Left);
- Assert.Equal(false, item.IsSelected);
- }
- [Fact]
- public void Can_Decrease_Number_Of_Materialized_Items_By_Removing_From_Source_Collection()
- {
- var items = new AvaloniaList<string>(Enumerable.Range(0, 20).Select(x => $"Item {x}"));
- var target = new ListBox
- {
- Template = ListBoxTemplate(),
- Items = items,
- ItemTemplate = new FuncDataTemplate<string>((x, _) => new TextBlock { Height = 10 })
- };
- Prepare(target);
- target.Scroll.Offset = new Vector(0, 1);
- items.RemoveRange(0, 11);
- }
- private void RaisePressedEvent(ListBox listBox, ListBoxItem item, MouseButton mouseButton)
- {
- _mouse.Click(listBox, item, mouseButton);
- }
- [Fact]
- public void ListBox_After_Scroll_IndexOutOfRangeException_Shouldnt_Be_Thrown()
- {
- var items = Enumerable.Range(0, 11).Select(x => $"{x}").ToArray();
- var target = new ListBox
- {
- Template = ListBoxTemplate(),
- Items = items,
- ItemTemplate = new FuncDataTemplate<string>((x, _) => new TextBlock { Height = 11 })
- };
- Prepare(target);
- var panel = target.Presenter.Panel as IVirtualizingPanel;
- var listBoxItems = panel.Children.OfType<ListBoxItem>();
- //virtualization should have created exactly 10 items
- Assert.Equal(10, listBoxItems.Count());
- Assert.Equal("0", listBoxItems.First().DataContext);
- Assert.Equal("9", listBoxItems.Last().DataContext);
- //instead pixeloffset > 0 there could be pretty complex sequence for repro
- //it involves add/remove/scroll to end multiple actions
- //which i can't find so far :(, but this is the simplest way to add it to unit test
- panel.PixelOffset = 1;
- //here scroll to end -> IndexOutOfRangeException is thrown
- target.Scroll.Offset = new Vector(0, 2);
- Assert.True(true);
- }
- [Fact]
- public void LayoutManager_Should_Measure_Arrange_All()
- {
- var virtualizationMode = ItemVirtualizationMode.Simple;
- using (UnitTestApplication.Start(TestServices.StyledWindow))
- {
- var items = new AvaloniaList<string>(Enumerable.Range(1, 7).Select(v => v.ToString()));
- var wnd = new Window() { SizeToContent = SizeToContent.WidthAndHeight };
- wnd.IsVisible = true;
- var target = new ListBox();
- wnd.Content = target;
- var lm = wnd.LayoutManager;
- target.Height = 110;
- target.Width = 50;
- target.DataContext = items;
- target.VirtualizationMode = virtualizationMode;
- target.ItemTemplate = new FuncDataTemplate<object>((c, _) =>
- {
- var tb = new TextBlock() { Height = 10, Width = 30 };
- tb.Bind(TextBlock.TextProperty, new Data.Binding());
- return tb;
- }, true);
- lm.ExecuteInitialLayoutPass();
- target.Items = items;
- lm.ExecuteLayoutPass();
- items.Insert(3, "3+");
- lm.ExecuteLayoutPass();
- items.Insert(4, "4+");
- lm.ExecuteLayoutPass();
- //RESET
- items.Clear();
- foreach (var i in Enumerable.Range(1, 7))
- {
- items.Add(i.ToString());
- }
- //working bit better with this line no outof memory or remaining to arrange/measure ???
- //lm.ExecuteLayoutPass();
- items.Insert(2, "2+");
- lm.ExecuteLayoutPass();
- //after few more layout cycles layoutmanager shouldn't hold any more visual for measure/arrange
- lm.ExecuteLayoutPass();
- lm.ExecuteLayoutPass();
- var flags = System.Reflection.BindingFlags.Instance | System.Reflection.BindingFlags.NonPublic;
- var toMeasure = lm.GetType().GetField("_toMeasure", flags).GetValue(lm) as System.Collections.Generic.IEnumerable<Layout.ILayoutable>;
- var toArrange = lm.GetType().GetField("_toArrange", flags).GetValue(lm) as System.Collections.Generic.IEnumerable<Layout.ILayoutable>;
- Assert.Equal(0, toMeasure.Count());
- Assert.Equal(0, toArrange.Count());
- }
- }
- [Fact]
- public void Clicking_Item_Should_Raise_BringIntoView_For_Correct_Control()
- {
- // Issue #3934
- var items = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToArray();
- var target = new ListBox
- {
- Template = ListBoxTemplate(),
- Items = items,
- ItemTemplate = new FuncDataTemplate<string>((x, _) => new TextBlock { Height = 10 }),
- SelectionMode = SelectionMode.AlwaysSelected,
- VirtualizationMode = ItemVirtualizationMode.None,
- };
- Prepare(target);
- // First an item that is not index 0 must be selected.
- _mouse.Click(target.Presenter.Panel.Children[1]);
- Assert.Equal(new IndexPath(1), target.Selection.AnchorIndex);
- // We're going to be clicking on item 9.
- var item = (ListBoxItem)target.Presenter.Panel.Children[9];
- var raised = 0;
- // Make sure a RequestBringIntoView event is raised for item 9. It won't be handled
- // by the ScrollContentPresenter as the item is already visible, so we don't need
- // handledEventsToo: true. Issue #3934 failed here because item 0 was being scrolled
- // into view due to SelectionMode.AlwaysSelected.
- target.AddHandler(Control.RequestBringIntoViewEvent, (s, e) =>
- {
- Assert.Same(item, e.TargetObject);
- ++raised;
- });
- // Click item 9.
- _mouse.Click(item);
- Assert.Equal(1, raised);
- }
- private FuncControlTemplate ListBoxTemplate()
- {
- return new FuncControlTemplate<ListBox>((parent, scope) =>
- new ScrollViewer
- {
- Name = "PART_ScrollViewer",
- Template = ScrollViewerTemplate(),
- Content = new ItemsPresenter
- {
- Name = "PART_ItemsPresenter",
- [~ItemsPresenter.ItemsProperty] = parent.GetObservable(ItemsControl.ItemsProperty).ToBinding(),
- [~ItemsPresenter.ItemsPanelProperty] = parent.GetObservable(ItemsControl.ItemsPanelProperty).ToBinding(),
- [~ItemsPresenter.VirtualizationModeProperty] = parent.GetObservable(ListBox.VirtualizationModeProperty).ToBinding(),
- }.RegisterInNameScope(scope)
- }.RegisterInNameScope(scope));
- }
- private FuncControlTemplate ListBoxItemTemplate()
- {
- return new FuncControlTemplate<ListBoxItem>((parent, scope) =>
- new ContentPresenter
- {
- Name = "PART_ContentPresenter",
- [!ContentPresenter.ContentProperty] = parent[!ListBoxItem.ContentProperty],
- [!ContentPresenter.ContentTemplateProperty] = parent[!ListBoxItem.ContentTemplateProperty],
- }.RegisterInNameScope(scope));
- }
- private FuncControlTemplate ScrollViewerTemplate()
- {
- return new FuncControlTemplate<ScrollViewer>((parent, scope) =>
- new Panel
- {
- Children =
- {
- new ScrollContentPresenter
- {
- Name = "PART_ContentPresenter",
- [~ScrollContentPresenter.ContentProperty] = parent.GetObservable(ScrollViewer.ContentProperty).ToBinding(),
- [~~ScrollContentPresenter.ExtentProperty] = parent[~~ScrollViewer.ExtentProperty],
- [~~ScrollContentPresenter.OffsetProperty] = parent[~~ScrollViewer.OffsetProperty],
- [~~ScrollContentPresenter.ViewportProperty] = parent[~~ScrollViewer.ViewportProperty],
- }.RegisterInNameScope(scope),
- new ScrollBar
- {
- Name = "verticalScrollBar",
- [~ScrollBar.MaximumProperty] = parent[~ScrollViewer.VerticalScrollBarMaximumProperty],
- [~~ScrollBar.ValueProperty] = parent[~~ScrollViewer.VerticalScrollBarValueProperty],
- }
- }
- });
- }
- private void Prepare(ListBox target)
- {
- // The ListBox needs to be part of a rooted visual tree.
- var root = new TestRoot();
- root.Child = target;
- // Apply the template to the ListBox itself.
- target.ApplyTemplate();
- // Then to its inner ScrollViewer.
- var scrollViewer = (ScrollViewer)target.GetVisualChildren().Single();
- scrollViewer.ApplyTemplate();
- // Then make the ScrollViewer create its child.
- ((ContentPresenter)scrollViewer.Presenter).UpdateChild();
- // Now the ItemsPresenter should be reigstered, so apply its template.
- target.Presenter.ApplyTemplate();
- // Because ListBox items are virtualized we need to do a layout to make them appear.
- target.Measure(new Size(100, 100));
- target.Arrange(new Rect(0, 0, 100, 100));
- // Now set and apply the item templates.
- foreach (ListBoxItem item in target.Presenter.Panel.Children)
- {
- item.Template = ListBoxItemTemplate();
- item.ApplyTemplate();
- item.Presenter.ApplyTemplate();
- ((ContentPresenter)item.Presenter).UpdateChild();
- }
- // The items were created before the template was applied, so now we need to go back
- // and re-arrange everything.
- foreach (IControl i in target.GetSelfAndVisualDescendants())
- {
- i.InvalidateMeasure();
- }
- target.Measure(new Size(100, 100));
- target.Arrange(new Rect(0, 0, 100, 100));
- }
- private class Item
- {
- public Item(string value)
- {
- Value = value;
- }
- public string Value { get; }
- }
- }
- }
|