| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058 | 
							- 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.Headless;
 
- 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);
 
-             var focusManager = ((IInputRoot)target.VisualRoot!).FocusManager;
 
-             Assert.Equal(panel.Children[1], focusManager?.GetFocusedElement());
 
-         }
 
-         [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);
 
-             var focusManager = ((IInputRoot)target.VisualRoot!).FocusManager;
 
-             Assert.Equal(panel.Children[2], focusManager?.GetFocusedElement());
 
-         }
 
-         [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 HeadlessFontManagerStub(),
 
-                     keyboardDevice: () => new KeyboardDevice(),
 
-                     keyboardNavigation: new KeyboardNavigationHandler(),
 
-                     inputManager: new InputManager(),
 
-                     renderInterface: new HeadlessPlatformRenderInterface(),
 
-                     textShaperImpl: new HeadlessTextShaperStub()));
 
-         }
 
-         private class ItemsControlWithContainer : ItemsControl
 
-         {
 
-             protected override Type StyleKeyOverride => typeof(ItemsControl);
 
-             protected internal override Control CreateContainerForItemOverride(object? item, int index, object? recycleKey)
 
-             {
 
-                 return new ContainerControl();
 
-             }
 
-             protected internal override bool NeedsContainerOverride(object? item, int index, out object? recycleKey)
 
-             {
 
-                 return NeedsContainer<ContainerControl>(item, out recycleKey);
 
-             }
 
-         }
 
-         private class ContainerControl : ContentControl
 
-         {
 
-             protected override Type StyleKeyOverride => typeof(ContentControl);
 
-         }
 
-         private record Item(string Caption, string? Value = null);
 
-     }
 
- }
 
 
  |