| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055 |
- using System;
- using System.Collections;
- using System.Collections.Generic;
- using System.Collections.ObjectModel;
- using System.Collections.Specialized;
- using System.Linq;
- using Avalonia.Collections;
- using Avalonia.Controls.Presenters;
- using Avalonia.Controls.Primitives;
- using Avalonia.Controls.Templates;
- using Avalonia.Data;
- using Avalonia.Input;
- using Avalonia.Layout;
- using Avalonia.LogicalTree;
- using Avalonia.Markup.Xaml.Templates;
- using Avalonia.Styling;
- using Avalonia.UnitTests;
- using Avalonia.VisualTree;
- using Xunit;
- #nullable enable
- namespace Avalonia.Controls.UnitTests
- {
- public class ItemsControlTests
- {
- [Fact]
- public void Setting_ItemsSource_Should_Populate_Items()
- {
- using var app = Start();
- var target = CreateTarget(itemsSource: new[] { "foo", "bar" });
- Assert.NotSame(target.ItemsSource, target.Items);
- Assert.Equal(target.ItemsSource, target.Items);
- }
- [Fact]
- public void Cannot_Set_ItemsSource_With_Items_Present()
- {
- using var app = Start();
- var target = CreateTarget();
- target.Items.Add("foo");
- Assert.Throws<InvalidOperationException>(() => target.ItemsSource = new[] { "baz" });
- }
- [Fact]
- public void Cannot_Modify_Items_When_ItemsSource_Set()
- {
- using var app = Start();
- var target = CreateTarget(itemsSource: Array.Empty<string>());
- Assert.Throws<InvalidOperationException>(() => target.Items.Add("foo"));
- }
- [Fact]
- public void Should_Use_ItemTemplate_To_Create_Control()
- {
- using var app = Start();
- var target = CreateTarget(
- itemsSource: new[] { "Foo" },
- itemTemplate: new FuncDataTemplate<string>((_, __) => new Canvas()));
- var container = GetContainer(target);
- Assert.IsType<Canvas>(container.Child);
- }
- [Fact]
- public void ItemTemplate_Can_Be_Changed()
- {
- using var app = Start();
- var target = CreateTarget(
- itemsSource: new[] { "Foo" },
- itemTemplate: new FuncDataTemplate<string>((_, __) => new Canvas()));
- var container = GetContainer(target);
- Assert.IsType<Canvas>(container.Child);
- target.ItemTemplate = new FuncDataTemplate<string>((_, __) => new Border());
- Layout(target);
- container = GetContainer(target);
- Assert.IsType<Border>(container.Child);
- }
- [Fact]
- public void Panel_Should_Have_TemplatedParent_Set_To_ItemsControl()
- {
- using var app = Start();
- var target = CreateTarget(itemsSource: new[] { "Foo" });
- Assert.Equal(target, target.ItemsPanelRoot?.TemplatedParent);
- }
- [Fact]
- public void Panel_Should_Have_ItemsHost_Set_To_True()
- {
- using var app = Start();
- var target = CreateTarget(itemsSource: new[] { "Foo" });
- Assert.True(target.ItemsPanelRoot?.IsItemsHost);
- }
- [Fact]
- public void Container_Should_Have_TemplatedParent_Set_To_Null()
- {
- using var app = Start();
- var target = CreateTarget(itemsSource: new[] { "Foo" });
- var container = GetContainer(target);
- Assert.Null(container.TemplatedParent);
- }
- [Fact]
- public void Container_Should_Have_Theme_Set_To_ItemContainerTheme()
- {
- using var app = Start();
- var theme = new ControlTheme { TargetType = typeof(ContentPresenter) };
- var target = CreateTarget(
- itemsSource: new[] { "Foo" },
- itemContainerTheme: theme);
- var container = GetContainer(target);
- Assert.Same(container.Theme, theme);
- }
- [Fact]
- public void Container_Should_Have_LogicalParent_Set_To_ItemsControl()
- {
- using var app = UnitTestApplication.Start(TestServices.StyledWindow);
- var target = new ItemsControl();
- var root = CreateRoot(target);
- var templatedParent = new Button();
- target.TemplatedParent = templatedParent;
- target.Template = CreateItemsControlTemplate();
- target.ItemsSource = new[] { "Foo" };
- root.LayoutManager.ExecuteInitialLayoutPass();
- var container = GetContainer(target);
- Assert.Equal(target, container.Parent);
- }
- [Fact]
- public void Control_Item_Should_Be_Logical_Child_Before_ApplyTemplate()
- {
- using var app = Start();
- var child = new Control();
- var target = CreateTarget(items: new[] { child }, performLayout: false);
- Assert.False(target.IsMeasureValid);
- Assert.Empty(target.GetVisualChildren());
- Assert.Equal(child.Parent, target);
- Assert.Equal(child.GetLogicalParent(), target);
- Assert.Equal(new[] { child }, target.GetLogicalChildren());
- }
- [Fact]
- public void Control_Item_Should_Be_Logical_Child_After_Layout()
- {
- using var app = Start();
- var child = new Control();
- var target = CreateTarget(items: new[] { child });
- Assert.True(target.IsMeasureValid);
- Assert.Single(target.GetVisualChildren());
- Assert.Equal(target, child.Parent);
- Assert.Equal(target, child.GetLogicalParent());
- Assert.Equal(new[] { child }, target.GetLogicalChildren());
- }
- [Fact]
- public void Added_Container_Should_Have_LogicalParent_Set_To_ItemsControl()
- {
- using var app = Start();
- var items = new ObservableCollection<Border>();
- var target = CreateTarget(itemsSource: items);
- var item = new Border();
- items.Add(item);
- Assert.Equal(target, item.Parent);
- }
- [Fact]
- public void Control_Item_Can_Be_Removed_From_Logical_Children_Before_ApplyTemplate()
- {
- using var app = Start();
- var child = new Control();
- var target = CreateTarget(items: new[] { child }, performLayout: false);
- Assert.False(target.IsMeasureValid);
- Assert.Empty(target.GetVisualChildren());
- Assert.Single(target.GetLogicalChildren());
- target.Items.RemoveAt(0);
- Assert.Null(child.Parent);
- Assert.Null(child.GetLogicalParent());
- Assert.Empty(target.GetLogicalChildren());
- }
- [Fact]
- public void Clearing_Items_Should_Clear_Child_Controls_Parent_Before_ApplyTemplate()
- {
- using var app = Start();
- var child = new Control();
- var target = CreateTarget(items: new[] { child }, performLayout: false);
- Assert.False(target.IsMeasureValid);
- Assert.Empty(target.GetVisualChildren());
- Assert.Single(target.GetLogicalChildren());
- target.Items.Clear();
- Assert.Null(child.Parent);
- Assert.Null(child.GetLogicalParent());
- }
- [Fact]
- public void Assigning_ItemsSource_Should_Not_Fire_LogicalChildren_CollectionChanged_Before_ApplyTemplate()
- {
- using var app = Start();
- var child = new Control();
- var target = CreateTarget(itemsSource: new[] { child }, performLayout: false);
- var called = false;
- ((ILogical)target).LogicalChildren.CollectionChanged += (s, e) => called = true;
- var list = new AvaloniaList<Control>(new[] { child });
- target.ItemsSource = list;
- Assert.False(called);
- }
- [Fact]
- public void Removing_ItemsSource_Items_Should_Not_Fire_LogicalChildren_CollectionChanged_Before_ApplyTemplate()
- {
- using var app = Start();
- var items = new AvaloniaList<string> { "Foo", "Bar" };
- var target = CreateTarget(itemsSource: items, performLayout: false);
- var called = false;
- ((ILogical)target).LogicalChildren.CollectionChanged += (s, e) => called = true;
- items.Remove("Bar");
- Assert.False(called);
- }
- [Fact]
- public void Changing_ItemsSource_Should_Not_Fire_LogicalChildren_CollectionChanged_Before_ApplyTemplate()
- {
- using var app = Start();
- var child = new Control();
- var target = CreateTarget(itemsSource: new[] { child }, performLayout: false);
- var called = false;
- ((ILogical)target).LogicalChildren.CollectionChanged += (s, e) => called = true;
- var list = new AvaloniaList<Control>();
- target.ItemsSource = list;
- list.Add(child);
- Assert.False(called);
- }
- [Fact]
- public void Clearing_Items_Should_Clear_Child_Controls_Parent()
- {
- using var app = Start();
- var child = new Control();
- var target = CreateTarget(items: new[] { child });
- target.Items.Clear();
- Assert.Null(child.Parent);
- Assert.Null(((ILogical)child).LogicalParent);
- }
- [Fact]
- public void Adding_Control_Item_Should_Make_Control_Appear_In_LogicalChildren()
- {
- using var app = Start();
- var child = new Control();
- var target = CreateTarget(items: new[] { child }, performLayout: false);
- // Should appear both before and after applying template.
- Assert.Equal(new ILogical[] { child }, target.GetLogicalChildren());
- Layout(target);
- Assert.Equal(new ILogical[] { child }, target.GetLogicalChildren());
- }
- [Fact]
- public void Adding_String_Item_Should_Make_ContentPresenter_Appear_In_LogicalChildren()
- {
- using var app = Start();
- var target = CreateTarget(itemsSource: new[] { "Foo " });
- var logical = (ILogical)target;
- Assert.Equal(1, logical.LogicalChildren.Count);
- Assert.IsType<ContentPresenter>(logical.LogicalChildren[0]);
- }
- [Fact]
- public void Adding_Items_Should_Fire_LogicalChildren_CollectionChanged()
- {
- using var app = Start();
- var target = CreateTarget();
- var called = false;
- target.Template = CreateItemsControlTemplate();
- target.ApplyTemplate();
- ((ILogical)target).LogicalChildren.CollectionChanged += (s, e) =>
- called = e.Action == NotifyCollectionChangedAction.Add;
- var child = new Control();
- target.Items.Add(child);
- Assert.True(called);
- }
- [Fact]
- public void Clearing_Items_Should_Fire_LogicalChildren_CollectionChanged()
- {
- using var app = Start();
- var child = new Control();
- var target = CreateTarget(items: new[] { child });
- var called = false;
- ((ILogical)target).LogicalChildren.CollectionChanged += (s, e) =>
- called = e.Action == NotifyCollectionChangedAction.Remove;
- target.Items.Clear();
- Assert.True(called);
- }
- [Fact]
- public void LogicalChildren_Should_Not_Change_Instance_When_Template_Changed()
- {
- using var app = Start();
- var target = CreateTarget();
- var before = ((ILogical)target).LogicalChildren;
- target.Template = null;
- target.Template = CreateItemsControlTemplate();
- Layout(target);
- var after = ((ILogical)target).LogicalChildren;
- Assert.NotNull(before);
- Assert.NotNull(after);
- Assert.Same(before, after);
- }
- [Fact]
- public void Should_Clear_Containers_When_ItemsPresenter_Changes()
- {
- using var app = Start();
- var target = CreateTarget(itemsSource: new[] { "foo", "bar" });
- var panel = Assert.IsAssignableFrom<Panel>(target.Presenter?.Panel);
- Assert.Equal(2, panel.Children.Count());
- target.Template = CreateItemsControlTemplate();
- target.ApplyTemplate();
- Assert.Empty(panel.Children);
- }
- [Fact]
- public void Empty_Class_Should_Initially_Be_Applied()
- {
- using var app = Start();
- var target = CreateTarget(performLayout: false);
- Assert.Contains(":empty", target.Classes);
- }
- [Fact]
- public void Empty_Class_Should_Be_Cleared_When_Items_Added()
- {
- using var app = Start();
- var target = CreateTarget(items: new[] { 1, 2, 3 }, performLayout: false);
- Assert.DoesNotContain(":empty", target.Classes);
- }
- [Fact]
- public void Empty_Class_Should_Be_Cleared_When_ItemsSource_Items_Added()
- {
- using var app = Start();
- var target = CreateTarget(itemsSource: new[] { 1, 2, 3 }, performLayout: false);
- Assert.DoesNotContain(":empty", target.Classes);
- }
- [Fact]
- public void Empty_Class_Should_Be_Set_When_ItemsSource_Collection_Cleared()
- {
- using var app = Start();
- var target = CreateTarget(itemsSource: new[] { 1, 2, 3 });
- target.ItemsSource = new int[0];
- Assert.Contains(":empty", target.Classes);
- }
- [Fact]
- public void Item_Count_Should_Be_Set_When_ItemsSource_Set()
- {
- using var app = Start();
- var target = CreateTarget(itemsSource: new[] { 1, 2, 3 });
- Assert.Equal(3, target.ItemCount);
- }
- [Fact]
- public void Item_Count_Should_Be_Set_When_Items_Changed()
- {
- using var app = Start();
- var items = new ObservableCollection<int>() { 1, 2, 3 };
- var target = CreateTarget(items: new[] { 1, 2, 3 });
- target.Items.Add(4);
- Assert.Equal(4, target.ItemCount);
- target.Items.Clear();
- Assert.Equal(0, target.ItemCount);
- }
- [Fact]
- public void Item_Count_Should_Be_Set_When_ItemsSource_Items_Changed()
- {
- using var app = Start();
- var items = new ObservableCollection<int>() { 1, 2, 3 };
- var target = CreateTarget(itemsSource: items);
- items.Add(4);
- Assert.Equal(4, target.ItemCount);
- items.Clear();
- Assert.Equal(0, target.ItemCount);
- }
- [Fact]
- public void Empty_Class_Should_Be_Set_When_Items_Collection_Cleared()
- {
- using var app = Start();
- var items = new ObservableCollection<int>() { 1, 2, 3 };
- var target = CreateTarget(itemsSource: items);
- items.Clear();
- Assert.Contains(":empty", target.Classes);
- }
- [Fact]
- public void Empty_Class_Should_Not_Be_Set_When_ItemsSource_Collection_Count_Increases()
- {
- using var app = Start();
- var items = new ObservableCollection<int>() { };
- var target = CreateTarget(itemsSource: items);
- items.Add(1);
- Assert.DoesNotContain(":empty", target.Classes);
- }
- [Fact]
- public void Single_Item_Class_Should_Be_Set_When_ItemsSource_Collection_Count_Increases_To_One()
- {
- using var app = Start();
- var items = new ObservableCollection<int>() { };
- var target = CreateTarget(itemsSource: items);
- items.Add(1);
- Assert.Contains(":singleitem", target.Classes);
- }
- [Fact]
- public void Empty_Class_Should_Not_Be_Set_When_ItemsSource_Collection_Cleared()
- {
- using var app = Start();
- var items = new ObservableCollection<int>() { 1, 2, 3 };
- var target = CreateTarget(itemsSource: items);
- items.Clear();
- Assert.DoesNotContain(":singleitem", target.Classes);
- }
- [Fact]
- public void Single_Item_Class_Should_Not_Be_Set_When_Items_Collection_Count_Increases_Beyond_One()
- {
- using var app = Start();
- var items = new ObservableCollection<int>() { 1 };
- var target = CreateTarget(itemsSource: items);
- items.Add(2);
- Assert.DoesNotContain(":singleitem", target.Classes);
- }
- [Fact]
- public void DataContexts_Should_Be_Correctly_Set()
- {
- using var app = Start();
- var items = new object[]
- {
- "Foo",
- new Item("Bar"),
- new TextBlock { Text = "Baz" },
- new ListBoxItem { Content = "Qux" },
- };
- var dataTemplate = new FuncDataTemplate<Item>((x, __) => new Button { Content = x });
- var target = CreateTarget(
- dataContext: "Base",
- itemsSource: items,
- dataTemplates: new[] { dataTemplate });
- var panel = Assert.IsAssignableFrom<Panel>(target.ItemsPanelRoot);
- var dataContexts = panel.Children
- .Do(x => (x as ContentPresenter)?.UpdateChild())
- .Cast<Control>()
- .Select(x => x.DataContext)
- .ToList();
- Assert.Equal(
- new object[] { items[0], items[1], "Base", "Base" },
- dataContexts);
- }
- [Fact]
- public void Control_Item_Should_Not_Be_NameScope()
- {
- using var app = Start();
- var items = new object[] { new TextBlock() };
- var target = CreateTarget(itemsSource: items);
- var item = target.LogicalChildren[0];
- Assert.Null(NameScope.GetNameScope((TextBlock)item));
- }
- [Fact]
- public void Focuses_Next_Item_On_Key_Down()
- {
- using var app = Start();
- var items = new object[]
- {
- new Button(),
- new Button(),
- };
- var target = CreateTarget(itemsSource: items);
- GetContainer<Button>(target).Focus();
- target.RaiseEvent(new KeyEventArgs
- {
- RoutedEvent = InputElement.KeyDownEvent,
- Key = Key.Down,
- });
- var panel = Assert.IsAssignableFrom<Panel>(target.ItemsPanelRoot);
- Assert.Equal(panel.Children[1], FocusManager.Instance!.Current);
- }
- [Fact]
- public void Does_Not_Focus_Non_Focusable_Item_On_Key_Down()
- {
- using var app = Start();
- var items = new object[]
- {
- new Button(),
- new Button { Focusable = false },
- new Button(),
- };
- var target = CreateTarget(itemsSource: items);
- GetContainer<Button>(target).Focus();
- target.RaiseEvent(new KeyEventArgs
- {
- RoutedEvent = InputElement.KeyDownEvent,
- Key = Key.Down,
- });
- var panel = Assert.IsAssignableFrom<Panel>(target.ItemsPanelRoot);
- Assert.Equal(panel.Children[2], FocusManager.Instance!.Current);
- }
- [Fact]
- public void Detaching_Then_Reattaching_To_Logical_Tree_Twice_Does_Not_Throw()
- {
- // # Issue 3487
- using var app = Start();
- var target = CreateTarget(
- itemsSource: new[] { "foo", "bar" },
- itemTemplate: new FuncDataTemplate<string>((_, __) => new Canvas()));
- var root = Assert.IsType<TestRoot>(target.GetVisualRoot());
- root.Child = null;
- root.Child = target;
- root.LayoutManager.ExecuteLayoutPass();
- root.Child = null;
- root.Child = target;
- }
- [Fact]
- public void Should_Use_DisplayMemberBinding()
- {
- using var app = Start();
- var target = CreateTarget(
- itemsSource: new[] { "Foo" },
- displayMemberBinding: new Binding("Length"));
- var container = GetContainer(target);
- var textBlock = Assert.IsType<TextBlock>(container.Child);
- Assert.Equal(textBlock.Text, "3");
- }
- [Fact]
- public void DisplayMemberBinding_Can_Be_Changed()
- {
- using var app = Start();
- var target = CreateTarget(
- itemsSource: new[] { new Item("Foo", "Bar") },
- displayMemberBinding: new Binding("Value"));
- var container = GetContainer(target);
- var textBlock = Assert.IsType<TextBlock>(container.Child);
- Assert.Equal(textBlock.Text, "Bar");
- target.DisplayMemberBinding = new Binding("Caption");
- Layout(target);
- container = GetContainer(target);
- textBlock = Assert.IsType<TextBlock>(container.Child);
-
- Assert.Equal(textBlock.Text, "Foo");
- }
- [Fact]
- public void Cannot_Set_Both_DisplayMemberBinding_And_ItemTemplate_1()
- {
- using var app = Start();
- var target = CreateTarget(
- displayMemberBinding: new Binding("Length"));
- Assert.Throws<InvalidOperationException>(() =>
- target.ItemTemplate = new FuncDataTemplate<string>((_, _) => new TextBlock()));
- }
- [Fact]
- public void Cannot_Set_Both_DisplayMemberBinding_And_ItemTemplate_2()
- {
- using var app = Start();
- var target = CreateTarget(
- itemTemplate: new FuncDataTemplate<string>((_, _) => new TextBlock()));
- Assert.Throws<InvalidOperationException>(() => target.DisplayMemberBinding = new Binding("Length"));
- }
- [Fact]
- public void ContainerPrepared_Is_Raised_For_Each_Control_Item_Container()
- {
- using var app = Start();
- var items = new AvaloniaList<string>();
- var target = CreateTarget();
- var result = new List<Control>();
- var index = 0;
- target.ContainerPrepared += (s, e) =>
- {
- Assert.Equal(index++, e.Index);
- result.Add(e.Container);
- };
- target.Items.Add(new Button());
- target.Items.Add(new Button());
- target.Items.Add(new Button());
- Assert.Equal(3, result.Count);
- Assert.Equal(target.GetRealizedContainers(), result);
- }
- [Fact]
- public void ContainerPrepared_Is_Raised_For_Each_Item_Container()
- {
- using var app = Start();
- var items = new AvaloniaList<string>();
- var target = CreateTarget();
- var result = new List<Control>();
- var index = 0;
- target.ContainerPrepared += (s, e) =>
- {
- Assert.Equal(index++, e.Index);
- result.Add(e.Container);
- };
- target.Items.Add("Foo");
- target.Items.Add("Bar");
- target.Items.Add("Baz");
- Assert.Equal(3, result.Count);
- Assert.Equal(target.GetRealizedContainers(), result);
- }
- [Fact]
- public void ContainerPrepared_Is_Raised_For_Each_ItemsSource_Item_Container_On_Layout()
- {
- using var app = Start();
- var items = new AvaloniaList<string>();
- var target = CreateTarget(itemsSource: items);
- var result = new List<Control>();
- var index = 0;
- target.ContainerPrepared += (s, e) =>
- {
- Assert.Equal(index++, e.Index);
- result.Add(e.Container);
- };
- items.AddRange(new[] { "Foo", "Bar", "Baz" });
- Assert.Equal(3, result.Count);
- Assert.Equal(target.GetRealizedContainers(), result);
- }
- [Fact]
- public void ContainerIndexChanged_Is_Raised_When_Item_Added()
- {
- using var app = Start();
- var target = CreateTarget(items: new[] { "Foo", "Bar", "Baz" });
- var result = new List<Control>();
- var index = 1;
- target.ContainerIndexChanged += (s, e) =>
- {
- Assert.Equal(index++, e.OldIndex);
- Assert.Equal(index, e.NewIndex);
- result.Add(e.Container);
- };
- target.Items.Insert(1, "Qux");
- Assert.Equal(2, result.Count);
- Assert.Equal(target.GetRealizedContainers().Skip(2), result);
- }
- [Fact]
- public void ContainerClearing_Is_Raised_When_Item_Removed()
- {
- using var app = Start();
- var target = CreateTarget(items: new[] { "Foo", "Bar", "Baz" });
- var expected = target.ContainerFromIndex(1);
- var raised = 0;
- target.ContainerClearing += (s, e) =>
- {
- Assert.Same(expected, e.Container);
- ++raised;
- };
- target.Items.RemoveAt(1);
- Assert.Equal(1, raised);
- }
- [Fact]
- public void Handles_Recycling_Control_Items_Inside_Containers()
- {
- // Issue #10825
- using var app = Start();
- // The items must be controls but not of the container type.
- var items = Enumerable.Range(0, 100).Select(x => new TextBlock
- {
- Text = $"Item {x}",
- Width = 100,
- Height = 100,
- }).ToList();
- // Virtualization is required
- var itemsPanel = new FuncTemplate<Panel?>(() => new VirtualizingStackPanel());
- // Create an ItemsControl which uses containers, and provide a scroll viewer.
- var target = CreateTarget<ItemsControlWithContainer>(
- items: items,
- itemsPanel: itemsPanel,
- scrollViewer: true);
- var scroll = target.FindAncestorOfType<ScrollViewer>();
- Assert.NotNull(scroll);
- Assert.Equal(10, target.GetRealizedContainers().Count());
- // Scroll so that half a container is visible: an extra container is generated.
- scroll.Offset = new(0, 2050);
- Layout(target);
- // Scroll so that the extra container is no longer needed and recycled.
- scroll.Offset = new(0, 2100);
- Layout(target);
- // Scroll back: issue #10825 triggered.
- scroll.Offset = new(0, 2000);
- Layout(target);
- }
- [Fact]
- public void ItemIsOwnContainer_Content_Should_Not_Be_Cleared_When_Removed()
- {
- // Issue #11128.
- using var app = Start();
- var item = new ContentPresenter { Content = "foo" };
- var target = CreateTarget(items: new[] { item });
- target.Items.RemoveAt(0);
- Assert.Equal("foo", item.Content);
- }
- private static ItemsControl CreateTarget(
- object? dataContext = null,
- IBinding? displayMemberBinding = null,
- IList? items = null,
- IList? itemsSource = null,
- ControlTheme? itemContainerTheme = null,
- IDataTemplate? itemTemplate = null,
- IEnumerable<IDataTemplate>? dataTemplates = null,
- bool performLayout = true)
- {
- return CreateTarget<ItemsControl>(
- dataContext: dataContext,
- displayMemberBinding: displayMemberBinding,
- items: items,
- itemsSource: itemsSource,
- itemContainerTheme: itemContainerTheme,
- itemTemplate: itemTemplate,
- dataTemplates: dataTemplates,
- performLayout: performLayout);
- }
- private static T CreateTarget<T>(
- object? dataContext = null,
- IBinding? displayMemberBinding = null,
- IList? items = null,
- IList? itemsSource = null,
- ControlTheme? itemContainerTheme = null,
- IDataTemplate? itemTemplate = null,
- ITemplate<Panel?>? itemsPanel = null,
- IEnumerable<IDataTemplate>? dataTemplates = null,
- bool performLayout = true,
- bool scrollViewer = false)
- where T : ItemsControl, new()
- {
- var target = new T
- {
- DataContext = dataContext,
- DisplayMemberBinding = displayMemberBinding,
- ItemContainerTheme = itemContainerTheme,
- ItemTemplate = itemTemplate,
- ItemsSource = itemsSource,
- };
- if (items is not null)
- {
- foreach (var item in items)
- target.Items.Add(item);
- }
- if (itemsPanel is not null)
- target.ItemsPanel = itemsPanel;
- var scroll = scrollViewer ? new ScrollViewer { Content = target } : null;
- var root = CreateRoot(scroll ?? (Control)target);
- if (dataTemplates is not null)
- {
- foreach (var dataTemplate in dataTemplates)
- root.DataTemplates.Add(dataTemplate);
- }
- if (performLayout)
- root.LayoutManager.ExecuteInitialLayoutPass();
- return target;
- }
- private static TestRoot CreateRoot(Control child)
- {
- return new TestRoot
- {
- Resources =
- {
- { typeof(ContentControl), CreateContentControlTheme() },
- { typeof(ItemsControl), CreateItemsControlTheme() },
- { typeof(ScrollViewer), CreateScrollViewerTheme() },
- },
- Child = child,
- };
- }
- private static ControlTheme CreateContentControlTheme()
- {
- return new ControlTheme(typeof(ContentControl))
- {
- Setters =
- {
- new Setter(TreeView.TemplateProperty, CreateContentControlTemplate()),
- },
- };
- }
- private static FuncControlTemplate CreateContentControlTemplate()
- {
- return new FuncControlTemplate<ContentControl>((parent, scope) =>
- new ContentPresenter
- {
- Name = "PART_ContentPresenter",
- [!ContentPresenter.ContentProperty] = parent[!ListBoxItem.ContentProperty],
- [!ContentPresenter.ContentTemplateProperty] = parent[!ListBoxItem.ContentTemplateProperty],
- }.RegisterInNameScope(scope));
- }
- private static ControlTheme CreateItemsControlTheme()
- {
- return new ControlTheme(typeof(ItemsControl))
- {
- Setters =
- {
- new Setter(TreeView.TemplateProperty, CreateItemsControlTemplate()),
- },
- };
- }
- private static FuncControlTemplate CreateItemsControlTemplate()
- {
- return new FuncControlTemplate<ItemsControl>((parent, scope) =>
- {
- return new Border
- {
- Background = new Media.SolidColorBrush(0xffffffff),
- Child = new ItemsPresenter
- {
- Name = "PART_ItemsPresenter",
- [~ItemsPresenter.ItemsPanelProperty] = parent[~ItemsControl.ItemsPanelProperty],
- }.RegisterInNameScope(scope)
- };
- });
- }
- private static ControlTheme CreateScrollViewerTheme()
- {
- return new ControlTheme(typeof(ScrollViewer))
- {
- Setters =
- {
- new Setter(TreeView.TemplateProperty, CreateScrollViewerTemplate()),
- },
- };
- }
- private static FuncControlTemplate CreateScrollViewerTemplate()
- {
- return new FuncControlTemplate<ScrollViewer>((parent, scope) =>
- new Panel
- {
- Children =
- {
- new ScrollContentPresenter
- {
- Name = "PART_ContentPresenter",
- [~ScrollContentPresenter.ContentProperty] = parent.GetObservable(ScrollViewer.ContentProperty).ToBinding(),
- }.RegisterInNameScope(scope),
- new ScrollBar
- {
- Name = "verticalScrollBar",
- }
- }
- });
- }
- private static void Layout(Control c)
- {
- (c.GetVisualRoot() as ILayoutRoot)?.LayoutManager.ExecuteLayoutPass();
- }
- private static ContentPresenter GetContainer(ItemsControl target, int index = 0)
- {
- return Assert.IsType<ContentPresenter>(target.GetRealizedContainers().ElementAt(index));
- }
- private static T GetContainer<T>(ItemsControl target, int index = 0)
- {
- return Assert.IsType<T>(target.GetRealizedContainers().ElementAt(index));
- }
- public static IDisposable Start()
- {
- return UnitTestApplication.Start(
- TestServices.MockThreadingInterface.With(
- focusManager: new FocusManager(),
- fontManagerImpl: new MockFontManagerImpl(),
- keyboardDevice: () => new KeyboardDevice(),
- keyboardNavigation: new KeyboardNavigationHandler(),
- inputManager: new InputManager(),
- renderInterface: new MockPlatformRenderInterface(),
- textShaperImpl: new MockTextShaperImpl()));
- }
- private class ItemsControlWithContainer : ItemsControl, IStyleable
- {
- Type IStyleable.StyleKey => typeof(ItemsControl);
- protected internal override Control CreateContainerForItemOverride()
- {
- return new ContainerControl();
- }
- protected internal override bool IsItemItsOwnContainerOverride(Control item)
- {
- return item is ContainerControl;
- }
- }
- private class ContainerControl : ContentControl, IStyleable
- {
- Type IStyleable.StyleKey => typeof(ContentControl);
- }
- private record Item(string Caption, string? Value = null);
- }
- }
|